Skip to content

Commit 662d549

Browse files
committed
[Profile] Implement new API __llvm_profile_dump
The API is intended to be used by user to do fine grained (per-region) control of profile dumping. Differential Revision: http://reviews.llvm.org/D23106 git-svn-id: https://llvm.org/svn/llvm-project/compiler-rt/trunk@278092 91177308-0d34-0410-b5e6-96231b3b80d8
1 parent 9780916 commit 662d549

File tree

5 files changed

+120
-0
lines changed

5 files changed

+120
-0
lines changed

lib/profile/InstrProfiling.c

+11
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,16 @@ COMPILER_RT_VISIBILITY uint64_t __llvm_profile_get_magic(void) {
2727
: (INSTR_PROF_RAW_MAGIC_32);
2828
}
2929

30+
static unsigned ProfileDumped = 0;
31+
32+
COMPILER_RT_VISIBILITY unsigned lprofProfileDumped() {
33+
return ProfileDumped;
34+
}
35+
36+
COMPILER_RT_VISIBILITY void lprofSetProfileDumped() {
37+
ProfileDumped = 1;
38+
}
39+
3040
/* Return the number of bytes needed to add to SizeInBytes to make it
3141
* the result a multiple of 8.
3242
*/
@@ -68,4 +78,5 @@ COMPILER_RT_VISIBILITY void __llvm_profile_reset_counters(void) {
6878
}
6979
}
7080
}
81+
ProfileDumped = 0;
7182
}

lib/profile/InstrProfiling.h

+22
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,28 @@ void INSTR_PROF_VALUE_PROF_FUNC(
117117
*/
118118
int __llvm_profile_write_file(void);
119119

120+
/*!
121+
* \brief this is a wrapper interface to \c __llvm_profile_write_file.
122+
* After this interface is invoked, a arleady dumped flag will be set
123+
* so that profile won't be dumped again during program exit.
124+
* Invocation of interface __llvm_profile_reset_counters will clear
125+
* the flag. This interface is designed to be used to collect profile
126+
* data from user selected hot regions. The use model is
127+
* __llvm_profile_reset_counters();
128+
* ... hot region 1
129+
* __llvm_profile_dump();
130+
* .. some other code
131+
* __llvm_profile_reset_counters();
132+
* ... hot region 2
133+
* __llvm_profile_dump();
134+
*
135+
* It is expected that on-line profile merging is on with \c %m specifier
136+
* used in profile filename . If merging is not turned on, user is expected
137+
* to invoke __llvm_profile_set_filename to specify different profile names
138+
* for different regions before dumping to avoid profile write clobbering.
139+
*/
140+
int __llvm_profile_dump(void);
141+
120142
/*!
121143
* \brief Set the filename for writing instrumentation data.
122144
*

lib/profile/InstrProfilingFile.c

+18
Original file line numberDiff line numberDiff line change
@@ -522,6 +522,12 @@ int __llvm_profile_write_file(void) {
522522
const char *Filename;
523523
char *FilenameBuf;
524524

525+
if (lprofProfileDumped()) {
526+
PROF_NOTE("Profile data not written to file: %s.\n",
527+
"already written");
528+
return 0;
529+
}
530+
525531
Length = getCurFilenameLength();
526532
FilenameBuf = (char *)COMPILER_RT_ALLOCA(Length + 1);
527533
Filename = getCurFilename(FilenameBuf);
@@ -548,6 +554,18 @@ int __llvm_profile_write_file(void) {
548554
return rc;
549555
}
550556

557+
COMPILER_RT_VISIBILITY
558+
int __llvm_profile_dump(void) {
559+
if (!doMerging())
560+
PROF_WARN("Later invocation of __llvm_profile_dump can lead to clobbering "
561+
" of previously dumped profile data : %s. Either use \%m "
562+
"in profile name or change profile name before dumping.\n",
563+
"online profile merging is not on");
564+
int rc = __llvm_profile_write_file();
565+
lprofSetProfileDumped();
566+
return rc;
567+
}
568+
551569
static void writeFileWithoutReturn(void) { __llvm_profile_write_file(); }
552570

553571
COMPILER_RT_VISIBILITY

lib/profile/InstrProfilingInternal.h

+7
Original file line numberDiff line numberDiff line change
@@ -163,6 +163,13 @@ void lprofSetupValueProfiler();
163163
* to dump merged profile data into its own profile file. */
164164
uint64_t lprofGetLoadModuleSignature();
165165

166+
/*
167+
* Return non zero value if the profile data has already been
168+
* dumped to the file.
169+
*/
170+
unsigned lprofProfileDumped();
171+
void lprofSetProfileDumped();
172+
166173
COMPILER_RT_VISIBILITY extern char *(*GetEnvHook)(const char *);
167174
COMPILER_RT_VISIBILITY extern void (*FreeHook)(void *);
168175
COMPILER_RT_VISIBILITY extern uint8_t *DynamicBufferIOBuffer;

test/profile/instrprof-dump.c

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
/*
2+
RUN: rm -fr %t.profdir
3+
RUN: %clang_profgen=%t.profdir/default_%m.profraw -o %t -O2 %s
4+
RUN: %run %t 2>&1 | FileCheck %s --check-prefix=NO_EXIT_WRITE
5+
RUN: llvm-profdata merge -o %t.profdata %t.profdir
6+
RUN: %clang_profuse=%t.profdata -o - -S -emit-llvm %s | FileCheck %s --check-prefix=PROF
7+
8+
NO_EXIT_WRITE: Profile data not written to file: already written
9+
*/
10+
11+
int __llvm_profile_dump(void);
12+
void __llvm_profile_reset_counters(void);
13+
int foo(int);
14+
int bar(int);
15+
int skip(int);
16+
17+
int main(int argc, const char *argv[]) {
18+
int Ret = foo(0); /* region 1 */
19+
__llvm_profile_dump();
20+
21+
/* not profiled -- cleared later. */
22+
skip(0); /* skipped region */
23+
24+
__llvm_profile_reset_counters();
25+
Ret += bar(0); /* region 2 */
26+
__llvm_profile_dump();
27+
28+
skip(1);
29+
30+
__llvm_profile_reset_counters();
31+
/* foo's profile will be merged. */
32+
foo(1); /* region 3 */
33+
__llvm_profile_dump();
34+
35+
return Ret;
36+
}
37+
38+
__attribute__((noinline)) int foo(int X) {
39+
/* PROF: define {{.*}} @foo({{.*}}!prof ![[ENT:[0-9]+]]
40+
PROF: br i1 %{{.*}}, label %{{.*}}, label %{{.*}}, !prof ![[PD1:[0-9]+]]
41+
*/
42+
return X <= 0 ? -X : X;
43+
}
44+
45+
__attribute__((noinline)) int skip(int X) {
46+
/* PROF: define {{.*}} @skip(
47+
PROF: br i1 %{{.*}}, label %{{.*}}, label %{{[^,]+$}}
48+
*/
49+
return X <= 0 ? -X : X;
50+
}
51+
52+
__attribute__((noinline)) int bar(int X) {
53+
/* PROF-LABEL: define {{.*}} @bar(
54+
PROF: br i1 %{{.*}}, label %{{.*}}, label %{{.*}}, !prof ![[PD2:[0-9]+]]
55+
*/
56+
return X <= 0 ? -X : X;
57+
}
58+
59+
/*
60+
PROF: ![[ENT]] = !{!"function_entry_count", i64 2}
61+
PROF: ![[PD1]] = !{!"branch_weights", i32 2, i32 2}
62+
*/

0 commit comments

Comments
 (0)