2014-08-13 10:39:27 +01:00
|
|
|
/*
|
2018-01-24 17:04:36 +00:00
|
|
|
* Bare bones profiler showing how a profiler module should be structured.
|
|
|
|
*
|
|
|
|
* Compilation:
|
|
|
|
* - Linux: gcc -fPIC -shared -o libmono-profiler-sample.so sample.c `pkg-config --cflags mono-2`
|
|
|
|
* - OS X: clang -undefined suppress -flat_namespace -o mono-profiler-sample.dylib sample.c `pkg-config --cflags mono-2`
|
2015-08-26 07:17:56 -04:00
|
|
|
*
|
2018-01-24 17:04:36 +00:00
|
|
|
* If you're using a custom prefix for your Mono installation (e.g. /opt/mono),
|
|
|
|
* pkg-config must be invoked like this: PKG_CONFIG_PATH=/opt/mono pkg-config --cflags mono-2
|
2017-10-19 20:04:20 +00:00
|
|
|
*
|
2018-01-24 17:04:36 +00:00
|
|
|
* Install the resulting shared library where the dynamic loader can find it,
|
|
|
|
* e.g. /usr/local/lib or /opt/mono (custom prefix).
|
2015-08-26 07:17:56 -04:00
|
|
|
*
|
2018-01-24 17:04:36 +00:00
|
|
|
* To use the module: mono --profile=sample hello.exe
|
2014-08-13 10:39:27 +01:00
|
|
|
*/
|
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
#include <mono/metadata/profiler.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Defining a type called _MonoProfiler will complete the opaque MonoProfiler
|
|
|
|
* type, which is used throughout the profiler API.
|
|
|
|
*/
|
2014-08-13 10:39:27 +01:00
|
|
|
struct _MonoProfiler {
|
2018-01-24 17:04:36 +00:00
|
|
|
/* Handle obtained from mono_profiler_create (). */
|
|
|
|
MonoProfilerHandle handle;
|
|
|
|
|
|
|
|
/* Counts the number of calls observed. */
|
2017-10-19 20:04:20 +00:00
|
|
|
unsigned long long ncalls;
|
2014-08-13 10:39:27 +01:00
|
|
|
};
|
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
/*
|
|
|
|
* Use static storage for the profiler structure for simplicity. The structure
|
|
|
|
* can be allocated dynamically as well, if needed.
|
|
|
|
*/
|
|
|
|
static MonoProfiler profiler;
|
2017-10-19 20:04:20 +00:00
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
/*
|
|
|
|
* Callback invoked after the runtime finishes shutting down. Managed code can
|
|
|
|
* no longer run and most runtime services are unavailable.
|
|
|
|
*/
|
2014-08-13 10:39:27 +01:00
|
|
|
static void
|
2018-01-24 17:04:36 +00:00
|
|
|
sample_shutdown_end (MonoProfiler *prof)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
2018-01-24 17:04:36 +00:00
|
|
|
printf ("Total number of calls: %llu\n", prof->ncalls);
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
/*
|
|
|
|
* Method enter callback invoked on entry to all instrumented methods.
|
|
|
|
*/
|
2014-08-13 10:39:27 +01:00
|
|
|
static void
|
2017-10-19 20:04:20 +00:00
|
|
|
sample_method_enter (MonoProfiler *prof, MonoMethod *method, MonoProfilerCallContext *ctx)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
|
|
|
prof->ncalls++;
|
|
|
|
}
|
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
/*
|
|
|
|
* Filter callback that decides which methods to instrument and how.
|
|
|
|
*/
|
2017-10-19 20:04:20 +00:00
|
|
|
static MonoProfilerCallInstrumentationFlags
|
2018-01-24 17:04:36 +00:00
|
|
|
sample_call_instrumentation_filter (MonoProfiler *prof, MonoMethod *method)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
2017-10-19 20:04:20 +00:00
|
|
|
return MONO_PROFILER_CALL_INSTRUMENTATION_ENTER;
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
/*
|
|
|
|
* The entry point function invoked by the Mono runtime.
|
|
|
|
*/
|
2014-08-13 10:39:27 +01:00
|
|
|
void
|
2017-10-19 20:04:20 +00:00
|
|
|
mono_profiler_init_sample (const char *desc)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
2018-01-24 17:04:36 +00:00
|
|
|
profiler.handle = mono_profiler_create (&profiler);
|
2014-08-13 10:39:27 +01:00
|
|
|
|
2018-01-24 17:04:36 +00:00
|
|
|
mono_profiler_set_runtime_shutdown_end_callback (profiler.handle, sample_shutdown_end);
|
|
|
|
mono_profiler_set_call_instrumentation_filter_callback (profiler.handle, sample_call_instrumentation_filter);
|
|
|
|
mono_profiler_set_method_enter_callback (profiler.handle, sample_method_enter);
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|