447 lines
11 KiB
C
447 lines
11 KiB
C
/**
|
|
* \file
|
|
* Tracing facilities for the Mono Runtime.
|
|
*
|
|
* Author:
|
|
* Paolo Molaro (lupus@ximian.com)
|
|
* Dietmar Maurer (dietmar@ximian.com)
|
|
*
|
|
* (C) 2002 Ximian, Inc.
|
|
* Copyright 2011 Xamarin, Inc (http://www.xamarin.com)
|
|
* Licensed under the MIT license. See LICENSE file in the project root for full license information.
|
|
*/
|
|
|
|
#include <config.h>
|
|
#ifdef HAVE_ALLOCA_H
|
|
#include <alloca.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#include <string.h>
|
|
#include "mini.h"
|
|
#include "mini-runtime.h"
|
|
#include <mono/metadata/debug-helpers.h>
|
|
#include <mono/utils/mono-time.h>
|
|
#include <mono/utils/mono-memory-model.h>
|
|
#include "trace.h"
|
|
#include <mono/metadata/callspec.h>
|
|
|
|
#if defined (HOST_ANDROID) || (defined (TARGET_IOS) && defined (TARGET_IOS))
|
|
# undef printf
|
|
# define printf(...) g_log("mono", G_LOG_LEVEL_MESSAGE, __VA_ARGS__)
|
|
# undef fprintf
|
|
# define fprintf(__ignore, ...) g_log ("mono-gc", G_LOG_LEVEL_MESSAGE, __VA_ARGS__)
|
|
#endif
|
|
|
|
static MonoCallSpec trace_spec;
|
|
|
|
static volatile gint32 output_lock = 0;
|
|
|
|
gboolean mono_trace_eval_exception (MonoClass *klass)
|
|
{
|
|
return mono_callspec_eval_exception (klass, &trace_spec);
|
|
}
|
|
|
|
gboolean mono_trace_eval (MonoMethod *method)
|
|
{
|
|
return mono_callspec_eval (method, &trace_spec);
|
|
}
|
|
|
|
MonoCallSpec *mono_trace_set_options (const char *options)
|
|
{
|
|
char *errstr;
|
|
if (!mono_callspec_parse (options, &trace_spec, &errstr)) {
|
|
fprintf (stderr, "%s\n", errstr);
|
|
g_free (errstr);
|
|
return NULL;
|
|
}
|
|
|
|
return &trace_spec;
|
|
}
|
|
|
|
static
|
|
#ifdef HAVE_KW_THREAD
|
|
__thread
|
|
#endif
|
|
int indent_level = 0;
|
|
static guint64 start_time = 0;
|
|
|
|
static double seconds_since_start (void)
|
|
{
|
|
guint64 diff = mono_100ns_ticks () - start_time;
|
|
return diff/10000000.0;
|
|
}
|
|
|
|
static void indent (int diff) {
|
|
if (diff < 0)
|
|
indent_level += diff;
|
|
if (start_time == 0)
|
|
start_time = mono_100ns_ticks ();
|
|
printf ("[%p: %.5f %d] ", (void*)mono_native_thread_id_get (), seconds_since_start (), indent_level);
|
|
if (diff > 0)
|
|
indent_level += diff;
|
|
}
|
|
|
|
static char *
|
|
string_to_utf8 (MonoString *s)
|
|
{
|
|
char *as;
|
|
GError *gerror = NULL;
|
|
|
|
g_assert (s);
|
|
|
|
if (!s->length)
|
|
return g_strdup ("");
|
|
|
|
as = g_utf16_to_utf8 (mono_string_chars (s), s->length, NULL, NULL, &gerror);
|
|
if (gerror) {
|
|
/* Happens with StringBuilders */
|
|
g_error_free (gerror);
|
|
return g_strdup ("<INVALID UTF8>");
|
|
}
|
|
else
|
|
return as;
|
|
}
|
|
|
|
/*
|
|
* cpos (ebp + arg_info[n].offset) points to the beginning of the
|
|
* stack slot for this argument. On little-endian systems, we can
|
|
* simply dereference it. On big-endian systems, we need to adjust
|
|
* cpos upward first if the datatype we're referencing is smaller than
|
|
* a stack slot. Also - one can't assume that gpointer is also the
|
|
* size of a stack slot - use SIZEOF_REGISTER instead. The following
|
|
* helper macro tries to keep down the mess of all the pointer
|
|
* calculations.
|
|
*/
|
|
#if (G_BYTE_ORDER == G_LITTLE_ENDIAN)
|
|
#define arg_in_stack_slot(cpos, type) ((type *)(cpos))
|
|
#else
|
|
#define arg_in_stack_slot(cpos, type) ((type *)((sizeof(type) < SIZEOF_REGISTER) ? (((gssize)(cpos)) + SIZEOF_REGISTER - sizeof(type)) : (gssize)(cpos)))
|
|
#endif
|
|
|
|
void
|
|
mono_trace_enter_method (MonoMethod *method, char *ebp)
|
|
{
|
|
int i, j;
|
|
MonoClass *klass;
|
|
MonoObject *o;
|
|
MonoJitArgumentInfo *arg_info;
|
|
MonoMethodSignature *sig;
|
|
char *fname;
|
|
MonoGenericSharingContext *gsctx = NULL;
|
|
|
|
if (!trace_spec.enabled)
|
|
return;
|
|
|
|
while (output_lock != 0 || mono_atomic_cas_i32 (&output_lock, 1, 0) != 0)
|
|
mono_thread_info_yield ();
|
|
|
|
fname = mono_method_full_name (method, TRUE);
|
|
indent (1);
|
|
printf ("ENTER: %s(", fname);
|
|
g_free (fname);
|
|
|
|
if (!ebp) {
|
|
printf (") ip: %p\n", MONO_RETURN_ADDRESS_N (1));
|
|
goto unlock;
|
|
}
|
|
|
|
sig = mono_method_signature (method);
|
|
|
|
arg_info = (MonoJitArgumentInfo *)alloca (sizeof (MonoJitArgumentInfo) * (sig->param_count + 1));
|
|
|
|
if (method->is_inflated) {
|
|
/* FIXME: Might be better to pass the ji itself */
|
|
MonoJitInfo *ji = mini_jit_info_table_find (mono_domain_get (), (char *)MONO_RETURN_ADDRESS (), NULL);
|
|
if (ji) {
|
|
gsctx = mono_jit_info_get_generic_sharing_context (ji);
|
|
if (gsctx && gsctx->is_gsharedvt) {
|
|
/* Needs a ctx to get precise method */
|
|
printf (") <gsharedvt>\n");
|
|
goto unlock;
|
|
}
|
|
}
|
|
}
|
|
|
|
mono_arch_get_argument_info (sig, sig->param_count, arg_info);
|
|
|
|
if (MONO_TYPE_ISSTRUCT (mono_method_signature (method)->ret)) {
|
|
g_assert (!mono_method_signature (method)->ret->byref);
|
|
|
|
printf ("VALUERET:%p, ", *((gpointer *)(ebp + 8)));
|
|
}
|
|
|
|
if (mono_method_signature (method)->hasthis) {
|
|
gpointer *this_obj = (gpointer *)(ebp + arg_info [0].offset);
|
|
if (method->klass->valuetype) {
|
|
printf ("value:%p, ", *arg_in_stack_slot(this_obj, gpointer *));
|
|
} else {
|
|
o = *arg_in_stack_slot(this_obj, MonoObject *);
|
|
|
|
if (o) {
|
|
klass = o->vtable->klass;
|
|
|
|
if (klass == mono_defaults.string_class) {
|
|
MonoString *s = (MonoString*)o;
|
|
char *as = string_to_utf8 (s);
|
|
|
|
printf ("this:[STRING:%p:%s], ", o, as);
|
|
g_free (as);
|
|
} else {
|
|
printf ("this:%p[%s.%s %s], ", o, klass->name_space, klass->name, o->vtable->domain->friendly_name);
|
|
}
|
|
} else
|
|
printf ("this:NULL, ");
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < mono_method_signature (method)->param_count; ++i) {
|
|
gpointer *cpos = (gpointer *)(ebp + arg_info [i + 1].offset);
|
|
int size = arg_info [i + 1].size;
|
|
|
|
MonoType *type = mono_method_signature (method)->params [i];
|
|
|
|
if (type->byref) {
|
|
printf ("[BYREF:%p], ", *arg_in_stack_slot(cpos, gpointer *));
|
|
} else switch (mini_get_underlying_type (type)->type) {
|
|
|
|
case MONO_TYPE_I:
|
|
case MONO_TYPE_U:
|
|
printf ("%p, ", *arg_in_stack_slot(cpos, gpointer *));
|
|
break;
|
|
case MONO_TYPE_BOOLEAN:
|
|
case MONO_TYPE_CHAR:
|
|
case MONO_TYPE_I1:
|
|
case MONO_TYPE_U1:
|
|
printf ("%d, ", *arg_in_stack_slot(cpos, gint8));
|
|
break;
|
|
case MONO_TYPE_I2:
|
|
case MONO_TYPE_U2:
|
|
printf ("%d, ", *arg_in_stack_slot(cpos, gint16));
|
|
break;
|
|
case MONO_TYPE_I4:
|
|
case MONO_TYPE_U4:
|
|
printf ("%d, ", *arg_in_stack_slot(cpos, int));
|
|
break;
|
|
case MONO_TYPE_STRING: {
|
|
MonoString *s = *arg_in_stack_slot(cpos, MonoString *);
|
|
if (s) {
|
|
char *as;
|
|
|
|
g_assert (((MonoObject *)s)->vtable->klass == mono_defaults.string_class);
|
|
as = string_to_utf8 (s);
|
|
|
|
printf ("[STRING:%p:%s], ", s, as);
|
|
g_free (as);
|
|
} else
|
|
printf ("[STRING:null], ");
|
|
break;
|
|
}
|
|
case MONO_TYPE_CLASS:
|
|
case MONO_TYPE_OBJECT: {
|
|
o = *arg_in_stack_slot(cpos, MonoObject *);
|
|
if (o) {
|
|
klass = o->vtable->klass;
|
|
|
|
if (klass == mono_defaults.string_class) {
|
|
char *as = string_to_utf8 ((MonoString*)o);
|
|
|
|
printf ("[STRING:%p:%s], ", o, as);
|
|
g_free (as);
|
|
} else if (klass == mono_defaults.int32_class) {
|
|
printf ("[INT32:%p:%d], ", o, *(gint32 *)((char *)o + sizeof (MonoObject)));
|
|
} else if (klass == mono_defaults.runtimetype_class) {
|
|
printf ("[TYPE:%s], ", mono_type_full_name (((MonoReflectionType*)o)->type));
|
|
} else
|
|
printf ("[%s.%s:%p], ", klass->name_space, klass->name, o);
|
|
} else {
|
|
printf ("%p, ", *arg_in_stack_slot(cpos, gpointer));
|
|
}
|
|
break;
|
|
}
|
|
case MONO_TYPE_PTR:
|
|
case MONO_TYPE_FNPTR:
|
|
case MONO_TYPE_ARRAY:
|
|
case MONO_TYPE_SZARRAY:
|
|
printf ("%p, ", *arg_in_stack_slot(cpos, gpointer));
|
|
break;
|
|
case MONO_TYPE_I8:
|
|
case MONO_TYPE_U8:
|
|
printf ("0x%016llx, ", (long long)*arg_in_stack_slot(cpos, gint64));
|
|
break;
|
|
case MONO_TYPE_R4:
|
|
printf ("%f, ", *arg_in_stack_slot(cpos, float));
|
|
break;
|
|
case MONO_TYPE_R8:
|
|
printf ("%f, ", *arg_in_stack_slot(cpos, double));
|
|
break;
|
|
case MONO_TYPE_VALUETYPE:
|
|
printf ("[");
|
|
for (j = 0; j < size; j++)
|
|
printf ("%02x,", *((guint8*)cpos +j));
|
|
printf ("], ");
|
|
break;
|
|
default:
|
|
printf ("XX, ");
|
|
}
|
|
}
|
|
|
|
printf (")\n");
|
|
fflush (stdout);
|
|
|
|
unlock:
|
|
mono_atomic_store_release (&output_lock, 0);
|
|
}
|
|
|
|
void
|
|
mono_trace_leave_method (MonoMethod *method, ...)
|
|
{
|
|
MonoType *type;
|
|
char *fname;
|
|
va_list ap;
|
|
MonoGenericSharingContext *gsctx;
|
|
|
|
if (!trace_spec.enabled)
|
|
return;
|
|
|
|
while (output_lock != 0 || mono_atomic_cas_i32 (&output_lock, 1, 0) != 0)
|
|
mono_thread_info_yield ();
|
|
|
|
va_start(ap, method);
|
|
|
|
fname = mono_method_full_name (method, TRUE);
|
|
indent (-1);
|
|
printf ("LEAVE: %s", fname);
|
|
g_free (fname);
|
|
|
|
if (method->is_inflated) {
|
|
/* FIXME: Might be better to pass the ji itself */
|
|
MonoJitInfo *ji = mini_jit_info_table_find (mono_domain_get (), (char *)MONO_RETURN_ADDRESS (), NULL);
|
|
if (ji) {
|
|
gsctx = mono_jit_info_get_generic_sharing_context (ji);
|
|
if (gsctx && gsctx->is_gsharedvt) {
|
|
/* Needs a ctx to get precise method */
|
|
printf (") <gsharedvt>\n");
|
|
goto unlock;
|
|
}
|
|
}
|
|
}
|
|
|
|
type = mini_get_underlying_type (mono_method_signature (method)->ret);
|
|
|
|
switch (type->type) {
|
|
case MONO_TYPE_VOID:
|
|
break;
|
|
case MONO_TYPE_BOOLEAN: {
|
|
int eax = va_arg (ap, int);
|
|
if (eax)
|
|
printf ("TRUE:%d", eax);
|
|
else
|
|
printf ("FALSE");
|
|
|
|
break;
|
|
}
|
|
case MONO_TYPE_CHAR:
|
|
case MONO_TYPE_I1:
|
|
case MONO_TYPE_U1:
|
|
case MONO_TYPE_I2:
|
|
case MONO_TYPE_U2:
|
|
case MONO_TYPE_I4:
|
|
case MONO_TYPE_U4:
|
|
case MONO_TYPE_I:
|
|
case MONO_TYPE_U: {
|
|
int eax = va_arg (ap, int);
|
|
printf ("result=%d", eax);
|
|
break;
|
|
}
|
|
case MONO_TYPE_STRING: {
|
|
MonoString *s = va_arg (ap, MonoString *);
|
|
;
|
|
if (s) {
|
|
char *as;
|
|
|
|
g_assert (((MonoObject *)s)->vtable->klass == mono_defaults.string_class);
|
|
as = string_to_utf8 (s);
|
|
printf ("[STRING:%p:%s]", s, as);
|
|
g_free (as);
|
|
} else
|
|
printf ("[STRING:null], ");
|
|
break;
|
|
}
|
|
case MONO_TYPE_CLASS:
|
|
case MONO_TYPE_OBJECT: {
|
|
MonoObject *o = va_arg (ap, MonoObject *);
|
|
|
|
if (o) {
|
|
if (o->vtable->klass == mono_defaults.boolean_class) {
|
|
printf ("[BOOLEAN:%p:%d]", o, *((guint8 *)o + sizeof (MonoObject)));
|
|
} else if (o->vtable->klass == mono_defaults.int32_class) {
|
|
printf ("[INT32:%p:%d]", o, *((gint32 *)((char *)o + sizeof (MonoObject))));
|
|
} else if (o->vtable->klass == mono_defaults.int64_class) {
|
|
printf ("[INT64:%p:%lld]", o, (long long)*((gint64 *)((char *)o + sizeof (MonoObject))));
|
|
} else
|
|
printf ("[%s.%s:%p]", o->vtable->klass->name_space, o->vtable->klass->name, o);
|
|
} else
|
|
printf ("[OBJECT:%p]", o);
|
|
|
|
break;
|
|
}
|
|
case MONO_TYPE_PTR:
|
|
case MONO_TYPE_FNPTR:
|
|
case MONO_TYPE_ARRAY:
|
|
case MONO_TYPE_SZARRAY: {
|
|
gpointer p = va_arg (ap, gpointer);
|
|
printf ("result=%p", p);
|
|
break;
|
|
}
|
|
case MONO_TYPE_I8: {
|
|
gint64 l = va_arg (ap, gint64);
|
|
printf ("lresult=0x%16llx", (long long)l);
|
|
break;
|
|
}
|
|
case MONO_TYPE_U8: {
|
|
gint64 l = va_arg (ap, gint64);
|
|
printf ("lresult=0x%16llx", (long long)l);
|
|
break;
|
|
}
|
|
case MONO_TYPE_R4:
|
|
case MONO_TYPE_R8: {
|
|
double f = va_arg (ap, double);
|
|
printf ("FP=%f", f);
|
|
break;
|
|
}
|
|
case MONO_TYPE_VALUETYPE: {
|
|
guint8 *p = (guint8 *)va_arg (ap, gpointer);
|
|
int j, size, align;
|
|
size = mono_type_size (type, &align);
|
|
printf ("[");
|
|
for (j = 0; p && j < size; j++)
|
|
printf ("%02x,", p [j]);
|
|
printf ("]");
|
|
break;
|
|
}
|
|
default:
|
|
printf ("(unknown return type %x)", mono_method_signature (method)->ret->type);
|
|
}
|
|
|
|
//printf (" ip: %p\n", MONO_RETURN_ADDRESS_N (1));
|
|
printf ("\n");
|
|
fflush (stdout);
|
|
|
|
unlock:
|
|
mono_atomic_store_release (&output_lock, 0);
|
|
}
|
|
|
|
void
|
|
mono_trace_enable (gboolean enable)
|
|
{
|
|
trace_spec.enabled = enable;
|
|
}
|
|
|
|
gboolean
|
|
mono_trace_is_enabled ()
|
|
{
|
|
return trace_spec.enabled;
|
|
}
|