Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[runtime] Move xamarin_create_managed_ref internal call to managed code. #11271

Merged
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 0 additions & 2 deletions runtime/monovm-bridge.m
Original file line number Diff line number Diff line change
Expand Up @@ -124,8 +124,6 @@
nsvalue_class = get_class_from_name (platform_image, foundation, "NSValue", true);
nsstring_class = get_class_from_name (platform_image, foundation, "NSString", true);

xamarin_add_internal_call ("Foundation.NSObject::xamarin_create_managed_ref", (const void *) xamarin_create_managed_ref);

runtime_initialize = mono_class_get_method_from_name (runtime_class, "Initialize", 1);

if (runtime_initialize == NULL)
Expand Down
47 changes: 9 additions & 38 deletions runtime/runtime.m
Original file line number Diff line number Diff line change
Expand Up @@ -479,7 +479,7 @@ void xamarin_framework_peer_unlock ()
// compiler warning (no 'xamarinGetGChandle' selector found).
@protocol XamarinExtendedObject
-(GCHandle) xamarinGetGCHandle;
-(void) xamarinSetGCHandle: (GCHandle) gc_handle flags: (enum XamarinGCHandleFlags) flags;
-(bool) xamarinSetGCHandle: (GCHandle) gc_handle flags: (enum XamarinGCHandleFlags) flags;
-(enum XamarinGCHandleFlags) xamarinGetFlags;
-(void) xamarinSetFlags: (enum XamarinGCHandleFlags) flags;
@end
Expand All @@ -496,16 +496,20 @@ -(void) xamarinSetFlags: (enum XamarinGCHandleFlags) flags;
return rv;
}

static inline void
static inline bool
set_gchandle (id self, GCHandle gc_handle, enum XamarinGCHandleFlags flags)
{
bool rv;

// COOP: we call a selector, and that must only be done in SAFE mode.
MONO_ASSERT_GC_UNSAFE;

MONO_ENTER_GC_SAFE;
id<XamarinExtendedObject> xself = self;
[xself xamarinSetGCHandle: gc_handle flags: flags];
rv = [xself xamarinSetGCHandle: gc_handle flags: flags];
MONO_EXIT_GC_SAFE;

return rv;
}

static inline GCHandle
Expand Down Expand Up @@ -1878,11 +1882,11 @@ -(void) xamarinSetFlags: (enum XamarinGCHandleFlags) flags;
set_gchandle (self, INVALID_GCHANDLE, XamarinGCHandleFlags_None);
}

void
bool
xamarin_set_gchandle_with_flags (id self, GCHandle gchandle, enum XamarinGCHandleFlags flags)
{
// COOP: no managed memory access: any mode
set_gchandle (self, gchandle, flags);
return set_gchandle (self, gchandle, flags);
}

#if defined(DEBUG_REF_COUNTING)
Expand Down Expand Up @@ -1989,39 +1993,6 @@ -(void) xamarinSetFlags: (enum XamarinGCHandleFlags) flags;
[self release];
}

void
xamarin_create_managed_ref (id self, gpointer managed_object, bool retain, bool user_type)
{
// COOP: This is an icall, so at entry we're in unsafe mode.
// COOP: we stay in unsafe mode (since we write to the managed memory) unless calling a selector (which must be done in safe mode)
MONO_ASSERT_GC_UNSAFE;

GCHandle gchandle;

#if defined(DEBUG_REF_COUNTING)
PRINT ("monotouch_create_managed_ref (%s Handle=%p) retainCount=%d; HasManagedRef=%i GCHandle=%i IsUserType=%i managed_object=%p\n",
class_getName ([self class]), self, get_safe_retainCount (self), user_type ? xamarin_has_managed_ref (self) : 666, user_type ? get_gchandle_without_flags (self) : (void*) 666, user_type, managed_object);
#endif

if (user_type) {
gchandle = get_gchandle_without_flags (self);
if (!gchandle) {
xamarin_create_gchandle (self, managed_object, XamarinGCHandleFlags_HasManagedRef, !retain);
} else {
#if defined(DEBUG_REF_COUNTING)
xamarin_assertion_message ("GCHandle already exists for %p: %d\n", self, gchandle);
#endif
}
}

if (retain) {
MONO_ENTER_GC_SAFE;
[self retain];
MONO_EXIT_GC_SAFE;
}
mt_dummy_use (managed_object);
}

/*
* Block support
*/
Expand Down
11 changes: 10 additions & 1 deletion runtime/trampolines.m
Original file line number Diff line number Diff line change
Expand Up @@ -735,7 +735,7 @@
}

static const char *associated_key = "x"; // the string value doesn't matter, only the pointer value.
void
bool
xamarin_set_gchandle_trampoline (id self, SEL sel, GCHandle gc_handle, enum XamarinGCHandleFlags flags)
{
// COOP: Called by ObjC (when the setGCHandle:flags: selector is called on an object).
Expand All @@ -745,6 +745,13 @@
/* This is for types registered using the dynamic registrar */
XamarinAssociatedObject *obj;
obj = objc_getAssociatedObject (self, associated_key);

// Check if we're setting the initial value, in which case we don't want to overwrite
if (obj != NULL && obj->gc_handle != INVALID_GCHANDLE && ((flags & XamarinGCHandleFlags_InitialSet) == XamarinGCHandleFlags_InitialSet))
return false;

flags = (enum XamarinGCHandleFlags) (flags & ~XamarinGCHandleFlags_InitialSet); // Remove the InitialSet flag, we don't want to store it.

if (obj == NULL && gc_handle != INVALID_GCHANDLE) {
obj = [[XamarinAssociatedObject alloc] init];
obj->gc_handle = gc_handle;
Expand Down Expand Up @@ -774,6 +781,8 @@
CFDictionarySetValue (gchandle_hash, self, entry);
}
pthread_mutex_unlock (&gchandle_hash_lock);

return true;
}

GCHandle
Expand Down
3 changes: 1 addition & 2 deletions runtime/xamarin/runtime.h
Original file line number Diff line number Diff line change
Expand Up @@ -218,9 +218,8 @@ GCHandle xamarin_get_gchandle (id self);
void xamarin_free_gchandle (id self, GCHandle gchandle);
void xamarin_clear_gchandle (id self);
GCHandle xamarin_get_gchandle_with_flags (id self, enum XamarinGCHandleFlags *flags);
void xamarin_set_gchandle_with_flags (id self, GCHandle gchandle, enum XamarinGCHandleFlags flags);
bool xamarin_set_gchandle_with_flags (id self, GCHandle gchandle, enum XamarinGCHandleFlags flags);
void xamarin_create_gchandle (id self, void *managed_object, enum XamarinGCHandleFlags flags, bool force_weak);
void xamarin_create_managed_ref (id self, void * managed_object, bool retain, bool user_type);
void xamarin_release_managed_ref (id self, bool user_type);
void xamarin_notify_dealloc (id self, GCHandle gchandle);

Expand Down
4 changes: 3 additions & 1 deletion runtime/xamarin/trampolines.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,12 @@
extern "C" {
#endif

// Must be kept in sync with the same enum in NSObject2.cs
enum XamarinGCHandleFlags : uint32_t {
XamarinGCHandleFlags_None = 0,
XamarinGCHandleFlags_WeakGCHandle = 1,
XamarinGCHandleFlags_HasManagedRef = 2,
XamarinGCHandleFlags_InitialSet = 4,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What happened with 3?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These are flags, so 1 + 2 = 3.

};

void * xamarin_trampoline (id self, SEL sel, ...);
Expand All @@ -39,7 +41,7 @@ long long xamarin_static_longret_trampoline (id self, SEL sel, ...);
id xamarin_copyWithZone_trampoline1 (id self, SEL sel, NSZone *zone);
id xamarin_copyWithZone_trampoline2 (id self, SEL sel, NSZone *zone);
GCHandle xamarin_get_gchandle_trampoline (id self, SEL sel);
void xamarin_set_gchandle_trampoline (id self, SEL sel, GCHandle gc_handle, enum XamarinGCHandleFlags flags);
bool xamarin_set_gchandle_trampoline (id self, SEL sel, GCHandle gc_handle, enum XamarinGCHandleFlags flags);
enum XamarinGCHandleFlags xamarin_get_flags_trampoline (id self, SEL sel);
void xamarin_set_flags_trampoline (id self, SEL sel, enum XamarinGCHandleFlags flags);

Expand Down
29 changes: 25 additions & 4 deletions src/Foundation/NSObject2.cs
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,14 @@ internal enum Flags : byte {
IsCustomType = 128,
}

// Must be kept in sync with the same enum in trampolines.h
enum XamarinGCHandleFlags : uint {
None = 0,
WeakGCHandle = 1,
HasManagedRef = 2,
InitialSet = 4,
}

[StructLayout (LayoutKind.Sequential)]
struct objc_super {
public IntPtr Handle;
Expand Down Expand Up @@ -205,9 +213,6 @@ internal static IntPtr Initialize ()
[DllImport ("__Internal")]
static extern void xamarin_release_managed_ref (IntPtr handle, bool user_type);

[MethodImplAttribute (MethodImplOptions.InternalCall)]
static extern void xamarin_create_managed_ref (IntPtr handle, NSObject obj, bool retain, bool user_type);

#if !XAMCORE_3_0
public static bool IsNewRefcountEnabled ()
{
Expand Down Expand Up @@ -267,10 +272,26 @@ private void InitializeObject (bool alloced) {
CreateManagedRef (!alloced || native_ref);
}

[DllImport ("__Internal")]
static extern bool xamarin_set_gchandle_with_flags (IntPtr handle, IntPtr gchandle, XamarinGCHandleFlags flags);

void CreateManagedRef (bool retain)
{
HasManagedRef = true;
xamarin_create_managed_ref (handle, this, retain, Runtime.IsUserType (handle));
bool isUserType = Runtime.IsUserType (handle);
if (isUserType) {
var flags = XamarinGCHandleFlags.HasManagedRef | XamarinGCHandleFlags.InitialSet | XamarinGCHandleFlags.WeakGCHandle;
var gchandle = GCHandle.Alloc (this, GCHandleType.WeakTrackResurrection);
var h = GCHandle.ToIntPtr (gchandle);
if (!xamarin_set_gchandle_with_flags (handle, h, flags)) {
// A GCHandle already existed: this shouldn't happen, but let's handle it anyway.
Runtime.NSLog ("Tried to create a managed reference from an object that already has a managed reference (type: {0})", GetType ());
gchandle.Free ();
}
}

if (retain)
DangerousRetain ();
}

void ReleaseManagedRef ()
Expand Down
9 changes: 7 additions & 2 deletions tools/common/StaticRegistrar.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2511,7 +2511,7 @@ string GetObjCSignature (ObjCMethod method, List<Exception> exceptions)
else if (method.CurrentTrampoline == Trampoline.SetFlags)
return "-(void) xamarinSetFlags: (enum XamarinGCHandleFlags) flags";
else if (method.CurrentTrampoline == Trampoline.SetGCHandle)
return "-(void) xamarinSetGCHandle: (GCHandle) gchandle flags: (enum XamarinGCHandleFlags) flags";
return "-(bool) xamarinSetGCHandle: (GCHandle) gchandle flags: (enum XamarinGCHandleFlags) flags";
else if (method.CurrentTrampoline == Trampoline.CopyWithZone1 || method.CurrentTrampoline == Trampoline.CopyWithZone2)
return "-(id) copyWithZone: (NSZone *)zone";

Expand Down Expand Up @@ -3141,11 +3141,16 @@ void Specialize (AutoIndentStringBuilder sb, ObjCMethod method, List<Exception>
sb.WriteLine ();
return;
case Trampoline.SetGCHandle:
sb.WriteLine ("-(void) xamarinSetGCHandle: (GCHandle) gc_handle flags: (enum XamarinGCHandleFlags) flags");
sb.WriteLine ("-(bool) xamarinSetGCHandle: (GCHandle) gc_handle flags: (enum XamarinGCHandleFlags) flags");
sb.WriteLine ("{");
sb.WriteLine ("if (((flags & XamarinGCHandleFlags_InitialSet) == XamarinGCHandleFlags_InitialSet) && __monoObjectGCHandle.gc_handle != INVALID_GCHANDLE) {");
sb.WriteLine ("return false;");
sb.WriteLine ("}");
sb.WriteLine ("flags = (enum XamarinGCHandleFlags) (flags & ~XamarinGCHandleFlags_InitialSet);"); // Remove the InitialSet flag, we don't want to store it.
sb.WriteLine ("__monoObjectGCHandle.gc_handle = gc_handle;");
sb.WriteLine ("__monoObjectGCHandle.flags = flags;");
sb.WriteLine ("__monoObjectGCHandle.native_object = self;");
sb.WriteLine ("return true;");
sb.WriteLine ("}");
sb.WriteLine ();
return;
Expand Down
1 change: 0 additions & 1 deletion tools/mtouch/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,6 @@ UNREFERENCED_SYMBOLS = \
_xamarin_IntPtr_objc_msgSend_IntPtr \
_xamarin_IntPtr_objc_msgSendSuper_IntPtr \
_xamarin_release_managed_ref \
_xamarin_create_managed_ref \
_xamarin_CGPoint__VNNormalizedFaceBoundingBoxPointForLandmarkPoint_Vector2_CGRect_nuint_nuint_string \
_xamarin_CGPoint__VNImagePointForFaceLandmarkPoint_Vector2_CGRect_nuint_nuint_string \
_CloseZStream \
Expand Down