From 77780b0e86c4ce36c06f1ffb54223db577d22a9e Mon Sep 17 00:00:00 2001 From: Petr Onderka Date: Thu, 11 Nov 2021 10:25:32 +0100 Subject: [PATCH 1/2] Handle empty string parameter value in MetadataLoadContext --- .../Ecma/EcmaDefaultValueProcessing.cs | 6 ++++- .../src/SampleMetadata/SampleMetadata.cs | 2 ++ .../src/Tests/Parameter/ParameterTests.cs | 23 +++++++++++++++++++ 3 files changed, 30 insertions(+), 1 deletion(-) diff --git a/src/libraries/System.Reflection.MetadataLoadContext/src/System/Reflection/TypeLoading/General/Ecma/EcmaDefaultValueProcessing.cs b/src/libraries/System.Reflection.MetadataLoadContext/src/System/Reflection/TypeLoading/General/Ecma/EcmaDefaultValueProcessing.cs index 75e0e0e26da1ea..eb0afc9cd1f796 100644 --- a/src/libraries/System.Reflection.MetadataLoadContext/src/System/Reflection/TypeLoading/General/Ecma/EcmaDefaultValueProcessing.cs +++ b/src/libraries/System.Reflection.MetadataLoadContext/src/System/Reflection/TypeLoading/General/Ecma/EcmaDefaultValueProcessing.cs @@ -15,7 +15,11 @@ internal static class EcmaDefaultValueProcessing throw new BadImageFormatException(); Constant constantValue = metadataReader.GetConstant(constantHandle); - if (constantValue.Value.IsNil) + // Partition II section 24.2.4: + // The first entry in both these heaps is the empty 'blob' that consists of the single byte 0x00. + // + // This means zero value is valid for string and is used to represent the empty string. + if (constantValue.Value.IsNil && constantValue.TypeCode != ConstantTypeCode.String) throw new BadImageFormatException(); BlobReader reader = metadataReader.GetBlobReader(constantValue.Value); diff --git a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs index 9835e14e9d4fe6..274a1e85ea89c4 100644 --- a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs +++ b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs @@ -198,6 +198,8 @@ public void Foo3(int i = 42) { } public void Foo4(short s = -34) { } public void Foo5(decimal d = 1234m) { } public void Foo6([DateTimeConstant(ticks: 8736726782)] DateTime dt) { } + public void Foo7(string s1 = "foo", string s2 = "") { } + public void Foo8(Action a = null) { } } public class ParametersWithPseudoCustomtAttributes diff --git a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs index 52fda61fae11bf..598a151f573e5b 100644 --- a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs +++ b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs @@ -66,6 +66,29 @@ public static void TestRawDefaultValue6() Assert.Equal(8736726782, ((DateTime)dv).Ticks); } + [Fact] + public static void TestRawDefaultValue7() + { + var parameters = typeof(ParametersWithDefaultValues).Project().GetTypeInfo().GetDeclaredMethod("Foo7").GetParameters(); + ParameterInfo p1 = parameters[0]; + Assert.True(p1.HasDefaultValue); + object dv1 = p1.RawDefaultValue; + Assert.Equal("foo", dv1); + ParameterInfo p2 = parameters[1]; + Assert.True(p2.HasDefaultValue); + object dv2 = p2.RawDefaultValue; + Assert.Equal("", dv2); + } + + [Fact] + public static void TestRawDefaultValue8() + { + ParameterInfo p = typeof(ParametersWithDefaultValues).Project().GetTypeInfo().GetDeclaredMethod("Foo8").GetParameters()[0]; + Assert.True(p.HasDefaultValue); + object dv = p.RawDefaultValue; + Assert.Null(dv); + } + [Fact] [ActiveIssue("https://github.com/mono/mono/issues/15340", TestRuntimes.Mono)] public static void TestPseudoCustomAttributes() From c039ed5e1d7ca17c24df29f82ac3704cb5647fed Mon Sep 17 00:00:00 2001 From: Petr Onderka Date: Thu, 18 Nov 2021 12:15:35 +0100 Subject: [PATCH 2/2] Added test for null string default parameter --- .../tests/src/SampleMetadata/SampleMetadata.cs | 2 +- .../tests/src/Tests/Parameter/ParameterTests.cs | 3 +++ 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs index 274a1e85ea89c4..15d46738c3516e 100644 --- a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs +++ b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/SampleMetadata/SampleMetadata.cs @@ -198,7 +198,7 @@ public void Foo3(int i = 42) { } public void Foo4(short s = -34) { } public void Foo5(decimal d = 1234m) { } public void Foo6([DateTimeConstant(ticks: 8736726782)] DateTime dt) { } - public void Foo7(string s1 = "foo", string s2 = "") { } + public void Foo7(string s1 = "foo", string s2 = "", string s3 = null) { } public void Foo8(Action a = null) { } } diff --git a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs index 598a151f573e5b..df4f18f2cfd3ba 100644 --- a/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs +++ b/src/libraries/System.Reflection.MetadataLoadContext/tests/src/Tests/Parameter/ParameterTests.cs @@ -78,6 +78,9 @@ public static void TestRawDefaultValue7() Assert.True(p2.HasDefaultValue); object dv2 = p2.RawDefaultValue; Assert.Equal("", dv2); + ParameterInfo p3 = parameters[2]; + Assert.True(p3.HasDefaultValue); + Assert.Null(p3.RawDefaultValue); } [Fact]