-
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathprofile.e
418 lines (397 loc) · 14.9 KB
/
profile.e
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
--
-- profile.e
--
-- Dump execution profile results to ex.pro
--
-- This file is part of p.exw. It is invoked after code interpretation,
-- if "with profile[_time]" was encountered somewhere in the source.
--
-- ptab is created in p.exw just before interpretation and is a 1:1 map
-- for filenames/filepaths/allfiles, initially a 0 for each and every
-- source line read in and compiled.
--
--
-- If showTally is 1, it sums the profile_time percentages as actually printed.
-- Admittedly this is of limited use, except to unequivocally state that::
--
-- ""Naturally this will not necessarily add up to exactly 100% due to rounding.
-- There is an obvious bias in that once-only initialisation code is likely to
-- contribute >0 but <=0.005 more often than not, and perfectly possible that
-- could be true of all lines, leading to a grand total of 0.00. It is equally
-- possible that every line only just rounds up, leading to a total of 199.99%.
-- OTOH, eg 99.93 does give some confidence the numbers are mostly accurate.""
--
constant showTally = 0
-- As per showTally, each "end function/procedure/type" can show a sub-total:
constant showSubTotals = 1
global procedure dump_profile()
integer fn, pathno
atom c, ttot, atot, rtot, stot
object ct
sequence path, name, texti, ctab, pti
string p72, tj, stim
integer ch, mark, tlim
-- puts(1,"writing profile results to ex.pro...")
puts(1,"writing profile results to "¤t_dir()&"\\ex.pro...")
fn = open("ex.pro","w")
if sequence(opstat) then
puts(fn,"opstats: [Meaning is dependent on your mods to p.exw, for an example see Branch().]\n")
for i=1 to length(opNames) do
name = opNames[i]
c = opstat[i]
--DEV 64bit..?
if c=#3FFFFFFF then
printf(fn,"%-13s ** maxed out **\n",{name})
else
printf(fn,"%-13s %10d\n",{name,c})
end if
end for
puts(fn,"\n")
end if
if profileon=OptProfileTime then
puts(fn,"-- Phix execution profile_time listing.\n")
puts(fn,"-- Left margin shows percentage of time spent executing line.\n")
puts(fn,"-- ( spaces means 0, 0.00 means >0 and <=0.005,\n")
puts(fn,"-- \"> markers\" on lines >=0.10 are for Edita\\F4 )\n")
rtot = 0
tlim = 1
stim = sprintf("%7.2f|",tlim/10)
elsif profileon=OptProfile then
puts(fn,"-- Phix execution profile_count listing.\n")
puts(fn,"-- Left margin shows count of times line was executed.\n")
tlim = 10000
else
close(fn)
return
end if
-- While ttot is only needed for OptProfileTime, skip is good for both.
ttot = 0
ctab = repeat(0,length(ptab))
for i=1 to length(ptab) do
atot = 0
pti = ptab[i]
for j=1 to length(pti) do
atot += pti[j]
end for
if atot then
ttot += atot
ctab[i] = atot
else
ptab[i] = {} -- skip output if unused
end if
end for
-- second pass to skip insignificant includes
if profileon=OptProfileTime then
if ttot then
for i=1 to length(ptab) do
c = ctab[i]/ttot
if c<0.005 then -- < 0.5%
ptab[i] = c
end if
end for
else
ptab = repeat(0,length(ptab))
end if
end if
mark = 0
stot = 0
for i=1 to length(allfiles) do
name = filenames[i]
pathno = name[1]
name = name[2]
path = filepaths[pathno]
if equal(expandedYet[i],0) then
text = allfiles[i]
exptext[i] = expandIntoLines()
expandedYet[i] = linestarts
end if
texti = exptext[i]
ct = ptab[i]
if sequence(ct)
and length(ct)>0 then
ctab = ct
puts(fn,"--"&path&name&":\n"&repeat('=',length(path)+length(name)+2)&"\n")
for j=1 to length(texti) do
c = ctab[j]
if c then
if profileon=OptProfileTime then
p72 = sprintf("%7.2f|",100*c/ttot)
if showTally then
-- sum percentages as actually printed
c = 0
for k=1 to length(p72)-1 do
ch = p72[k]
if ch>='0' then
-- if ch>'0' then
-- mark = 1
-- end if
c = c*10+(ch-'0')
elsif ch='.' then
for l=k+1 to length(p72)-1 do
ch = p72[l]
-- if ch>'0' then
-- mark = 1
-- end if
c += (ch-'0')/power(10,l-k)
end for
exit
end if
end for
rtot += c
end if
-- if not equal(p72," 0.00|") then
-- if compare(p72," 0.10|")>=0 then
if compare(p72,stim)>=0 then
mark = 1
end if
else
p72 = sprintf("%7d|",c)
-- for k=1 to length(p72)-1 do
-- if p72[k]>'0' then
-- mark = 1
-- exit
-- end if
-- end for
if c>=tlim/10 then
mark = 1
end if
end if
stot += c
else
p72 = " |"
end if
if mark and p72[1]=' ' then
p72[1] = '>'
mark = 0
end if
tj = texti[j]
puts(fn,p72&tj)
if showSubTotals and length(tj)>=1 and find(tj[1],"gpfte") then
if match("global procedure",tj)=1
or match("global function",tj)=1
or match("global type",tj)=1
or match("procedure",tj)=1
or match("function",tj)=1
or match("type",tj)=1 then
stot = 0
elsif match("end procedure",tj)=1
or match("end function",tj)=1
or match("end type",tj)=1 then
if stot then
if profileon=OptProfileTime then
if not showTally then -- stot already a %age, see above
stot = 100*stot/ttot
end if
p72 = sprintf("=======|\n%7.2f|\n",stot)
else
p72 = sprintf("=======|\n%7d|\n",stot)
end if
if stot>=tlim and p72[10]=' ' then
p72[10] = '>'
end if
puts(fn,p72)
end if
stot = 0
end if
end if
end for
elsif atom(ct) then
-- See above, where we did:
-- if c<0.005 then -- < 0.5%
-- ptab[i] = c
-- if profileon=OptProfileTime then -- added 13/1/09
printf(fn,"%s%s - skipped (insignificant: %3.2f%%)\n",{path,name,ct*100})
-- else
-- printf(fn,"%s%s - skipped (execution count of zero)\n",{path,name})
-- end if
else
-- See above, where we did:
-- ptab[i] = {} -- skip output if unused
-- (because ptab[i] was nowt but zeroes)
puts(fn,path&name&" - skipped\n")
end if
end for
if showTally and profileon=OptProfileTime then
printf(fn,"=======\n%7.2f (Total of rounded percentages as printed)\n",rtot)
end if
close(fn)
end procedure
--dev
--/*
integer profileon = 0
--NO, we need this in the optable:
global procedure profile(integer level)
-- This just sets profileon, it is opLnp/opLnpt (profile/profile_time) which do the real work.
if level<0 or level>1 then
--e90atpmb01 ; argument to profile() must be 0 or 1
?9/0
end if
profileon := level
end procedure
#ilASM{ jmp :fin
; (see profile.e, incomplete)
opProfile: ; profile([edi]) -- (where [edi] is 0|1)
;---------
trc opProfile
; This just sets profileon, it is opLnp/opLnpt (profile/profile_time) which do the real work.
mov eax,[edi]
cmp eax,h4
jl @f
opProfile92: ; exception here mapped to e92vhnbaavedifeh
cmp byte[ebx+eax*4-1],0x12
jne e90atpmb01 ; argument to profile() must be 0 or 1
fld qword[ebx+eax*4]
fldcw [down53]
fistp qword[FltWrk]
fldcw [near53]
mov eax,[FltWrk]
@@:
cmp eax,1
ja e90atpmb01 ; argument to profile must be 0 or 1
mov [profileon],eax
ret
-- ::e91pie4
-- pop eax
-- ::e91pie
-- ?9/0
%:opLnp -- profile(count times executed) this line. Only valid when interpreting
-------
mov ecx,[profileon]
test eax,eax
-- jle :e91pie -- profile internal error (lineno<=0)
test ecx,ecx
jz @f
mov esi,[ebp+24] -- symtab
mov ecx,[ebp+8] -- current routine
mov esi,[esi+ecx*4-4] -- symtab[current_routine]
shl esi,2 -- convert to real address
mov esi,[esi+8] -- symtab[current_routine][S_FPno(=3)]
-- test esi,esi
-- jle e91pie
mov edi,[ptab] -- raw address
-- cmp esi,[edi-12] -- check length
-- jg e91pie
mov edi,[edi+esi*4-4] -- ptab[fileno]
shl edi,2
-- cmp eax,[edi-12] -- check length
-- jg e91pie
--DEV prevent this going over #3FFFFFFF
inc dword[edi+eax*4-4] -- ptab[fileno][lineno]
@@:
ret
:%opLnpt -- profile_time this line. Only valid when interpreting
--------
-- Times are recorded on move to new line; first and last are
-- not properly accounted for (on this *asm*! backend).
-- NB: the asm b/end uses rdtsc rather than time()
mov ecx,[profileon]
test ecx,ecx
jz @f
push eax ; save lineno [DEV *4?]
-- first, update [ptaddr] with diff from [pteax/edx].
cmp [pteax],0 -- DEV should this be [ptaddr]?
jne :lnpt_eax_not_zero
cmp [ptedx],0
jz :lnpt_end_if
::lnpt_eax_not_zero
rdtsc
sub eax,[pteax]
sbb edx,[ptedx]
and edx,0x7FFFFFFF
push edx
push eax
fild qword[esp]
add esp,8
mov edx,[ptaddr]
mov edx,[edx]
cmp edx,h4
jle :lnpt_new_float
shl edx,2
cmp byte[edx-1],0x12
je :e91pie4
fld qword[edx]
faddp
fstp qword[edx]
jmp :lnpt_end_if
::lnpt_new_float
--DEV check which:
mov edx,[ptaddr]
mov edi,[ptaddr]
call :%pStoreFlt
mov [edx],eax
::lnpt_end_if
rdtsc
mov [pteax],eax
mov [ptedx],edx
mov ecx,[ebp+8] -- current routine
mov esi,[ebp+24] -- symtab
mov esi,[esi+ecx*4-4] -- symtab[current_routine]
-- shl esi,2 -- convert to real address
mov esi,[ebx+esi*4+8] -- symtab[current_routine][S_FPno(=3)]
-- test esi,esi
-- jle :e91pie4
mov edi,[ptab] -- raw address
-- cmp esi,[edi-12] -- check length
-- jg :e91pie4
mov edi,[edi+esi*4-4] -- ptab[fileno]
shl edi,2
pop eax -- lineno
-- cmp eax,[edi-12] -- check length
-- jg :e91pie
-- save addr of ptab[fno][line] instead of fno&line:
lea edx,[edi+eax*4-4]
mov [ptaddr],edx -- (updateme at next opLnpt)
@@:
ret
--*/
-- Final note:
-- There is no way to profile a pre-built executable, since allfiles is
-- not available, and hence both creating ptab and later printing it
-- is problematic at best.
-- I have considered this and decided it is impractical and pointless.
-- I will however list my findings:
-- First job would be to change lineinfo() in pemit.e, specifically
-- the 'if not bind then' condition. Obviously I am not about to do
-- that on any 'official' release due to the performance overhead.
-- It is critical that ptab matches source file lengths at the time
-- of compilation. The best way to do this is save an ltab, which
-- contains the needed lengths of each ptab element. A reasonable
-- place for this is near T_fileset. The user app needs to create
-- a ptab, before any profile or profile_time statement. You would
-- need some way to pass this to the interpreter as opInterp does.
--DEV newEmit:
-- [Tip: you can retrieve symtab/ltab via opGetST, and would need
-- a new op like opSetDbg to pass over the new ptab.]
-- If the sources have been edited since the bind, then at best,
-- costs will show against the wrong lines; were you to base the
-- lengths of ptab on the current source, and they were now shorter,
-- you'd get a memory corruption [opLnp/t do no bounds checking!]
-- probably followed by some indecipherable machine error, and/or
-- not altogether unlikely a machine lockup.
-- At the end of the program, the app could recreate allfiles and
-- invoke this. It would have to define/set the following globals:
-- global sequence ptab, allfiles, filenames, filepaths
-- (expandedYet/expfiles can be recreated here)
-- global integer profileon
-- global constant OptProfile=1, OptProfileTime=2
-- Obviously, profileon must match that at compilation time, or
-- else the results are undefined, almost certainly meaningless.
-- Of course you should not forget that "with profile" interferes
-- with program execution, eg if I have:
-- integer i,j,k
-- j = i
-- k = j+2
-- then without profile, I get:
-- mov eax,[i]
-- mov [j],eax
-- add eax,2
-- mov [k],eax
-- whereas with it, I get:
-- call opLnp
-- mov eax,[i]
-- mov [j],eax
-- call opLnp
-- mov eax,[j]
-- add eax,2
-- mov [k],eax
-- (verify this by running "p -d! test" and examining list.asm)