diff --git a/src/AST/ClassExtensions.cs b/src/AST/ClassExtensions.cs index df33d0eaf0..d8faf8066e 100644 --- a/src/AST/ClassExtensions.cs +++ b/src/AST/ClassExtensions.cs @@ -5,7 +5,7 @@ namespace CppSharp.AST { - public static class ClassExtensions + public static class ClassExtensions { public static IEnumerable GetFunctionOverloads(this Class @class, Function function) @@ -211,6 +211,11 @@ public static bool HasDependentValueFieldInLayout(this Class @class) b => b.Class).Any(HasDependentValueFieldInLayout); } + public static int GetSize(this ClassLayout layout) => + /// There's at least one ABI (System V) that gives to empty structs + /// size 1 in C++ and size 0 in C. The former causes crashes in older versions of Mono. + layout.Size == 1 && layout.DataSize == 0 ? 0 : layout.Size; + private static bool IsValueDependent(Type type) { var desugared = type.Desugar(); diff --git a/src/Generator/Generators/CSharp/CSharpSources.cs b/src/Generator/Generators/CSharp/CSharpSources.cs index df09a91a8c..5287b28a26 100644 --- a/src/Generator/Generators/CSharp/CSharpSources.cs +++ b/src/Generator/Generators/CSharp/CSharpSources.cs @@ -524,7 +524,7 @@ public void GenerateClassInternals(Class @class) { PushBlock(BlockKind.InternalsClass); if (!Options.GenerateSequentialLayout || @class.IsUnion) - WriteLine($"[StructLayout(LayoutKind.Explicit, Size = {@class.Layout.Size})]"); + WriteLine($"[StructLayout(LayoutKind.Explicit, Size = {@class.Layout.GetSize()})]"); else if (@class.MaxFieldAlignment > 0) WriteLine($"[StructLayout(LayoutKind.Sequential, Pack = {@class.MaxFieldAlignment})]"); @@ -2885,7 +2885,7 @@ public void GenerateFunctionCall(string functionName, List parameters ((Method) method.OriginalFunction).IsConstructor) { WriteLine($@"Marshal.AllocHGlobal({ - ((Class) method.OriginalNamespace).Layout.Size});"); + ((Class) method.OriginalNamespace).Layout.GetSize()});"); names.Insert(0, Helpers.ReturnIdentifier); } WriteLine("{0}({1});", functionName, string.Join(", ", names)); diff --git a/src/Generator/Generators/CSharp/CSharpTypePrinter.cs b/src/Generator/Generators/CSharp/CSharpTypePrinter.cs index 7ca2642e08..daf76cdcb7 100644 --- a/src/Generator/Generators/CSharp/CSharpTypePrinter.cs +++ b/src/Generator/Generators/CSharp/CSharpTypePrinter.cs @@ -100,7 +100,7 @@ public override TypePrinterResult VisitArrayType(ArrayType array, return new TypePrinterResult { Type = "fixed byte", - NameSuffix = $"[{array.Size * @class.Layout.Size}]" + NameSuffix = $"[{array.Size * @class.Layout.GetSize()}]" }; } diff --git a/src/Generator/Passes/CheckAbiParameters.cs b/src/Generator/Passes/CheckAbiParameters.cs index 396a4b0120..331ca68146 100644 --- a/src/Generator/Passes/CheckAbiParameters.cs +++ b/src/Generator/Passes/CheckAbiParameters.cs @@ -25,19 +25,6 @@ namespace CppSharp.Passes /// public class CheckAbiParameters : TranslationUnitPass { - public override bool VisitClassDecl(Class @class) - { - if (!base.VisitClassDecl(@class)) - return false; - - if (@class.IsDependent || @class.Layout.Fields.Count > 0 || @class.Fields.Count > 0) - return false; - - @class.Layout.Size = @class.Layout.DataSize = 0; - - return true; - } - public override bool VisitFunctionDecl(Function function) { if (!VisitDeclaration(function))