2014-01-07 06:27:03 +08:00
|
|
|
/*===- PGOProfiling.c - Support library for PGO instrumentation -----------===*\
|
|
|
|
|*
|
|
|
|
|* The LLVM Compiler Infrastructure
|
|
|
|
|*
|
|
|
|
|* This file is distributed under the University of Illinois Open Source
|
|
|
|
|* License. See LICENSE.TXT for details.
|
|
|
|
|*
|
|
|
|
\*===----------------------------------------------------------------------===*/
|
|
|
|
|
2014-01-07 09:46:49 +08:00
|
|
|
#include <inttypes.h>
|
2014-01-07 06:27:03 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#ifndef _MSC_VER
|
|
|
|
#include <stdint.h>
|
|
|
|
#else
|
|
|
|
typedef unsigned int uint32_t;
|
|
|
|
typedef unsigned int uint64_t;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static FILE *OutputFile = NULL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A list of functions to write out the data.
|
|
|
|
*/
|
|
|
|
typedef void (*writeout_fn)();
|
|
|
|
|
|
|
|
struct writeout_fn_node {
|
|
|
|
writeout_fn fn;
|
|
|
|
struct writeout_fn_node *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct writeout_fn_node *writeout_fn_head = NULL;
|
|
|
|
static struct writeout_fn_node *writeout_fn_tail = NULL;
|
|
|
|
|
|
|
|
void llvm_pgo_emit(const char *MangledName, uint32_t NumCounters,
|
|
|
|
uint64_t *Counters) {
|
|
|
|
uint32_t i;
|
|
|
|
fprintf(OutputFile, "%s %u\n", MangledName, NumCounters);
|
|
|
|
for (i = 0; i < NumCounters; ++i)
|
2014-01-07 07:17:27 +08:00
|
|
|
fprintf(OutputFile, "%" PRIu64 "\n", Counters[i]);
|
2014-01-07 06:27:03 +08:00
|
|
|
fprintf(OutputFile, "\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
void llvm_pgo_register_writeout_function(writeout_fn fn) {
|
|
|
|
struct writeout_fn_node *new_node = malloc(sizeof(struct writeout_fn_node));
|
|
|
|
new_node->fn = fn;
|
|
|
|
new_node->next = NULL;
|
|
|
|
|
|
|
|
if (!writeout_fn_head) {
|
|
|
|
writeout_fn_head = writeout_fn_tail = new_node;
|
|
|
|
} else {
|
|
|
|
writeout_fn_tail->next = new_node;
|
|
|
|
writeout_fn_tail = new_node;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void llvm_pgo_writeout_files() {
|
2014-02-21 03:52:32 +08:00
|
|
|
const char *OutputName = getenv("LLVM_PROFILE_FILE");
|
|
|
|
if (OutputName == NULL || OutputName[0] == '\0')
|
|
|
|
OutputName = "default.profdata";
|
|
|
|
OutputFile = fopen(OutputName, "w");
|
2014-01-07 06:27:03 +08:00
|
|
|
if (!OutputFile) return;
|
|
|
|
|
|
|
|
while (writeout_fn_head) {
|
|
|
|
struct writeout_fn_node *node = writeout_fn_head;
|
|
|
|
writeout_fn_head = writeout_fn_head->next;
|
|
|
|
node->fn();
|
|
|
|
free(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(OutputFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
void llvm_pgo_init(writeout_fn wfn) {
|
|
|
|
static int atexit_ran = 0;
|
|
|
|
|
|
|
|
if (wfn)
|
|
|
|
llvm_pgo_register_writeout_function(wfn);
|
|
|
|
|
|
|
|
if (atexit_ran == 0) {
|
|
|
|
atexit_ran = 1;
|
|
|
|
atexit(llvm_pgo_writeout_files);
|
|
|
|
}
|
|
|
|
}
|