-
Notifications
You must be signed in to change notification settings - Fork 4.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Reflection API Incorrectly Boxes (U)IntPtr Constants as (U)Int32 #104270
Comments
Tagging subscribers to this area: @dotnet/area-system-reflection |
Const |
The runtime is capable of loading an assembly containing a field defined as |
I believe that the field initialization clause should merely serve as an expression of a literal value, rather than determining the field's type. To illustrate this with a thought experiment: even if the field initialization clause provided a string representing a C++ constant expression, and a tool calculated the field's value based on this C++ source code, I wouldn't find it strange. |
Note: according to ECMA-335 §II.16.2 Field init metadata:
The original purpose was to allow conversion from underlying numeric types to enum types. However, it doesn't specify the desired behavior of other casts. It's not clear what's the expected behavior here. |
Also note that currently there's no compiler emitting this, and probably not understanding it. |
Verified repo; for the first field, the IL is:
so for constants yes it does change to a int\uint. Also, for .NET Framework, this code fails compile with CS0283 The type 'IntPtr' cannot be declared const where that link does not call out the native pointer types. This is not a reflection issue; changing area to runtime, although this is likely an ECMA issue and where we supported this to compile, unlike .NET Framework. |
Tagging subscribers to this area: @dotnet/area-system-runtime |
@steveharter Thanks for the investigation. I've also confirmed your findings using a variety of tools. The Constants table ( I personally don't like inconsistencies in this domain, makes for too much confusion. The fact that .NET Framework doesn't support this scenario lowers the concern about making it do what the OP expects. The code to update is below. We could make a check to see if the field is runtime/src/coreclr/System.Private.CoreLib/src/System/Reflection/MdConstant.cs Lines 108 to 126 in 8432f0d
|
I don't think we should change anything in the runtime here. I took a slightly deeper look, and if you change the repro code to use |
Handling of constant and default values in reflection have seen number of fixes over the years in .NET Core. Some of the earlier fixes include: We have systemic problem where Roslyn encodes the constants and default values as it sees a fit, but reflection is not aware of all possible encodings produced by Roslyn and it may produce unexpected results for some of them. We took number of fixes in this area and the corner case behavior deviated from .NET Framework, so I do not see a problem with fixing more corner cases. However, until somebody sits down and ensures that the reflection behavior is sensible for everything that's possible to write in C#, we are unlikely to be in 100% happy place. Note that there is a fresh from the press proposal to allow encoding even more types as constants in C#: dotnet/csharplang#8257 . If this proposal is accepted, we will have a work to do here anyway.
Nit: I see this as Roslyn and reflection interaction issue. Nothing in the runtime outside of reflection cares about encoding of C# constants. |
Tagging subscribers to this area: @dotnet/area-system-reflection |
Do note the signed-ness mismatch in original post: the literal field is declared as native uint, but the initializer is declared as int32. Recent roslyn and C# versions allow n(u)int constant being initialized with int32 of the same size:
In this case, both |
However, initializing nuint with negative int32 is a different case. I'm also not sure whether roslyn can understand |
@huoyaoyuan I don't see this locally (compiler version: '4.11.0-2.24304.1`) - can you verify please?
|
The original post is about the case that never emitted by current compiler, but allowed in runtime. Personally I don't think the runtime should allow this case. However, getting the value of |
The original post is about what is emitted by the compiler, unless I'm missing something. For example, this: |
Let me clarify this, with information from offline discussion. Currently when the runtime encounters non-enum type in
This is achievable in C# with
This is not achievable in C#. Currently runtime allows loading this, and getting the value of |
Description
The reflection API is expected to box constant literal fields of type (U)IntPtr as their correct types. However, when using reflection to retrieve the value of these constants, they are being boxed as (U)Int32 instead of (U)IntPtr.
Reproduction Steps
Expected behavior
Actual behavior
Regression?
No response
Known Workarounds
No response
Configuration
No response
Other information
This issue may present a theoretical security risk, although the practical threat is likely negligible.
The text was updated successfully, but these errors were encountered: