diff --git a/.github/actions/create/action.yml b/.github/actions/create/action.yml index d899c3399..3811663ce 100644 --- a/.github/actions/create/action.yml +++ b/.github/actions/create/action.yml @@ -19,9 +19,8 @@ runs: uses: actions/setup-dotnet@v4 with: dotnet-version: | - 6.0.x - 7.0.x 8.0.x + 9.0.x - name: Compile native library run: dotnet fsi GenerateGirTestLib.fsx diff --git a/src/Generation/Generator/Classes.cs b/src/Generation/Generator/Classes.cs index 2db0b8115..a8cd94c35 100644 --- a/src/Generation/Generator/Classes.cs +++ b/src/Generation/Generator/Classes.cs @@ -20,6 +20,7 @@ public static void Generate(IEnumerable classes, string path) //Standard generators new Generator.Internal.ClassMethods(publisher), new Generator.Internal.ClassStruct(publisher), + new Generator.Internal.ClassHandle(publisher), new Generator.Public.ClassConstructors(publisher), new Generator.Public.ClassMethods(publisher), new Generator.Public.ClassFunctions(publisher), diff --git a/src/Generation/Generator/Generator/Internal/ClassHandle.cs b/src/Generation/Generator/Generator/Internal/ClassHandle.cs new file mode 100644 index 000000000..3d20482e1 --- /dev/null +++ b/src/Generation/Generator/Generator/Internal/ClassHandle.cs @@ -0,0 +1,32 @@ +using Generator.Model; + +namespace Generator.Generator.Internal; + +internal class ClassHandle : Generator +{ + private readonly Publisher _publisher; + + public ClassHandle(Publisher publisher) + { + _publisher = publisher; + } + + public void Generate(GirModel.Class obj) + { + if (obj.Fundamental) + return; + + if (obj.Parent is null) + return; //Do not generate a handle for GObject.Object itself + + var source = Renderer.Internal.ClassHandle.Render(obj); + var codeUnit = new CodeUnit( + Project: Namespace.GetCanonicalName(obj.Namespace), + Name: Class.GetInternalHandleName(obj), + Source: source, + IsInternal: true + ); + + _publisher.Publish(codeUnit); + } +} diff --git a/src/Generation/Generator/Generator/Public/Class/ClassFramework.cs b/src/Generation/Generator/Generator/Public/Class/ClassFramework.cs index 7e883efcb..f7fea64df 100644 --- a/src/Generation/Generator/Generator/Public/Class/ClassFramework.cs +++ b/src/Generation/Generator/Generator/Public/Class/ClassFramework.cs @@ -16,6 +16,9 @@ public void Generate(GirModel.Class obj) if (obj.Fundamental) return; + if (obj.Parent is null) + return; //Do not generate Framework for GObject.Object itsel + var source = Renderer.Public.ClassFramework.Render(obj); var codeUnit = new CodeUnit( Project: Namespace.GetCanonicalName(obj.Namespace), diff --git a/src/Generation/Generator/Generator/Public/Interface/InterfaceFramework.cs b/src/Generation/Generator/Generator/Public/Interface/InterfaceFramework.cs deleted file mode 100644 index 89b4241e1..000000000 --- a/src/Generation/Generator/Generator/Public/Interface/InterfaceFramework.cs +++ /dev/null @@ -1,26 +0,0 @@ -using Generator.Model; - -namespace Generator.Generator.Public; - -internal class InterfaceFramework : Generator -{ - private readonly Publisher _publisher; - - public InterfaceFramework(Publisher publisher) - { - _publisher = publisher; - } - - public void Generate(GirModel.Interface obj) - { - var source = Renderer.Public.InterfaceFramework.Render(obj); - var codeUnit = new CodeUnit( - Project: Namespace.GetCanonicalName(obj.Namespace), - Name: $"{obj.Name}.Framework", - Source: source, - IsInternal: false - ); - - _publisher.Publish(codeUnit); - } -} diff --git a/src/Generation/Generator/Interfaces.cs b/src/Generation/Generator/Interfaces.cs index 4cfcc4c8f..8a4174499 100644 --- a/src/Generation/Generator/Interfaces.cs +++ b/src/Generation/Generator/Interfaces.cs @@ -11,7 +11,6 @@ public static void Generate(IEnumerable interfaces, string p var generators = new List>() { new Generator.Internal.InterfaceMethods(publisher), - new Generator.Public.InterfaceFramework(publisher), new Generator.Public.InterfaceMethods(publisher), new Generator.Public.InterfaceProperties(publisher), diff --git a/src/Generation/Generator/Model/Class.cs b/src/Generation/Generator/Model/Class.cs index d44d48a9a..c7a49c558 100644 --- a/src/Generation/Generator/Model/Class.cs +++ b/src/Generation/Generator/Model/Class.cs @@ -10,7 +10,19 @@ public static string GetFullyQualifiedInternalStructName(GirModel.Class @class) public static string GetInternalStructName(GirModel.Class @class) => @class.Name + "Data"; - + + public static string GetInternalHandleName(GirModel.Class @class) + => @class.Name + "Handle"; + + public static string GetFullyQualifiedInternalHandleName(GirModel.Class @class) + => Namespace.GetInternalName(@class.Namespace) + "." + GetInternalHandleName(@class); + + public static string GetFullyQualifiedPublicName(GirModel.Class @class) + => Namespace.GetPublicName(@class.Namespace) + "." + @class.Name; + + public static string GetFullyQualifiedInternalName(GirModel.Class @class) + => Namespace.GetInternalName(@class.Namespace) + "." + @class.Name; + public static bool HidesConstructor(GirModel.Class? cls, GirModel.Constructor constructor) { if (cls is null) @@ -84,4 +96,16 @@ private static bool ParameterMatch(GirModel.Parameter[] p1, GirModel.Parameter[] return true; } + + public static bool IsInitiallyUnowned(GirModel.Class cls) => IsNamedInitiallyUnowned(cls.Name) || InheritsInitiallyUnowned(cls); + + private static bool InheritsInitiallyUnowned(GirModel.Class @class) + { + if (@class.Parent is null) + return false; + + return IsNamedInitiallyUnowned(@class.Parent.Name) || InheritsInitiallyUnowned(@class.Parent); + } + + private static bool IsNamedInitiallyUnowned(string name) => name == "InitiallyUnowned"; } diff --git a/src/Generation/Generator/Renderer/Internal/Callback/CallbackCallHandler.cs b/src/Generation/Generator/Renderer/Internal/Callback/CallbackCallHandler.cs index ced3dfc4b..b0020bfbb 100644 --- a/src/Generation/Generator/Renderer/Internal/Callback/CallbackCallHandler.cs +++ b/src/Generation/Generator/Renderer/Internal/Callback/CallbackCallHandler.cs @@ -22,7 +22,7 @@ namespace {Namespace.GetInternalName(callback.Namespace)}; // AUTOGENERATED FILE - DO NOT MODIFY -// +/// /// Call Handler for {callback.Name}. A call annotation indicates the closure should /// be valid for the duration of the call. This handler does not implement any special /// memory management. diff --git a/src/Generation/Generator/Renderer/Internal/Class/ClassHandle.cs b/src/Generation/Generator/Renderer/Internal/Class/ClassHandle.cs new file mode 100644 index 000000000..2143ecf74 --- /dev/null +++ b/src/Generation/Generator/Renderer/Internal/Class/ClassHandle.cs @@ -0,0 +1,101 @@ +using System; +using Generator.Model; + +namespace Generator.Renderer.Internal; + +internal static class ClassHandle +{ + public static string Render(GirModel.Class cls) + { + return cls.Final + ? RenderFinalClassHandle(cls) + : RenderStandardClassHandle(cls); + } + + private static string RenderFinalClassHandle(GirModel.Class cls) + { + var handleName = Class.GetInternalHandleName(cls); + + return $$""" + using System; + using System.Linq; + using System.Runtime.InteropServices; + using System.Runtime.Versioning; + + #nullable enable + + namespace {{Namespace.GetInternalName(cls.Namespace)}}; + + public partial class {{handleName}} : {{RenderParent(cls)}} + { + public {{handleName}}(IntPtr handle, bool ownsHandle) : base(handle, ownsHandle){ } + + public static {{handleName}} Create(bool owned, GObject.ConstructArgument[] constructArguments) + { + // We can't check if a reference is floating via "g_object_is_floating" here + // as the function could be "lying" depending on the intent of framework writers. + // E.g. A Gtk.Window created via "g_object_new_with_properties" returns an unowned + // reference which is not marked as floating as the gtk toolkit "owns" it. + // For this reason we just delegate the problem to the caller and require a + // definition whether the ownership of the new object will be transferred to us or not. + + var ptr = GObject.Internal.Object.NewWithProperties( + objectType: {{Class.GetFullyQualifiedInternalName(cls)}}.GetGType(), + nProperties: (uint) constructArguments.Length, + names: constructArguments.Select(x => x.Name).ToArray(), + values: GObject.Internal.ValueArray2OwnedHandle.Create(constructArguments.Select(x => x.Value).ToArray()) + ); + + return new {{handleName}}(ptr, owned); + } + } + """; + } + + private static string RenderStandardClassHandle(GirModel.Class cls) + { + var handleName = Class.GetInternalHandleName(cls); + + return $$""" + using System; + using System.Linq; + using System.Runtime.InteropServices; + using System.Runtime.Versioning; + + #nullable enable + + namespace {{Namespace.GetInternalName(cls.Namespace)}}; + + public partial class {{handleName}} : {{RenderParent(cls)}} + { + public {{handleName}}(IntPtr handle, bool ownsHandle) : base(handle, ownsHandle){ } + + public static new {{handleName}} For(bool owned, GObject.ConstructArgument[] constructArguments) where T : {{Class.GetFullyQualifiedPublicName(cls)}}, GObject.GTypeProvider + { + // We can't check if a reference is floating via "g_object_is_floating" here + // as the function could be "lying" depending on the intent of framework writers. + // E.g. A Gtk.Window created via "g_object_new_with_properties" returns an unowned + // reference which is not marked as floating as the gtk toolkit "owns" it. + // For this reason we just delegate the problem to the caller and require a + // definition whether the ownership of the new object will be transferred to us or not. + + var ptr = GObject.Internal.Object.NewWithProperties( + objectType: T.GetGType(), + nProperties: (uint) constructArguments.Length, + names: constructArguments.Select(x => x.Name).ToArray(), + values: GObject.Internal.ValueArray2OwnedHandle.Create(constructArguments.Select(x => x.Value).ToArray()) + ); + + return new {{handleName}}(ptr, owned); + } + } + """; + } + + private static string RenderParent(GirModel.Class cls) + { + return cls.Parent is null + ? throw new Exception("Class is missing parent") + : Class.GetFullyQualifiedInternalHandleName(cls.Parent); + } +} diff --git a/src/Generation/Generator/Renderer/Internal/ClassMethods.cs b/src/Generation/Generator/Renderer/Internal/Class/ClassMethods.cs similarity index 100% rename from src/Generation/Generator/Renderer/Internal/ClassMethods.cs rename to src/Generation/Generator/Renderer/Internal/Class/ClassMethods.cs diff --git a/src/Generation/Generator/Renderer/Internal/ClassStruct.cs b/src/Generation/Generator/Renderer/Internal/Class/ClassStruct.cs similarity index 100% rename from src/Generation/Generator/Renderer/Internal/ClassStruct.cs rename to src/Generation/Generator/Renderer/Internal/Class/ClassStruct.cs diff --git a/src/Generation/Generator/Renderer/Internal/Framework/FrameworkTypeRegistration.cs b/src/Generation/Generator/Renderer/Internal/Framework/FrameworkTypeRegistration.cs index 9047ca51e..adb01e365 100644 --- a/src/Generation/Generator/Renderer/Internal/Framework/FrameworkTypeRegistration.cs +++ b/src/Generation/Generator/Renderer/Internal/Framework/FrameworkTypeRegistration.cs @@ -15,6 +15,8 @@ public static string Render(GirModel.Namespace ns) using System.Diagnostics; using System.Linq; using System.Runtime.InteropServices; +using GObject; +using GObject.Internal; namespace {Namespace.GetInternalName(ns)}; @@ -40,24 +42,24 @@ internal static void RegisterTypes() .Join(Environment.NewLine)} }} - private static void Register(Func getType, params OSPlatform[] supportedPlatforms) where T : class + private static void Register(params OSPlatform[] supportedPlatforms) where T : InstanceFactory, GTypeProvider {{ - try - {{ + try + {{ if(supportedPlatforms.Any(RuntimeInformation.IsOSPlatform)) - GObject.Internal.TypeDictionary.Add(typeof(T), new GObject.Type(getType())); - }} - catch(System.Exception e) - {{ - Debug.WriteLine($""Could not register type '{{nameof(T)}}': {{e.Message}}""); - }} - }} + GObject.Internal.DynamicInstanceFactory.Register(T.GetGType(), T.Create); + }} + catch(System.Exception e) + {{ + Debug.WriteLine($""Could not register type: {{e.Message}}""); + }} + }} }}"; } private static string RenderRegistration(GirModel.ComplexType type) { - return @$"Register<{ComplexType.GetFullyQualified(type)}>(Internal.{type.Name}.{Function.GetGType}{RenderPlatforms(type as GirModel.PlatformDependent)});"; + return $"Register<{ComplexType.GetFullyQualified(type)}>({RenderPlatforms(type as GirModel.PlatformDependent)});"; } private static string RenderPlatforms(GirModel.PlatformDependent? platformDependent) @@ -76,6 +78,6 @@ private static string RenderPlatforms(GirModel.PlatformDependent? platformDepend if (platformDependent.SupportsWindows) statements.Add("OSPlatform.Windows"); - return ", " + string.Join(", ", statements); + return string.Join(", ", statements); } } diff --git a/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Class.cs b/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Class.cs index 422237dda..7cfeb2aba 100644 --- a/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Class.cs +++ b/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Class.cs @@ -38,12 +38,14 @@ private static void Default(ParameterToManagedData parameterData) var parameterName = Model.Parameter.GetName(parameterData.Parameter); var callName = Model.Parameter.GetConvertedName(parameterData.Parameter); + var type = Model.ComplexType.GetFullyQualified(cls); + var wrapHandle = parameterData.Parameter.Nullable - ? "GObject.Internal.ObjectWrapper.WrapNullableHandle" - : "GObject.Internal.ObjectWrapper.WrapHandle"; - + ? $"({type}?) GObject.Internal.InstanceWrapper.WrapNullableHandle" + : $"({type}) GObject.Internal.InstanceWrapper.WrapHandle"; + parameterData.SetSignatureName(() => parameterName); - parameterData.SetExpression(() => $"var {callName} = {wrapHandle}<{Model.ComplexType.GetFullyQualified(cls)}>({parameterName}, {Model.Transfer.IsOwnedRef(parameterData.Parameter.Transfer).ToString().ToLower()});"); + parameterData.SetExpression(() => $"var {callName} = {wrapHandle}<{type}>({parameterName}, {Model.Transfer.IsOwnedRef(parameterData.Parameter.Transfer).ToString().ToLower()});"); parameterData.SetCallName(() => callName); } } diff --git a/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Interface.cs b/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Interface.cs index 9ba89fcd3..873e26cb8 100644 --- a/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Interface.cs +++ b/src/Generation/Generator/Renderer/Internal/ParameterToManagedExpression/Converter/Interface.cs @@ -19,7 +19,7 @@ public void Initialize(ParameterToManagedData parameterData, IEnumerable signatureName); - parameterData.SetExpression(() => $"var {callName} = GObject.Internal.ObjectWrapper.WrapInterfaceHandle<{Model.Interface.GetFullyQualifiedImplementationName(iface)}>({signatureName}, {Model.Transfer.IsOwnedRef(parameterData.Parameter.Transfer).ToString().ToLower()});"); + parameterData.SetExpression(() => $"var {callName} = ({Model.Type.GetPublicNameFullyQuallified(iface)}) GObject.Internal.InstanceWrapper.WrapHandle<{Model.Interface.GetFullyQualifiedImplementationName(iface)}>({signatureName}, {Model.Transfer.IsOwnedRef(parameterData.Parameter.Transfer).ToString().ToLower()});"); parameterData.SetCallName(() => callName); } } diff --git a/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Class.cs b/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Class.cs index 28c2b0578..40658fa58 100644 --- a/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Class.cs +++ b/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Class.cs @@ -14,7 +14,7 @@ public string GetString(GirModel.ReturnType returnType, string fromVariableName) throw new NotImplementedException($"{returnType.AnyType}: class return type which is no pointer can not be converted to native"); return returnType.Nullable - ? fromVariableName + "?.Handle ?? IntPtr.Zero" - : fromVariableName + ".Handle"; + ? fromVariableName + "?.Handle.DangerousGetHandle() ?? IntPtr.Zero" + : fromVariableName + ".Handle.DangerousGetHandle()"; } } diff --git a/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Interface.cs b/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Interface.cs index f2445de21..3f42dc83b 100644 --- a/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Interface.cs +++ b/src/Generation/Generator/Renderer/Internal/ReturnTypeToNativeExpression/Converter/Interface.cs @@ -8,7 +8,7 @@ public bool Supports(GirModel.AnyType type) public string GetString(GirModel.ReturnType returnType, string fromVariableName) { return returnType.Nullable - ? fromVariableName + "?.Handle ?? IntPtr.Zero" - : fromVariableName + ".Handle"; + ? fromVariableName + "?.Handle.DangerousGetHandle() ?? IntPtr.Zero" + : fromVariableName + ".Handle.DangerousGetHandle()"; } } diff --git a/src/Generation/Generator/Renderer/Public/Class/ClassFramework.cs b/src/Generation/Generator/Renderer/Public/Class/ClassFramework.cs index 0afbf52eb..08decf41f 100644 --- a/src/Generation/Generator/Renderer/Public/Class/ClassFramework.cs +++ b/src/Generation/Generator/Renderer/Public/Class/ClassFramework.cs @@ -13,7 +13,6 @@ public static string Render(GirModel.Class cls) return $@" using System; -using GObject; using System.Runtime.InteropServices; using System.Runtime.Versioning; @@ -26,7 +25,8 @@ namespace {Namespace.GetPublicName(cls.Namespace)}; {PlatformSupportAttribute.Render(cls as GirModel.PlatformDependent)} public {@sealed}partial class {cls.Name} {RenderInheritance(cls)} {{ - {RenderParentConstructors(cls)} + {$"protected internal { cls.Name }({Class.GetFullyQualifiedInternalHandleName(cls)} handle) : base(handle) {{ }}"} + {RenderPublicConstructor(cls)} }}"; } @@ -36,7 +36,7 @@ private static string RenderInheritance(GirModel.Class cls) var interfaces = cls.Implements.Select(ComplexType.GetFullyQualified); var elements = new List(interfaces); - + if (parentClass is not null) elements.Insert(0, parentClass); @@ -45,46 +45,10 @@ private static string RenderInheritance(GirModel.Class cls) : $": {string.Join(", ", elements)}"; } - private static string RenderParentConstructors(GirModel.Class cls) - { - if (cls.Parent is null) - return string.Empty; - - var constructors = new List() - { - $@"protected internal { cls.Name }(IntPtr ptr, bool ownedRef) : base(ptr, ownedRef) {{}}", - }; - - if (IsInitiallyUnowned(cls)) - { - constructors.Add($@" -// As initially unowned objects always start with a floating reference -// we can safely set the ""owned"" parameter to false. -protected internal {cls.Name}(params ConstructArgument[] constructArguments) : base(owned: false, constructArguments) {{}}"); - constructors.Add($"public {cls.Name}() : this(Array.Empty()) {{}}"); - } - else if (InheritsInitiallyUnowned(cls)) - { - constructors.Add($"protected internal {cls.Name}(params ConstructArgument[] constructArguments) : base(constructArguments) {{}}"); - constructors.Add($"public {cls.Name}() : this(Array.Empty()) {{}}"); - } - else - { - constructors.Add($"protected internal {cls.Name}(bool owned, params ConstructArgument[] constructArguments) : base(owned, constructArguments) {{}}"); - } - - return constructors.Join(Environment.NewLine); - } - - private static bool IsInitiallyUnowned(GirModel.Class cls) => IsNamedInitiallyUnowned(cls.Name); - - private static bool InheritsInitiallyUnowned(GirModel.Class @class) + private static string RenderPublicConstructor(GirModel.Class cls) { - if (@class.Parent is null) - return false; - - return IsNamedInitiallyUnowned(@class.Parent.Name) || InheritsInitiallyUnowned(@class.Parent); + return Class.IsInitiallyUnowned(cls) + ? $" public {cls.Name}() : this({Class.GetFullyQualifiedInternalHandleName(cls)}.For<{cls.Name}>(false, Array.Empty())) {{ }}" + : $" public {cls.Name}() : this({Class.GetFullyQualifiedInternalHandleName(cls)}.For<{cls.Name}>(true, Array.Empty())) {{ }}"; } - - private static bool IsNamedInitiallyUnowned(string name) => name == "InitiallyUnowned"; } diff --git a/src/Generation/Generator/Renderer/Public/Class/ClassFunctions.cs b/src/Generation/Generator/Renderer/Public/Class/ClassFunctions.cs index b1bfddcc7..3766e91e3 100644 --- a/src/Generation/Generator/Renderer/Public/Class/ClassFunctions.cs +++ b/src/Generation/Generator/Renderer/Public/Class/ClassFunctions.cs @@ -20,13 +20,37 @@ namespace {Namespace.GetPublicName(cls.Namespace)}; // AUTOGENERATED FILE - DO NOT MODIFY -public partial class {cls.Name} : GObject.GTypeProvider +public partial class {cls.Name} : GObject.GTypeProvider, GObject.InstanceFactory {{ - {FunctionRenderer.Render(cls.TypeFunction)} + + static GObject.Type GTypeProvider.GetGType() + {{ + return {RenderGetGType(cls.TypeFunction)}; + }} + + static object GObject.InstanceFactory.Create(IntPtr handle, bool ownsHandle) + {{ + return CreateIntern(handle, ownsHandle); + }} + + private static {cls.Name} CreateIntern(IntPtr handle, bool ownsHandle) + {{ + return {RenderObjectFactory(cls)}; + }} {cls.Functions .Select(FunctionRenderer.Render) .Join(Environment.NewLine)} }}"; } + + private static string RenderGetGType(GirModel.Function function) + { + return $"{Namespace.GetInternalName(function.Namespace)}.{function.Parent!.Name}.{Function.GetName(function)}()"; + } + + private static string RenderObjectFactory(GirModel.Class cls) + { + return $"new {cls.Name}(new {Class.GetFullyQualifiedInternalHandleName(cls)}(handle, ownsHandle))"; + } } diff --git a/src/Generation/Generator/Renderer/Public/Class/InterfaceProperties/ClassInterfacePropertiesRenderer.Accessor.cs b/src/Generation/Generator/Renderer/Public/Class/InterfaceProperties/ClassInterfacePropertiesRenderer.Accessor.cs index c8cf52628..df7bb41c9 100644 --- a/src/Generation/Generator/Renderer/Public/Class/InterfaceProperties/ClassInterfacePropertiesRenderer.Accessor.cs +++ b/src/Generation/Generator/Renderer/Public/Class/InterfaceProperties/ClassInterfacePropertiesRenderer.Accessor.cs @@ -36,14 +36,14 @@ private static string RenderAccessor(GirModel.Interface @interface, GirModel.Pro private static string GetGetter(GirModel.Interface @interface, GirModel.Property property) { return Property.SupportsAccessorGetMethod(property, out var getter) - ? $"{Namespace.GetInternalName(@interface.Namespace)}.{@interface.Name}.{Method.GetInternalName(getter)}(Handle)" + ? $"{Namespace.GetInternalName(@interface.Namespace)}.{@interface.Name}.{Method.GetInternalName(getter)}(Handle.DangerousGetHandle())" : $"{ComplexType.GetFullyQualified(@interface)}.{Property.GetDescriptorName(property)}.Get(this)"; } private static string GetSetter(GirModel.Interface @interface, GirModel.Property property) { return Property.SupportsAccessorSetMethod(property, out var setter) - ? $"{Namespace.GetInternalName(@interface.Namespace)}.{@interface.Name}.{Method.GetInternalName(setter)}(Handle, value)" + ? $"{Namespace.GetInternalName(@interface.Namespace)}.{@interface.Name}.{Method.GetInternalName(setter)}(Handle.DangerousGetHandle(), value)" : $"{ComplexType.GetFullyQualified(@interface)}.{Property.GetDescriptorName(property)}.Set(this, value)"; } } diff --git a/src/Generation/Generator/Renderer/Public/Class/Properties/ClassPropertiesRenderer.Accessor.cs b/src/Generation/Generator/Renderer/Public/Class/Properties/ClassPropertiesRenderer.Accessor.cs index 0edefbd33..468cd32ec 100644 --- a/src/Generation/Generator/Renderer/Public/Class/Properties/ClassPropertiesRenderer.Accessor.cs +++ b/src/Generation/Generator/Renderer/Public/Class/Properties/ClassPropertiesRenderer.Accessor.cs @@ -28,14 +28,14 @@ private static string RenderAccessor(GirModel.ComplexType complexType, GirModel. private static string GetGetter(GirModel.ComplexType complexType, GirModel.Property property) { return Property.SupportsAccessorGetMethod(property, out var getter) - ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(getter)}(Handle)" + ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(getter)}(base.Handle.DangerousGetHandle())" : $"{Property.GetDescriptorName(property)}.Get(this)"; } private static string GetSetter(GirModel.ComplexType complexType, GirModel.Property property) { return Property.SupportsAccessorSetMethod(property, out var setter) - ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(setter)}(Handle, value)" + ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(setter)}(base.Handle.DangerousGetHandle(), value)" : $"{Property.GetDescriptorName(property)}.Set(this, value)"; } } diff --git a/src/Generation/Generator/Renderer/Public/Constructor/Converter/Class.cs b/src/Generation/Generator/Renderer/Public/Constructor/Converter/Class.cs index dc190c18d..c4db50228 100644 --- a/src/Generation/Generator/Renderer/Public/Constructor/Converter/Class.cs +++ b/src/Generation/Generator/Renderer/Public/Constructor/Converter/Class.cs @@ -24,6 +24,6 @@ private static string CreateExpression(GirModel.Constructor constructor, string return cls.Fundamental ? $"new {cls.Name}({fromVariableName})" - : $"new {cls.Name}({fromVariableName}, {ownedRef.ToString().ToLower()})"; + : $"{cls.Name}.CreateIntern({fromVariableName}, {ownedRef.ToString().ToLower()})"; } } diff --git a/src/Generation/Generator/Renderer/Public/ForeignTypedRecord.cs b/src/Generation/Generator/Renderer/Public/ForeignTypedRecord.cs index e3531734c..3dd6d970b 100644 --- a/src/Generation/Generator/Renderer/Public/ForeignTypedRecord.cs +++ b/src/Generation/Generator/Renderer/Public/ForeignTypedRecord.cs @@ -24,31 +24,33 @@ namespace {Namespace.GetPublicName(record.Namespace)}; // AUTOGENERATED FILE - DO NOT MODIFY {PlatformSupportAttribute.Render(record as GirModel.PlatformDependent)} -public partial class {name} : GLib.BoxedRecord, IDisposable +public partial class {name} : GLib.BoxedRecord, GObject.GTypeProvider, GObject.InstanceFactory, IDisposable {{ public {internalHandleName} Handle {{ get; }} public {name}({internalHandleName} handle) {{ Handle = handle; - Initialize(); }} - //TODO: This is a workaround constructor as long as we are - //not having https://github.com/gircore/gir.core/issues/397 - private {name}(IntPtr ptr, bool ownsHandle) : this(ownsHandle - ? new {Model.ForeignTypedRecord.GetFullyQuallifiedOwnedHandle(record)}(ptr) - : new {Model.ForeignTypedRecord.GetFullyQuallifiedUnownedHandle(record)}(ptr).OwnedCopy()){{ }} + static object GObject.InstanceFactory.Create(IntPtr handle, bool ownsHandle) + {{ + var safeHandle = ownsHandle + ? new {Record.GetFullyQualifiedInternalOwnedHandle(record)}(handle) + : {Record.GetFullyQualifiedInternalOwnedHandle(record)}.FromUnowned(handle); - // Implement this to perform additional steps in the constructor - partial void Initialize(); + return new {name}(safeHandle); + }} System.IntPtr GLib.BoxedRecord.GetHandle() {{ return Handle.DangerousGetHandle(); }} - {FunctionRenderer.Render(record.TypeFunction)} + static GObject.Type GObject.GTypeProvider.GetGType() + {{ + return {RenderGetGType(record.TypeFunction!)}; + }} public void Dispose() {{ @@ -56,4 +58,9 @@ public void Dispose() }} }}"; } + + private static string RenderGetGType(GirModel.Function function) + { + return $"{Namespace.GetInternalName(function.Namespace)}.{function.Parent!.Name}.{Function.GetName(function)}()"; + } } diff --git a/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Class.cs b/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Class.cs index b63b94d5c..fb3fe298e 100644 --- a/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Class.cs +++ b/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Class.cs @@ -9,6 +9,13 @@ public bool Supports(GirModel.Type type) public string GetExpression(GirModel.InstanceParameter instanceParameter) { - return "this.Handle"; + var cls = (GirModel.Class) instanceParameter.Type; + + if (cls.Fundamental) + return "this.Handle"; + + return cls.Parent is null + ? "this.Handle.DangerousGetHandle()" + : "base.Handle.DangerousGetHandle()"; } } diff --git a/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Interface.cs b/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Interface.cs index 67d3c54a4..f45347a5f 100644 --- a/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Interface.cs +++ b/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Interface.cs @@ -9,6 +9,6 @@ public bool Supports(GirModel.Type type) public string GetExpression(GirModel.InstanceParameter instanceParameter) { - return "this.Handle"; + return "this.Handle.DangerousGetHandle()"; } } diff --git a/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Pointer.cs b/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Pointer.cs index ad03d3ba3..042959537 100644 --- a/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Pointer.cs +++ b/src/Generation/Generator/Renderer/Public/InstanceParameterToNativeExpression/Converter/Pointer.cs @@ -9,6 +9,6 @@ public bool Supports(GirModel.Type type) public string GetExpression(GirModel.InstanceParameter instanceParameter) { - return "this.Handle"; + return "this.Handle.DangerousGetHandle()"; } } diff --git a/src/Generation/Generator/Renderer/Public/Interface/InterfaceFramework.cs b/src/Generation/Generator/Renderer/Public/Interface/InterfaceFramework.cs deleted file mode 100644 index 981cbbac4..000000000 --- a/src/Generation/Generator/Renderer/Public/Interface/InterfaceFramework.cs +++ /dev/null @@ -1,40 +0,0 @@ -using Generator.Model; - -namespace Generator.Renderer.Public; - -internal static class InterfaceFramework -{ - public static string Render(GirModel.Interface iface) - { - return $@" -using System; -using GObject; -using System.Runtime.InteropServices; -using System.Runtime.Versioning; - -#nullable enable - -namespace {Namespace.GetPublicName(iface.Namespace)}; - -// AUTOGENERATED FILE - DO NOT MODIFY - -{PlatformSupportAttribute.Render(iface as GirModel.PlatformDependent)} -public partial interface {iface.Name} : GLib.IHandle -{{ -/* - IntPtr GLib.IHandle.Handle - {{ - get - {{ - System.Diagnostics.Debug.Assert( - condition: GetType().IsAssignableTo(typeof(GObject.Object)), - message: $""Interface '{{nameof({iface.Name})}}' can only be implemented on GObject-based classes"" - ); - - return ((GObject.Object)this).Handle; - }} - }} -*/ -}}"; - } -} diff --git a/src/Generation/Generator/Renderer/Public/Interface/InterfaceProperties.cs b/src/Generation/Generator/Renderer/Public/Interface/InterfaceProperties.cs index 6a5222402..ce5dfab02 100644 --- a/src/Generation/Generator/Renderer/Public/Interface/InterfaceProperties.cs +++ b/src/Generation/Generator/Renderer/Public/Interface/InterfaceProperties.cs @@ -12,14 +12,19 @@ public static string Render(GirModel.Interface @interface) return $@" using System; using GObject; +using GObject.Internal; using System.Runtime.InteropServices; + #nullable enable + namespace {Namespace.GetPublicName(@interface.Namespace)} {{ // AUTOGENERATED FILE - DO NOT MODIFY public partial interface {@interface.Name} {{ + ObjectHandle Handle {{ get; }} + {@interface.Properties .Where(Property.IsEnabled) .Select(x => Render(@interface, x)) diff --git a/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationFramework.cs b/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationFramework.cs index ca5f45776..33dadbbd3 100644 --- a/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationFramework.cs +++ b/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationFramework.cs @@ -19,11 +19,16 @@ namespace {Namespace.GetPublicName(@interface.Namespace)}; // AUTOGENERATED FILE - DO NOT MODIFY {PlatformSupportAttribute.Render(@interface as GirModel.PlatformDependent)} -public sealed partial class {Interface.GetImplementationName(@interface)} : GObject.Object, {@interface.Name} +public sealed partial class {Interface.GetImplementationName(@interface)} : GObject.Object, {@interface.Name}, GObject.InstanceFactory {{ - public {Interface.GetImplementationName(@interface)}(IntPtr handle, bool ownedRef) : base(handle, ownedRef) + public {Interface.GetImplementationName(@interface)}(GObject.Internal.ObjectHandle handle) : base(handle) {{ }} + + static object GObject.InstanceFactory.Create(IntPtr handle, bool ownsHandle) + {{ + return new {Interface.GetImplementationName(@interface)}(new GObject.Internal.ObjectHandle(handle, ownsHandle)); + }} }}"; } } diff --git a/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationProperties.cs b/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationProperties.cs index 80dff3caa..7362b4cc0 100644 --- a/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationProperties.cs +++ b/src/Generation/Generator/Renderer/Public/InterfaceImplementation/InterfaceImplementationProperties.cs @@ -76,14 +76,14 @@ private static string RenderAccessor(GirModel.ComplexType complexType, GirModel. private static string GetGetter(GirModel.ComplexType complexType, GirModel.Property property) { return Property.SupportsAccessorGetMethod(property, out var getter) - ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(getter)}(Handle)" + ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(getter)}(Handle.DangerousGetHandle())" : $"{ComplexType.GetFullyQualified(complexType)}.{Property.GetDescriptorName(property)}.Get(this)"; } private static string GetSetter(GirModel.ComplexType complexType, GirModel.Property property) { return Property.SupportsAccessorSetMethod(property, out var setter) - ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(setter)}(Handle, value)" + ? $"{Namespace.GetInternalName(complexType.Namespace)}.{complexType.Name}.{Method.GetInternalName(setter)}(Handle.DangerousGetHandle(), value)" : $"{ComplexType.GetFullyQualified(complexType)}.{Property.GetDescriptorName(property)}.Set(this, value)"; } } diff --git a/src/Generation/Generator/Renderer/Public/OpaqueTypedRecord.cs b/src/Generation/Generator/Renderer/Public/OpaqueTypedRecord.cs index ffd186001..3d86685dd 100644 --- a/src/Generation/Generator/Renderer/Public/OpaqueTypedRecord.cs +++ b/src/Generation/Generator/Renderer/Public/OpaqueTypedRecord.cs @@ -30,7 +30,7 @@ namespace {Namespace.GetPublicName(record.Namespace)}; // AUTOGENERATED FILE - DO NOT MODIFY {PlatformSupportAttribute.Render(record as GirModel.PlatformDependent)} -public sealed partial class {name} : GLib.BoxedRecord, IEquatable<{name}>, IDisposable +public sealed partial class {name} : GLib.BoxedRecord, GObject.GTypeProvider, GObject.InstanceFactory, IEquatable<{name}>, IDisposable {{ public {internalHandleName} Handle {{ get; }} @@ -38,23 +38,25 @@ public sealed partial class {name} : GLib.BoxedRecord, IEquatable<{name}>, IDisp {{ Handle = handle; Handle.SetMemoryPressure(); - Initialize(); }} - //TODO: This is a workaround constructor as long as we are - //not having https://github.com/gircore/gir.core/issues/397 - private {name}(IntPtr ptr, bool ownsHandle) : this(ownsHandle - ? new {Model.OpaqueTypedRecord.GetFullyQuallifiedOwnedHandle(record)}(ptr) - : new {Model.OpaqueTypedRecord.GetFullyQuallifiedUnownedHandle(record)}(ptr).OwnedCopy()){{ }} + static object GObject.InstanceFactory.Create(IntPtr handle, bool ownsHandle) + {{ + var safeHandle = ownsHandle + ? new {Record.GetFullyQualifiedInternalOwnedHandle(record)}(handle) + : {Record.GetFullyQualifiedInternalOwnedHandle(record)}.FromUnowned(handle); - // Implement this to perform additional steps in the constructor - partial void Initialize(); + return new {name}(safeHandle); + }} {record.Constructors .Select(ConstructorRenderer.Render) .Join(Environment.NewLine)} - {FunctionRenderer.Render(record.TypeFunction)} + static GObject.Type GObject.GTypeProvider.GetGType() + {{ + return {RenderGetGType(record.TypeFunction!)}; + }} {record.Functions .Select(FunctionRenderer.Render) @@ -100,4 +102,9 @@ public void Dispose() }} }}"; } + + private static string RenderGetGType(GirModel.Function function) + { + return $"{Namespace.GetInternalName(function.Namespace)}.{function.Parent!.Name}.{Function.GetName(function)}()"; + } } diff --git a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Class.cs b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Class.cs index 04dc12a0b..2c74bf846 100644 --- a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Class.cs +++ b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Class.cs @@ -10,12 +10,22 @@ public bool Supports(GirModel.AnyType type) => type.Is(); public void Initialize(ParameterToNativeData parameter, IEnumerable _) + { + var cls = (GirModel.Class) parameter.Parameter.AnyTypeOrVarArgs.AsT0.AsT0; + + if (cls.Fundamental) + Fundamental(parameter); + else + Default(parameter); + } + + private static void Fundamental(ParameterToNativeData parameter) { if (parameter.Parameter.Direction != GirModel.Direction.In) - throw new NotImplementedException($"{parameter.Parameter.AnyTypeOrVarArgs}: class parameter with direction != in not yet supported"); + throw new NotImplementedException($"{parameter.Parameter.AnyTypeOrVarArgs}: fundamental class parameter with direction != in not yet supported"); if (!parameter.Parameter.IsPointer) - throw new NotImplementedException($"{parameter.Parameter.AnyTypeOrVarArgs}: class parameter which is no pointer can not be converted to native"); + throw new NotImplementedException($"{parameter.Parameter.AnyTypeOrVarArgs}: fundamental class parameter which is no pointer can not be converted to native"); var parameterName = Model.Parameter.GetName(parameter.Parameter); var callParameter = parameter.Parameter.Nullable @@ -37,4 +47,33 @@ public void Initialize(ParameterToNativeData parameter, IEnumerable parameterName); parameter.SetCallName(() => callParameter); } + + private static void Default(ParameterToNativeData parameter) + { + if (parameter.Parameter.Direction != GirModel.Direction.In) + throw new NotImplementedException($"{parameter.Parameter.AnyTypeOrVarArgs}: class parameter with direction != in not yet supported"); + + if (!parameter.Parameter.IsPointer) + throw new NotImplementedException($"{parameter.Parameter.AnyTypeOrVarArgs}: class parameter which is no pointer can not be converted to native"); + + var parameterName = Model.Parameter.GetName(parameter.Parameter); + var callParameter = parameter.Parameter.Nullable + ? parameterName + "?.Handle.DangerousGetHandle() ?? IntPtr.Zero" + : parameterName + ".Handle.DangerousGetHandle()"; + + // If there is an ownership transfer, the called function will not add + // a ref but will hold onto the object and later remove a ref. + // However, the original owned ref still exists (e.g. the managed object's handle) + // so we need to add an extra ref to account for the remaining owner. + if (Transfer.IsOwnedRef(parameter.Parameter.Transfer)) + { + var addRefExpression = parameter.Parameter.Nullable + ? $"if({parameterName}?.Handle is not null) GObject.Internal.Object.Ref({parameterName}.Handle.DangerousGetHandle());" + : $"GObject.Internal.Object.Ref({parameterName}.Handle.DangerousGetHandle());"; + parameter.SetExpression(() => addRefExpression); + } + + parameter.SetSignatureName(() => parameterName); + parameter.SetCallName(() => callParameter); + } } diff --git a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/ClassArray.cs b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/ClassArray.cs index a5755b51d..24f302bb1 100644 --- a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/ClassArray.cs +++ b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/ClassArray.cs @@ -21,6 +21,6 @@ public void Initialize(ParameterToNativeData parameter, IEnumerable parameterName); parameter.SetCallName(() => nativeVariableName); - parameter.SetExpression(() => $"var {nativeVariableName} = {parameterName}.Select(cls => cls.Handle).ToArray();"); + parameter.SetExpression(() => $"var {nativeVariableName} = {parameterName}.Select(cls => cls.Handle.DangerousGetHandle()).ToArray();"); } } diff --git a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Interface.cs b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Interface.cs index e3bfc6347..00f68c932 100644 --- a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Interface.cs +++ b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/Interface.cs @@ -16,8 +16,8 @@ public void Initialize(ParameterToNativeData parameter, IEnumerable parameterName); parameter.SetCallName(() => callParameter); @@ -29,8 +29,8 @@ public void Initialize(ParameterToNativeData parameter, IEnumerable addRefExpression); } } diff --git a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/InterfaceArray.cs b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/InterfaceArray.cs index 955cfef36..fb746f2f4 100644 --- a/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/InterfaceArray.cs +++ b/src/Generation/Generator/Renderer/Public/ParameterToNativeExpression/Converter/InterfaceArray.cs @@ -15,6 +15,6 @@ public void Initialize(ParameterToNativeData parameter, IEnumerable parameterName); parameter.SetCallName(() => nativevariableName); - parameter.SetExpression(() => $"var {nativevariableName} = {parameterName}.Select(iface => (iface as GObject.Object).Handle).ToArray();"); + parameter.SetExpression(() => $"var {nativevariableName} = {parameterName}.Select(iface => (iface as GObject.Object).Handle.DangerousGetHandle()).ToArray();"); } } diff --git a/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Class.cs b/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Class.cs index b793509df..cfbee21f0 100644 --- a/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Class.cs +++ b/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Class.cs @@ -36,8 +36,9 @@ private static string Fundamental(GirModel.Class cls, GirModel.ReturnType return private static string Standard(GirModel.Class cls, GirModel.ReturnType returnType, string fromVariableName) { + var type = ComplexType.GetFullyQualified(cls); return returnType.Nullable - ? $"GObject.Internal.ObjectWrapper.WrapNullableHandle<{ComplexType.GetFullyQualified(cls)}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})" - : $"GObject.Internal.ObjectWrapper.WrapHandle<{ComplexType.GetFullyQualified(cls)}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})"; + ? $"({type}?) GObject.Internal.InstanceWrapper.WrapNullableHandle<{type}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})" + : $"({type}) GObject.Internal.InstanceWrapper.WrapHandle<{type}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})"; } } diff --git a/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Interface.cs b/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Interface.cs index c02863c33..29856cab8 100644 --- a/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Interface.cs +++ b/src/Generation/Generator/Renderer/Public/ReturnTypeToManagedExpression/Converter/Interface.cs @@ -21,8 +21,8 @@ public void Initialize(ReturnTypeToManagedData data, IEnumerable({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})" - : $"GObject.Internal.ObjectWrapper.WrapInterfaceHandle<{Model.Interface.GetFullyQualifiedImplementationName(@interface)}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})"; + ? $"({Model.Type.GetPublicNameFullyQuallified(@interface)}?) GObject.Internal.InstanceWrapper.WrapNullableHandle<{Model.Interface.GetFullyQualifiedImplementationName(@interface)}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})" + : $"({Model.Type.GetPublicNameFullyQuallified(@interface)}) GObject.Internal.InstanceWrapper.WrapHandle<{Model.Interface.GetFullyQualifiedImplementationName(@interface)}>({fromVariableName}, {Transfer.IsOwnedRef(returnType.Transfer).ToString().ToLower()})"; }); } } diff --git a/src/Generation/Generator/Renderer/Public/TypedRecord.cs b/src/Generation/Generator/Renderer/Public/TypedRecord.cs index ae8a65f34..4b104a159 100644 --- a/src/Generation/Generator/Renderer/Public/TypedRecord.cs +++ b/src/Generation/Generator/Renderer/Public/TypedRecord.cs @@ -31,34 +31,36 @@ namespace {Namespace.GetPublicName(record.Namespace)}; // AUTOGENERATED FILE - DO NOT MODIFY {PlatformSupportAttribute.Render(record as GirModel.PlatformDependent)} -public sealed partial class {name} : GLib.BoxedRecord, IEquatable<{name}>, IDisposable +public sealed partial class {name} : GLib.BoxedRecord, GObject.GTypeProvider, GObject.InstanceFactory, IEquatable<{name}>, IDisposable {{ public {internalHandleName} Handle {{ get; }} public {name}({internalHandleName} handle) {{ Handle = handle; - Initialize(); }} public {name}() : this({Model.TypedRecord.GetFullyQuallifiedManagedHandle(record)}.Create()) {{ }} - //TODO: This is a workaround constructor as long as we are - //not having https://github.com/gircore/gir.core/issues/397 - private {name}(IntPtr ptr, bool ownsHandle) : this(ownsHandle - ? new {Model.OpaqueTypedRecord.GetFullyQuallifiedOwnedHandle(record)}(ptr) - : new {Model.OpaqueTypedRecord.GetFullyQuallifiedUnownedHandle(record)}(ptr).OwnedCopy()){{ }} + static object GObject.InstanceFactory.Create(IntPtr handle, bool ownsHandle) + {{ + var safeHandle = ownsHandle + ? new {Record.GetFullyQualifiedInternalOwnedHandle(record)}(handle) + : {Record.GetFullyQualifiedInternalOwnedHandle(record)}.FromUnowned(handle); - // Implement this to perform additional steps in the constructor - partial void Initialize(); + return new {name}(safeHandle); + }} {record.Constructors .Select(ConstructorRenderer.Render) .Join(Environment.NewLine)} - {FunctionRenderer.Render(record.TypeFunction)} + static GObject.Type GObject.GTypeProvider.GetGType() + {{ + return {RenderGetGType(record.TypeFunction!)}; + }} {record.Fields .Select(f => RenderField(record, f)) @@ -108,6 +110,11 @@ public void Dispose() }} }}"; } + + private static string RenderGetGType(GirModel.Function function) + { + return $"{Namespace.GetInternalName(function.Namespace)}.{function.Parent!.Name}.{Function.GetName(function)}()"; + } private static string RenderField(GirModel.Record record, GirModel.Field field) { diff --git a/src/Libs/GObject-2.0/Public/GTypeProvider.cs b/src/Libs/GLib-2.0/GObject/Public/GTypeProvider.cs similarity index 76% rename from src/Libs/GObject-2.0/Public/GTypeProvider.cs rename to src/Libs/GLib-2.0/GObject/Public/GTypeProvider.cs index d7e0a5d99..b1f92a56e 100644 --- a/src/Libs/GObject-2.0/Public/GTypeProvider.cs +++ b/src/Libs/GLib-2.0/GObject/Public/GTypeProvider.cs @@ -2,7 +2,5 @@ public interface GTypeProvider { -#if NET7_0_OR_GREATER static abstract Type GetGType(); -#endif } diff --git a/src/Libs/GLib-2.0/GObject/Public/InstanceFactory.cs b/src/Libs/GLib-2.0/GObject/Public/InstanceFactory.cs new file mode 100644 index 000000000..d815d74de --- /dev/null +++ b/src/Libs/GLib-2.0/GObject/Public/InstanceFactory.cs @@ -0,0 +1,8 @@ +using System; + +namespace GObject; + +public interface InstanceFactory +{ + static abstract object Create(IntPtr handle, bool ownsHandle); +} diff --git a/src/Libs/GLib-2.0/Public/IHandle.cs b/src/Libs/GLib-2.0/Public/IHandle.cs deleted file mode 100644 index 86ce1145d..000000000 --- a/src/Libs/GLib-2.0/Public/IHandle.cs +++ /dev/null @@ -1,8 +0,0 @@ -using System; - -namespace GLib; - -public interface IHandle -{ - IntPtr Handle { get; } -} diff --git a/src/Libs/GObject-2.0/Internal/BoxedWrapper.cs b/src/Libs/GObject-2.0/Internal/BoxedWrapper.cs index 373efac44..77c547eec 100644 --- a/src/Libs/GObject-2.0/Internal/BoxedWrapper.cs +++ b/src/Libs/GObject-2.0/Internal/BoxedWrapper.cs @@ -1,6 +1,4 @@ using System; -using System.Linq; -using System.Reflection; namespace GObject.Internal; @@ -8,34 +6,13 @@ public class BoxedWrapper { public static object WrapHandle(IntPtr handle, bool ownsHandle, Type gtype) { - System.Type trueType = TypeDictionary.GetSystemType(gtype); - if (handle == IntPtr.Zero) - throw new NullReferenceException($"Failed to wrap handle as type <{trueType}>. Null handle passed to WrapHandle."); - - // Get constructor for the true type - var ctr = GetBoxedConstructor(trueType); - - if (ctr is null) - throw new Exception($"Type {trueType} does not define an IntPtr constructor. This could mean improperly defined bindings"); - - var result = ctr.Invoke(new object[] { handle, ownsHandle }); - - if (result == null) - throw new Exception($"Type {trueType}'s factory method returned a null object. This could mean improperly defined bindings"); - - return result; - } - - private static ConstructorInfo? GetBoxedConstructor(System.Type type) - { - // Create using 'IntPtr, ownsHandle' constructor - ConstructorInfo? ctor = type.GetConstructor( - System.Reflection.BindingFlags.NonPublic - | System.Reflection.BindingFlags.Public - | System.Reflection.BindingFlags.Instance, - null, new[] { typeof(IntPtr), typeof(bool) }, null - ); - return ctor; + throw new NullReferenceException("Failed to wrap boxed handle as a NULL handle was given."); + + //Using GObject.Object as a fallback type is not strictly correct as this handler is used for boxed + //records. As boxed records are known through the type system the fallback is never actually used. + var createInstance = DynamicInstanceFactory.GetInstanceFactory(gtype); + + return createInstance(handle, ownsHandle); } } diff --git a/src/Libs/GObject-2.0/Internal/CreateInstance.cs b/src/Libs/GObject-2.0/Internal/CreateInstance.cs new file mode 100644 index 000000000..fd8ec3df4 --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/CreateInstance.cs @@ -0,0 +1,5 @@ +using System; + +namespace GObject.Internal; + +public delegate object CreateInstance(IntPtr handle, bool ownsHandle); diff --git a/src/Libs/GObject-2.0/Internal/DynamicInstanceFactory.cs b/src/Libs/GObject-2.0/Internal/DynamicInstanceFactory.cs new file mode 100644 index 000000000..11f103aa1 --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/DynamicInstanceFactory.cs @@ -0,0 +1,57 @@ +using System; +using System.Collections.Generic; +using System.Runtime.CompilerServices; + +namespace GObject.Internal; + +public static class DynamicInstanceFactory +{ + private static readonly Dictionary InstanceFactories = new(); + + public static void Register(Type type, CreateInstance handleWrapper) + { + InstanceFactories.Add(type, handleWrapper); + } + + internal static object Create(IntPtr handle, bool ownsHandle) where TFallback : InstanceFactory, GTypeProvider + { + var type = GetType(handle); + var createInstance = GetInstanceFactory(type); + return createInstance(handle, ownsHandle); + } + + internal static CreateInstance GetInstanceFactory(Type gtype) where TFallback : InstanceFactory, GTypeProvider + { + if (InstanceFactories.TryGetValue(gtype, out CreateInstance? factory)) + return factory; + + var fallbackType = TFallback.GetGType(); + + // If the gtype is not found this could mean it is some anonymous subclass + // which is not public. So we look for the parent type, because this one could be known. + // If the parent is fundamental (e.g. GObject.Object) this is too unspecific to create. + var parent = GObject.Functions.TypeParent(gtype); + if (!Functions.IsFundamental(parent) && Functions.TypeIsA(parent, fallbackType)) + if(InstanceFactories.TryGetValue(parent, out factory)) + return factory; + + // If the gtype is not found this could mean it implements some known interface + foreach(var iface in GObject.Functions.TypeInterfaces(gtype)) + if(Functions.TypeIsA(iface, fallbackType)) + if(InstanceFactories.TryGetValue(iface, out factory)) + return factory; + + return TFallback.Create; + } + + private static unsafe Type GetType(IntPtr handle) + { + var gclass = Unsafe.AsRef((void*) handle).GClass; + var gtype = Unsafe.AsRef((void*) gclass).GType; + + if (gtype == 0) + throw new Exception("Could not retrieve type from class struct - is the struct valid?"); + + return new Type(gtype); + } +} diff --git a/src/Libs/GObject-2.0/Internal/Functions.cs b/src/Libs/GObject-2.0/Internal/Functions.cs index 2a027e4a3..57e929a21 100644 --- a/src/Libs/GObject-2.0/Internal/Functions.cs +++ b/src/Libs/GObject-2.0/Internal/Functions.cs @@ -14,9 +14,9 @@ public partial class Functions /// Returns whether the given type is a fundamental type. /// /// True if the type is fundamental otherwise false. - public static bool IsFundamental(nuint type) + public static bool IsFundamental(Type type) { //255 << 2 corresponds to G_TYPE_FUNDAMENTAL_MAX - return type <= (255 << 2); + return type.Value <= (255 << 2); } } diff --git a/src/Libs/GObject-2.0/Internal/GTypeProviderHelper.cs b/src/Libs/GObject-2.0/Internal/GTypeProviderHelper.cs deleted file mode 100644 index c118e82aa..000000000 --- a/src/Libs/GObject-2.0/Internal/GTypeProviderHelper.cs +++ /dev/null @@ -1,20 +0,0 @@ -using System; - -namespace GObject.Internal; - -#if NET6_0 - -//TODO: Remove this class once support for dotnet 6 is dropped -public static class GTypeProviderHelper -{ - public static Type GetGType() where T : GObject.Object, GTypeProvider - { - var getGTypeMethod = typeof(T).GetMethod(nameof(GObject.Object.GetGType)); - - if (getGTypeMethod is null) - throw new Exception($"Method {nameof(GObject.Object.GetGType)} not found on {typeof(T).Name}"); - - return (Type) (getGTypeMethod.Invoke(null, null) ?? throw new Exception($"Method {nameof(GObject.Object.GetGType)} on {typeof(T).Name} did not return a result")); - } -} -#endif diff --git a/src/Libs/GObject-2.0/Internal/InstanceCache.cs b/src/Libs/GObject-2.0/Internal/InstanceCache.cs new file mode 100644 index 000000000..63308cd10 --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/InstanceCache.cs @@ -0,0 +1,76 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Diagnostics.CodeAnalysis; +using System.Runtime.InteropServices; + +namespace GObject.Internal; + +internal static class InstanceCache +{ + private static readonly object Lock = new(); + private static readonly Dictionary Cache = new(); + internal static int ObjectCount + { + get + { + lock (Lock) + { + return Cache.Count; + } + } + } + + public static bool TryGetObject(IntPtr handle, [NotNullWhen(true)] out GObject.Object? obj) + { + lock (Lock) + { + if (Cache.TryGetValue(handle, out ToggleRef? toggleRef)) + { + if (toggleRef.Object is not null) + { + obj = toggleRef.Object; + return true; + } + } + } + + obj = null; + return false; + } + + public static unsafe void Add(IntPtr handle, GObject.Object obj) + { + lock (Cache) + { + Cache[handle] = new ToggleRef(obj); + ToggleRegistration.AddToggleRef(handle, &ToggleNotify); + Object.Unref(handle); + } + + Debug.WriteLine($"Handle {handle}: Added object of type '{obj.GetType()}' to {nameof(InstanceCache)}"); + } + + public static unsafe void Remove(IntPtr handle) + { + lock (Cache) + { + if (Cache.Remove(handle)) + ToggleRegistration.RemoveToggleRef(handle, &ToggleNotify); + } + + Debug.WriteLine($"Handle {handle}: Removed object from {nameof(InstanceCache)}."); + } + + [UnmanagedCallersOnly] + private static void ToggleNotify(IntPtr data, IntPtr @object, int isLastRef) + { + lock (Lock) + { + if (Cache.TryGetValue(@object, out var toggleRef)) + toggleRef.ToggleReference(isLastRef != 0); + else + Debug.WriteLine($"Handle {@object}: Could not toggle to {isLastRef} as there is no toggle reference."); + } + } +} diff --git a/src/Libs/GObject-2.0/Internal/InstanceWrapper.cs b/src/Libs/GObject-2.0/Internal/InstanceWrapper.cs new file mode 100644 index 000000000..6585b3d36 --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/InstanceWrapper.cs @@ -0,0 +1,24 @@ +using System; + +namespace GObject.Internal; + +public static class InstanceWrapper +{ + public static object? WrapNullableHandle(IntPtr handle, bool ownedRef) where TFallback : InstanceFactory, GTypeProvider + { + return handle == IntPtr.Zero + ? null + : WrapHandle(handle, ownedRef); + } + + public static object WrapHandle(IntPtr handle, bool ownedRef) where TFallback : InstanceFactory, GTypeProvider + { + if (handle == IntPtr.Zero) + throw new NullReferenceException("Failed to wrap handle: Null handle passed to WrapHandle."); + + if (InstanceCache.TryGetObject(handle, out var obj)) + return obj; + + return DynamicInstanceFactory.Create(handle, ownedRef); + } +} diff --git a/src/Libs/GObject-2.0/Internal/ObjectHandle.cs b/src/Libs/GObject-2.0/Internal/ObjectHandle.cs index c7b8070d1..483cb40e8 100644 --- a/src/Libs/GObject-2.0/Internal/ObjectHandle.cs +++ b/src/Libs/GObject-2.0/Internal/ObjectHandle.cs @@ -1,23 +1,72 @@ using System; +using System.Diagnostics; +using System.Linq; using System.Runtime.InteropServices; namespace GObject.Internal; public class ObjectHandle : SafeHandle { - public IntPtr Handle => IsInvalid ? IntPtr.Zero : DangerousGetHandle(); + public override bool IsInvalid => handle == IntPtr.Zero; - public ObjectHandle(IntPtr handle, object obj, bool ownedRef) : base(IntPtr.Zero, true) + public ObjectHandle(IntPtr handle, bool ownsHandle) : base(IntPtr.Zero, true) { - ObjectMapper.Map(handle, obj, ownedRef); SetHandle(handle); + OwnReference(ownsHandle); } - public sealed override bool IsInvalid => handle == IntPtr.Zero; + private void OwnReference(bool ownedRef) + { + if (!ownedRef) + { + // - Unowned GObjects need to be refed to bind them to this instance + // - Unowned InitiallyUnowned floating objects need to be ref_sinked + // - Unowned InitiallyUnowned non-floating objects need to be refed + // As ref_sink behaves like ref in case of non floating instances we use it for all 3 cases + Object.RefSink(handle); + } + else + { + //In case we own the ref because the ownership was fully transfered to us we + //do not need to ref the object at all. + + Debug.Assert(!Internal.Object.IsFloating(handle), $"Handle {handle}: Owned floating references are not possible."); + } + } + + internal void Cache(GObject.Object obj) + { + Debug.Assert(handle == obj.Handle.DangerousGetHandle(), "Must cache the instance of this handle."); - protected sealed override bool ReleaseHandle() + InstanceCache.Add(handle, obj); + } + + protected override bool ReleaseHandle() { - ObjectMapper.Unmap(handle); + RemoveMemoryPressure(); + InstanceCache.Remove(handle); return true; } + + protected internal virtual void AddMemoryPressure() { } + protected virtual void RemoveMemoryPressure() { } + + public static ObjectHandle For(bool owned, ConstructArgument[] constructArguments) where T : GObject.Object, GTypeProvider + { + // We can't check if a reference is floating via "g_object_is_floating" here + // as the function could be "lying" depending on the intent of framework writers. + // E.g. A Gtk.Window created via "g_object_new_with_properties" returns an unowned + // reference which is not marked as floating as the gtk toolkit "owns" it. + // For this reason we just delegate the problem to the caller and require a + // definition whether the ownership of the new object will be transferred to us or not. + + var ptr = Object.NewWithProperties( + objectType: T.GetGType(), + nProperties: (uint) constructArguments.Length, + names: constructArguments.Select(x => x.Name).ToArray(), + values: GObject.Internal.ValueArray2OwnedHandle.Create(constructArguments.Select(x => x.Value).ToArray()) + ); + + return new ObjectHandle(ptr, owned); + } } diff --git a/src/Libs/GObject-2.0/Internal/ObjectMapper.ToggleRef.cs b/src/Libs/GObject-2.0/Internal/ObjectMapper.ToggleRef.cs deleted file mode 100644 index cd54ed862..000000000 --- a/src/Libs/GObject-2.0/Internal/ObjectMapper.ToggleRef.cs +++ /dev/null @@ -1,81 +0,0 @@ -using System; -using System.Diagnostics; - -namespace GObject.Internal; - -public partial class ObjectMapper -{ - private class ToggleRef - { - private object _reference; - private readonly IntPtr _handle; - - public object? Object - { - get - { - if (_reference is not WeakReference weakRef) - return _reference; - - if (weakRef.Target is { } target) - return target; - - return null; - } - } - - /// - /// Initializes a toggle ref. The given object must be already owned by C# as the owned - /// reference is exchanged with a toggling reference meaning the toggle reference is taking control - /// over the reference. - /// This object saves a strong reference to the given object which prevents it from beeing garbage - /// collected. This strong reference is hold as long as there are other than our own toggling ref - /// on the given object. - /// If our toggeling ref is the last ref on the given object the strong reference is changed into a - /// weak reference. This allows the garbage collector to free the C# object which must result in the - /// call of the Dispose method of the ToggleRef. The Dispose mehtod removes the added toggle reference - /// and thus frees the last reference to the C object. - /// - public ToggleRef(IntPtr handle, object obj, bool ownedRef) - { - _reference = obj; - _handle = handle; - - OwnReference(ownedRef); - } - - private void OwnReference(bool ownedRef) - { - if (!ownedRef) - { - // - Unowned GObjects need to be refed to bind them to this instance - // - Unowned InitiallyUnowned floating objects need to be ref_sinked - // - Unowned InitiallyUnowned non-floating objects need to be refed - // As ref_sink behaves like ref in case of non floating instances we use it for all 3 cases - Internal.Object.RefSink(_handle); - } - else - { - //In case we own the ref because the ownership was fully transfered to us we - //do not need to ref the object at all. - - Debug.Assert(!Internal.Object.IsFloating(_handle), $"Handle {_handle}: Owned floating references are not possible."); - } - } - - internal void ToggleReference(bool isLastRef) - { - if (!isLastRef && _reference is WeakReference weakRef) - { - if (weakRef.Target is { } weakObj) - _reference = weakObj; - else - throw new Exception($"Handle {_handle}: Could not toggle reference to strong. It got garbage collected."); - } - else if (isLastRef && _reference is not WeakReference) - { - _reference = new WeakReference(_reference); - } - } - } -} diff --git a/src/Libs/GObject-2.0/Internal/ObjectMapper.ToggleRegistration.cs b/src/Libs/GObject-2.0/Internal/ObjectMapper.ToggleRegistration.cs deleted file mode 100644 index 0cc784e44..000000000 --- a/src/Libs/GObject-2.0/Internal/ObjectMapper.ToggleRegistration.cs +++ /dev/null @@ -1,32 +0,0 @@ -using System; -using System.Runtime.InteropServices; - -namespace GObject.Internal; - -public partial class ObjectMapper -{ - private static unsafe class ToggleRegistration - { - internal static void AddToggleRef(IntPtr handle) - { - AddToggleRef(handle, &ToggleNotify, IntPtr.Zero); - Internal.Object.Unref(handle); - } - - internal static void RemoveToggleRef(IntPtr handle) - { - var sourceFunc = new GLib.Internal.SourceFuncAsyncHandler(() => - { - RemoveToggleRef(handle, &ToggleNotify, IntPtr.Zero); - return false; - }); - GLib.Internal.MainContext.Invoke(GLib.Internal.MainContextUnownedHandle.NullHandle, sourceFunc.NativeCallback, IntPtr.Zero); - } - - [DllImport(ImportResolver.Library, EntryPoint = "g_object_add_toggle_ref")] - private static extern void AddToggleRef(IntPtr @object, delegate* unmanaged toggleNotify, IntPtr data); - - [DllImport(ImportResolver.Library, EntryPoint = "g_object_remove_toggle_ref")] - private static extern void RemoveToggleRef(IntPtr @object, delegate* unmanaged toggleNotify, IntPtr data); - } -} diff --git a/src/Libs/GObject-2.0/Internal/ObjectMapper.cs b/src/Libs/GObject-2.0/Internal/ObjectMapper.cs deleted file mode 100644 index 80bc5c5f4..000000000 --- a/src/Libs/GObject-2.0/Internal/ObjectMapper.cs +++ /dev/null @@ -1,77 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Diagnostics; -using System.Diagnostics.CodeAnalysis; -using System.Runtime.InteropServices; -using GLib; - -namespace GObject.Internal; - -public static partial class ObjectMapper -{ - private static readonly object Lock = new(); - private static readonly Dictionary WrapperObjects = new(); - - public static int ObjectCount - { - get - { - lock (Lock) - { - return WrapperObjects.Count; - } - } - } - - public static bool TryGetObject(IntPtr handle, [NotNullWhen(true)] out T? obj) where T : class, IHandle - { - lock (Lock) - { - if (WrapperObjects.TryGetValue(handle, out ToggleRef? weakRef)) - { - if (weakRef.Object is not null) - { - obj = (T) weakRef.Object; - return true; - } - } - } - - obj = null; - return false; - } - - public static void Map(IntPtr handle, object obj, bool ownedRef) - { - lock (Lock) - { - WrapperObjects[handle] = new ToggleRef(handle, obj, ownedRef); - ToggleRegistration.AddToggleRef(handle); - } - - Debug.WriteLine($"Handle {handle}: Mapped object of type '{obj.GetType()}' as owned ref '{ownedRef}'."); - } - - public static void Unmap(IntPtr handle) - { - lock (Lock) - { - if (WrapperObjects.Remove(handle)) - ToggleRegistration.RemoveToggleRef(handle); - } - - Debug.WriteLine($"Handle {handle}: Unmapped object."); - } - - [UnmanagedCallersOnly] - private static void ToggleNotify(IntPtr data, IntPtr @object, int isLastRef) - { - lock (Lock) - { - if (WrapperObjects.TryGetValue(@object, out var toggleRef)) - toggleRef.ToggleReference(isLastRef != 0); - else - Debug.WriteLine($"Handle {@object}: Could not toggle to {isLastRef} as there is no toggle reference."); - } - } -} diff --git a/src/Libs/GObject-2.0/Internal/ObjectWrapper.cs b/src/Libs/GObject-2.0/Internal/ObjectWrapper.cs deleted file mode 100644 index 48640d48b..000000000 --- a/src/Libs/GObject-2.0/Internal/ObjectWrapper.cs +++ /dev/null @@ -1,103 +0,0 @@ -using System; -using System.Diagnostics; -using System.Reflection; -using System.Runtime.CompilerServices; -using GLib; - -namespace GObject.Internal; - -public static class ObjectWrapper -{ - public static T? WrapNullableHandle(IntPtr handle, bool ownedRef) where T : class, IHandle - { - return handle == IntPtr.Zero - ? null - : WrapHandle(handle, ownedRef); - } - - public static T WrapHandle(IntPtr handle, bool ownedRef) where T : class, IHandle - { - Debug.Assert( - condition: typeof(T).IsClass && typeof(T).IsAssignableTo(typeof(GObject.Object)), - message: "Type 'T' must be a GObject-based class" - ); - - if (handle == IntPtr.Zero) - throw new NullReferenceException($"Failed to wrap handle as type <{typeof(T).FullName}>. Null handle passed to WrapHandle."); - - if (ObjectMapper.TryGetObject(handle, out T? obj)) - return obj; - - //In case of classes prefer the type reported by the gobject type system over - //the expected type as often an API returns a less derived class in it's public - //API then the actual one. - Type gtype = GetTypeFromInstance(handle); - - Debug.Assert( - condition: Functions.TypeName(gtype.Value).ConvertToString() == Functions.TypeNameFromInstance(new TypeInstanceUnownedHandle(handle)).ConvertToString(), - message: "GType name of instance and class do not match" - ); - - System.Type trueType = TypeDictionary.GetSystemType(gtype); - ConstructorInfo? ctor = GetObjectConstructor(trueType); - - if (ctor == null) - throw new Exception($"Type {typeof(T).FullName} does not define an IntPtr constructor. This could mean improperly defined bindings"); - - return (T) ctor.Invoke(new object[] { handle, ownedRef }); - } - - public static T? WrapNullableInterfaceHandle(IntPtr handle, bool ownedRef) where T : class, IHandle - { - return handle == IntPtr.Zero - ? null - : WrapInterfaceHandle(handle, ownedRef); - } - - public static T WrapInterfaceHandle(IntPtr handle, bool ownedRef) where T : class, IHandle - { - Debug.Assert( - condition: typeof(T).IsClass && typeof(T).IsAssignableTo(typeof(GObject.Object)), - message: "Type 'T' must be a GObject-based class" - ); - - if (handle == IntPtr.Zero) - throw new NullReferenceException($"Failed to wrap handle as type <{typeof(T).FullName}>. Null handle passed to WrapHandle."); - - if (ObjectMapper.TryGetObject(handle, out T? obj)) - return obj; - - //In case of interfaces prefer the given type over the type reported by the gobject - //type system as the reported type is probably not part of the public API. Otherwise the - //class itself would be returned and not an interface. - ConstructorInfo? ctor = GetObjectConstructor(typeof(T)); - - if (ctor == null) - throw new Exception($"Type {typeof(T).FullName} does not define an IntPtr constructor. This could mean improperly defined bindings"); - - return (T) ctor.Invoke(new object[] { handle, ownedRef }); - } - - private static unsafe Type GetTypeFromInstance(IntPtr handle) - { - var gclass = Unsafe.AsRef((void*) handle).GClass; - var gtype = Unsafe.AsRef((void*) gclass).GType; - - if (gtype == 0) - throw new Exception("Could not retrieve type from class struct - is the struct valid?"); - - return new Type(gtype); - } - - private static ConstructorInfo? GetObjectConstructor(System.Type type) - { - // Create using 'IntPtr' constructor - ConstructorInfo? ctor = type.GetConstructor( - System.Reflection.BindingFlags.NonPublic - | System.Reflection.BindingFlags.Public - | System.Reflection.BindingFlags.Instance, - null, new[] { typeof(IntPtr), typeof(bool) }, null - ); - return ctor; - } -} diff --git a/src/Libs/GObject-2.0/Internal/SubclassRegistrar.cs b/src/Libs/GObject-2.0/Internal/SubclassRegistrar.cs new file mode 100644 index 000000000..1dd86934e --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/SubclassRegistrar.cs @@ -0,0 +1,77 @@ +using System.Diagnostics; + +namespace GObject.Internal; + +/// +/// Registers a custom subclass with the GObject type system. +/// +public static class SubclassRegistrar +{ + public static Type Register() + where TSubclass : InstanceFactory + where TParent : GTypeProvider + { + var newType = RegisterNewGType(); + DynamicInstanceFactory.Register(newType, TSubclass.Create); + + return newType; + } + + private static Type RegisterNewGType() + where TParent : GTypeProvider + { + var parentType = TParent.GetGType(); + var parentTypeInfo = TypeQueryOwnedHandle.Create(); + Functions.TypeQuery(parentType, parentTypeInfo); + + if (parentTypeInfo.GetType() == 0) + throw new TypeRegistrationException("Could not query parent type"); + + Debug.WriteLine($"Registering new type {typeof(TSubclass).FullName} with parent {typeof(TParent).FullName}"); + + // Create TypeInfo + //TODO: Callbacks for "ClassInit" and "InstanceInit" are disabled because if multiple instances + //of the same type are created, the typeInfo object can get garbagec collected in the mean time + //and with it the instances of "DoClassInit" and "DoInstanceInit". If the callback occurs the + //runtime can't do the call anymore and crashes with: + //A callback was made on a garbage collected delegate of type 'GObject-2.0!GObject.Internal.InstanceInitFunc::Invoke' + //Fix this by caching the garbage collected instances somehow + var handle = TypeInfoOwnedHandle.Create(); + handle.SetClassSize((ushort) parentTypeInfo.GetClassSize()); + handle.SetInstanceSize((ushort) parentTypeInfo.GetInstanceSize()); + //handle.SetClassInit(); + //handle.SetInstanceInit(); + + var qualifiedName = QualifyName(typeof(TSubclass)); + var typeid = Functions.TypeRegisterStatic(parentType, + GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(qualifiedName), handle, 0); + + if (typeid == 0) + throw new TypeRegistrationException("Type Registration Failed!"); + + return new Type(typeid); + } + + private static string QualifyName(System.Type type) + => type.ToString() + .Replace(".", string.Empty) + .Replace("+", string.Empty) + .Replace("`", string.Empty) + .Replace("[", "_") + .Replace("]", string.Empty) + .Replace(" ", string.Empty) + .Replace(",", "_"); + + /* TODO: Enable if init functions are supported again + // Default Handler for class initialisation. + private static void DoClassInit(IntPtr gClass, IntPtr classData) + { + Console.WriteLine("Subclass type class initialised!"); + } + // Default Handler for instance initialisation. + private static void DoInstanceInit(IntPtr gClass, IntPtr classData) + { + Console.WriteLine("Subclass instance initialised!"); + } + */ +} diff --git a/src/Libs/GObject-2.0/Internal/ToggleRef.cs b/src/Libs/GObject-2.0/Internal/ToggleRef.cs new file mode 100644 index 000000000..dd445d7ef --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/ToggleRef.cs @@ -0,0 +1,46 @@ +using System; + +namespace GObject.Internal; + +internal class ToggleRef +{ + private object _reference; + + public GObject.Object? Object + { + get + { + if (_reference is WeakReference weakRef) + return (GObject.Object?) weakRef.Target; + + return (GObject.Object) _reference; + } + } + + /// + /// This object saves a strong reference to the given object which prevents it from beeing garbage + /// collected. This strong reference is hold as long as there are other references than the toggling ref + /// on the given object. + /// If the toggeling ref is the last ref on the given object the strong reference is changed into a + /// weak reference. This is signaled via a call to "ToggleReference". + /// + public ToggleRef(GObject.Object obj) + { + _reference = obj; + } + + internal void ToggleReference(bool isLastRef) + { + if (!isLastRef && _reference is WeakReference weakRef) + { + if (weakRef.Target is { } weakObj) + _reference = weakObj; + else + throw new Exception("Could not toggle reference to strong. It got garbage collected."); + } + else if (isLastRef && _reference is not WeakReference) + { + _reference = new WeakReference(_reference); + } + } +} diff --git a/src/Libs/GObject-2.0/Internal/ToggleRegistration.cs b/src/Libs/GObject-2.0/Internal/ToggleRegistration.cs new file mode 100644 index 000000000..f063a5d3a --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/ToggleRegistration.cs @@ -0,0 +1,28 @@ +using System; +using System.Runtime.InteropServices; + +namespace GObject.Internal; + +internal static unsafe class ToggleRegistration +{ + internal static void AddToggleRef(IntPtr handle, delegate* unmanaged toggleNotify) + { + AddToggleRef(handle, toggleNotify, IntPtr.Zero); + } + + internal static void RemoveToggleRef(IntPtr handle, delegate* unmanaged toggleNotify) + { + var sourceFunc = new GLib.Internal.SourceFuncAsyncHandler(() => + { + RemoveToggleRef(handle, toggleNotify, IntPtr.Zero); + return false; + }); + GLib.Internal.MainContext.Invoke(GLib.Internal.MainContextUnownedHandle.NullHandle, sourceFunc.NativeCallback, IntPtr.Zero); + } + + [DllImport(ImportResolver.Library, EntryPoint = "g_object_add_toggle_ref")] + private static extern void AddToggleRef(IntPtr @object, delegate* unmanaged toggleNotify, IntPtr data); + + [DllImport(ImportResolver.Library, EntryPoint = "g_object_remove_toggle_ref")] + private static extern void RemoveToggleRef(IntPtr @object, delegate* unmanaged toggleNotify, IntPtr data); +} diff --git a/src/Libs/GObject-2.0/Internal/TypeDictionary.cs b/src/Libs/GObject-2.0/Internal/TypeDictionary.cs deleted file mode 100644 index 82088b822..000000000 --- a/src/Libs/GObject-2.0/Internal/TypeDictionary.cs +++ /dev/null @@ -1,99 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Diagnostics; - -namespace GObject.Internal; - -/// -/// This exception is thrown when a is not found in -/// the type dictionary. -/// -public class TypeNotFoundException : Exception -{ - public TypeNotFoundException(System.Type managedType) - : base($"Type {managedType.FullName} not registered in type dictionary") { } -} - - - -/// -/// The global type dictionary which maps between the .NET Type System and -/// the GType dynamic type system. -/// -public static class TypeDictionary -{ - private static readonly Dictionary _systemTypeDict = new(); - private static readonly Dictionary _reverseTypeDict = new(); - - /// - /// Add a new mapping of (System.Type, GObject.Type) to the type dictionary. - /// - /// A managed type that has not already been registered - /// The gtype retrieved from the object's get type method or from registration. - public static void Add(System.Type systemType, Type type) - { - // Check we have not already registered - Debug.Assert( - condition: !_systemTypeDict.ContainsKey(systemType), - message: $"Type {nameof(systemType)} is already registered in the type dictionary." - ); - - _systemTypeDict[systemType] = type; - _reverseTypeDict[type] = systemType; - } - - /// - /// For a given managed GObject-based class, retrieve the corresponding gtype. - /// - /// The type of a class that is equal or derived from - /// The equivalent GType - /// The given type is not registered in the type dictionary. The caller should register it themselves. - internal static Type GetGType(System.Type type) - { - Debug.Assert( - condition: type.IsAssignableTo(typeof(GObject.Object)), - message: $"Parameter {type} is not a GObject or subclass of GObject" - ); - - if (!_systemTypeDict.TryGetValue(type, out Type gType)) - throw new TypeNotFoundException(type); - - return gType; - } - - /// - /// For a given gtype, retrieve the corresponding managed type. - /// - /// A type from the GType type system - /// The equivalent managed type - internal static System.Type GetSystemType(Type gtype) - { - if (_reverseTypeDict.TryGetValue(gtype, out System.Type? sysType)) - return sysType; - - // If gtype is not in the type dictionary, walk up the - // tree until we find a type that is. As all objects are - // descended from GObject, we will eventually find a parent - // type that is registered. - - while (!_reverseTypeDict.TryGetValue(gtype, out sysType)) - { - gtype = new Type(Functions.TypeParent(gtype.Value)); - if (gtype.Value == (nuint) BasicType.Invalid || - gtype.Value == (nuint) BasicType.None) - throw new Exception("Could not retrieve parent type - is the typeid valid?"); - } - - // Store for future lookups - _reverseTypeDict[gtype] = sysType; - - return sysType; - } - - // These may be unneeded - keep for now - internal static bool ContainsGType(Type gtype) - => _reverseTypeDict.ContainsKey(gtype); - - internal static bool ContainsSystemType(System.Type type) - => _systemTypeDict.ContainsKey(type); -} diff --git a/src/Libs/GObject-2.0/Internal/TypeRegistrar.cs b/src/Libs/GObject-2.0/Internal/TypeRegistrar.cs deleted file mode 100644 index 6c71d6e0e..000000000 --- a/src/Libs/GObject-2.0/Internal/TypeRegistrar.cs +++ /dev/null @@ -1,72 +0,0 @@ -using System; -using System.Diagnostics; -using System.Runtime.InteropServices; - -namespace GObject.Internal; - -/// -/// Thrown when type registration with GType fails -/// -internal class TypeRegistrationException : Exception -{ - public TypeRegistrationException(string message) : base(message) { } -} - -/// -/// A set of utility functions to register new types with the -/// GType dynamic type system. -/// -public static class TypeRegistrar -{ - /// - /// Registers with GType a new child class of 'parentType'. - /// - /// The name of the class - /// The parent class to derive from - /// The newly registered type - /// The type could not be registered - internal static Type RegisterGType(string qualifiedName, Type parentType) - { - var typeQuery = TypeQueryOwnedHandle.Create(); - Functions.TypeQuery(parentType, typeQuery); - - if (typeQuery.GetType() == 0) - throw new TypeRegistrationException("Could not query parent type"); - - Debug.WriteLine($"Registering new type {qualifiedName} with parent {parentType.ToString()}"); - - // Create TypeInfo - //TODO: Callbacks for "ClassInit" and "InstanceInit" are disabled because if multiple instances - //of the same type are created, the typeInfo object can get garbagec collected in the mean time - //and with it the instances of "DoClassInit" and "DoInstanceInit". If the callback occurs the - //runtime can't do the call anymore and crashes with: - //A callback was made on a garbage collected delegate of type 'GObject-2.0!GObject.Internal.InstanceInitFunc::Invoke' - //Fix this by caching the garbage collected instances somehow - var handle = TypeInfoOwnedHandle.Create(); - handle.SetClassSize((ushort) typeQuery.GetClassSize()); - handle.SetInstanceSize((ushort) typeQuery.GetInstanceSize()); - //handle.SetClassInit(); - //handle.SetInstanceInit(); - - var typeid = Functions.TypeRegisterStatic(parentType, GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(qualifiedName), handle, 0); - - if (typeid == 0) - throw new TypeRegistrationException("Type Registration Failed!"); - - return new Type(typeid); - } - - /* TODO: Enable if init functions are supported again - // Default Handler for class initialisation. - private static void DoClassInit(IntPtr gClass, IntPtr classData) - { - Console.WriteLine("Subclass type class initialised!"); - } - - // Default Handler for instance initialisation. - private static void DoInstanceInit(IntPtr gClass, IntPtr classData) - { - Console.WriteLine("Subclass instance initialised!"); - } - */ -} diff --git a/src/Libs/GObject-2.0/Internal/TypeRegistrationException.cs b/src/Libs/GObject-2.0/Internal/TypeRegistrationException.cs new file mode 100644 index 000000000..52180a160 --- /dev/null +++ b/src/Libs/GObject-2.0/Internal/TypeRegistrationException.cs @@ -0,0 +1,11 @@ +using System; + +namespace GObject.Internal; + +/// +/// Thrown when type registration with GType fails +/// +public class TypeRegistrationException : Exception +{ + internal TypeRegistrationException(string message) : base(message) { } +} diff --git a/src/Libs/GObject-2.0/Public/Closure.cs b/src/Libs/GObject-2.0/Public/Closure.cs index 70a64a03d..1082493df 100644 --- a/src/Libs/GObject-2.0/Public/Closure.cs +++ b/src/Libs/GObject-2.0/Public/Closure.cs @@ -1,6 +1,5 @@ using System; using System.Diagnostics; -using System.Linq; using System.Runtime.InteropServices; namespace GObject; diff --git a/src/Libs/GObject-2.0/Public/ConstructArgument.cs b/src/Libs/GObject-2.0/Public/ConstructArgument.cs index 416c2ace6..c8ba98416 100644 --- a/src/Libs/GObject-2.0/Public/ConstructArgument.cs +++ b/src/Libs/GObject-2.0/Public/ConstructArgument.cs @@ -5,23 +5,17 @@ namespace GObject; /// /// Define the value of GProperty which can be used at the construct time. /// -public sealed class ConstructArgument : IDisposable +public sealed class ConstructArgument(string name, Value value) : IDisposable { /// /// The GProperty name to set at the construct time. /// - public string Name { get; } + public string Name { get; } = name; /// /// The value of the property. /// - public Value Value { get; } - - public ConstructArgument(string name, Value value) - { - Name = name; - Value = value; - } + public Value Value { get; } = value; public void Dispose() { diff --git a/src/Libs/GObject-2.0/Public/IObject.cs b/src/Libs/GObject-2.0/Public/IObject.cs deleted file mode 100644 index 1569b7111..000000000 --- a/src/Libs/GObject-2.0/Public/IObject.cs +++ /dev/null @@ -1,9 +0,0 @@ -using System; -using System.ComponentModel; - -namespace GObject; - -public interface IObject -{ - -} diff --git a/src/Libs/GObject-2.0/Public/Object.Registration.cs b/src/Libs/GObject-2.0/Public/Object.Registration.cs deleted file mode 100644 index 68c98aa1f..000000000 --- a/src/Libs/GObject-2.0/Public/Object.Registration.cs +++ /dev/null @@ -1,58 +0,0 @@ -using GObject.Internal; - -namespace GObject; - -public partial class Object -{ - private Type GetGTypeOrRegister(System.Type type) - { - if (TypeDictionary.ContainsSystemType(type)) - return TypeDictionary.GetGType(type); - - // We are not in the type dictionary, which means we are - // an unregistered managed subclass type. There are two ways - // to register subclasses: Dynamically (reflection) and - // Statically (by source generation). - - // Static registration happens on application startup in the - // module initialiser if source generator support is present - // at build time (note: not implemented as of 29/04/21) - - // Therefore, we can assume static registration did not go ahead - // and we should resort to the fallback reflection-based registration - // implemented below. We should therefore register ourselves - // and every type we inherit from that has also not been registered. - - FallbackRegistrationStrategy.RegisterSubclassRecursive(type); - - return TypeDictionary.GetGType(type); - } - - private static class FallbackRegistrationStrategy - { - private static string QualifyName(System.Type type) - => type.ToString() - .Replace(".", string.Empty) - .Replace("+", string.Empty) - .Replace("`", string.Empty) - .Replace("[", "_") - .Replace("]", string.Empty) - .Replace(" ", string.Empty) - .Replace(",", "_"); - - public static void RegisterSubclassRecursive(System.Type type) - { - System.Type baseType = type.BaseType!; - if (!TypeDictionary.ContainsSystemType(baseType)) - RegisterSubclassRecursive(baseType); - - // Do actual registration - Type gtype = TypeRegistrar.RegisterGType( - qualifiedName: QualifyName(type), - parentType: TypeDictionary.GetGType(baseType) - ); - - TypeDictionary.Add(type, gtype); - } - } -} diff --git a/src/Libs/GObject-2.0/Public/Object.Signals.cs b/src/Libs/GObject-2.0/Public/Object.Signals.cs index fe7bb11c4..7b90b9298 100644 --- a/src/Libs/GObject-2.0/Public/Object.Signals.cs +++ b/src/Libs/GObject-2.0/Public/Object.Signals.cs @@ -11,7 +11,7 @@ public partial class Object internal void SignalConnectClosure(SignalDefinition signalDefinition, Delegate callback, Closure closure, bool after, string? detail) { var detailQuark = GLib.Functions.QuarkFromString(detail); - var handlerId = Internal.Functions.SignalConnectClosureById(Handle, signalDefinition.Id, detailQuark, closure.Handle, after); + var handlerId = Internal.Functions.SignalConnectClosureById(Handle.DangerousGetHandle(), signalDefinition.Id, detailQuark, closure.Handle, after); if (handlerId.Value == 0) throw new Exception($"Could not connect to event {signalDefinition.ManagedName}"); @@ -24,7 +24,7 @@ internal void Disconnect(SignalDefinition signalDefinition, Delegate callback) if (!_signalStore.TryGetValue((signalDefinition, callback), out var tuple)) return; - Internal.Functions.SignalHandlerDisconnect(Handle, tuple.Item1); + Internal.Functions.SignalHandlerDisconnect(Handle.DangerousGetHandle(), tuple.Item1); tuple.Item2.Dispose(); _signalStore.Remove((signalDefinition, callback)); } @@ -33,7 +33,7 @@ private void DisposeClosures() { foreach (var item in _signalStore.Values) { - Internal.Functions.SignalHandlerDisconnect(Handle, item.Item1); + Internal.Functions.SignalHandlerDisconnect(Handle.DangerousGetHandle(), item.Item1); item.Item2.Dispose(); } diff --git a/src/Libs/GObject-2.0/Public/Object.cs b/src/Libs/GObject-2.0/Public/Object.cs index 5b4fd48ba..114b3dc5a 100644 --- a/src/Libs/GObject-2.0/Public/Object.cs +++ b/src/Libs/GObject-2.0/Public/Object.cs @@ -1,77 +1,24 @@ using System; -using System.ComponentModel; using System.Diagnostics; -using System.Linq; -using System.Runtime.CompilerServices; -using GLib; using GObject.Internal; namespace GObject; -public partial class Object : IObject, IDisposable, IHandle +public partial class Object : IDisposable { - private readonly ObjectHandle _handle; + public ObjectHandle Handle { get; } - public IntPtr Handle => _handle.Handle; - - /// - /// Initializes a wrapper for an existing object - /// - /// - /// Defines if the handle is owned by us. If not owned by us it is refed to keep it around. - protected Object(IntPtr handle, bool ownedRef) - { - _handle = new ObjectHandle(handle, this, ownedRef); - Initialize(); - } - - /// - /// Constructs a new object - /// - /// True if the ownership of the resulting resulting handle will be transfered. Otherwise false. - /// - /// This constructor is protected to be sure that there is no caller (enduser) keeping a reference to - /// the construct parameters as the contained values are freed at the end of this constructor. - /// If certain constructors are needed they need to be implemented with concrete constructor arguments in - /// a higher layer. - protected Object(bool owned, ConstructArgument[] constructArguments) - { - Type gtype = GetGTypeOrRegister(GetType()); - - IntPtr handle = Internal.Object.NewWithProperties( - objectType: gtype, - nProperties: (uint) constructArguments.Length, - names: GetNames(constructArguments), - values: ValueArray2OwnedHandle.Create(constructArguments.Select(x => x.Value).ToArray()) - ); - - // We can't check if a reference is floating via "g_object_is_floating" here - // as the function could be "lying" depending on the intent of framework writers. - // E.g. A Gtk.Window created via "g_object_new_with_properties" returns an unowned - // reference which is not marked as floating as the gtk toolkit "owns" it. - // For this reason we just delegate the problem to the caller and require a - // definition wether the ownership of the new object will be transered to us or not. - _handle = new ObjectHandle(handle, this, owned); - - Initialize(); - } - - private string[] GetNames(ConstructArgument[] constructParameters) - => constructParameters.Select(x => x.Name).ToArray(); - - /// - /// Does common initialization tasks. - /// Wrapper and subclasses can override here to perform immediate initialization. - /// - protected virtual void Initialize() + protected Object(ObjectHandle handle) { - Debug.WriteLine($"Handle {_handle.Handle}: Initialising object of type {GetType()}."); + Handle = handle; + Handle.Cache(this); + Handle.AddMemoryPressure(); } - public virtual void Dispose() + public void Dispose() { - Debug.WriteLine($"Handle {_handle.Handle}: Disposing object of type {GetType()}."); + Debug.WriteLine($"Handle {Handle.DangerousGetHandle()}: Disposing object of type {GetType()}."); DisposeClosures(); - _handle.Dispose(); + Handle.Dispose(); } } diff --git a/src/Libs/GObject-2.0/Public/Property.cs b/src/Libs/GObject-2.0/Public/Property.cs index 18e9a98d1..c048d3b47 100644 --- a/src/Libs/GObject-2.0/Public/Property.cs +++ b/src/Libs/GObject-2.0/Public/Property.cs @@ -56,7 +56,7 @@ public void Set(K obj, T value) if (obj is not Object o) throw new ArgumentException($"Can't set property {ManagedName} for object of type {typeof(K).Name} as it is not derived from {nameof(Object)}."); - var type = GetPropertyType(o.Handle); + var type = GetPropertyType(o.Handle.DangerousGetHandle()); using var gvalue = new Value(type); gvalue.Set(value); diff --git a/src/Libs/GObject-2.0/Public/ReturningSignalTSender.cs b/src/Libs/GObject-2.0/Public/ReturningSignalTSender.cs index ddd5237d1..1ff7d0f5f 100644 --- a/src/Libs/GObject-2.0/Public/ReturningSignalTSender.cs +++ b/src/Libs/GObject-2.0/Public/ReturningSignalTSender.cs @@ -20,11 +20,7 @@ public ReturningSignal(string unmanagedName, string managedName) private uint GetId() { -#if NET7_0_OR_GREATER var gType = TSender.GetGType(); -#else - var gType = Internal.GTypeProviderHelper.GetGType(); -#endif return Internal.Functions.SignalLookup(GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(UnmanagedName), gType); } diff --git a/src/Libs/GObject-2.0/Public/ReturningSignalTSenderTSignalArgs.cs b/src/Libs/GObject-2.0/Public/ReturningSignalTSenderTSignalArgs.cs index 00711d487..3b7ed7061 100644 --- a/src/Libs/GObject-2.0/Public/ReturningSignalTSenderTSignalArgs.cs +++ b/src/Libs/GObject-2.0/Public/ReturningSignalTSenderTSignalArgs.cs @@ -21,11 +21,7 @@ public ReturningSignal(string unmanagedName, string managedName) private uint GetId() { -#if NET7_0_OR_GREATER var gType = TSender.GetGType(); -#else - var gType = Internal.GTypeProviderHelper.GetGType(); -#endif return Internal.Functions.SignalLookup(GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(UnmanagedName), gType); } diff --git a/src/Libs/GObject-2.0/Public/SignalTSender.cs b/src/Libs/GObject-2.0/Public/SignalTSender.cs index ac8979739..fa313c627 100644 --- a/src/Libs/GObject-2.0/Public/SignalTSender.cs +++ b/src/Libs/GObject-2.0/Public/SignalTSender.cs @@ -20,11 +20,7 @@ public Signal(string unmanagedName, string managedName) private uint GetId() { -#if NET7_0_OR_GREATER var gType = TSender.GetGType(); -#else - var gType = Internal.GTypeProviderHelper.GetGType(); -#endif return Internal.Functions.SignalLookup(GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(UnmanagedName), gType); } diff --git a/src/Libs/GObject-2.0/Public/SignalTSenderTSignalArgs.cs b/src/Libs/GObject-2.0/Public/SignalTSenderTSignalArgs.cs index eef4a30e5..72ae7f059 100644 --- a/src/Libs/GObject-2.0/Public/SignalTSenderTSignalArgs.cs +++ b/src/Libs/GObject-2.0/Public/SignalTSenderTSignalArgs.cs @@ -21,11 +21,7 @@ public Signal(string unmanagedName, string managedName) private uint GetId() { -#if NET7_0_OR_GREATER var gType = TSender.GetGType(); -#else - var gType = Internal.GTypeProviderHelper.GetGType(); -#endif return Internal.Functions.SignalLookup(GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(UnmanagedName), gType); } diff --git a/src/Libs/GObject-2.0/Public/Value.cs b/src/Libs/GObject-2.0/Public/Value.cs index 723a17345..c39e0ebb3 100644 --- a/src/Libs/GObject-2.0/Public/Value.cs +++ b/src/Libs/GObject-2.0/Public/Value.cs @@ -1,6 +1,5 @@ using System; using System.Runtime.CompilerServices; -using System.Runtime.InteropServices; using GLib.Internal; using GObject.Internal; diff --git a/src/Libs/Gdk-4.0/Public/Clipboard.cs b/src/Libs/Gdk-4.0/Public/Clipboard.cs index 20442b7f3..cb4c8f00f 100644 --- a/src/Libs/Gdk-4.0/Public/Clipboard.cs +++ b/src/Libs/Gdk-4.0/Public/Clipboard.cs @@ -18,7 +18,7 @@ public partial class Clipboard return; } - var readValue = Internal.Clipboard.ReadTextFinish(sourceObject.Handle, res.Handle, out var error); + var readValue = Internal.Clipboard.ReadTextFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); @@ -27,7 +27,7 @@ public partial class Clipboard }); Internal.Clipboard.ReadTextAsync( - clipboard: Handle, + clipboard: Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero diff --git a/src/Libs/GdkPixbuf-2.0/Internal/PixbufHandle.cs b/src/Libs/GdkPixbuf-2.0/Internal/PixbufHandle.cs new file mode 100644 index 000000000..474d566d0 --- /dev/null +++ b/src/Libs/GdkPixbuf-2.0/Internal/PixbufHandle.cs @@ -0,0 +1,19 @@ +using System; + +namespace GdkPixbuf.Internal; + +public partial class PixbufHandle +{ + private long _size; + + protected override void AddMemoryPressure() + { + _size = (long) Pixbuf.GetByteLength(handle); + GC.AddMemoryPressure(_size); + } + + protected override void RemoveMemoryPressure() + { + GC.RemoveMemoryPressure(_size); + } +} diff --git a/src/Libs/GdkPixbuf-2.0/Public/Pixbuf.cs b/src/Libs/GdkPixbuf-2.0/Public/Pixbuf.cs deleted file mode 100644 index 8b78c952f..000000000 --- a/src/Libs/GdkPixbuf-2.0/Public/Pixbuf.cs +++ /dev/null @@ -1,26 +0,0 @@ -using System; -using GLib; - -namespace GdkPixbuf; - -public partial class Pixbuf -{ - #region Fields - - private long _size; - - #endregion - - protected override void Initialize() - { - base.Initialize(); - _size = (long) Internal.Pixbuf.GetByteLength(Handle); - GC.AddMemoryPressure(_size); - } - - public override void Dispose() - { - base.Dispose(); - GC.RemoveMemoryPressure(_size); - } -} diff --git a/src/Libs/GdkPixbuf-2.0/Public/PixbufLoader.cs b/src/Libs/GdkPixbuf-2.0/Public/PixbufLoader.cs deleted file mode 100644 index c65291a99..000000000 --- a/src/Libs/GdkPixbuf-2.0/Public/PixbufLoader.cs +++ /dev/null @@ -1,33 +0,0 @@ -using System; -using GLib; - -namespace GdkPixbuf; - -public partial class PixbufLoader -{ - public static Pixbuf FromBytes(byte[] data) - { - IntPtr handle = Internal.PixbufLoader.New(); - - try - { - using var bytes = Bytes.New(data); - - Internal.PixbufLoader.WriteBytes(handle, bytes.Handle, out var error); - - if (!error.IsInvalid) - throw new GException(error); - - Internal.PixbufLoader.Close(handle, out error); - - if (!error.IsInvalid) - throw new GException(error); - - return new Pixbuf(Internal.PixbufLoader.GetPixbuf(handle), false); - } - finally - { - GObject.Internal.Object.Unref(handle); - } - } -} diff --git a/src/Libs/Gio-2.0/Public/DBusConnection.cs b/src/Libs/Gio-2.0/Public/DBusConnection.cs index 860366edf..cb021fcb9 100644 --- a/src/Libs/Gio-2.0/Public/DBusConnection.cs +++ b/src/Libs/Gio-2.0/Public/DBusConnection.cs @@ -13,7 +13,7 @@ public static DBusConnection Get(BusType busType) if (!error.IsInvalid) throw new GException(error); - return GObject.Internal.ObjectWrapper.WrapHandle(handle, true); + return (DBusConnection) GObject.Internal.InstanceWrapper.WrapHandle(handle, true); } public Task CallAsync(string busName, string objectPath, string interfaceName, string methodName, @@ -29,7 +29,7 @@ public Task CallAsync(string busName, string objectPath, string interfa return; } - var ret = Internal.DBusConnection.CallFinish(sourceObject.Handle, res.Handle, out var error); + var ret = Internal.DBusConnection.CallFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GException(error)); @@ -37,7 +37,7 @@ public Task CallAsync(string busName, string objectPath, string interfa tcs.SetResult(new Variant(ret)); }); - Internal.DBusConnection.Call(Handle, + Internal.DBusConnection.Call(Handle.DangerousGetHandle(), GLib.Internal.NullableUtf8StringOwnedHandle.Create(busName), GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(objectPath), GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(interfaceName), GLib.Internal.NonNullableUtf8StringOwnedHandle.Create(methodName), (GLib.Internal.VariantHandle?) parameters?.Handle ?? GLib.Internal.VariantUnownedHandle.NullHandle, GLib.Internal.VariantTypeUnownedHandle.NullHandle, DBusCallFlags.None, -1, IntPtr.Zero, callbackHandler.NativeCallback, IntPtr.Zero); diff --git a/src/Libs/Gtk-4.0/Public/AlertDialog.cs b/src/Libs/Gtk-4.0/Public/AlertDialog.cs index ee9a25bff..725a91546 100644 --- a/src/Libs/Gtk-4.0/Public/AlertDialog.cs +++ b/src/Libs/Gtk-4.0/Public/AlertDialog.cs @@ -18,7 +18,7 @@ public Task ChooseAsync(Window parent) return; } - var chooseValue = Internal.AlertDialog.ChooseFinish(sourceObject.Handle, res.Handle, out var error); + var chooseValue = Internal.AlertDialog.ChooseFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); @@ -27,8 +27,8 @@ public Task ChooseAsync(Window parent) }); Internal.AlertDialog.Choose( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero diff --git a/src/Libs/Gtk-4.0/Public/FileDialog.cs b/src/Libs/Gtk-4.0/Public/FileDialog.cs index 35606506b..dbbd73142 100644 --- a/src/Libs/Gtk-4.0/Public/FileDialog.cs +++ b/src/Libs/Gtk-4.0/Public/FileDialog.cs @@ -18,7 +18,7 @@ public partial class FileDialog return; } - var fileValue = Internal.FileDialog.OpenFinish(sourceObject.Handle, res.Handle, out var error); + var fileValue = Internal.FileDialog.OpenFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); @@ -29,8 +29,8 @@ public partial class FileDialog }); Internal.FileDialog.Open( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero @@ -52,17 +52,17 @@ public partial class FileDialog return; } - var listValue = Internal.FileDialog.OpenMultipleFinish(sourceObject.Handle, res.Handle, out var error); + var listValue = Internal.FileDialog.OpenMultipleFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); else - tcs.SetResult(GObject.Internal.ObjectWrapper.WrapNullableInterfaceHandle(listValue, true)); + tcs.SetResult(GObject.Internal.InstanceWrapper.WrapNullableHandle(listValue, true)); }); Internal.FileDialog.OpenMultiple( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero @@ -84,7 +84,7 @@ public partial class FileDialog return; } - var fileValue = Internal.FileDialog.SaveFinish(sourceObject.Handle, res.Handle, out var error); + var fileValue = Internal.FileDialog.SaveFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); @@ -95,8 +95,8 @@ public partial class FileDialog }); Internal.FileDialog.Save( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero @@ -119,7 +119,7 @@ public partial class FileDialog return; } - var fileValue = Internal.FileDialog.SelectFolderFinish(sourceObject.Handle, res.Handle, out var error); + var fileValue = Internal.FileDialog.SelectFolderFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); @@ -130,8 +130,8 @@ public partial class FileDialog }); Internal.FileDialog.SelectFolder( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero @@ -153,17 +153,17 @@ public partial class FileDialog return; } - var listValue = Internal.FileDialog.SelectMultipleFoldersFinish(sourceObject.Handle, res.Handle, out var error); + var listValue = Internal.FileDialog.SelectMultipleFoldersFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); else - tcs.SetResult(GObject.Internal.ObjectWrapper.WrapNullableInterfaceHandle(listValue, true)); + tcs.SetResult(GObject.Internal.InstanceWrapper.WrapNullableHandle(listValue, true)); }); Internal.FileDialog.SelectMultipleFolders( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero diff --git a/src/Libs/Gtk-4.0/Public/UriLauncher.cs b/src/Libs/Gtk-4.0/Public/UriLauncher.cs index 514d8e1f2..39d3bee6e 100644 --- a/src/Libs/Gtk-4.0/Public/UriLauncher.cs +++ b/src/Libs/Gtk-4.0/Public/UriLauncher.cs @@ -18,7 +18,7 @@ public Task LaunchAsync(Window parent) return; } - var launchValue = Internal.UriLauncher.LaunchFinish(sourceObject.Handle, res.Handle, out var error); + var launchValue = Internal.UriLauncher.LaunchFinish(sourceObject.Handle.DangerousGetHandle(), res.Handle.DangerousGetHandle(), out var error); if (!error.IsInvalid) tcs.SetException(new GLib.GException(error)); @@ -27,8 +27,8 @@ public Task LaunchAsync(Window parent) }); Internal.UriLauncher.Launch( - self: Handle, - parent: parent.Handle, + self: Handle.DangerousGetHandle(), + parent: parent.Handle.DangerousGetHandle(), cancellable: IntPtr.Zero, callback: callbackHandler.NativeCallback, userData: IntPtr.Zero diff --git a/src/Properties/GirCore.Libraries.props b/src/Properties/GirCore.Libraries.props index 3daf1db32..2cfa07837 100644 --- a/src/Properties/GirCore.Libraries.props +++ b/src/Properties/GirCore.Libraries.props @@ -1,7 +1,7 @@ - net6.0;net7.0;net8.0 + net8.0;net9.0 enable true true diff --git a/src/Properties/GirCore.Tooling.props b/src/Properties/GirCore.Tooling.props index 02d834b74..e72df6640 100644 --- a/src/Properties/GirCore.Tooling.props +++ b/src/Properties/GirCore.Tooling.props @@ -1,7 +1,7 @@ - net8.0 + net9.0 enable true diff --git a/src/Samples/Adw-1/Window/Window.csproj b/src/Samples/Adw-1/Window/Window.csproj index 20b422599..76c54e551 100644 --- a/src/Samples/Adw-1/Window/Window.csproj +++ b/src/Samples/Adw-1/Window/Window.csproj @@ -6,7 +6,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/GdkPixbuf-2.0/TestLoading/TestLoading.csproj b/src/Samples/GdkPixbuf-2.0/TestLoading/TestLoading.csproj index d1445e488..613f3366a 100644 --- a/src/Samples/GdkPixbuf-2.0/TestLoading/TestLoading.csproj +++ b/src/Samples/GdkPixbuf-2.0/TestLoading/TestLoading.csproj @@ -2,7 +2,7 @@ Exe - net6.0 + net8.0 diff --git a/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/Program.cs b/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/Program.cs index 4343a3c5c..eef637431 100644 --- a/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/Program.cs +++ b/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/Program.cs @@ -2,6 +2,7 @@ using System.IO; using System.Threading.Tasks; using GdkPixbuf; +using GLib; namespace TestMemoryLeaks; @@ -28,7 +29,7 @@ public static void Main(string[] args) { tasks[i] = Task.Run(() => { - PixbufLoader.FromBytes(imageBytes); + FromBytes(imageBytes); }); } Task.WaitAll(tasks); @@ -55,7 +56,7 @@ public static void Main(string[] args) Console.WriteLine("Bytes finalizer: Memory can go up. GC.Collect() is called in the end which must free everything up."); for (int i = 0; i < cycles; i++) { - var p = PixbufLoader.FromBytes(imageBytes); + var p = FromBytes(imageBytes); } Done(); @@ -72,12 +73,23 @@ public static void Main(string[] args) for (int i = 0; i < cycles; i++) { - var p = PixbufLoader.FromBytes(imageBytes); + var p = FromBytes(imageBytes); p.Dispose(); } Done(); } + private static Pixbuf FromBytes(byte[] data) + { + using var bytes = Bytes.New(data); + var loader = PixbufLoader.New(); + loader.WriteBytes(bytes); + loader.Close(); + + return loader.GetPixbuf() ?? throw new Exception("No Pixbuf created."); + + } + private static void Done() { Console.WriteLine("DONE."); diff --git a/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/TestMemoryLeaks.csproj b/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/TestMemoryLeaks.csproj index b6d5df728..4246e8406 100644 --- a/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/TestMemoryLeaks.csproj +++ b/src/Samples/GdkPixbuf-2.0/TestMemoryLeaks/TestMemoryLeaks.csproj @@ -2,7 +2,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/Gio-2.0/DBus/DBus.csproj b/src/Samples/Gio-2.0/DBus/DBus.csproj index 34cf605ee..9350572d7 100644 --- a/src/Samples/Gio-2.0/DBus/DBus.csproj +++ b/src/Samples/Gio-2.0/DBus/DBus.csproj @@ -2,7 +2,7 @@ Exe - net6.0 + net8.0 DBus diff --git a/src/Samples/Gst-1.0/VideoPlayback/VideoPlayback.csproj b/src/Samples/Gst-1.0/VideoPlayback/VideoPlayback.csproj index 76ff17fdc..d8e8ea578 100644 --- a/src/Samples/Gst-1.0/VideoPlayback/VideoPlayback.csproj +++ b/src/Samples/Gst-1.0/VideoPlayback/VideoPlayback.csproj @@ -2,7 +2,7 @@ Exe - net6.0 + net8.0 GStreamer diff --git a/src/Samples/Gtk-4.0/AboutDialog/AboutDialog.csproj b/src/Samples/Gtk-4.0/AboutDialog/AboutDialog.csproj index 9c0ac4494..e26f560ba 100644 --- a/src/Samples/Gtk-4.0/AboutDialog/AboutDialog.csproj +++ b/src/Samples/Gtk-4.0/AboutDialog/AboutDialog.csproj @@ -2,7 +2,7 @@ Exe - net6.0 + net8.0 diff --git a/src/Samples/Gtk-4.0/Builder/Builder.csproj b/src/Samples/Gtk-4.0/Builder/Builder.csproj index 1e8b5c8d3..ad10aeb98 100644 --- a/src/Samples/Gtk-4.0/Builder/Builder.csproj +++ b/src/Samples/Gtk-4.0/Builder/Builder.csproj @@ -6,7 +6,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/Gtk-4.0/DrawingArea/DrawingArea.csproj b/src/Samples/Gtk-4.0/DrawingArea/DrawingArea.csproj index e7da48baf..fddd153a0 100644 --- a/src/Samples/Gtk-4.0/DrawingArea/DrawingArea.csproj +++ b/src/Samples/Gtk-4.0/DrawingArea/DrawingArea.csproj @@ -1,7 +1,7 @@ - net6.0 + net8.0 Exe diff --git a/src/Samples/Gtk-4.0/FontDialog/FontDialog.csproj b/src/Samples/Gtk-4.0/FontDialog/FontDialog.csproj index f767743ba..70494dac4 100644 --- a/src/Samples/Gtk-4.0/FontDialog/FontDialog.csproj +++ b/src/Samples/Gtk-4.0/FontDialog/FontDialog.csproj @@ -6,7 +6,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/Gtk-4.0/GridView/GridView.csproj b/src/Samples/Gtk-4.0/GridView/GridView.csproj index 46e44ff30..c24d63834 100644 --- a/src/Samples/Gtk-4.0/GridView/GridView.csproj +++ b/src/Samples/Gtk-4.0/GridView/GridView.csproj @@ -10,7 +10,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/Gtk-4.0/ListView/ListView.csproj b/src/Samples/Gtk-4.0/ListView/ListView.csproj index 950a67f89..216634ccf 100644 --- a/src/Samples/Gtk-4.0/ListView/ListView.csproj +++ b/src/Samples/Gtk-4.0/ListView/ListView.csproj @@ -6,7 +6,7 @@ Exe - net6.0 + net8.0 enable enable diff --git a/src/Samples/Gtk-4.0/Window/Window.csproj b/src/Samples/Gtk-4.0/Window/Window.csproj index f767743ba..70494dac4 100644 --- a/src/Samples/Gtk-4.0/Window/Window.csproj +++ b/src/Samples/Gtk-4.0/Window/Window.csproj @@ -6,7 +6,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/GtkSource-5/GtkSourceView/GtkSourceView.csproj b/src/Samples/GtkSource-5/GtkSourceView/GtkSourceView.csproj index 738d81bbc..a8bab9547 100644 --- a/src/Samples/GtkSource-5/GtkSourceView/GtkSourceView.csproj +++ b/src/Samples/GtkSource-5/GtkSourceView/GtkSourceView.csproj @@ -7,7 +7,7 @@ Exe - net6.0 + net8.0 enable diff --git a/src/Samples/WebKit-6.0/JavascriptCall/JavascriptCall.csproj b/src/Samples/WebKit-6.0/JavascriptCall/JavascriptCall.csproj index 400d5f32c..aa0f214c2 100644 --- a/src/Samples/WebKit-6.0/JavascriptCall/JavascriptCall.csproj +++ b/src/Samples/WebKit-6.0/JavascriptCall/JavascriptCall.csproj @@ -7,7 +7,7 @@ Exe - net7.0 + net8.0 enable diff --git a/src/Samples/WebKit-6.0/JavascriptCallback/JavascriptCallback.csproj b/src/Samples/WebKit-6.0/JavascriptCallback/JavascriptCallback.csproj index 400d5f32c..aa0f214c2 100644 --- a/src/Samples/WebKit-6.0/JavascriptCallback/JavascriptCallback.csproj +++ b/src/Samples/WebKit-6.0/JavascriptCallback/JavascriptCallback.csproj @@ -7,7 +7,7 @@ Exe - net7.0 + net8.0 enable diff --git a/src/Tests/Generation/GirLoader.Tests/GirLoader.Tests.csproj b/src/Tests/Generation/GirLoader.Tests/GirLoader.Tests.csproj index 60a04d5cd..76386f65d 100644 --- a/src/Tests/Generation/GirLoader.Tests/GirLoader.Tests.csproj +++ b/src/Tests/Generation/GirLoader.Tests/GirLoader.Tests.csproj @@ -1,6 +1,6 @@ - net8.0 + net9.0 diff --git a/src/Tests/Libs/GObject-2.0.Tests/Classes/GenericTypeRegistrationTests.cs b/src/Tests/Libs/GObject-2.0.Tests/Classes/GenericTypeRegistrationTests.cs index fcf6cbd26..8d1477897 100644 --- a/src/Tests/Libs/GObject-2.0.Tests/Classes/GenericTypeRegistrationTests.cs +++ b/src/Tests/Libs/GObject-2.0.Tests/Classes/GenericTypeRegistrationTests.cs @@ -1,35 +1,60 @@ using System; +using GObject.Internal; using Microsoft.VisualStudio.TestTools.UnitTesting; namespace GObject.Tests.Classes; -internal class GenericClass : GObject.Object +internal class GenericClass : GObject.Object, GTypeProvider, InstanceFactory { - public GenericClass() : base(true, Array.Empty()) + private static readonly Type GType = SubclassRegistrar.Register(); + static Type GTypeProvider.GetGType() => GType; + static object InstanceFactory.Create(IntPtr handle, bool ownsHandle) { + return new GenericClass(handle, ownsHandle); } + + public GenericClass() : base(ObjectHandle.For(true, [])){ } + private GenericClass(IntPtr ptr, bool ownsHandle) : base(new ObjectHandle(ptr, ownsHandle)) { } } -internal class GenericClass : GObject.Object +internal class GenericClass : GObject.Object, GTypeProvider, InstanceFactory { - public GenericClass() : base(true, Array.Empty()) + private static readonly Type GType = SubclassRegistrar.Register, Object>(); + static Type GTypeProvider.GetGType() => GType; + static object InstanceFactory.Create(IntPtr handle, bool ownsHandle) { + return new GenericClass(handle, ownsHandle); } + + public GenericClass() : base(ObjectHandle.For>(true, [])){ } + private GenericClass(IntPtr ptr, bool ownsHandle) : base(new ObjectHandle(ptr, ownsHandle)) { } } -internal class GenericClass : GObject.Object +internal class GenericClass : GObject.Object, GTypeProvider, InstanceFactory { - public GenericClass() : base(true, Array.Empty()) + private static readonly Type GType = SubclassRegistrar.Register, Object>(); + static Type GTypeProvider.GetGType() => GType; + static object InstanceFactory.Create(IntPtr handle, bool ownsHandle) { + return new GenericClass(handle, ownsHandle); } + + public GenericClass() : base(ObjectHandle.For>(true, [])){ } + private GenericClass(IntPtr ptr, bool ownsHandle) : base(new ObjectHandle(ptr, ownsHandle)) { } } [TestClass, TestCategory("UnitTest")] public class GenericTypeRegistrationTests { - private class NestedGenericClass : GObject.Object + private class NestedGenericClass : GObject.Object, GTypeProvider, InstanceFactory { - public NestedGenericClass() : base(true, Array.Empty()) + private static readonly Type GType = SubclassRegistrar.Register, Object>(); + static Type GTypeProvider.GetGType() => GType; + static object InstanceFactory.Create(IntPtr handle, bool ownsHandle) { + return new NestedGenericClass(handle, ownsHandle); } + + public NestedGenericClass() : base(ObjectHandle.For>(true, [])){ } + private NestedGenericClass(IntPtr ptr, bool ownsHandle) : base(new ObjectHandle(ptr, ownsHandle)) { } } [TestMethod] diff --git a/src/Tests/Libs/GObject-2.0.Tests/Records/TypeTest.cs b/src/Tests/Libs/GObject-2.0.Tests/Records/TypeTest.cs index 2ca4b1a40..9e2cc58cb 100644 --- a/src/Tests/Libs/GObject-2.0.Tests/Records/TypeTest.cs +++ b/src/Tests/Libs/GObject-2.0.Tests/Records/TypeTest.cs @@ -22,9 +22,9 @@ public void FundamentalTypesCanBeDetected() Internal.Functions.IsFundamental(255 << 2).Should().BeTrue(); //Fundamental types - Internal.Functions.IsFundamental(Object.GetGType().Value).Should().BeTrue(); + Internal.Functions.IsFundamental(Internal.Object.GetGType()).Should().BeTrue(); //Non fundamental types - Internal.Functions.IsFundamental(Binding.GetGType().Value).Should().BeFalse(); + Internal.Functions.IsFundamental(Internal.Binding.GetGType()).Should().BeFalse(); } } diff --git a/src/Tests/Libs/GirTest-0.1.Tests/CallbackTest.cs b/src/Tests/Libs/GirTest-0.1.Tests/CallbackTest.cs index ed4e9d369..8571dda77 100644 --- a/src/Tests/Libs/GirTest-0.1.Tests/CallbackTest.cs +++ b/src/Tests/Libs/GirTest-0.1.Tests/CallbackTest.cs @@ -140,7 +140,7 @@ GObject.Type Callback() [TestMethod] public void SupportsCallbackWithObjectReturn() { - var testObject = new TestClass(); + var testObject = ExecutorImpl.New(); GObject.Object Callback() { return testObject; @@ -150,11 +150,6 @@ GObject.Object Callback() obj.Should().Be(testObject); } - private class TestClass : GObject.Object - { - public TestClass() : base(true, System.Array.Empty()) { } - } - [TestMethod] public void SupportsCallbackWithInterfaceReturn() { diff --git a/src/Tests/Libs/GirTest-0.1.Tests/ClassTest.cs b/src/Tests/Libs/GirTest-0.1.Tests/ClassTest.cs index 62244fc67..cec4b6ed3 100644 --- a/src/Tests/Libs/GirTest-0.1.Tests/ClassTest.cs +++ b/src/Tests/Libs/GirTest-0.1.Tests/ClassTest.cs @@ -50,15 +50,15 @@ public void CanTransferOwnershipOfInterfaces() var obj = ClassTester.New(); var executor = GirTest.ExecutorImpl.New(); - var instanceData = Marshal.PtrToStructure(executor.Handle); + var instanceData = Marshal.PtrToStructure(executor.Handle.DangerousGetHandle()); instanceData.RefCount.Should().Be(1); obj.TakeExecutor(executor); - instanceData = Marshal.PtrToStructure(executor.Handle); + instanceData = Marshal.PtrToStructure(executor.Handle.DangerousGetHandle()); instanceData.RefCount.Should().Be(2); obj.FreeExecutor(); - instanceData = Marshal.PtrToStructure(executor.Handle); + instanceData = Marshal.PtrToStructure(executor.Handle.DangerousGetHandle()); instanceData.RefCount.Should().Be(1); } @@ -66,9 +66,9 @@ public void CanTransferOwnershipOfInterfaces() public void TestManualGObjectDisposal() { var obj = ClassTester.New(); - GObject.Internal.ObjectMapper.ObjectCount.Should().Be(1); + GObject.Internal.InstanceCache.ObjectCount.Should().Be(1); obj.Dispose(); - GObject.Internal.ObjectMapper.ObjectCount.Should().Be(0); + GObject.Internal.InstanceCache.ObjectCount.Should().Be(0); } [TestMethod] @@ -80,18 +80,18 @@ public void TestAutomaticGObjectDisposal() CollectAfter(() => { var obj = ClassTester.New(); - GObject.Internal.ObjectMapper.ObjectCount.Should().Be(1); + GObject.Internal.InstanceCache.ObjectCount.Should().Be(1); weakReference.Target = obj; }); - GObject.Internal.ObjectMapper.ObjectCount.Should().Be(0); + GObject.Internal.InstanceCache.ObjectCount.Should().Be(0); weakReference.IsAlive.Should().BeFalse(); CollectAfter(() => { var obj = ClassTester.New(); - GObject.Internal.ObjectMapper.ObjectCount.Should().Be(1); + GObject.Internal.InstanceCache.ObjectCount.Should().Be(1); strongReference = obj; weakReference.Target = obj; diff --git a/src/Tests/Libs/GirTest-0.1.Tests/OpaqueTypedRecordTest.cs b/src/Tests/Libs/GirTest-0.1.Tests/OpaqueTypedRecordTest.cs index 070a57275..c6c1070ca 100644 --- a/src/Tests/Libs/GirTest-0.1.Tests/OpaqueTypedRecordTest.cs +++ b/src/Tests/Libs/GirTest-0.1.Tests/OpaqueTypedRecordTest.cs @@ -303,7 +303,7 @@ public void SupportsWrapHandle() var wrapped = (OpaqueTypedRecordTester) GObject.Internal.BoxedWrapper.WrapHandle( handle: recordTester.Handle.DangerousGetHandle(), ownsHandle: false, - gtype: OpaqueTypedRecordTester.GetGType() + gtype: Internal.OpaqueTypedRecordTester.GetGType() ); wrapped.Handle.DangerousGetHandle().Should().Be(recordTester.Handle.DangerousGetHandle()); diff --git a/src/Tests/Libs/GirTest-0.1.Tests/TypedRecordTest.cs b/src/Tests/Libs/GirTest-0.1.Tests/TypedRecordTest.cs index 72462292e..8c1c0b741 100644 --- a/src/Tests/Libs/GirTest-0.1.Tests/TypedRecordTest.cs +++ b/src/Tests/Libs/GirTest-0.1.Tests/TypedRecordTest.cs @@ -303,7 +303,7 @@ public void SupportsWrapHandle() var wrapped = (TypedRecordTester) GObject.Internal.BoxedWrapper.WrapHandle( handle: recordTester.Handle.DangerousGetHandle(), ownsHandle: false, - gtype: TypedRecordTester.GetGType() + gtype: Internal.TypedRecordTester.GetGType() ); wrapped.Handle.DangerousGetHandle().Should().Be(recordTester.Handle.DangerousGetHandle());