2020-10-30 17:32:40 +01:00
|
|
|
local table_sort = table.sort
|
|
|
|
local string_rep = string.rep
|
|
|
|
local string_format = string.format
|
|
|
|
local debug_getinfo = debug.getinfo
|
2020-12-29 00:08:53 +01:00
|
|
|
local Color = require 'utils.color_presets'
|
2020-10-30 17:32:40 +01:00
|
|
|
|
2020-11-04 17:14:30 +01:00
|
|
|
local Profiler = {
|
|
|
|
-- Call
|
|
|
|
CallTree = nil,
|
|
|
|
IsRunning = false
|
2020-10-30 17:32:40 +01:00
|
|
|
}
|
|
|
|
|
2020-12-29 00:08:53 +01:00
|
|
|
-- we can have this on runtime,
|
|
|
|
-- but never ever can a player run this without notifying us.
|
|
|
|
local allowed = {
|
|
|
|
['Gerkiz'] = true,
|
|
|
|
['mewmew'] = true
|
|
|
|
}
|
|
|
|
|
2020-11-04 17:14:30 +01:00
|
|
|
local ignoredFunctions = {
|
|
|
|
[debug.sethook] = true
|
2020-10-30 17:32:40 +01:00
|
|
|
}
|
|
|
|
|
2020-11-04 17:14:30 +01:00
|
|
|
local namedSources = {
|
|
|
|
['[string "local n, v = "serpent", "0.30" -- (C) 2012-17..."]'] = 'serpent'
|
2020-10-30 17:32:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
local function startCommand(command)
|
2020-12-29 00:08:53 +01:00
|
|
|
local player = game.player
|
|
|
|
if player then
|
|
|
|
if player ~= nil then
|
|
|
|
if not player.admin then
|
|
|
|
local p = player.print
|
|
|
|
p('[ERROR] Only admins are allowed to run this command!', Color.fail)
|
|
|
|
return
|
|
|
|
else
|
|
|
|
if allowed[player.name] then
|
|
|
|
Profiler.Start(command.parameter ~= nil)
|
|
|
|
elseif _DEBUG then
|
|
|
|
Profiler.Start(command.parameter ~= nil)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2020-10-30 17:32:40 +01:00
|
|
|
end
|
|
|
|
local function stopCommand(command)
|
2020-12-29 00:08:53 +01:00
|
|
|
local player = game.player
|
|
|
|
if player then
|
|
|
|
if player ~= nil then
|
|
|
|
if not player.admin then
|
|
|
|
local p = player.print
|
|
|
|
p('[ERROR] Only admins are allowed to run this command!', Color.fail)
|
|
|
|
return
|
|
|
|
else
|
|
|
|
if allowed[player.name] then
|
|
|
|
Profiler.Stop(command.parameter ~= nil, nil)
|
|
|
|
elseif _DEBUG then
|
|
|
|
Profiler.Stop(command.parameter ~= nil, nil)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2020-10-30 17:32:40 +01:00
|
|
|
end
|
|
|
|
ignoredFunctions[startCommand] = true
|
|
|
|
ignoredFunctions[stopCommand] = true
|
|
|
|
|
2020-11-04 17:14:30 +01:00
|
|
|
commands.add_command('startProfiler', 'Starts profiling', startCommand)
|
|
|
|
commands.add_command('stopProfiler', 'Stops profiling', stopCommand)
|
2020-10-30 17:32:40 +01:00
|
|
|
|
|
|
|
--local assert_raw = assert
|
|
|
|
--function assert(expr, ...)
|
|
|
|
-- if not expr then
|
|
|
|
-- Profiler.Stop(false, "Assertion failed")
|
|
|
|
-- end
|
|
|
|
-- assert_raw(expr, ...)
|
|
|
|
--end
|
|
|
|
local error_raw = error
|
2021-03-24 20:14:55 +01:00
|
|
|
--luacheck: ignore error
|
2020-10-30 17:32:40 +01:00
|
|
|
function error(...)
|
2020-11-04 17:14:30 +01:00
|
|
|
Profiler.Stop(false, 'Error raised')
|
|
|
|
error_raw(...)
|
2020-10-30 17:32:40 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
function Profiler.Start(excludeCalledMs)
|
2020-11-04 17:14:30 +01:00
|
|
|
if Profiler.IsRunning then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
local create_profiler = game.create_profiler
|
|
|
|
|
|
|
|
Profiler.IsRunning = true
|
|
|
|
|
|
|
|
Profiler.CallTree = {
|
|
|
|
name = 'root',
|
|
|
|
calls = 0,
|
|
|
|
profiler = create_profiler(),
|
|
|
|
next = {}
|
|
|
|
}
|
|
|
|
|
|
|
|
-- Array of Call
|
|
|
|
local stack = {[0] = Profiler.CallTree}
|
|
|
|
local stack_count = 0
|
|
|
|
|
|
|
|
debug.sethook(
|
|
|
|
function(event)
|
|
|
|
local info = debug_getinfo(2, 'nSf')
|
|
|
|
|
|
|
|
if ignoredFunctions[info.func] then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if event == 'call' or event == 'tail call' then
|
|
|
|
local prevCall = stack[stack_count]
|
|
|
|
if excludeCalledMs then
|
|
|
|
prevCall.profiler.stop()
|
|
|
|
end
|
|
|
|
|
|
|
|
local what = info.what
|
|
|
|
local name
|
|
|
|
if what == 'C' then
|
|
|
|
name = string_format('C function %q', info.name or 'anonymous')
|
|
|
|
else
|
|
|
|
local source = info.short_src
|
|
|
|
local namedSource = namedSources[source]
|
|
|
|
if namedSource ~= nil then
|
|
|
|
source = namedSource
|
|
|
|
elseif string.sub(source, 1, 1) == '@' then
|
|
|
|
source = string.sub(source, 1)
|
|
|
|
end
|
|
|
|
name = string_format('%q in %q, line %d', info.name or 'anonymous', source, info.linedefined)
|
|
|
|
end
|
|
|
|
|
|
|
|
local prevCall_next = prevCall.next
|
|
|
|
if prevCall_next == nil then
|
|
|
|
prevCall_next = {}
|
|
|
|
prevCall.next = prevCall_next
|
|
|
|
end
|
|
|
|
|
|
|
|
local currCall = prevCall_next[name]
|
|
|
|
local profilerStartFunc
|
|
|
|
if currCall == nil then
|
|
|
|
local prof = create_profiler()
|
|
|
|
currCall = {
|
|
|
|
name = name,
|
|
|
|
calls = 1,
|
|
|
|
profiler = prof
|
|
|
|
}
|
|
|
|
prevCall_next[name] = currCall
|
|
|
|
profilerStartFunc = prof.reset
|
|
|
|
else
|
|
|
|
currCall.calls = currCall.calls + 1
|
|
|
|
profilerStartFunc = currCall.profiler.restart
|
|
|
|
end
|
|
|
|
|
|
|
|
stack_count = stack_count + 1
|
|
|
|
stack[stack_count] = currCall
|
|
|
|
|
|
|
|
profilerStartFunc()
|
|
|
|
end
|
|
|
|
|
|
|
|
if event == 'return' or event == 'tail call' then
|
|
|
|
if stack_count > 0 then
|
|
|
|
stack[stack_count].profiler.stop()
|
|
|
|
stack[stack_count] = nil
|
|
|
|
stack_count = stack_count - 1
|
|
|
|
|
|
|
|
if excludeCalledMs then
|
|
|
|
stack[stack_count].profiler.restart()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end,
|
|
|
|
'cr'
|
|
|
|
)
|
2020-10-30 17:32:40 +01:00
|
|
|
end
|
|
|
|
ignoredFunctions[Profiler.Start] = true
|
|
|
|
|
|
|
|
local function DumpTree(averageMs)
|
2020-11-04 17:14:30 +01:00
|
|
|
local function sort_Call(a, b)
|
|
|
|
return a.calls > b.calls
|
|
|
|
end
|
|
|
|
local fullStr = {''}
|
|
|
|
local str = fullStr
|
|
|
|
local line = 1
|
|
|
|
|
|
|
|
local function recurse(curr, depth)
|
|
|
|
local sort = {}
|
|
|
|
local i = 1
|
|
|
|
for k, v in pairs(curr) do
|
|
|
|
sort[i] = v
|
|
|
|
i = i + 1
|
|
|
|
end
|
|
|
|
table_sort(sort, sort_Call)
|
|
|
|
|
2021-03-24 20:14:55 +01:00
|
|
|
for ii = 1, #sort do
|
|
|
|
local call = sort[ii]
|
2020-11-04 17:14:30 +01:00
|
|
|
|
|
|
|
if line >= 19 then --Localised string can only have up to 20 parameters
|
|
|
|
local newStr = {''} --So nest them!
|
|
|
|
str[line + 1] = newStr
|
|
|
|
str = newStr
|
|
|
|
line = 1
|
|
|
|
end
|
|
|
|
|
|
|
|
if averageMs then
|
|
|
|
call.profiler.divide(call.calls)
|
|
|
|
end
|
|
|
|
|
2020-12-29 00:08:53 +01:00
|
|
|
str[line + 1] = string_format('\n%s%dx %s. %s ', string_rep('\t', depth), call.calls, call.name, averageMs and 'Average' or 'Total')
|
2020-11-04 17:14:30 +01:00
|
|
|
str[line + 2] = call.profiler
|
|
|
|
line = line + 2
|
|
|
|
|
|
|
|
local next = call.next
|
|
|
|
if next ~= nil then
|
|
|
|
recurse(next, depth + 1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if Profiler.CallTree.next ~= nil then
|
|
|
|
recurse(Profiler.CallTree.next, 0)
|
|
|
|
return fullStr
|
|
|
|
end
|
|
|
|
return 'No calls'
|
2020-10-30 17:32:40 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
function Profiler.Stop(averageMs, message)
|
2020-11-04 17:14:30 +01:00
|
|
|
if not Profiler.IsRunning then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
debug.sethook()
|
|
|
|
|
|
|
|
local text = {'', '\n\n----------PROFILER DUMP----------\n', DumpTree(averageMs), '\n\n----------PROFILER STOPPED----------\n'}
|
|
|
|
if message ~= nil then
|
|
|
|
text[#text + 1] = string.format('Reason: %s\n', message)
|
|
|
|
end
|
|
|
|
log(text)
|
|
|
|
Profiler.CallTree = nil
|
|
|
|
Profiler.IsRunning = false
|
2020-10-30 17:32:40 +01:00
|
|
|
end
|
|
|
|
ignoredFunctions[Profiler.Stop] = true
|
|
|
|
|
2020-11-04 17:14:30 +01:00
|
|
|
return Profiler
|