1 | // Licensed to the .NET Foundation under one or more agreements. |
2 | // The .NET Foundation licenses this file to you under the MIT license. |
3 | // See the LICENSE file in the project root for more information. |
4 | |
5 | #include "stdafx.h" |
6 | #include "jitperf.h" |
7 | #include "perflog.h" |
8 | #include "clrhost.h" |
9 | #include "contract.h" |
10 | #include "utilcode.h" |
11 | #include "sstring.h" |
12 | |
13 | //============================================================================= |
14 | // ALL THE JIT PERF STATS GATHERING CODE IS COMPILED ONLY IF THE ENABLE_JIT_PERF WAS DEFINED. |
15 | #if defined(ENABLE_JIT_PERF) |
16 | |
17 | __int64 g_JitCycles = 0; |
18 | size_t g_NonJitCycles = 0; |
19 | CRITSEC_COOKIE g_csJit; |
20 | __int64 g_tlsJitCycles = 0; |
21 | int g_fJitPerfOn; |
22 | |
23 | size_t g_dwTlsx86CodeSize = 0; |
24 | size_t g_TotalILCodeSize = 0; |
25 | size_t g_Totalx86CodeSize = 0; |
26 | size_t g_TotalMethodsJitted = 0; |
27 | |
28 | void OutputStats () |
29 | { |
30 | CONTRACTL { |
31 | NOTHROW; |
32 | GC_NOTRIGGER; |
33 | CANNOT_TAKE_LOCK; |
34 | } CONTRACTL_END; |
35 | |
36 | LARGE_INTEGER cycleFreq; |
37 | if (QueryPerformanceFrequency (&cycleFreq)) |
38 | { |
39 | double dJitC = (double) g_JitCycles; |
40 | double dNonJitC = (double) g_NonJitCycles; |
41 | double dFreq = (double)cycleFreq.QuadPart; |
42 | double compileSpeed = (double)g_TotalILCodeSize/(dJitC/dFreq); |
43 | |
44 | PERFLOG((W("Jit Cycles" ), (dJitC - dNonJitC), CYCLES)); |
45 | PERFLOG((W("Jit Time" ), (dJitC - dNonJitC)/dFreq, SECONDS)); |
46 | PERFLOG((W("Non Jit Cycles" ), dNonJitC, CYCLES)); |
47 | PERFLOG((W("Non Jit Time" ), dNonJitC/dFreq, SECONDS)); |
48 | PERFLOG((W("Total Jit Cycles" ), dJitC, CYCLES)); |
49 | PERFLOG((W("Total Jit Time" ), dJitC/dFreq, SECONDS)); |
50 | PERFLOG((W("Methods Jitted" ), (UINT_PTR)g_TotalMethodsJitted, COUNT)); |
51 | PERFLOG((W("IL Code Compiled" ), (UINT_PTR)g_TotalILCodeSize, BYTES)); |
52 | PERFLOG((W("X86 Code Emitted" ), (UINT_PTR)g_Totalx86CodeSize, BYTES)); |
53 | // Included the perf counter description in this case because its not obvious what we are reporting. |
54 | PERFLOG((W("ExecTime" ), compileSpeed/1000, KBYTES_PER_SEC, W("IL Code compiled/sec" ))); |
55 | } |
56 | } |
57 | |
58 | void InitJitPerf(void) |
59 | { |
60 | CONTRACTL { |
61 | NOTHROW; |
62 | GC_NOTRIGGER; |
63 | CANNOT_TAKE_LOCK; |
64 | } CONTRACTL_END; |
65 | |
66 | InlineSString<4> lpszValue; |
67 | g_fJitPerfOn = WszGetEnvironmentVariable (W("JIT_PERF_OUTPUT" ), lpszValue); |
68 | if (g_fJitPerfOn) |
69 | { |
70 | g_csJit = ClrCreateCriticalSection(CrstJitPerf,CRST_UNSAFE_ANYMODE); |
71 | } |
72 | } |
73 | |
74 | void DoneJitPerfStats() |
75 | { |
76 | CONTRACTL { |
77 | NOTHROW; |
78 | GC_NOTRIGGER; |
79 | CANNOT_TAKE_LOCK; |
80 | } CONTRACTL_END; |
81 | |
82 | if (g_fJitPerfOn) |
83 | { |
84 | ClrDeleteCriticalSection(g_csJit); |
85 | |
86 | // Output stats to stdout and if necessary to the perf automation file. |
87 | OutputStats(); |
88 | } |
89 | |
90 | |
91 | } |
92 | |
93 | void StartNonJITPerfWorker(LARGE_INTEGER * pCycleStart) |
94 | { |
95 | CONTRACTL { |
96 | NOTHROW; |
97 | GC_NOTRIGGER; |
98 | CANNOT_TAKE_LOCK; |
99 | } CONTRACTL_END; |
100 | |
101 | pCycleStart->QuadPart = 0; |
102 | |
103 | size_t pTlsNonJitCycles = (size_t) ClrFlsGetValue (TlsIdx_JitPerf); |
104 | if ((pTlsNonJitCycles & 1) == 0 ) { /* odd value indicates we are in the EE */ |
105 | ClrFlsSetValue(TlsIdx_JitPerf, (LPVOID)(pTlsNonJitCycles + 1)); |
106 | QueryPerformanceCounter(pCycleStart); |
107 | } |
108 | } |
109 | |
110 | void StopNonJITPerfWorker(LARGE_INTEGER * pCycleStart) |
111 | { |
112 | CONTRACTL { |
113 | NOTHROW; |
114 | GC_NOTRIGGER; |
115 | CANNOT_TAKE_LOCK; |
116 | } CONTRACTL_END; |
117 | |
118 | LARGE_INTEGER CycleStop; |
119 | |
120 | if (pCycleStart->QuadPart != 0 && QueryPerformanceCounter(&CycleStop) ) { |
121 | size_t pTlsNonJitCycles = (size_t)ClrFlsGetValue (TlsIdx_JitPerf); |
122 | pTlsNonJitCycles += static_cast<size_t>(CycleStop.QuadPart - pCycleStart->QuadPart); |
123 | pTlsNonJitCycles &= ~1; /* even indicate we are not in EE */ |
124 | ClrFlsSetValue(TlsIdx_JitPerf, (LPVOID)(pTlsNonJitCycles)); |
125 | } |
126 | } |
127 | |
128 | |
129 | #endif //ENABLE_JIT_PERF |
130 | |
131 | |
132 | |