You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
{{ message }}
This repository has been archived by the owner on Mar 30, 2019. It is now read-only.
I’ve found that with Configuration.EnableReleaseOnFinalizer set to true, I occasionally get an exception if the last use of a ComObject is passing it into a generated interop method. For example, when using SharpDX versions 3.1.1 or 4.2.0, the following call to d2drendertarget.FillGeometry occasionally throws an exception. I believe this is because the finalizer runs and decrements the ref count to zero before the native method acquires its own ref:
// Some managed method in my C# codepublicvoidFillGeo(MyGeometryTypegeometry,D2D.Brushd2dbrush){D2D.Geometryd2dgeometry=geometry.CreateD2DGeometry();this.d2drendertarget.FillGeometry(d2dgeometry,d2dbrush);}
// generated FillGeometry interop method in SharpDX (comments mine)publicunsafevoidDrawGeometry(SharpDX.Direct2D1.Geometrygeometry,SharpDX.Direct2D1.Brushbrush,System.SinglestrokeWidth,SharpDX.Direct2D1.StrokeStylestrokeStyle){System.IntPtrgeometry_=System.IntPtr.Zero;System.IntPtrbrush_=System.IntPtr.Zero;System.IntPtrstrokeStyle_=System.IntPtr.Zero;geometry_=SharpDX.CppObject.ToCallbackPtr<SharpDX.Direct2D1.Geometry>(geometry);// At this point, “geometry” is eligible for finalization, its last use was in the above line. // If the finalizer runs now, it will decrement the native Geometry object's ref count to zerobrush_=SharpDX.CppObject.ToCallbackPtr<SharpDX.Direct2D1.Brush>(brush);strokeStyle_=SharpDX.CppObject.ToCallbackPtr<SharpDX.Direct2D1.StrokeStyle>(strokeStyle);// The native method called here will try to dereference "geometry_" but the native object it // points at might have been destroyed (if the managed object's finalizer already ran)SharpDX.Direct2D1.LocalInterop.CalliStdCallvoid(this._nativePointer,(void*)geometry_,(void*)brush_,strokeWidth,(void*)strokeStyle_,(*(void***)this._nativePointer)[22]);}
This case can be trivially resolved with a using block in my code, but it seems like the point of Configuration.EnableReleaseOnFinalizer is that explicitly disposing is not required. I'm working around this issue with using blocks and GC.KeepAlive for the cases I've found in my codebase, but resolving this more generally would be great.
Would it be reasonable to have SharpGen generate GC.KeepAlive calls to delay finalization for objects whose native pointer is passed into a native method call? i.e. the generated method would look something like:
Hey can you open an issue for this on the SharpGenTools repo? That's where the code generator lives. I'm busy with work right now but I'd definitely be interested in adding this into the generator.
I’ve found that with Configuration.EnableReleaseOnFinalizer set to true, I occasionally get an exception if the last use of a ComObject is passing it into a generated interop method. For example, when using SharpDX versions 3.1.1 or 4.2.0, the following call to
d2drendertarget.FillGeometry
occasionally throws an exception. I believe this is because the finalizer runs and decrements the ref count to zero before the native method acquires its own ref:This case can be trivially resolved with a using block in my code, but it seems like the point of Configuration.EnableReleaseOnFinalizer is that explicitly disposing is not required. I'm working around this issue with using blocks and GC.KeepAlive for the cases I've found in my codebase, but resolving this more generally would be great.
Would it be reasonable to have SharpGen generate GC.KeepAlive calls to delay finalization for objects whose native pointer is passed into a native method call? i.e. the generated method would look something like:
The text was updated successfully, but these errors were encountered: