13cab2bb3Spatrick /*===- GCDAProfiling.c - Support library for GCDA file emission -----------===*\
23cab2bb3Spatrick |*
33cab2bb3Spatrick |* Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
43cab2bb3Spatrick |* See https://llvm.org/LICENSE.txt for license information.
53cab2bb3Spatrick |* SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
63cab2bb3Spatrick |*
73cab2bb3Spatrick |*===----------------------------------------------------------------------===*|
83cab2bb3Spatrick |*
93cab2bb3Spatrick |* This file implements the call back routines for the gcov profiling
103cab2bb3Spatrick |* instrumentation pass. Link against this library when running code through
113cab2bb3Spatrick |* the -insert-gcov-profiling LLVM pass.
123cab2bb3Spatrick |*
133cab2bb3Spatrick |* We emit files in a corrupt version of GCOV's "gcda" file format. These files
143cab2bb3Spatrick |* are only close enough that LCOV will happily parse them. Anything that lcov
153cab2bb3Spatrick |* ignores is missing.
163cab2bb3Spatrick |*
173cab2bb3Spatrick |* TODO: gcov is multi-process safe by having each exit open the existing file
183cab2bb3Spatrick |* and append to it. We'd like to achieve that and be thread-safe too.
193cab2bb3Spatrick |*
203cab2bb3Spatrick \*===----------------------------------------------------------------------===*/
213cab2bb3Spatrick
223cab2bb3Spatrick #if !defined(__Fuchsia__)
233cab2bb3Spatrick
243cab2bb3Spatrick #include <errno.h>
253cab2bb3Spatrick #include <fcntl.h>
26d89ec533Spatrick #include <stdint.h>
273cab2bb3Spatrick #include <stdio.h>
283cab2bb3Spatrick #include <stdlib.h>
293cab2bb3Spatrick #include <string.h>
303cab2bb3Spatrick
313cab2bb3Spatrick #if defined(_WIN32)
323cab2bb3Spatrick #define WIN32_LEAN_AND_MEAN
333cab2bb3Spatrick #include <windows.h>
343cab2bb3Spatrick #include "WindowsMMap.h"
353cab2bb3Spatrick #else
363cab2bb3Spatrick #include <sys/file.h>
373cab2bb3Spatrick #include <sys/mman.h>
383cab2bb3Spatrick #include <sys/types.h>
393cab2bb3Spatrick #include <unistd.h>
403cab2bb3Spatrick #endif
413cab2bb3Spatrick
423cab2bb3Spatrick #include "InstrProfiling.h"
433cab2bb3Spatrick #include "InstrProfilingUtil.h"
443cab2bb3Spatrick
453cab2bb3Spatrick /* #define DEBUG_GCDAPROFILING */
463cab2bb3Spatrick
471f9cb04fSpatrick enum {
481f9cb04fSpatrick GCOV_DATA_MAGIC = 0x67636461, // "gcda"
491f9cb04fSpatrick
501f9cb04fSpatrick GCOV_TAG_FUNCTION = 0x01000000,
511f9cb04fSpatrick GCOV_TAG_COUNTER_ARCS = 0x01a10000,
521f9cb04fSpatrick // GCOV_TAG_OBJECT_SUMMARY superseded GCOV_TAG_PROGRAM_SUMMARY in GCC 9.
531f9cb04fSpatrick GCOV_TAG_OBJECT_SUMMARY = 0xa1000000,
541f9cb04fSpatrick GCOV_TAG_PROGRAM_SUMMARY = 0xa3000000,
551f9cb04fSpatrick };
561f9cb04fSpatrick
573cab2bb3Spatrick /*
583cab2bb3Spatrick * --- GCOV file format I/O primitives ---
593cab2bb3Spatrick */
603cab2bb3Spatrick
613cab2bb3Spatrick /*
623cab2bb3Spatrick * The current file name we're outputting. Used primarily for error logging.
633cab2bb3Spatrick */
643cab2bb3Spatrick static char *filename = NULL;
653cab2bb3Spatrick
663cab2bb3Spatrick /*
673cab2bb3Spatrick * The current file we're outputting.
683cab2bb3Spatrick */
693cab2bb3Spatrick static FILE *output_file = NULL;
703cab2bb3Spatrick
713cab2bb3Spatrick /*
723cab2bb3Spatrick * Buffer that we write things into.
733cab2bb3Spatrick */
743cab2bb3Spatrick #define WRITE_BUFFER_SIZE (128 * 1024)
753cab2bb3Spatrick static unsigned char *write_buffer = NULL;
763cab2bb3Spatrick static uint64_t cur_buffer_size = 0;
773cab2bb3Spatrick static uint64_t cur_pos = 0;
783cab2bb3Spatrick static uint64_t file_size = 0;
793cab2bb3Spatrick static int new_file = 0;
801f9cb04fSpatrick static int gcov_version;
813cab2bb3Spatrick #if defined(_WIN32)
823cab2bb3Spatrick static HANDLE mmap_handle = NULL;
833cab2bb3Spatrick #endif
843cab2bb3Spatrick static int fd = -1;
853cab2bb3Spatrick
86*810390e3Srobert typedef void (*fn_ptr)(void);
873cab2bb3Spatrick
883cab2bb3Spatrick typedef void* dynamic_object_id;
893cab2bb3Spatrick // The address of this variable identifies a given dynamic object.
903cab2bb3Spatrick static dynamic_object_id current_id;
913cab2bb3Spatrick #define CURRENT_ID (¤t_id)
923cab2bb3Spatrick
933cab2bb3Spatrick struct fn_node {
943cab2bb3Spatrick dynamic_object_id id;
953cab2bb3Spatrick fn_ptr fn;
963cab2bb3Spatrick struct fn_node* next;
973cab2bb3Spatrick };
983cab2bb3Spatrick
993cab2bb3Spatrick struct fn_list {
1003cab2bb3Spatrick struct fn_node *head, *tail;
1013cab2bb3Spatrick };
1023cab2bb3Spatrick
1033cab2bb3Spatrick /*
1043cab2bb3Spatrick * A list of functions to write out the data, shared between all dynamic objects.
1053cab2bb3Spatrick */
1063cab2bb3Spatrick struct fn_list writeout_fn_list;
1073cab2bb3Spatrick
1083cab2bb3Spatrick /*
1093cab2bb3Spatrick * A list of reset functions, shared between all dynamic objects.
1103cab2bb3Spatrick */
1113cab2bb3Spatrick struct fn_list reset_fn_list;
1123cab2bb3Spatrick
fn_list_insert(struct fn_list * list,fn_ptr fn)1133cab2bb3Spatrick static void fn_list_insert(struct fn_list* list, fn_ptr fn) {
1143cab2bb3Spatrick struct fn_node* new_node = malloc(sizeof(struct fn_node));
1153cab2bb3Spatrick new_node->fn = fn;
1163cab2bb3Spatrick new_node->next = NULL;
1173cab2bb3Spatrick new_node->id = CURRENT_ID;
1183cab2bb3Spatrick
1193cab2bb3Spatrick if (!list->head) {
1203cab2bb3Spatrick list->head = list->tail = new_node;
1213cab2bb3Spatrick } else {
1223cab2bb3Spatrick list->tail->next = new_node;
1233cab2bb3Spatrick list->tail = new_node;
1243cab2bb3Spatrick }
1253cab2bb3Spatrick }
1263cab2bb3Spatrick
fn_list_remove(struct fn_list * list)1273cab2bb3Spatrick static void fn_list_remove(struct fn_list* list) {
1283cab2bb3Spatrick struct fn_node* curr = list->head;
1293cab2bb3Spatrick struct fn_node* prev = NULL;
1303cab2bb3Spatrick struct fn_node* next = NULL;
1313cab2bb3Spatrick
1323cab2bb3Spatrick while (curr) {
1333cab2bb3Spatrick next = curr->next;
1343cab2bb3Spatrick
1353cab2bb3Spatrick if (curr->id == CURRENT_ID) {
1363cab2bb3Spatrick if (curr == list->head) {
1373cab2bb3Spatrick list->head = next;
1383cab2bb3Spatrick }
1393cab2bb3Spatrick
1403cab2bb3Spatrick if (curr == list->tail) {
1413cab2bb3Spatrick list->tail = prev;
1423cab2bb3Spatrick }
1433cab2bb3Spatrick
1443cab2bb3Spatrick if (prev) {
1453cab2bb3Spatrick prev->next = next;
1463cab2bb3Spatrick }
1473cab2bb3Spatrick
1483cab2bb3Spatrick free(curr);
1493cab2bb3Spatrick } else {
1503cab2bb3Spatrick prev = curr;
1513cab2bb3Spatrick }
1523cab2bb3Spatrick
1533cab2bb3Spatrick curr = next;
1543cab2bb3Spatrick }
1553cab2bb3Spatrick }
1563cab2bb3Spatrick
resize_write_buffer(uint64_t size)1573cab2bb3Spatrick static void resize_write_buffer(uint64_t size) {
1583cab2bb3Spatrick if (!new_file) return;
1593cab2bb3Spatrick size += cur_pos;
1603cab2bb3Spatrick if (size <= cur_buffer_size) return;
1613cab2bb3Spatrick size = (size - 1) / WRITE_BUFFER_SIZE + 1;
1623cab2bb3Spatrick size *= WRITE_BUFFER_SIZE;
1633cab2bb3Spatrick write_buffer = realloc(write_buffer, size);
1643cab2bb3Spatrick cur_buffer_size = size;
1653cab2bb3Spatrick }
1663cab2bb3Spatrick
write_bytes(const char * s,size_t len)1673cab2bb3Spatrick static void write_bytes(const char *s, size_t len) {
1683cab2bb3Spatrick resize_write_buffer(len);
1693cab2bb3Spatrick memcpy(&write_buffer[cur_pos], s, len);
1703cab2bb3Spatrick cur_pos += len;
1713cab2bb3Spatrick }
1723cab2bb3Spatrick
write_32bit_value(uint32_t i)1733cab2bb3Spatrick static void write_32bit_value(uint32_t i) {
1743cab2bb3Spatrick write_bytes((char*)&i, 4);
1753cab2bb3Spatrick }
1763cab2bb3Spatrick
write_64bit_value(uint64_t i)1773cab2bb3Spatrick static void write_64bit_value(uint64_t i) {
1783cab2bb3Spatrick // GCOV uses a lo-/hi-word format even on big-endian systems.
1793cab2bb3Spatrick // See also GCOVBuffer::readInt64 in LLVM.
1803cab2bb3Spatrick uint32_t lo = (uint32_t) i;
1813cab2bb3Spatrick uint32_t hi = (uint32_t) (i >> 32);
1823cab2bb3Spatrick write_32bit_value(lo);
1833cab2bb3Spatrick write_32bit_value(hi);
1843cab2bb3Spatrick }
1853cab2bb3Spatrick
read_32bit_value(void)186*810390e3Srobert static uint32_t read_32bit_value(void) {
1873cab2bb3Spatrick uint32_t val;
1883cab2bb3Spatrick
1893cab2bb3Spatrick if (new_file)
1903cab2bb3Spatrick return (uint32_t)-1;
1913cab2bb3Spatrick
1923cab2bb3Spatrick val = *(uint32_t*)&write_buffer[cur_pos];
1933cab2bb3Spatrick cur_pos += 4;
1943cab2bb3Spatrick return val;
1953cab2bb3Spatrick }
1963cab2bb3Spatrick
read_64bit_value(void)197*810390e3Srobert static uint64_t read_64bit_value(void) {
1983cab2bb3Spatrick // GCOV uses a lo-/hi-word format even on big-endian systems.
1993cab2bb3Spatrick // See also GCOVBuffer::readInt64 in LLVM.
2003cab2bb3Spatrick uint32_t lo = read_32bit_value();
2013cab2bb3Spatrick uint32_t hi = read_32bit_value();
2023cab2bb3Spatrick return ((uint64_t)hi << 32) | ((uint64_t)lo);
2033cab2bb3Spatrick }
2043cab2bb3Spatrick
mangle_filename(const char * orig_filename)2053cab2bb3Spatrick static char *mangle_filename(const char *orig_filename) {
2063cab2bb3Spatrick char *new_filename;
2073cab2bb3Spatrick size_t prefix_len;
2083cab2bb3Spatrick int prefix_strip;
2093cab2bb3Spatrick const char *prefix = lprofGetPathPrefix(&prefix_strip, &prefix_len);
2103cab2bb3Spatrick
2113cab2bb3Spatrick if (prefix == NULL)
2123cab2bb3Spatrick return strdup(orig_filename);
2133cab2bb3Spatrick
2143cab2bb3Spatrick new_filename = malloc(prefix_len + 1 + strlen(orig_filename) + 1);
2153cab2bb3Spatrick lprofApplyPathPrefix(new_filename, orig_filename, prefix, prefix_len,
2163cab2bb3Spatrick prefix_strip);
2173cab2bb3Spatrick
2183cab2bb3Spatrick return new_filename;
2193cab2bb3Spatrick }
2203cab2bb3Spatrick
map_file(void)221*810390e3Srobert static int map_file(void) {
2223cab2bb3Spatrick fseek(output_file, 0L, SEEK_END);
2233cab2bb3Spatrick file_size = ftell(output_file);
2243cab2bb3Spatrick
2251f9cb04fSpatrick /* A size of 0 means the file has been created just now (possibly by another
2261f9cb04fSpatrick * process in lock-after-open race condition). No need to mmap. */
2273cab2bb3Spatrick if (file_size == 0)
2283cab2bb3Spatrick return -1;
2293cab2bb3Spatrick
2303cab2bb3Spatrick #if defined(_WIN32)
2313cab2bb3Spatrick HANDLE mmap_fd;
2323cab2bb3Spatrick if (fd == -1)
2333cab2bb3Spatrick mmap_fd = INVALID_HANDLE_VALUE;
2343cab2bb3Spatrick else
2353cab2bb3Spatrick mmap_fd = (HANDLE)_get_osfhandle(fd);
2363cab2bb3Spatrick
2373cab2bb3Spatrick mmap_handle = CreateFileMapping(mmap_fd, NULL, PAGE_READWRITE, DWORD_HI(file_size), DWORD_LO(file_size), NULL);
2383cab2bb3Spatrick if (mmap_handle == NULL) {
2393cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot create file mapping: %lu\n",
2403cab2bb3Spatrick filename, GetLastError());
2413cab2bb3Spatrick return -1;
2423cab2bb3Spatrick }
2433cab2bb3Spatrick
2443cab2bb3Spatrick write_buffer = MapViewOfFile(mmap_handle, FILE_MAP_WRITE, 0, 0, file_size);
2453cab2bb3Spatrick if (write_buffer == NULL) {
2463cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot map: %lu\n", filename,
2473cab2bb3Spatrick GetLastError());
2483cab2bb3Spatrick CloseHandle(mmap_handle);
2493cab2bb3Spatrick return -1;
2503cab2bb3Spatrick }
2513cab2bb3Spatrick #else
2523cab2bb3Spatrick write_buffer = mmap(0, file_size, PROT_READ | PROT_WRITE,
2533cab2bb3Spatrick MAP_FILE | MAP_SHARED, fd, 0);
2543cab2bb3Spatrick if (write_buffer == (void *)-1) {
2553cab2bb3Spatrick int errnum = errno;
2563cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot map: %s\n", filename,
2573cab2bb3Spatrick strerror(errnum));
2583cab2bb3Spatrick return -1;
2593cab2bb3Spatrick }
2603cab2bb3Spatrick #endif
2613cab2bb3Spatrick
2623cab2bb3Spatrick return 0;
2633cab2bb3Spatrick }
2643cab2bb3Spatrick
unmap_file(void)265*810390e3Srobert static void unmap_file(void) {
2663cab2bb3Spatrick #if defined(_WIN32)
2673cab2bb3Spatrick if (!UnmapViewOfFile(write_buffer)) {
2683cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot unmap mapped view: %lu\n", filename,
2693cab2bb3Spatrick GetLastError());
2703cab2bb3Spatrick }
2713cab2bb3Spatrick
2723cab2bb3Spatrick if (!CloseHandle(mmap_handle)) {
2733cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot close file mapping handle: %lu\n",
2743cab2bb3Spatrick filename, GetLastError());
2753cab2bb3Spatrick }
2763cab2bb3Spatrick
2773cab2bb3Spatrick mmap_handle = NULL;
2783cab2bb3Spatrick #else
279d89ec533Spatrick if (munmap(write_buffer, file_size) == -1) {
2803cab2bb3Spatrick int errnum = errno;
281d89ec533Spatrick fprintf(stderr, "profiling: %s: cannot munmap: %s\n", filename,
2823cab2bb3Spatrick strerror(errnum));
2833cab2bb3Spatrick }
2843cab2bb3Spatrick #endif
2853cab2bb3Spatrick
2863cab2bb3Spatrick write_buffer = NULL;
2873cab2bb3Spatrick file_size = 0;
2883cab2bb3Spatrick }
2893cab2bb3Spatrick
2903cab2bb3Spatrick /*
2913cab2bb3Spatrick * --- LLVM line counter API ---
2923cab2bb3Spatrick */
2933cab2bb3Spatrick
2943cab2bb3Spatrick /* A file in this case is a translation unit. Each .o file built with line
2953cab2bb3Spatrick * profiling enabled will emit to a different file. Only one file may be
2963cab2bb3Spatrick * started at a time.
2973cab2bb3Spatrick */
2983cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_gcda_start_file(const char * orig_filename,uint32_t version,uint32_t checksum)2991f9cb04fSpatrick void llvm_gcda_start_file(const char *orig_filename, uint32_t version,
3003cab2bb3Spatrick uint32_t checksum) {
3013cab2bb3Spatrick const char *mode = "r+b";
3023cab2bb3Spatrick filename = mangle_filename(orig_filename);
3033cab2bb3Spatrick
3043cab2bb3Spatrick /* Try just opening the file. */
3053cab2bb3Spatrick fd = open(filename, O_RDWR | O_BINARY);
3063cab2bb3Spatrick
3073cab2bb3Spatrick if (fd == -1) {
3081f9cb04fSpatrick /* Try creating the file. */
3091f9cb04fSpatrick fd = open(filename, O_RDWR | O_CREAT | O_EXCL | O_BINARY, 0644);
3101f9cb04fSpatrick if (fd != -1) {
3113cab2bb3Spatrick mode = "w+b";
3121f9cb04fSpatrick } else {
3133cab2bb3Spatrick /* Try creating the directories first then opening the file. */
3143cab2bb3Spatrick __llvm_profile_recursive_mkdir(filename);
3151f9cb04fSpatrick fd = open(filename, O_RDWR | O_CREAT | O_EXCL | O_BINARY, 0644);
3161f9cb04fSpatrick if (fd != -1) {
3171f9cb04fSpatrick mode = "w+b";
3181f9cb04fSpatrick } else {
3191f9cb04fSpatrick /* Another process may have created the file just now.
3201f9cb04fSpatrick * Try opening it without O_CREAT and O_EXCL. */
3211f9cb04fSpatrick fd = open(filename, O_RDWR | O_BINARY);
3223cab2bb3Spatrick if (fd == -1) {
3233cab2bb3Spatrick /* Bah! It's hopeless. */
3243cab2bb3Spatrick int errnum = errno;
3253cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot open: %s\n", filename,
3263cab2bb3Spatrick strerror(errnum));
3273cab2bb3Spatrick return;
3283cab2bb3Spatrick }
3293cab2bb3Spatrick }
3303cab2bb3Spatrick }
3311f9cb04fSpatrick }
3323cab2bb3Spatrick
3333cab2bb3Spatrick /* Try to flock the file to serialize concurrent processes writing out to the
3343cab2bb3Spatrick * same GCDA. This can fail if the filesystem doesn't support it, but in that
3353cab2bb3Spatrick * case we'll just carry on with the old racy behaviour and hope for the best.
3363cab2bb3Spatrick */
3373cab2bb3Spatrick lprofLockFd(fd);
3383cab2bb3Spatrick output_file = fdopen(fd, mode);
3393cab2bb3Spatrick
3403cab2bb3Spatrick /* Initialize the write buffer. */
3411f9cb04fSpatrick new_file = 0;
3423cab2bb3Spatrick write_buffer = NULL;
3433cab2bb3Spatrick cur_buffer_size = 0;
3443cab2bb3Spatrick cur_pos = 0;
3453cab2bb3Spatrick
3463cab2bb3Spatrick if (map_file() == -1) {
3471f9cb04fSpatrick /* The file has been created just now (file_size == 0) or mmap failed
3481f9cb04fSpatrick * unexpectedly. In the latter case, try to recover by clobbering. */
3493cab2bb3Spatrick new_file = 1;
3503cab2bb3Spatrick write_buffer = NULL;
3513cab2bb3Spatrick resize_write_buffer(WRITE_BUFFER_SIZE);
3523cab2bb3Spatrick memset(write_buffer, 0, WRITE_BUFFER_SIZE);
3533cab2bb3Spatrick }
3543cab2bb3Spatrick
3553cab2bb3Spatrick /* gcda file, version, stamp checksum. */
3561f9cb04fSpatrick {
3571f9cb04fSpatrick uint8_t c3 = version >> 24;
3581f9cb04fSpatrick uint8_t c2 = (version >> 16) & 255;
3591f9cb04fSpatrick uint8_t c1 = (version >> 8) & 255;
3601f9cb04fSpatrick gcov_version = c3 >= 'A' ? (c3 - 'A') * 100 + (c2 - '0') * 10 + c1 - '0'
3611f9cb04fSpatrick : (c3 - '0') * 10 + c1 - '0';
3621f9cb04fSpatrick }
3631f9cb04fSpatrick write_32bit_value(GCOV_DATA_MAGIC);
3641f9cb04fSpatrick write_32bit_value(version);
3653cab2bb3Spatrick write_32bit_value(checksum);
3663cab2bb3Spatrick
3673cab2bb3Spatrick #ifdef DEBUG_GCDAPROFILING
3683cab2bb3Spatrick fprintf(stderr, "llvmgcda: [%s]\n", orig_filename);
3693cab2bb3Spatrick #endif
3703cab2bb3Spatrick }
3713cab2bb3Spatrick
3723cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_gcda_emit_function(uint32_t ident,uint32_t func_checksum,uint32_t cfg_checksum)3731f9cb04fSpatrick void llvm_gcda_emit_function(uint32_t ident, uint32_t func_checksum,
3743cab2bb3Spatrick uint32_t cfg_checksum) {
3753cab2bb3Spatrick uint32_t len = 2;
3761f9cb04fSpatrick int use_extra_checksum = gcov_version >= 47;
3773cab2bb3Spatrick
3783cab2bb3Spatrick if (use_extra_checksum)
3793cab2bb3Spatrick len++;
3803cab2bb3Spatrick #ifdef DEBUG_GCDAPROFILING
3811f9cb04fSpatrick fprintf(stderr, "llvmgcda: function id=0x%08x\n", ident);
3823cab2bb3Spatrick #endif
3833cab2bb3Spatrick if (!output_file) return;
3843cab2bb3Spatrick
3853cab2bb3Spatrick /* function tag */
3861f9cb04fSpatrick write_32bit_value(GCOV_TAG_FUNCTION);
3873cab2bb3Spatrick write_32bit_value(len);
3883cab2bb3Spatrick write_32bit_value(ident);
3893cab2bb3Spatrick write_32bit_value(func_checksum);
3903cab2bb3Spatrick if (use_extra_checksum)
3913cab2bb3Spatrick write_32bit_value(cfg_checksum);
3923cab2bb3Spatrick }
3933cab2bb3Spatrick
3943cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_gcda_emit_arcs(uint32_t num_counters,uint64_t * counters)3953cab2bb3Spatrick void llvm_gcda_emit_arcs(uint32_t num_counters, uint64_t *counters) {
3963cab2bb3Spatrick uint32_t i;
3973cab2bb3Spatrick uint64_t *old_ctrs = NULL;
3983cab2bb3Spatrick uint32_t val = 0;
3993cab2bb3Spatrick uint64_t save_cur_pos = cur_pos;
4003cab2bb3Spatrick
4013cab2bb3Spatrick if (!output_file) return;
4023cab2bb3Spatrick
4031f9cb04fSpatrick val = read_32bit_value();
4043cab2bb3Spatrick
4053cab2bb3Spatrick if (val != (uint32_t)-1) {
4063cab2bb3Spatrick /* There are counters present in the file. Merge them. */
4071f9cb04fSpatrick if (val != GCOV_TAG_COUNTER_ARCS) {
4083cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot merge previous GCDA file: "
4093cab2bb3Spatrick "corrupt arc tag (0x%08x)\n",
4103cab2bb3Spatrick filename, val);
4113cab2bb3Spatrick return;
4123cab2bb3Spatrick }
4133cab2bb3Spatrick
4143cab2bb3Spatrick val = read_32bit_value();
4153cab2bb3Spatrick if (val == (uint32_t)-1 || val / 2 != num_counters) {
4163cab2bb3Spatrick fprintf(stderr, "profiling: %s: cannot merge previous GCDA file: "
4173cab2bb3Spatrick "mismatched number of counters (%d)\n",
4183cab2bb3Spatrick filename, val);
4193cab2bb3Spatrick return;
4203cab2bb3Spatrick }
4213cab2bb3Spatrick
4223cab2bb3Spatrick old_ctrs = malloc(sizeof(uint64_t) * num_counters);
4233cab2bb3Spatrick for (i = 0; i < num_counters; ++i)
4243cab2bb3Spatrick old_ctrs[i] = read_64bit_value();
4253cab2bb3Spatrick }
4263cab2bb3Spatrick
4273cab2bb3Spatrick cur_pos = save_cur_pos;
4283cab2bb3Spatrick
4293cab2bb3Spatrick /* Counter #1 (arcs) tag */
4301f9cb04fSpatrick write_32bit_value(GCOV_TAG_COUNTER_ARCS);
4313cab2bb3Spatrick write_32bit_value(num_counters * 2);
4323cab2bb3Spatrick for (i = 0; i < num_counters; ++i) {
4333cab2bb3Spatrick counters[i] += (old_ctrs ? old_ctrs[i] : 0);
4343cab2bb3Spatrick write_64bit_value(counters[i]);
4353cab2bb3Spatrick }
4363cab2bb3Spatrick
4373cab2bb3Spatrick free(old_ctrs);
4383cab2bb3Spatrick
4393cab2bb3Spatrick #ifdef DEBUG_GCDAPROFILING
4403cab2bb3Spatrick fprintf(stderr, "llvmgcda: %u arcs\n", num_counters);
4413cab2bb3Spatrick for (i = 0; i < num_counters; ++i)
4423cab2bb3Spatrick fprintf(stderr, "llvmgcda: %llu\n", (unsigned long long)counters[i]);
4433cab2bb3Spatrick #endif
4443cab2bb3Spatrick }
4453cab2bb3Spatrick
4463cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_gcda_summary_info(void)447*810390e3Srobert void llvm_gcda_summary_info(void) {
4483cab2bb3Spatrick uint32_t runs = 1;
4493cab2bb3Spatrick static uint32_t run_counted = 0; // We only want to increase the run count once.
4503cab2bb3Spatrick uint32_t val = 0;
4513cab2bb3Spatrick uint64_t save_cur_pos = cur_pos;
4523cab2bb3Spatrick
4533cab2bb3Spatrick if (!output_file) return;
4543cab2bb3Spatrick
4551f9cb04fSpatrick val = read_32bit_value();
4563cab2bb3Spatrick
4573cab2bb3Spatrick if (val != (uint32_t)-1) {
4583cab2bb3Spatrick /* There are counters present in the file. Merge them. */
459d89ec533Spatrick uint32_t gcov_tag =
460d89ec533Spatrick gcov_version >= 90 ? GCOV_TAG_OBJECT_SUMMARY : GCOV_TAG_PROGRAM_SUMMARY;
461d89ec533Spatrick if (val != gcov_tag) {
4621f9cb04fSpatrick fprintf(stderr,
4631f9cb04fSpatrick "profiling: %s: cannot merge previous run count: "
4643cab2bb3Spatrick "corrupt object tag (0x%08x)\n",
4653cab2bb3Spatrick filename, val);
4663cab2bb3Spatrick return;
4673cab2bb3Spatrick }
4683cab2bb3Spatrick
4693cab2bb3Spatrick val = read_32bit_value(); /* length */
4701f9cb04fSpatrick uint32_t prev_runs;
4711f9cb04fSpatrick if (gcov_version < 90) {
4721f9cb04fSpatrick read_32bit_value();
4731f9cb04fSpatrick read_32bit_value();
4741f9cb04fSpatrick prev_runs = read_32bit_value();
4751f9cb04fSpatrick } else {
4761f9cb04fSpatrick prev_runs = read_32bit_value();
4771f9cb04fSpatrick read_32bit_value();
4783cab2bb3Spatrick }
4791f9cb04fSpatrick for (uint32_t i = gcov_version < 90 ? 3 : 2; i < val; ++i)
4801f9cb04fSpatrick read_32bit_value();
4813cab2bb3Spatrick /* Add previous run count to new counter, if not already counted before. */
4823cab2bb3Spatrick runs = run_counted ? prev_runs : prev_runs + 1;
4833cab2bb3Spatrick }
4843cab2bb3Spatrick
4853cab2bb3Spatrick cur_pos = save_cur_pos;
4863cab2bb3Spatrick
4871f9cb04fSpatrick if (gcov_version >= 90) {
4881f9cb04fSpatrick write_32bit_value(GCOV_TAG_OBJECT_SUMMARY);
4891f9cb04fSpatrick write_32bit_value(2);
4903cab2bb3Spatrick write_32bit_value(runs);
4911f9cb04fSpatrick write_32bit_value(0); // sum_max
4921f9cb04fSpatrick } else {
4931f9cb04fSpatrick // Before gcov 4.8 (r190952), GCOV_TAG_SUMMARY_LENGTH was 9. r190952 set
4941f9cb04fSpatrick // GCOV_TAG_SUMMARY_LENGTH to 22. We simply use the smallest length which
4951f9cb04fSpatrick // can make gcov read "Runs:".
4961f9cb04fSpatrick write_32bit_value(GCOV_TAG_PROGRAM_SUMMARY);
4971f9cb04fSpatrick write_32bit_value(3);
4983cab2bb3Spatrick write_32bit_value(0);
4991f9cb04fSpatrick write_32bit_value(0);
5001f9cb04fSpatrick write_32bit_value(runs);
5011f9cb04fSpatrick }
5023cab2bb3Spatrick
5033cab2bb3Spatrick run_counted = 1;
5043cab2bb3Spatrick
5053cab2bb3Spatrick #ifdef DEBUG_GCDAPROFILING
5063cab2bb3Spatrick fprintf(stderr, "llvmgcda: %u runs\n", runs);
5073cab2bb3Spatrick #endif
5083cab2bb3Spatrick }
5093cab2bb3Spatrick
5103cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_gcda_end_file(void)511*810390e3Srobert void llvm_gcda_end_file(void) {
5123cab2bb3Spatrick /* Write out EOF record. */
5133cab2bb3Spatrick if (output_file) {
5143cab2bb3Spatrick write_bytes("\0\0\0\0\0\0\0\0", 8);
5153cab2bb3Spatrick
5163cab2bb3Spatrick if (new_file) {
5173cab2bb3Spatrick fwrite(write_buffer, cur_pos, 1, output_file);
5183cab2bb3Spatrick free(write_buffer);
5193cab2bb3Spatrick } else {
5203cab2bb3Spatrick unmap_file();
5213cab2bb3Spatrick }
5223cab2bb3Spatrick
5233cab2bb3Spatrick fflush(output_file);
5243cab2bb3Spatrick lprofUnlockFd(fd);
5253cab2bb3Spatrick fclose(output_file);
5263cab2bb3Spatrick output_file = NULL;
5273cab2bb3Spatrick write_buffer = NULL;
5283cab2bb3Spatrick }
5293cab2bb3Spatrick free(filename);
5303cab2bb3Spatrick
5313cab2bb3Spatrick #ifdef DEBUG_GCDAPROFILING
5323cab2bb3Spatrick fprintf(stderr, "llvmgcda: -----\n");
5333cab2bb3Spatrick #endif
5343cab2bb3Spatrick }
5353cab2bb3Spatrick
5363cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_register_writeout_function(fn_ptr fn)5373cab2bb3Spatrick void llvm_register_writeout_function(fn_ptr fn) {
5383cab2bb3Spatrick fn_list_insert(&writeout_fn_list, fn);
5393cab2bb3Spatrick }
5403cab2bb3Spatrick
5413cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_writeout_files(void)5423cab2bb3Spatrick void llvm_writeout_files(void) {
5433cab2bb3Spatrick struct fn_node *curr = writeout_fn_list.head;
5443cab2bb3Spatrick
5453cab2bb3Spatrick while (curr) {
5463cab2bb3Spatrick if (curr->id == CURRENT_ID) {
5473cab2bb3Spatrick curr->fn();
5483cab2bb3Spatrick }
5493cab2bb3Spatrick curr = curr->next;
5503cab2bb3Spatrick }
5513cab2bb3Spatrick }
5523cab2bb3Spatrick
5531f9cb04fSpatrick #ifndef _WIN32
5541f9cb04fSpatrick // __attribute__((destructor)) and destructors whose priorities are greater than
5551f9cb04fSpatrick // 100 run before this function and can thus be tracked. The priority is
5561f9cb04fSpatrick // compatible with GCC 7 onwards.
5571f9cb04fSpatrick #if __GNUC__ >= 9
5581f9cb04fSpatrick #pragma GCC diagnostic ignored "-Wprio-ctor-dtor"
5591f9cb04fSpatrick #endif
5601f9cb04fSpatrick __attribute__((destructor(100)))
5611f9cb04fSpatrick #endif
llvm_writeout_and_clear(void)5621f9cb04fSpatrick static void llvm_writeout_and_clear(void) {
5631f9cb04fSpatrick llvm_writeout_files();
5643cab2bb3Spatrick fn_list_remove(&writeout_fn_list);
5653cab2bb3Spatrick }
5663cab2bb3Spatrick
5673cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_register_reset_function(fn_ptr fn)5683cab2bb3Spatrick void llvm_register_reset_function(fn_ptr fn) {
5693cab2bb3Spatrick fn_list_insert(&reset_fn_list, fn);
5703cab2bb3Spatrick }
5713cab2bb3Spatrick
5723cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_delete_reset_function_list(void)5733cab2bb3Spatrick void llvm_delete_reset_function_list(void) { fn_list_remove(&reset_fn_list); }
5743cab2bb3Spatrick
5753cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_reset_counters(void)5763cab2bb3Spatrick void llvm_reset_counters(void) {
5773cab2bb3Spatrick struct fn_node *curr = reset_fn_list.head;
5783cab2bb3Spatrick
5793cab2bb3Spatrick while (curr) {
5803cab2bb3Spatrick if (curr->id == CURRENT_ID) {
5813cab2bb3Spatrick curr->fn();
5823cab2bb3Spatrick }
5833cab2bb3Spatrick curr = curr->next;
5843cab2bb3Spatrick }
5853cab2bb3Spatrick }
5863cab2bb3Spatrick
5873cab2bb3Spatrick #if !defined(_WIN32)
5883cab2bb3Spatrick COMPILER_RT_VISIBILITY
__gcov_fork()5893cab2bb3Spatrick pid_t __gcov_fork() {
5903cab2bb3Spatrick pid_t parent_pid = getpid();
5913cab2bb3Spatrick pid_t pid = fork();
5923cab2bb3Spatrick
5933cab2bb3Spatrick if (pid == 0) {
5943cab2bb3Spatrick pid_t child_pid = getpid();
5953cab2bb3Spatrick if (child_pid != parent_pid) {
5963cab2bb3Spatrick // The pid changed so we've a fork (one could have its own fork function)
5973cab2bb3Spatrick // Just reset the counters for this child process
5983cab2bb3Spatrick // threads.
5993cab2bb3Spatrick llvm_reset_counters();
6003cab2bb3Spatrick }
6013cab2bb3Spatrick }
6023cab2bb3Spatrick return pid;
6033cab2bb3Spatrick }
6043cab2bb3Spatrick #endif
6053cab2bb3Spatrick
6063cab2bb3Spatrick COMPILER_RT_VISIBILITY
llvm_gcov_init(fn_ptr wfn,fn_ptr rfn)607d89ec533Spatrick void llvm_gcov_init(fn_ptr wfn, fn_ptr rfn) {
6083cab2bb3Spatrick static int atexit_ran = 0;
6093cab2bb3Spatrick
6103cab2bb3Spatrick if (wfn)
6113cab2bb3Spatrick llvm_register_writeout_function(wfn);
6123cab2bb3Spatrick
6133cab2bb3Spatrick if (rfn)
6143cab2bb3Spatrick llvm_register_reset_function(rfn);
6153cab2bb3Spatrick
6163cab2bb3Spatrick if (atexit_ran == 0) {
6173cab2bb3Spatrick atexit_ran = 1;
6183cab2bb3Spatrick
6193cab2bb3Spatrick /* Make sure we write out the data and delete the data structures. */
6203cab2bb3Spatrick atexit(llvm_delete_reset_function_list);
6211f9cb04fSpatrick #ifdef _WIN32
6221f9cb04fSpatrick atexit(llvm_writeout_and_clear);
6231f9cb04fSpatrick #endif
6243cab2bb3Spatrick }
6253cab2bb3Spatrick }
6263cab2bb3Spatrick
__gcov_dump(void)627d89ec533Spatrick void __gcov_dump(void) {
628d89ec533Spatrick for (struct fn_node *f = writeout_fn_list.head; f; f = f->next)
629d89ec533Spatrick f->fn();
630d89ec533Spatrick }
631d89ec533Spatrick
__gcov_reset(void)632d89ec533Spatrick void __gcov_reset(void) {
633d89ec533Spatrick for (struct fn_node *f = reset_fn_list.head; f; f = f->next)
634d89ec533Spatrick f->fn();
635d89ec533Spatrick }
636d89ec533Spatrick
6373cab2bb3Spatrick #endif
638