xref: /freebsd-src/contrib/llvm-project/compiler-rt/lib/profile/GCDAProfiling.c (revision 753f127f3ace09432b2baeffd71a308760641a62)
10b57cec5SDimitry Andric /*===- GCDAProfiling.c - Support library for GCDA file emission -----------===*\
20b57cec5SDimitry Andric |*
30b57cec5SDimitry Andric |* Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric |* See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric |* SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric |*
70b57cec5SDimitry Andric |*===----------------------------------------------------------------------===*|
80b57cec5SDimitry Andric |*
90b57cec5SDimitry Andric |* This file implements the call back routines for the gcov profiling
100b57cec5SDimitry Andric |* instrumentation pass. Link against this library when running code through
110b57cec5SDimitry Andric |* the -insert-gcov-profiling LLVM pass.
120b57cec5SDimitry Andric |*
130b57cec5SDimitry Andric |* We emit files in a corrupt version of GCOV's "gcda" file format. These files
140b57cec5SDimitry Andric |* are only close enough that LCOV will happily parse them. Anything that lcov
150b57cec5SDimitry Andric |* ignores is missing.
160b57cec5SDimitry Andric |*
170b57cec5SDimitry Andric |* TODO: gcov is multi-process safe by having each exit open the existing file
180b57cec5SDimitry Andric |* and append to it. We'd like to achieve that and be thread-safe too.
190b57cec5SDimitry Andric |*
200b57cec5SDimitry Andric \*===----------------------------------------------------------------------===*/
210b57cec5SDimitry Andric 
220b57cec5SDimitry Andric #if !defined(__Fuchsia__)
230b57cec5SDimitry Andric 
240b57cec5SDimitry Andric #include <errno.h>
250b57cec5SDimitry Andric #include <fcntl.h>
26fe6060f1SDimitry Andric #include <stdint.h>
270b57cec5SDimitry Andric #include <stdio.h>
280b57cec5SDimitry Andric #include <stdlib.h>
290b57cec5SDimitry Andric #include <string.h>
300b57cec5SDimitry Andric 
310b57cec5SDimitry Andric #if defined(_WIN32)
320b57cec5SDimitry Andric #define WIN32_LEAN_AND_MEAN
330b57cec5SDimitry Andric #include <windows.h>
340b57cec5SDimitry Andric #include "WindowsMMap.h"
350b57cec5SDimitry Andric #else
360b57cec5SDimitry Andric #include <sys/file.h>
37d65cd7a5SDimitry Andric #include <sys/mman.h>
38d65cd7a5SDimitry Andric #include <sys/types.h>
39d65cd7a5SDimitry Andric #include <unistd.h>
400b57cec5SDimitry Andric #endif
410b57cec5SDimitry Andric 
420b57cec5SDimitry Andric #include "InstrProfiling.h"
430b57cec5SDimitry Andric #include "InstrProfilingUtil.h"
440b57cec5SDimitry Andric 
450b57cec5SDimitry Andric /* #define DEBUG_GCDAPROFILING */
460b57cec5SDimitry Andric 
475ffd83dbSDimitry Andric enum {
485ffd83dbSDimitry Andric   GCOV_DATA_MAGIC = 0x67636461, // "gcda"
495ffd83dbSDimitry Andric 
505ffd83dbSDimitry Andric   GCOV_TAG_FUNCTION = 0x01000000,
515ffd83dbSDimitry Andric   GCOV_TAG_COUNTER_ARCS = 0x01a10000,
525ffd83dbSDimitry Andric   // GCOV_TAG_OBJECT_SUMMARY superseded GCOV_TAG_PROGRAM_SUMMARY in GCC 9.
535ffd83dbSDimitry Andric   GCOV_TAG_OBJECT_SUMMARY = 0xa1000000,
545ffd83dbSDimitry Andric   GCOV_TAG_PROGRAM_SUMMARY = 0xa3000000,
555ffd83dbSDimitry Andric };
565ffd83dbSDimitry Andric 
570b57cec5SDimitry Andric /*
580b57cec5SDimitry Andric  * --- GCOV file format I/O primitives ---
590b57cec5SDimitry Andric  */
600b57cec5SDimitry Andric 
610b57cec5SDimitry Andric /*
620b57cec5SDimitry Andric  * The current file name we're outputting. Used primarily for error logging.
630b57cec5SDimitry Andric  */
640b57cec5SDimitry Andric static char *filename = NULL;
650b57cec5SDimitry Andric 
660b57cec5SDimitry Andric /*
670b57cec5SDimitry Andric  * The current file we're outputting.
680b57cec5SDimitry Andric  */
690b57cec5SDimitry Andric static FILE *output_file = NULL;
700b57cec5SDimitry Andric 
710b57cec5SDimitry Andric /*
720b57cec5SDimitry Andric  * Buffer that we write things into.
730b57cec5SDimitry Andric  */
740b57cec5SDimitry Andric #define WRITE_BUFFER_SIZE (128 * 1024)
750b57cec5SDimitry Andric static unsigned char *write_buffer = NULL;
760b57cec5SDimitry Andric static uint64_t cur_buffer_size = 0;
770b57cec5SDimitry Andric static uint64_t cur_pos = 0;
780b57cec5SDimitry Andric static uint64_t file_size = 0;
790b57cec5SDimitry Andric static int new_file = 0;
805ffd83dbSDimitry Andric static int gcov_version;
810b57cec5SDimitry Andric #if defined(_WIN32)
820b57cec5SDimitry Andric static HANDLE mmap_handle = NULL;
830b57cec5SDimitry Andric #endif
840b57cec5SDimitry Andric static int fd = -1;
850b57cec5SDimitry Andric 
86*81ad6265SDimitry Andric typedef void (*fn_ptr)(void);
870b57cec5SDimitry Andric 
880b57cec5SDimitry Andric typedef void* dynamic_object_id;
890b57cec5SDimitry Andric // The address of this variable identifies a given dynamic object.
900b57cec5SDimitry Andric static dynamic_object_id current_id;
910b57cec5SDimitry Andric #define CURRENT_ID (&current_id)
920b57cec5SDimitry Andric 
930b57cec5SDimitry Andric struct fn_node {
940b57cec5SDimitry Andric   dynamic_object_id id;
950b57cec5SDimitry Andric   fn_ptr fn;
960b57cec5SDimitry Andric   struct fn_node* next;
970b57cec5SDimitry Andric };
980b57cec5SDimitry Andric 
990b57cec5SDimitry Andric struct fn_list {
1000b57cec5SDimitry Andric   struct fn_node *head, *tail;
1010b57cec5SDimitry Andric };
1020b57cec5SDimitry Andric 
1030b57cec5SDimitry Andric /*
1040b57cec5SDimitry Andric  * A list of functions to write out the data, shared between all dynamic objects.
1050b57cec5SDimitry Andric  */
1060b57cec5SDimitry Andric struct fn_list writeout_fn_list;
1070b57cec5SDimitry Andric 
1080b57cec5SDimitry Andric /*
109d65cd7a5SDimitry Andric  *  A list of reset functions, shared between all dynamic objects.
110d65cd7a5SDimitry Andric  */
111d65cd7a5SDimitry Andric struct fn_list reset_fn_list;
112d65cd7a5SDimitry Andric 
fn_list_insert(struct fn_list * list,fn_ptr fn)1130b57cec5SDimitry Andric static void fn_list_insert(struct fn_list* list, fn_ptr fn) {
1140b57cec5SDimitry Andric   struct fn_node* new_node = malloc(sizeof(struct fn_node));
1150b57cec5SDimitry Andric   new_node->fn = fn;
1160b57cec5SDimitry Andric   new_node->next = NULL;
1170b57cec5SDimitry Andric   new_node->id = CURRENT_ID;
1180b57cec5SDimitry Andric 
1190b57cec5SDimitry Andric   if (!list->head) {
1200b57cec5SDimitry Andric     list->head = list->tail = new_node;
1210b57cec5SDimitry Andric   } else {
1220b57cec5SDimitry Andric     list->tail->next = new_node;
1230b57cec5SDimitry Andric     list->tail = new_node;
1240b57cec5SDimitry Andric   }
1250b57cec5SDimitry Andric }
1260b57cec5SDimitry Andric 
fn_list_remove(struct fn_list * list)1270b57cec5SDimitry Andric static void fn_list_remove(struct fn_list* list) {
1280b57cec5SDimitry Andric   struct fn_node* curr = list->head;
1290b57cec5SDimitry Andric   struct fn_node* prev = NULL;
1300b57cec5SDimitry Andric   struct fn_node* next = NULL;
1310b57cec5SDimitry Andric 
1320b57cec5SDimitry Andric   while (curr) {
1330b57cec5SDimitry Andric     next = curr->next;
1340b57cec5SDimitry Andric 
1350b57cec5SDimitry Andric     if (curr->id == CURRENT_ID) {
1360b57cec5SDimitry Andric       if (curr == list->head) {
1370b57cec5SDimitry Andric         list->head = next;
1380b57cec5SDimitry Andric       }
1390b57cec5SDimitry Andric 
1400b57cec5SDimitry Andric       if (curr == list->tail) {
1410b57cec5SDimitry Andric         list->tail = prev;
1420b57cec5SDimitry Andric       }
1430b57cec5SDimitry Andric 
1440b57cec5SDimitry Andric       if (prev) {
1450b57cec5SDimitry Andric         prev->next = next;
1460b57cec5SDimitry Andric       }
1470b57cec5SDimitry Andric 
1480b57cec5SDimitry Andric       free(curr);
1490b57cec5SDimitry Andric     } else {
1500b57cec5SDimitry Andric       prev = curr;
1510b57cec5SDimitry Andric     }
1520b57cec5SDimitry Andric 
1530b57cec5SDimitry Andric     curr = next;
1540b57cec5SDimitry Andric   }
1550b57cec5SDimitry Andric }
1560b57cec5SDimitry Andric 
resize_write_buffer(uint64_t size)1570b57cec5SDimitry Andric static void resize_write_buffer(uint64_t size) {
1580b57cec5SDimitry Andric   if (!new_file) return;
1590b57cec5SDimitry Andric   size += cur_pos;
1600b57cec5SDimitry Andric   if (size <= cur_buffer_size) return;
1610b57cec5SDimitry Andric   size = (size - 1) / WRITE_BUFFER_SIZE + 1;
1620b57cec5SDimitry Andric   size *= WRITE_BUFFER_SIZE;
1630b57cec5SDimitry Andric   write_buffer = realloc(write_buffer, size);
1640b57cec5SDimitry Andric   cur_buffer_size = size;
1650b57cec5SDimitry Andric }
1660b57cec5SDimitry Andric 
write_bytes(const char * s,size_t len)1670b57cec5SDimitry Andric static void write_bytes(const char *s, size_t len) {
1680b57cec5SDimitry Andric   resize_write_buffer(len);
1690b57cec5SDimitry Andric   memcpy(&write_buffer[cur_pos], s, len);
1700b57cec5SDimitry Andric   cur_pos += len;
1710b57cec5SDimitry Andric }
1720b57cec5SDimitry Andric 
write_32bit_value(uint32_t i)1730b57cec5SDimitry Andric static void write_32bit_value(uint32_t i) {
1740b57cec5SDimitry Andric   write_bytes((char*)&i, 4);
1750b57cec5SDimitry Andric }
1760b57cec5SDimitry Andric 
write_64bit_value(uint64_t i)1770b57cec5SDimitry Andric static void write_64bit_value(uint64_t i) {
1780b57cec5SDimitry Andric   // GCOV uses a lo-/hi-word format even on big-endian systems.
1790b57cec5SDimitry Andric   // See also GCOVBuffer::readInt64 in LLVM.
1800b57cec5SDimitry Andric   uint32_t lo = (uint32_t) i;
1810b57cec5SDimitry Andric   uint32_t hi = (uint32_t) (i >> 32);
1820b57cec5SDimitry Andric   write_32bit_value(lo);
1830b57cec5SDimitry Andric   write_32bit_value(hi);
1840b57cec5SDimitry Andric }
1850b57cec5SDimitry Andric 
read_32bit_value(void)186*81ad6265SDimitry Andric static uint32_t read_32bit_value(void) {
1870b57cec5SDimitry Andric   uint32_t val;
1880b57cec5SDimitry Andric 
1890b57cec5SDimitry Andric   if (new_file)
1900b57cec5SDimitry Andric     return (uint32_t)-1;
1910b57cec5SDimitry Andric 
1920b57cec5SDimitry Andric   val = *(uint32_t*)&write_buffer[cur_pos];
1930b57cec5SDimitry Andric   cur_pos += 4;
1940b57cec5SDimitry Andric   return val;
1950b57cec5SDimitry Andric }
1960b57cec5SDimitry Andric 
read_64bit_value(void)197*81ad6265SDimitry Andric static uint64_t read_64bit_value(void) {
1980b57cec5SDimitry Andric   // GCOV uses a lo-/hi-word format even on big-endian systems.
1990b57cec5SDimitry Andric   // See also GCOVBuffer::readInt64 in LLVM.
2000b57cec5SDimitry Andric   uint32_t lo = read_32bit_value();
2010b57cec5SDimitry Andric   uint32_t hi = read_32bit_value();
2020b57cec5SDimitry Andric   return ((uint64_t)hi << 32) | ((uint64_t)lo);
2030b57cec5SDimitry Andric }
2040b57cec5SDimitry Andric 
mangle_filename(const char * orig_filename)2050b57cec5SDimitry Andric static char *mangle_filename(const char *orig_filename) {
2060b57cec5SDimitry Andric   char *new_filename;
2070b57cec5SDimitry Andric   size_t prefix_len;
2080b57cec5SDimitry Andric   int prefix_strip;
2090b57cec5SDimitry Andric   const char *prefix = lprofGetPathPrefix(&prefix_strip, &prefix_len);
2100b57cec5SDimitry Andric 
2110b57cec5SDimitry Andric   if (prefix == NULL)
2120b57cec5SDimitry Andric     return strdup(orig_filename);
2130b57cec5SDimitry Andric 
2140b57cec5SDimitry Andric   new_filename = malloc(prefix_len + 1 + strlen(orig_filename) + 1);
2150b57cec5SDimitry Andric   lprofApplyPathPrefix(new_filename, orig_filename, prefix, prefix_len,
2160b57cec5SDimitry Andric                        prefix_strip);
2170b57cec5SDimitry Andric 
2180b57cec5SDimitry Andric   return new_filename;
2190b57cec5SDimitry Andric }
2200b57cec5SDimitry Andric 
map_file(void)221*81ad6265SDimitry Andric static int map_file(void) {
2220b57cec5SDimitry Andric   fseek(output_file, 0L, SEEK_END);
2230b57cec5SDimitry Andric   file_size = ftell(output_file);
2240b57cec5SDimitry Andric 
2255ffd83dbSDimitry Andric   /* A size of 0 means the file has been created just now (possibly by another
2265ffd83dbSDimitry Andric    * process in lock-after-open race condition). No need to mmap. */
2270b57cec5SDimitry Andric   if (file_size == 0)
2280b57cec5SDimitry Andric     return -1;
2290b57cec5SDimitry Andric 
2300b57cec5SDimitry Andric #if defined(_WIN32)
2310b57cec5SDimitry Andric   HANDLE mmap_fd;
2320b57cec5SDimitry Andric   if (fd == -1)
2330b57cec5SDimitry Andric     mmap_fd = INVALID_HANDLE_VALUE;
2340b57cec5SDimitry Andric   else
2350b57cec5SDimitry Andric     mmap_fd = (HANDLE)_get_osfhandle(fd);
2360b57cec5SDimitry Andric 
2370b57cec5SDimitry Andric   mmap_handle = CreateFileMapping(mmap_fd, NULL, PAGE_READWRITE, DWORD_HI(file_size), DWORD_LO(file_size), NULL);
2380b57cec5SDimitry Andric   if (mmap_handle == NULL) {
2390b57cec5SDimitry Andric     fprintf(stderr, "profiling: %s: cannot create file mapping: %lu\n",
2400b57cec5SDimitry Andric             filename, GetLastError());
2410b57cec5SDimitry Andric     return -1;
2420b57cec5SDimitry Andric   }
2430b57cec5SDimitry Andric 
2440b57cec5SDimitry Andric   write_buffer = MapViewOfFile(mmap_handle, FILE_MAP_WRITE, 0, 0, file_size);
2450b57cec5SDimitry Andric   if (write_buffer == NULL) {
2460b57cec5SDimitry Andric     fprintf(stderr, "profiling: %s: cannot map: %lu\n", filename,
2470b57cec5SDimitry Andric             GetLastError());
2480b57cec5SDimitry Andric     CloseHandle(mmap_handle);
2490b57cec5SDimitry Andric     return -1;
2500b57cec5SDimitry Andric   }
2510b57cec5SDimitry Andric #else
2520b57cec5SDimitry Andric   write_buffer = mmap(0, file_size, PROT_READ | PROT_WRITE,
2530b57cec5SDimitry Andric                       MAP_FILE | MAP_SHARED, fd, 0);
2540b57cec5SDimitry Andric   if (write_buffer == (void *)-1) {
2550b57cec5SDimitry Andric     int errnum = errno;
2560b57cec5SDimitry Andric     fprintf(stderr, "profiling: %s: cannot map: %s\n", filename,
2570b57cec5SDimitry Andric             strerror(errnum));
2580b57cec5SDimitry Andric     return -1;
2590b57cec5SDimitry Andric   }
2600b57cec5SDimitry Andric #endif
2610b57cec5SDimitry Andric 
2620b57cec5SDimitry Andric   return 0;
2630b57cec5SDimitry Andric }
2640b57cec5SDimitry Andric 
unmap_file(void)265*81ad6265SDimitry Andric static void unmap_file(void) {
2660b57cec5SDimitry Andric #if defined(_WIN32)
2670b57cec5SDimitry Andric   if (!UnmapViewOfFile(write_buffer)) {
2680b57cec5SDimitry Andric     fprintf(stderr, "profiling: %s: cannot unmap mapped view: %lu\n", filename,
2690b57cec5SDimitry Andric             GetLastError());
2700b57cec5SDimitry Andric   }
2710b57cec5SDimitry Andric 
2720b57cec5SDimitry Andric   if (!CloseHandle(mmap_handle)) {
2730b57cec5SDimitry Andric     fprintf(stderr, "profiling: %s: cannot close file mapping handle: %lu\n",
2740b57cec5SDimitry Andric             filename, GetLastError());
2750b57cec5SDimitry Andric   }
2760b57cec5SDimitry Andric 
2770b57cec5SDimitry Andric   mmap_handle = NULL;
2780b57cec5SDimitry Andric #else
279e8d8bef9SDimitry Andric   if (munmap(write_buffer, file_size) == -1) {
2800b57cec5SDimitry Andric     int errnum = errno;
281e8d8bef9SDimitry Andric     fprintf(stderr, "profiling: %s: cannot munmap: %s\n", filename,
2820b57cec5SDimitry Andric             strerror(errnum));
2830b57cec5SDimitry Andric   }
2840b57cec5SDimitry Andric #endif
2850b57cec5SDimitry Andric 
2860b57cec5SDimitry Andric   write_buffer = NULL;
2870b57cec5SDimitry Andric   file_size = 0;
2880b57cec5SDimitry Andric }
2890b57cec5SDimitry Andric 
2900b57cec5SDimitry Andric /*
2910b57cec5SDimitry Andric  * --- LLVM line counter API ---
2920b57cec5SDimitry Andric  */
2930b57cec5SDimitry Andric 
2940b57cec5SDimitry Andric /* A file in this case is a translation unit. Each .o file built with line
2950b57cec5SDimitry Andric  * profiling enabled will emit to a different file. Only one file may be
2960b57cec5SDimitry Andric  * started at a time.
2970b57cec5SDimitry Andric  */
2980b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_gcda_start_file(const char * orig_filename,uint32_t version,uint32_t checksum)2995ffd83dbSDimitry Andric void llvm_gcda_start_file(const char *orig_filename, uint32_t version,
3000b57cec5SDimitry Andric                           uint32_t checksum) {
3010b57cec5SDimitry Andric   const char *mode = "r+b";
3020b57cec5SDimitry Andric   filename = mangle_filename(orig_filename);
3030b57cec5SDimitry Andric 
3040b57cec5SDimitry Andric   /* Try just opening the file. */
3050b57cec5SDimitry Andric   fd = open(filename, O_RDWR | O_BINARY);
3060b57cec5SDimitry Andric 
3070b57cec5SDimitry Andric   if (fd == -1) {
3085ffd83dbSDimitry Andric     /* Try creating the file. */
3095ffd83dbSDimitry Andric     fd = open(filename, O_RDWR | O_CREAT | O_EXCL | O_BINARY, 0644);
3105ffd83dbSDimitry Andric     if (fd != -1) {
3110b57cec5SDimitry Andric       mode = "w+b";
3125ffd83dbSDimitry Andric     } else {
3130b57cec5SDimitry Andric       /* Try creating the directories first then opening the file. */
3140b57cec5SDimitry Andric       __llvm_profile_recursive_mkdir(filename);
3155ffd83dbSDimitry Andric       fd = open(filename, O_RDWR | O_CREAT | O_EXCL | O_BINARY, 0644);
3165ffd83dbSDimitry Andric       if (fd != -1) {
3175ffd83dbSDimitry Andric         mode = "w+b";
3185ffd83dbSDimitry Andric       } else {
3195ffd83dbSDimitry Andric         /* Another process may have created the file just now.
3205ffd83dbSDimitry Andric          * Try opening it without O_CREAT and O_EXCL. */
3215ffd83dbSDimitry Andric         fd = open(filename, O_RDWR | O_BINARY);
3220b57cec5SDimitry Andric         if (fd == -1) {
3230b57cec5SDimitry Andric           /* Bah! It's hopeless. */
3240b57cec5SDimitry Andric           int errnum = errno;
3250b57cec5SDimitry Andric           fprintf(stderr, "profiling: %s: cannot open: %s\n", filename,
3260b57cec5SDimitry Andric                   strerror(errnum));
3270b57cec5SDimitry Andric           return;
3280b57cec5SDimitry Andric         }
3290b57cec5SDimitry Andric       }
3300b57cec5SDimitry Andric     }
3315ffd83dbSDimitry Andric   }
3320b57cec5SDimitry Andric 
3330b57cec5SDimitry Andric   /* Try to flock the file to serialize concurrent processes writing out to the
3340b57cec5SDimitry Andric    * same GCDA. This can fail if the filesystem doesn't support it, but in that
3350b57cec5SDimitry Andric    * case we'll just carry on with the old racy behaviour and hope for the best.
3360b57cec5SDimitry Andric    */
3370b57cec5SDimitry Andric   lprofLockFd(fd);
3380b57cec5SDimitry Andric   output_file = fdopen(fd, mode);
3390b57cec5SDimitry Andric 
3400b57cec5SDimitry Andric   /* Initialize the write buffer. */
3415ffd83dbSDimitry Andric   new_file = 0;
3420b57cec5SDimitry Andric   write_buffer = NULL;
3430b57cec5SDimitry Andric   cur_buffer_size = 0;
3440b57cec5SDimitry Andric   cur_pos = 0;
3450b57cec5SDimitry Andric 
3460b57cec5SDimitry Andric   if (map_file() == -1) {
3475ffd83dbSDimitry Andric     /* The file has been created just now (file_size == 0) or mmap failed
3485ffd83dbSDimitry Andric      * unexpectedly. In the latter case, try to recover by clobbering. */
3490b57cec5SDimitry Andric     new_file = 1;
3500b57cec5SDimitry Andric     write_buffer = NULL;
3510b57cec5SDimitry Andric     resize_write_buffer(WRITE_BUFFER_SIZE);
3520b57cec5SDimitry Andric     memset(write_buffer, 0, WRITE_BUFFER_SIZE);
3530b57cec5SDimitry Andric   }
3540b57cec5SDimitry Andric 
3550b57cec5SDimitry Andric   /* gcda file, version, stamp checksum. */
3565ffd83dbSDimitry Andric   {
3575ffd83dbSDimitry Andric     uint8_t c3 = version >> 24;
3585ffd83dbSDimitry Andric     uint8_t c2 = (version >> 16) & 255;
3595ffd83dbSDimitry Andric     uint8_t c1 = (version >> 8) & 255;
3605ffd83dbSDimitry Andric     gcov_version = c3 >= 'A' ? (c3 - 'A') * 100 + (c2 - '0') * 10 + c1 - '0'
3615ffd83dbSDimitry Andric                              : (c3 - '0') * 10 + c1 - '0';
3625ffd83dbSDimitry Andric   }
3635ffd83dbSDimitry Andric   write_32bit_value(GCOV_DATA_MAGIC);
3645ffd83dbSDimitry Andric   write_32bit_value(version);
3650b57cec5SDimitry Andric   write_32bit_value(checksum);
3660b57cec5SDimitry Andric 
3670b57cec5SDimitry Andric #ifdef DEBUG_GCDAPROFILING
3680b57cec5SDimitry Andric   fprintf(stderr, "llvmgcda: [%s]\n", orig_filename);
3690b57cec5SDimitry Andric #endif
3700b57cec5SDimitry Andric }
3710b57cec5SDimitry Andric 
3720b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_gcda_emit_function(uint32_t ident,uint32_t func_checksum,uint32_t cfg_checksum)3735ffd83dbSDimitry Andric void llvm_gcda_emit_function(uint32_t ident, uint32_t func_checksum,
3740b57cec5SDimitry Andric                              uint32_t cfg_checksum) {
3750b57cec5SDimitry Andric   uint32_t len = 2;
3765ffd83dbSDimitry Andric   int use_extra_checksum = gcov_version >= 47;
3770b57cec5SDimitry Andric 
3780b57cec5SDimitry Andric   if (use_extra_checksum)
3790b57cec5SDimitry Andric     len++;
3800b57cec5SDimitry Andric #ifdef DEBUG_GCDAPROFILING
3815ffd83dbSDimitry Andric   fprintf(stderr, "llvmgcda: function id=0x%08x\n", ident);
3820b57cec5SDimitry Andric #endif
3830b57cec5SDimitry Andric   if (!output_file) return;
3840b57cec5SDimitry Andric 
3850b57cec5SDimitry Andric   /* function tag */
3865ffd83dbSDimitry Andric   write_32bit_value(GCOV_TAG_FUNCTION);
3870b57cec5SDimitry Andric   write_32bit_value(len);
3880b57cec5SDimitry Andric   write_32bit_value(ident);
3890b57cec5SDimitry Andric   write_32bit_value(func_checksum);
3900b57cec5SDimitry Andric   if (use_extra_checksum)
3910b57cec5SDimitry Andric     write_32bit_value(cfg_checksum);
3920b57cec5SDimitry Andric }
3930b57cec5SDimitry Andric 
3940b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_gcda_emit_arcs(uint32_t num_counters,uint64_t * counters)3950b57cec5SDimitry Andric void llvm_gcda_emit_arcs(uint32_t num_counters, uint64_t *counters) {
3960b57cec5SDimitry Andric   uint32_t i;
3970b57cec5SDimitry Andric   uint64_t *old_ctrs = NULL;
3980b57cec5SDimitry Andric   uint32_t val = 0;
3990b57cec5SDimitry Andric   uint64_t save_cur_pos = cur_pos;
4000b57cec5SDimitry Andric 
4010b57cec5SDimitry Andric   if (!output_file) return;
4020b57cec5SDimitry Andric 
4035ffd83dbSDimitry Andric   val = read_32bit_value();
4040b57cec5SDimitry Andric 
4050b57cec5SDimitry Andric   if (val != (uint32_t)-1) {
4060b57cec5SDimitry Andric     /* There are counters present in the file. Merge them. */
4075ffd83dbSDimitry Andric     if (val != GCOV_TAG_COUNTER_ARCS) {
4080b57cec5SDimitry Andric       fprintf(stderr, "profiling: %s: cannot merge previous GCDA file: "
4090b57cec5SDimitry Andric                       "corrupt arc tag (0x%08x)\n",
4100b57cec5SDimitry Andric               filename, val);
4110b57cec5SDimitry Andric       return;
4120b57cec5SDimitry Andric     }
4130b57cec5SDimitry Andric 
4140b57cec5SDimitry Andric     val = read_32bit_value();
4150b57cec5SDimitry Andric     if (val == (uint32_t)-1 || val / 2 != num_counters) {
4160b57cec5SDimitry Andric       fprintf(stderr, "profiling: %s: cannot merge previous GCDA file: "
4170b57cec5SDimitry Andric                       "mismatched number of counters (%d)\n",
4180b57cec5SDimitry Andric               filename, val);
4190b57cec5SDimitry Andric       return;
4200b57cec5SDimitry Andric     }
4210b57cec5SDimitry Andric 
4220b57cec5SDimitry Andric     old_ctrs = malloc(sizeof(uint64_t) * num_counters);
4230b57cec5SDimitry Andric     for (i = 0; i < num_counters; ++i)
4240b57cec5SDimitry Andric       old_ctrs[i] = read_64bit_value();
4250b57cec5SDimitry Andric   }
4260b57cec5SDimitry Andric 
4270b57cec5SDimitry Andric   cur_pos = save_cur_pos;
4280b57cec5SDimitry Andric 
4290b57cec5SDimitry Andric   /* Counter #1 (arcs) tag */
4305ffd83dbSDimitry Andric   write_32bit_value(GCOV_TAG_COUNTER_ARCS);
4310b57cec5SDimitry Andric   write_32bit_value(num_counters * 2);
4320b57cec5SDimitry Andric   for (i = 0; i < num_counters; ++i) {
4330b57cec5SDimitry Andric     counters[i] += (old_ctrs ? old_ctrs[i] : 0);
4340b57cec5SDimitry Andric     write_64bit_value(counters[i]);
4350b57cec5SDimitry Andric   }
4360b57cec5SDimitry Andric 
4370b57cec5SDimitry Andric   free(old_ctrs);
4380b57cec5SDimitry Andric 
4390b57cec5SDimitry Andric #ifdef DEBUG_GCDAPROFILING
4400b57cec5SDimitry Andric   fprintf(stderr, "llvmgcda:   %u arcs\n", num_counters);
4410b57cec5SDimitry Andric   for (i = 0; i < num_counters; ++i)
4420b57cec5SDimitry Andric     fprintf(stderr, "llvmgcda:   %llu\n", (unsigned long long)counters[i]);
4430b57cec5SDimitry Andric #endif
4440b57cec5SDimitry Andric }
4450b57cec5SDimitry Andric 
4460b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_gcda_summary_info(void)447*81ad6265SDimitry Andric void llvm_gcda_summary_info(void) {
4480b57cec5SDimitry Andric   uint32_t runs = 1;
4490b57cec5SDimitry Andric   static uint32_t run_counted = 0; // We only want to increase the run count once.
4500b57cec5SDimitry Andric   uint32_t val = 0;
4510b57cec5SDimitry Andric   uint64_t save_cur_pos = cur_pos;
4520b57cec5SDimitry Andric 
4530b57cec5SDimitry Andric   if (!output_file) return;
4540b57cec5SDimitry Andric 
4555ffd83dbSDimitry Andric   val = read_32bit_value();
4560b57cec5SDimitry Andric 
4570b57cec5SDimitry Andric   if (val != (uint32_t)-1) {
4580b57cec5SDimitry Andric     /* There are counters present in the file. Merge them. */
459fe6060f1SDimitry Andric     uint32_t gcov_tag =
460fe6060f1SDimitry Andric         gcov_version >= 90 ? GCOV_TAG_OBJECT_SUMMARY : GCOV_TAG_PROGRAM_SUMMARY;
461fe6060f1SDimitry Andric     if (val != gcov_tag) {
4625ffd83dbSDimitry Andric       fprintf(stderr,
4635ffd83dbSDimitry Andric               "profiling: %s: cannot merge previous run count: "
4640b57cec5SDimitry Andric               "corrupt object tag (0x%08x)\n",
4650b57cec5SDimitry Andric               filename, val);
4660b57cec5SDimitry Andric       return;
4670b57cec5SDimitry Andric     }
4680b57cec5SDimitry Andric 
4690b57cec5SDimitry Andric     val = read_32bit_value(); /* length */
4705ffd83dbSDimitry Andric     uint32_t prev_runs;
4715ffd83dbSDimitry Andric     if (gcov_version < 90) {
4725ffd83dbSDimitry Andric       read_32bit_value();
4735ffd83dbSDimitry Andric       read_32bit_value();
4745ffd83dbSDimitry Andric       prev_runs = read_32bit_value();
4755ffd83dbSDimitry Andric     } else {
4765ffd83dbSDimitry Andric       prev_runs = read_32bit_value();
4775ffd83dbSDimitry Andric       read_32bit_value();
4780b57cec5SDimitry Andric     }
4795ffd83dbSDimitry Andric     for (uint32_t i = gcov_version < 90 ? 3 : 2; i < val; ++i)
4805ffd83dbSDimitry Andric       read_32bit_value();
4810b57cec5SDimitry Andric     /* Add previous run count to new counter, if not already counted before. */
4820b57cec5SDimitry Andric     runs = run_counted ? prev_runs : prev_runs + 1;
4830b57cec5SDimitry Andric   }
4840b57cec5SDimitry Andric 
4850b57cec5SDimitry Andric   cur_pos = save_cur_pos;
4860b57cec5SDimitry Andric 
4875ffd83dbSDimitry Andric   if (gcov_version >= 90) {
4885ffd83dbSDimitry Andric     write_32bit_value(GCOV_TAG_OBJECT_SUMMARY);
4895ffd83dbSDimitry Andric     write_32bit_value(2);
4900b57cec5SDimitry Andric     write_32bit_value(runs);
4915ffd83dbSDimitry Andric     write_32bit_value(0); // sum_max
4925ffd83dbSDimitry Andric   } else {
4935ffd83dbSDimitry Andric     // Before gcov 4.8 (r190952), GCOV_TAG_SUMMARY_LENGTH was 9. r190952 set
4945ffd83dbSDimitry Andric     // GCOV_TAG_SUMMARY_LENGTH to 22. We simply use the smallest length which
4955ffd83dbSDimitry Andric     // can make gcov read "Runs:".
4965ffd83dbSDimitry Andric     write_32bit_value(GCOV_TAG_PROGRAM_SUMMARY);
4975ffd83dbSDimitry Andric     write_32bit_value(3);
4980b57cec5SDimitry Andric     write_32bit_value(0);
4995ffd83dbSDimitry Andric     write_32bit_value(0);
5005ffd83dbSDimitry Andric     write_32bit_value(runs);
5015ffd83dbSDimitry Andric   }
5020b57cec5SDimitry Andric 
5030b57cec5SDimitry Andric   run_counted = 1;
5040b57cec5SDimitry Andric 
5050b57cec5SDimitry Andric #ifdef DEBUG_GCDAPROFILING
5060b57cec5SDimitry Andric   fprintf(stderr, "llvmgcda:   %u runs\n", runs);
5070b57cec5SDimitry Andric #endif
5080b57cec5SDimitry Andric }
5090b57cec5SDimitry Andric 
5100b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_gcda_end_file(void)511*81ad6265SDimitry Andric void llvm_gcda_end_file(void) {
5120b57cec5SDimitry Andric   /* Write out EOF record. */
5130b57cec5SDimitry Andric   if (output_file) {
5140b57cec5SDimitry Andric     write_bytes("\0\0\0\0\0\0\0\0", 8);
5150b57cec5SDimitry Andric 
5160b57cec5SDimitry Andric     if (new_file) {
5170b57cec5SDimitry Andric       fwrite(write_buffer, cur_pos, 1, output_file);
5180b57cec5SDimitry Andric       free(write_buffer);
5190b57cec5SDimitry Andric     } else {
5200b57cec5SDimitry Andric       unmap_file();
5210b57cec5SDimitry Andric     }
5220b57cec5SDimitry Andric 
5230b57cec5SDimitry Andric     fflush(output_file);
5240b57cec5SDimitry Andric     lprofUnlockFd(fd);
5250b57cec5SDimitry Andric     fclose(output_file);
5260b57cec5SDimitry Andric     output_file = NULL;
5270b57cec5SDimitry Andric     write_buffer = NULL;
5280b57cec5SDimitry Andric   }
5290b57cec5SDimitry Andric   free(filename);
5300b57cec5SDimitry Andric 
5310b57cec5SDimitry Andric #ifdef DEBUG_GCDAPROFILING
5320b57cec5SDimitry Andric   fprintf(stderr, "llvmgcda: -----\n");
5330b57cec5SDimitry Andric #endif
5340b57cec5SDimitry Andric }
5350b57cec5SDimitry Andric 
5360b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_register_writeout_function(fn_ptr fn)5370b57cec5SDimitry Andric void llvm_register_writeout_function(fn_ptr fn) {
5380b57cec5SDimitry Andric   fn_list_insert(&writeout_fn_list, fn);
5390b57cec5SDimitry Andric }
5400b57cec5SDimitry Andric 
5410b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_writeout_files(void)5420b57cec5SDimitry Andric void llvm_writeout_files(void) {
5430b57cec5SDimitry Andric   struct fn_node *curr = writeout_fn_list.head;
5440b57cec5SDimitry Andric 
5450b57cec5SDimitry Andric   while (curr) {
5460b57cec5SDimitry Andric     if (curr->id == CURRENT_ID) {
5470b57cec5SDimitry Andric       curr->fn();
5480b57cec5SDimitry Andric     }
5490b57cec5SDimitry Andric     curr = curr->next;
5500b57cec5SDimitry Andric   }
5510b57cec5SDimitry Andric }
5520b57cec5SDimitry Andric 
5535ffd83dbSDimitry Andric #ifndef _WIN32
5545ffd83dbSDimitry Andric // __attribute__((destructor)) and destructors whose priorities are greater than
5555ffd83dbSDimitry Andric // 100 run before this function and can thus be tracked. The priority is
5565ffd83dbSDimitry Andric // compatible with GCC 7 onwards.
55716d6b3b3SDimitry Andric #if __GNUC__ >= 9
55816d6b3b3SDimitry Andric #pragma GCC diagnostic ignored "-Wprio-ctor-dtor"
55916d6b3b3SDimitry Andric #endif
5605ffd83dbSDimitry Andric __attribute__((destructor(100)))
5615ffd83dbSDimitry Andric #endif
llvm_writeout_and_clear(void)5625ffd83dbSDimitry Andric static void llvm_writeout_and_clear(void) {
5635ffd83dbSDimitry Andric   llvm_writeout_files();
5640b57cec5SDimitry Andric   fn_list_remove(&writeout_fn_list);
5650b57cec5SDimitry Andric }
5660b57cec5SDimitry Andric 
5670b57cec5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_register_reset_function(fn_ptr fn)568d65cd7a5SDimitry Andric void llvm_register_reset_function(fn_ptr fn) {
569d65cd7a5SDimitry Andric   fn_list_insert(&reset_fn_list, fn);
570d65cd7a5SDimitry Andric }
571d65cd7a5SDimitry Andric 
572d65cd7a5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_delete_reset_function_list(void)573d65cd7a5SDimitry Andric void llvm_delete_reset_function_list(void) { fn_list_remove(&reset_fn_list); }
574d65cd7a5SDimitry Andric 
575d65cd7a5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_reset_counters(void)576d65cd7a5SDimitry Andric void llvm_reset_counters(void) {
577d65cd7a5SDimitry Andric   struct fn_node *curr = reset_fn_list.head;
578d65cd7a5SDimitry Andric 
579d65cd7a5SDimitry Andric   while (curr) {
580d65cd7a5SDimitry Andric     if (curr->id == CURRENT_ID) {
581d65cd7a5SDimitry Andric       curr->fn();
582d65cd7a5SDimitry Andric     }
583d65cd7a5SDimitry Andric     curr = curr->next;
584d65cd7a5SDimitry Andric   }
585d65cd7a5SDimitry Andric }
586d65cd7a5SDimitry Andric 
587d65cd7a5SDimitry Andric #if !defined(_WIN32)
588d65cd7a5SDimitry Andric COMPILER_RT_VISIBILITY
__gcov_fork()589d65cd7a5SDimitry Andric pid_t __gcov_fork() {
590d65cd7a5SDimitry Andric   pid_t parent_pid = getpid();
591d65cd7a5SDimitry Andric   pid_t pid = fork();
592d65cd7a5SDimitry Andric 
593d65cd7a5SDimitry Andric   if (pid == 0) {
594d65cd7a5SDimitry Andric     pid_t child_pid = getpid();
595d65cd7a5SDimitry Andric     if (child_pid != parent_pid) {
596d65cd7a5SDimitry Andric       // The pid changed so we've a fork (one could have its own fork function)
597d65cd7a5SDimitry Andric       // Just reset the counters for this child process
598d65cd7a5SDimitry Andric       // threads.
599d65cd7a5SDimitry Andric       llvm_reset_counters();
600d65cd7a5SDimitry Andric     }
601d65cd7a5SDimitry Andric   }
602d65cd7a5SDimitry Andric   return pid;
603d65cd7a5SDimitry Andric }
604d65cd7a5SDimitry Andric #endif
605d65cd7a5SDimitry Andric 
606d65cd7a5SDimitry Andric COMPILER_RT_VISIBILITY
llvm_gcov_init(fn_ptr wfn,fn_ptr rfn)607e8d8bef9SDimitry Andric void llvm_gcov_init(fn_ptr wfn, fn_ptr rfn) {
6080b57cec5SDimitry Andric   static int atexit_ran = 0;
6090b57cec5SDimitry Andric 
6100b57cec5SDimitry Andric   if (wfn)
6110b57cec5SDimitry Andric     llvm_register_writeout_function(wfn);
6120b57cec5SDimitry Andric 
613d65cd7a5SDimitry Andric   if (rfn)
614d65cd7a5SDimitry Andric     llvm_register_reset_function(rfn);
615d65cd7a5SDimitry Andric 
6160b57cec5SDimitry Andric   if (atexit_ran == 0) {
6170b57cec5SDimitry Andric     atexit_ran = 1;
6180b57cec5SDimitry Andric 
6190b57cec5SDimitry Andric     /* Make sure we write out the data and delete the data structures. */
620d65cd7a5SDimitry Andric     atexit(llvm_delete_reset_function_list);
6215ffd83dbSDimitry Andric #ifdef _WIN32
6225ffd83dbSDimitry Andric     atexit(llvm_writeout_and_clear);
6235ffd83dbSDimitry Andric #endif
6240b57cec5SDimitry Andric   }
6250b57cec5SDimitry Andric }
6260b57cec5SDimitry Andric 
__gcov_dump(void)627e8d8bef9SDimitry Andric void __gcov_dump(void) {
628e8d8bef9SDimitry Andric   for (struct fn_node *f = writeout_fn_list.head; f; f = f->next)
629e8d8bef9SDimitry Andric     f->fn();
630e8d8bef9SDimitry Andric }
631e8d8bef9SDimitry Andric 
__gcov_reset(void)632e8d8bef9SDimitry Andric void __gcov_reset(void) {
633e8d8bef9SDimitry Andric   for (struct fn_node *f = reset_fn_list.head; f; f = f->next)
634e8d8bef9SDimitry Andric     f->fn();
635e8d8bef9SDimitry Andric }
636e8d8bef9SDimitry Andric 
6370b57cec5SDimitry Andric #endif
638