mono_domain_assembly_search (MonoAssemblyName *aname,
gpointer user_data);
+static MonoAssembly *
+mono_domain_assembly_postload_search (MonoAssemblyName *aname,
+ gpointer user_data);
+
static void
mono_domain_fire_assembly_load (MonoAssembly *assembly, gpointer user_data);
mono_install_assembly_refonly_preload_hook (mono_domain_assembly_preload, GUINT_TO_POINTER (TRUE));
mono_install_assembly_search_hook (mono_domain_assembly_search, GUINT_TO_POINTER (FALSE));
mono_install_assembly_refonly_search_hook (mono_domain_assembly_search, GUINT_TO_POINTER (TRUE));
+ mono_install_assembly_postload_search_hook (mono_domain_assembly_postload_search, GUINT_TO_POINTER (FALSE));
+ mono_install_assembly_postload_refonly_search_hook (mono_domain_assembly_postload_search, GUINT_TO_POINTER (TRUE));
mono_install_assembly_load_hook (mono_domain_fire_assembly_load, NULL);
mono_install_lookup_dynamic_token (mono_reflection_lookup_dynamic_token);
return (MonoReflectionAssembly *) mono_runtime_invoke (method, domain->domain, params, NULL);
}
+static MonoAssembly *
+mono_domain_assembly_postload_search (MonoAssemblyName *aname,
+ gpointer user_data)
+{
+ gboolean refonly = GPOINTER_TO_UINT (user_data);
+ MonoReflectionAssembly *assembly;
+ MonoDomain *domain = mono_domain_get ();
+ char *aname_str;
+
+ aname_str = mono_stringify_assembly_name (aname);
+
+ /* FIXME: We invoke managed code here, so there is a potential for deadlocks */
+ assembly = try_assembly_resolve (domain, mono_string_new (domain, aname_str), refonly);
+
+ g_free (aname_str);
+
+ if (assembly)
+ return assembly->assembly;
+ else
+ return NULL;
+}
+
/*
* LOCKING: assumes assemblies_lock in the domain is already locked.
*/
/* If defined, points to the bundled assembly information */
const MonoBundledAssembly **bundles;
-/* Reflection only private hook functions */
-static MonoAssembly* mono_assembly_refonly_invoke_search_hook (MonoAssemblyName *aname);
-
/* Loaded assembly binding info */
static GSList *loaded_assembly_bindings = NULL;
+static MonoAssembly*
+mono_assembly_invoke_search_hook_internal (MonoAssemblyName *aname, gboolean refonly, gboolean postload);
+
static gchar*
encode_public_tok (const guchar *token, gint32 len)
{
MonoAssembly *ass;
GList *loading;
- ass = refonly ? mono_assembly_refonly_invoke_search_hook (aname) : mono_assembly_invoke_search_hook (aname);
+ ass = mono_assembly_invoke_search_hook_internal (aname, refonly, FALSE);
if (ass)
return ass;
return TRUE;
}
+/*
+ * mono_stringify_assembly_name:
+ *
+ * Convert @aname into its string format. The returned string is dynamically
+ * allocated and should be freed by the caller.
+ */
+char*
+mono_stringify_assembly_name (MonoAssemblyName *aname)
+{
+ return g_strdup_printf (
+ "%s, Version=%d.%d.%d.%d, Culture=%s%s%s",
+ aname->name,
+ aname->major, aname->minor, aname->build, aname->revision,
+ aname->culture && *aname->culture? aname->culture: "neutral",
+ aname->public_key_token [0] ? ", PublicKeyToken=" : "",
+ aname->public_key_token [0] ? (char *)aname->public_key_token : "");
+}
+
static gchar*
assemblyref_public_tok (MonoImage *image, guint32 key_index, guint32 flags)
{
struct AssemblySearchHook {
AssemblySearchHook *next;
MonoAssemblySearchFunc func;
+ gboolean refonly;
+ gboolean postload;
gpointer user_data;
};
AssemblySearchHook *assembly_search_hook = NULL;
-static AssemblySearchHook *assembly_refonly_search_hook = NULL;
-MonoAssembly*
-mono_assembly_invoke_search_hook (MonoAssemblyName *aname)
+static MonoAssembly*
+mono_assembly_invoke_search_hook_internal (MonoAssemblyName *aname, gboolean refonly, gboolean postload)
{
AssemblySearchHook *hook;
for (hook = assembly_search_hook; hook; hook = hook->next) {
- MonoAssembly *ass = hook->func (aname, hook->user_data);
- if (ass)
- return ass;
+ if ((hook->refonly == refonly) && (hook->postload == postload)) {
+ MonoAssembly *ass = hook->func (aname, hook->user_data);
+ if (ass)
+ return ass;
+ }
}
return NULL;
}
-static MonoAssembly*
-mono_assembly_refonly_invoke_search_hook (MonoAssemblyName *aname)
+MonoAssembly*
+mono_assembly_invoke_search_hook (MonoAssemblyName *aname)
{
- AssemblySearchHook *hook;
-
- for (hook = assembly_refonly_search_hook; hook; hook = hook->next) {
- MonoAssembly *ass = hook->func (aname, hook->user_data);
- if (ass)
- return ass;
- }
-
- return NULL;
+ return mono_assembly_invoke_search_hook_internal (aname, FALSE, FALSE);
}
-void
-mono_install_assembly_search_hook (MonoAssemblySearchFunc func, gpointer user_data)
+static void
+mono_install_assembly_search_hook_internal (MonoAssemblySearchFunc func, gpointer user_data, gboolean refonly, gboolean postload)
{
AssemblySearchHook *hook;
hook = g_new0 (AssemblySearchHook, 1);
hook->func = func;
hook->user_data = user_data;
+ hook->refonly = refonly;
+ hook->postload = postload;
hook->next = assembly_search_hook;
assembly_search_hook = hook;
+}
+
+void
+mono_install_assembly_search_hook (MonoAssemblySearchFunc func, gpointer user_data)
+{
+ mono_install_assembly_search_hook_internal (func, user_data, FALSE, FALSE);
}
void
mono_install_assembly_refonly_search_hook (MonoAssemblySearchFunc func, gpointer user_data)
{
- AssemblySearchHook *hook;
+ mono_install_assembly_search_hook_internal (func, user_data, TRUE, FALSE);
+}
- g_return_if_fail (func != NULL);
+void
+mono_install_assembly_postload_search_hook (MonoAssemblySearchFunc func, gpointer user_data)
+{
+ mono_install_assembly_search_hook_internal (func, user_data, FALSE, TRUE);
+}
- hook = g_new0 (AssemblySearchHook, 1);
- hook->func = func;
- hook->user_data = user_data;
- hook->next = assembly_refonly_search_hook;
- assembly_refonly_search_hook = hook;
+void
+mono_install_assembly_postload_refonly_search_hook (MonoAssemblySearchFunc func, gpointer user_data)
+{
+ mono_install_assembly_search_hook_internal (func, user_data, TRUE, TRUE);
}
typedef struct AssemblyPreLoadHook AssemblyPreLoadHook;
return result;
}
- return NULL;
+ /* Try a postload search hook */
+ result = mono_assembly_invoke_search_hook_internal (aname, refonly, TRUE);
+
+ return result;
}
MonoAssembly*
MonoImage *mono_assembly_get_image (MonoAssembly *assembly);
gboolean mono_assembly_fill_assembly_name (MonoImage *image, MonoAssemblyName *aname);
gboolean mono_assembly_names_equal (MonoAssemblyName *l, MonoAssemblyName *r);
+char* mono_stringify_assembly_name (MonoAssemblyName *aname);
/* Installs a function which is called each time a new assembly is loaded. */
typedef void (*MonoAssemblyLoadFunc) (MonoAssembly *assembly, gpointer user_data);
MonoAssembly* mono_assembly_invoke_search_hook (MonoAssemblyName *aname);
+/*
+ * Installs a new search function which is used as a last resort when loading
+ * an assembly fails. This could invoke AssemblyResolve events.
+ */
+void
+mono_install_assembly_postload_search_hook (MonoAssemblySearchFunc func, gpointer user_data);
+
+void
+mono_install_assembly_postload_refonly_search_hook (MonoAssemblySearchFunc func, gpointer user_data);
+
+
/* Installs a function which is called before a new assembly is loaded
* The hook are invoked from last hooked to first. If any of them returns
* a non-null value, that will be the value returned in mono_assembly_load */