Xamarin Public Jenkins (auto-signing) 8fc30896db Imported Upstream version 5.12.0.220
Former-commit-id: c477e03582759447177c6d4bf412cd2355aad476
2018-04-24 09:31:23 +00:00

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;
}