# cython: c_string_type=unicode, c_string_encoding=utf8 from godot._hazmat.gdnative_api_struct cimport ( godot_pluginscript_language_data, godot_pluginscript_profiling_data, ) from godot._hazmat.gdapi cimport pythonscript_gdapi10 as gdapi10 import sys from collections import defaultdict from time import perf_counter # TODO: should be greatly improved by using cdef struct and godot_string_name everywhere class MethProfile: __slots__ = ( "call_count", "self_time", "total_time", "cur_frame_call_count", "cur_frame_self_time", "cur_frame_total_time", "last_frame_call_count", "last_frame_self_time", "last_frame_total_time", ) def __init__(self): self.call_count = 0 self.self_time = 0 self.total_time = 0 self.cur_frame_call_count = 0 self.cur_frame_self_time = 0 self.cur_frame_total_time = 0 self.last_frame_call_count = 0 self.last_frame_self_time = 0 self.last_frame_total_time = 0 class FuncCallProfile: __slots__ = ("signature", "start", "end", "out_of_func_time") def __init__(self, signature): self.signature = signature self.start = perf_counter() self.end = None # Time spend calling another function self.out_of_func_time = 0 def add_out_of_func(self, time): self.out_of_func_time += time def get_self_time(self): return self.get_total_time() - self.out_of_func_time def done(self): self.end = perf_counter() def get_total_time(self): return self.end - self.start class Profiler: def __init__(self): self.per_meth_profiling = defaultdict(MethProfile) self._profile_stack = [] @property def _per_thread_profile_stack(self): return self._profile_stack # TODO: Make this thread safe # Not sure if multithreading is supported by sys.setprofile anyway... # loc = threading.local() # key = 'profile_stack_%s' % id(self) # stack = getattr(loc, key, None) # if not stack: # stack = [] # setattr(loc, key, stack) # return stack def next_frame(self): for meth_profile in self.per_meth_profiling.values(): meth_profile.call_count = meth_profile.cur_frame_call_count meth_profile.self_time = meth_profile.cur_frame_self_time meth_profile.total_time = meth_profile.cur_frame_total_time meth_profile.last_frame_call_count = meth_profile.cur_frame_call_count meth_profile.last_frame_self_time = meth_profile.cur_frame_self_time meth_profile.last_frame_total_time = meth_profile.cur_frame_total_time meth_profile.cur_frame_call_count = 0 meth_profile.cur_frame_self_time = 0 meth_profile.cur_frame_total_time = 0 def get_profilefunc(self): def profilefunc(frame, event, arg): # TODO: improve this hack to avoid profiling builtins functions if frame.f_code.co_filename.startswith("<"): return if event in ("call", "c_call"): # TODO generate signature ahead of time and store it into the object signature = "{path}::{line}::{name}".format( path=frame.f_code.co_filename, line=frame.f_lineno, name=frame.f_code.co_name, ) self.per_meth_profiling[signature].cur_frame_call_count += 1 self._per_thread_profile_stack.append(FuncCallProfile(signature)) else: try: callprof = self._per_thread_profile_stack.pop() except IndexError: # `pybind_profiling_start` has been called before the # profiler was enable, so _per_thread_profile_stack lacks # it representation return callprof.done() signature = callprof.signature prof = self.per_meth_profiling[signature] prof.cur_frame_total_time += callprof.get_total_time() prof.cur_frame_self_time += callprof.get_self_time() if self._per_thread_profile_stack: self._per_thread_profile_stack[-1].add_out_of_func( callprof.get_total_time() ) return profilefunc cdef object profiler = None cdef api void pythonscript_profiling_start( godot_pluginscript_language_data *p_data ) with gil: global profiler profiler = Profiler() sys.setprofile(profiler.get_profilefunc()) cdef api void pythonscript_profiling_stop( godot_pluginscript_language_data *p_data ) with gil: global profiler profiler = None sys.setprofile(None) cdef api int pythonscript_profiling_get_accumulated_data( godot_pluginscript_language_data *p_data, godot_pluginscript_profiling_data *r_info, int p_info_max ) with gil: # Sort function to make sure we can display the most consuming ones sorted_and_limited = sorted( profiler.per_meth_profiling.items(), key=lambda x: -x[1].self_time )[:p_info_max] cdef int i cdef object signature cdef object profile for i, (signature, profile) in enumerate(sorted_and_limited): pyobj_to_godot_string_name(signature, &r_info[i].signature) r_info[i].call_count = profile.call_count r_info[i].total_time = int(profile.total_time * 1e6) r_info[i].self_time = int(profile.self_time * 1e6) return len(sorted_and_limited) cdef api int pythonscript_profiling_get_frame_data( godot_pluginscript_language_data *p_data, godot_pluginscript_profiling_data *r_info, int p_info_max ) with gil: # Sort function to make sure we can display the most consuming ones sorted_and_limited = sorted( profiler.per_meth_profiling.items(), key=lambda x: -x[1].last_frame_self_time )[:p_info_max] cdef int i cdef object signature cdef object profile for i, (signature, profile) in enumerate(sorted_and_limited): pyobj_to_godot_string_name(signature, &r_info[i].signature) r_info[i].call_count = profile.last_frame_call_count r_info[i].total_time = int(profile.last_frame_total_time * 1e6) r_info[i].self_time = int(profile.last_frame_self_time * 1e6) return len(sorted_and_limited) cdef api void pythonscript_profiling_frame( godot_pluginscript_language_data *p_data ) with gil: if profiler is not None: profiler.next_frame()