2024-04-21 19:15:22 +02:00
|
|
|
if (debug.sethook) then
|
|
|
|
local table_sort = table.sort
|
|
|
|
local string_rep = string.rep
|
|
|
|
local string_format = string.format
|
|
|
|
local debug_getinfo = debug.getinfo
|
|
|
|
local Color = require 'utils.color_presets'
|
|
|
|
local Task = require 'utils.task'
|
|
|
|
local Token = require 'utils.token'
|
|
|
|
local Event = require 'utils.event'
|
|
|
|
|
|
|
|
local Public = {
|
|
|
|
call_tree = nil,
|
|
|
|
is_running = false
|
|
|
|
}
|
|
|
|
|
|
|
|
local stop_profiler_token =
|
|
|
|
Token.register(
|
|
|
|
function()
|
|
|
|
Public.stop()
|
|
|
|
game.print('[PROFILER] Stopped!')
|
|
|
|
log('[PROFILER] Stopped!')
|
|
|
|
end
|
|
|
|
)
|
|
|
|
|
|
|
|
-- we can have this on runtime,
|
|
|
|
-- but never ever can a player run this without notifying us.
|
|
|
|
local allowed = {
|
|
|
|
['Gerkiz'] = true,
|
|
|
|
['mewmew'] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
local ignored_functions = {
|
|
|
|
[debug.sethook] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
local named_sources = {
|
|
|
|
['[string "local n, v = "serpent", "0.30" -- (C) 2012-17..."]'] = 'serpent'
|
|
|
|
}
|
|
|
|
|
|
|
|
local function start_command(command)
|
|
|
|
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
|
|
|
|
Public.start(command.parameter ~= nil)
|
|
|
|
elseif _DEBUG then
|
|
|
|
Public.start(command.parameter ~= nil)
|
|
|
|
end
|
2020-12-29 01:08:53 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2024-04-21 19:15:22 +02:00
|
|
|
local function stop_command(command)
|
|
|
|
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
|
|
|
|
Public.stop(command.parameter ~= nil, nil)
|
|
|
|
elseif _DEBUG then
|
|
|
|
Public.stop(command.parameter ~= nil, nil)
|
|
|
|
end
|
2020-12-29 01:08:53 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2024-04-21 19:15:22 +02:00
|
|
|
ignored_functions[start_command] = true
|
|
|
|
ignored_functions[stop_command] = true
|
|
|
|
|
|
|
|
commands.add_command('start_profiler', 'Starts profiling', start_command)
|
|
|
|
commands.add_command('stop_profiler', 'Stops profiling', stop_command)
|
|
|
|
|
|
|
|
--local assert_raw = assert
|
|
|
|
--function assert(expr, ...)
|
|
|
|
-- if not expr then
|
|
|
|
-- Public.stop(false, "Assertion failed")
|
|
|
|
-- end
|
|
|
|
-- assert_raw(expr, ...)
|
|
|
|
--end
|
|
|
|
local error_raw = error
|
|
|
|
--luacheck: ignore error
|
|
|
|
function error(...)
|
|
|
|
Public.stop(false, 'Error raised')
|
|
|
|
error_raw(...)
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
function Public.start(exclude_called_ms)
|
|
|
|
if Public.is_running then
|
|
|
|
return
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
local create_profiler = game.create_profiler
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
Public.is_running = true
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
Public.call_tree = {
|
|
|
|
name = 'root',
|
|
|
|
calls = 0,
|
|
|
|
profiler = create_profiler(),
|
|
|
|
next = {}
|
|
|
|
}
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
-- Array of Call
|
|
|
|
local stack = {[0] = Public.call_tree}
|
|
|
|
local stack_count = 0
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
debug.sethook(
|
|
|
|
function(event)
|
|
|
|
local info = debug_getinfo(2, 'nSf')
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
if ignored_functions[info.func] then
|
|
|
|
return
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
if event == 'call' or event == 'tail call' then
|
|
|
|
local prev_call = stack[stack_count]
|
|
|
|
if exclude_called_ms and prev_call then
|
|
|
|
prev_call.profiler.stop()
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
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 = named_sources[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
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
local prev_call_next = prev_call.next
|
|
|
|
if prev_call_next == nil then
|
|
|
|
prev_call_next = {}
|
|
|
|
prev_call.next = prev_call_next
|
|
|
|
end
|
|
|
|
|
|
|
|
local currCall = prev_call_next[name]
|
|
|
|
local profilerStartFunc
|
|
|
|
if currCall == nil then
|
|
|
|
local prof = create_profiler()
|
|
|
|
currCall = {
|
|
|
|
name = name,
|
|
|
|
calls = 1,
|
|
|
|
profiler = prof
|
|
|
|
}
|
|
|
|
prev_call_next[name] = currCall
|
|
|
|
profilerStartFunc = prof.reset
|
|
|
|
else
|
|
|
|
currCall.calls = currCall.calls + 1
|
|
|
|
profilerStartFunc = currCall.profiler.restart
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
stack_count = stack_count + 1
|
|
|
|
stack[stack_count] = currCall
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
profilerStartFunc()
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
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
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
if exclude_called_ms then
|
|
|
|
stack[stack_count].profiler.restart()
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
|
|
|
end
|
2024-04-21 19:15:22 +02:00
|
|
|
end,
|
|
|
|
'cr'
|
|
|
|
)
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
2024-04-21 19:15:22 +02:00
|
|
|
ignored_functions[Public.start] = true
|
|
|
|
|
|
|
|
local function dump_tree(averageMs)
|
|
|
|
local function sort_Call(a, b)
|
|
|
|
return a.calls > b.calls
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
2024-04-21 19:15:22 +02:00
|
|
|
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)
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
for ii = 1, #sort do
|
|
|
|
local call = sort[ii]
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02: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
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
if averageMs then
|
|
|
|
call.profiler.divide(call.calls)
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
str[line + 1] = string_format('\n%s%dx %s. %s ', string_rep('\t', depth), call.calls, call.name, averageMs and 'Average' or 'Total')
|
|
|
|
str[line + 2] = call.profiler
|
|
|
|
line = line + 2
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
local next = call.next
|
|
|
|
if next ~= nil then
|
|
|
|
recurse(next, depth + 1)
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
|
|
|
end
|
2024-04-21 19:15:22 +02:00
|
|
|
if Public.call_tree.next ~= nil then
|
|
|
|
recurse(Public.call_tree.next, 0)
|
|
|
|
return fullStr
|
|
|
|
end
|
|
|
|
return 'No calls'
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
2020-10-30 18:32:40 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
function Public.stop(averageMs, message)
|
|
|
|
if not Public.is_running then
|
|
|
|
return
|
|
|
|
end
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
debug.sethook()
|
2020-11-04 18:14:30 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
local text = {'', '\n\n----------PROFILER DUMP----------\n', dump_tree(averageMs), '\n\n----------PROFILER STOPPED----------\n'}
|
|
|
|
if message ~= nil then
|
|
|
|
text[#text + 1] = string.format('Reason: %s\n', message)
|
|
|
|
end
|
|
|
|
log(text)
|
|
|
|
Public.call_tree = nil
|
|
|
|
Public.is_running = false
|
|
|
|
end
|
|
|
|
ignored_functions[Public.stop] = true
|
|
|
|
|
2024-06-01 22:10:03 +02:00
|
|
|
if _PROFILE then
|
2024-04-21 19:15:22 +02:00
|
|
|
Event.on_init(
|
|
|
|
function()
|
|
|
|
game.print('[PROFILER] Started!')
|
|
|
|
log('[PROFILER] Started!')
|
|
|
|
Public.start()
|
|
|
|
Task.set_timeout_in_ticks(3600, stop_profiler_token)
|
|
|
|
end
|
|
|
|
)
|
2020-11-04 18:14:30 +02:00
|
|
|
end
|
2023-06-17 23:32:35 +02:00
|
|
|
|
2024-04-21 19:15:22 +02:00
|
|
|
return Public
|
2024-01-29 00:13:32 +02:00
|
|
|
end
|