-
Notifications
You must be signed in to change notification settings - Fork 1k
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
Compiler intrinsics #1685
Compiler intrinsics #1685
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,224 @@ | ||
# Compiler Intrinsics | ||
|
||
## Summary | ||
|
||
This proposal provides language constructs that expose low level IL opcodes that cannot currently | ||
be accessed efficiently, or at all: `ldftn`, `ldvirtftn`, `ldtoken` and `calli`. These low level op | ||
codes can be important in high performance code and developers need an efficient way to access | ||
them. | ||
|
||
## Motivation | ||
|
||
The motivations and background for this feature are described in the following issue (as is a | ||
potential implementation of the feature): | ||
|
||
https://github.com/dotnet/csharplang/issues/191 | ||
|
||
This alternate design proposal comes after reviewing a prototype implementation of the original | ||
proposal by @msjabby as well as the use throughout a significant code base. This design was done | ||
with significant input from @mjsabby, @tmat and @jkotas. | ||
|
||
## Detailed Design | ||
|
||
### Allow address of to target methods | ||
|
||
Method groups will now be allowed as arguments to an address-of expression. The type of such an | ||
expression will be `void*`. | ||
|
||
``` csharp | ||
class Util { | ||
public static void Log() { } | ||
} | ||
|
||
// ldftn Util.Log | ||
void* ptr = &Util.Log; | ||
``` | ||
|
||
Given there is no delegate conversion here the only mechanism for filtering members in the method | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do we think that this is going to be a significant bar for existing libraries to use this feature, if they've already got a number of method overloads that they want to distinguish between? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ah, I see you have a section on that below, hadn't gotten there yet. |
||
group is by static / instance access. If that cannot distinguish the members then a compile time | ||
error will occur. | ||
|
||
``` csharp | ||
class Util { | ||
public void Log() { } | ||
public void Log(string p1) { } | ||
public static void Log(int i) { }; | ||
} | ||
|
||
unsafe { | ||
// Error: Method group Log has more than one applicable candidate. | ||
void* ptr1 = &Log; | ||
|
||
// Okay: only one static member to consider here. | ||
void* ptr2 = &Util.Log; | ||
} | ||
``` | ||
|
||
The addressof expression in this context will be implemented in the following manner: | ||
|
||
- ldftn: when the method is non-virtual. | ||
- ldvirtftn: when the method is virtual. | ||
|
||
Restrictions of this feature: | ||
|
||
- Instance methods can only be specified when using an invocation expression on a value | ||
- Local functions cannot be used in `&`. The implementation details of these methods are | ||
deliberately not specified by the language. This includes whether they are static vs. instance or | ||
exactly what signature they are emitted with. | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We should productize the NativeCallableAttribute that will work together with this. This proposal together with NativeCallableAttribute will deliver complete solution for low-level managed/unmanaged interop for function pointers:
.NET Core supports the NativeCallableAttribute attribute internally, but the feature was not exposed in public contracts and documented. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What does this attribute do today in the internal support? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It adds unmanaged->managed->unmanaged transition to the method prolog/epilog. Marking the method with NativeCallableAttribute and taking its address is more efficient low-level equivalent of Marshal.GetFunctionPointerForDelegate. It avoids the delegate wrapping and unwrapping overhead. More details are in description for dotnet/coreclr#1566 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
They are not exactly equivalent. The first one doesn't support marshalling(you need to use blittable types), the second one support marshalling. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Chatted with @AaronRobinsonMSFT and I better understand how this could interact with the feature. Agree there are benefits. Seems easy to add if the runtime decides to productize this. Will add this to the "future consideration" section. |
||
### handleof | ||
|
||
The `handleof` contextual keyword will translate a field, member or type into their equivalent | ||
`RuntimeHandle` type using the `ldtoken` instruction. The exact type of the expression will | ||
depend on the kind of the name in `handleof`: | ||
|
||
- field: `RuntimeFieldHandle` | ||
- type: `RuntimeTypeHandle` | ||
- method: `RuntimeMethodHandle` | ||
|
||
The arguments to `handleof` are identical to `nameof`. It must be a simple name, qualified name, | ||
member access, base access with a specified member, or this access with a specified member. The | ||
argument expression identifies a code definition, but it is never evaluated. | ||
|
||
The `handleof` expression is evaluated at runtime and has a return type of `RuntimeHandle`. This | ||
can be executed in safe code as well as unsafe. | ||
|
||
``` | ||
RuntimeHandle stringHandle = handleof(string); | ||
``` | ||
|
||
Restrictions of this feature: | ||
|
||
- Properties cannot be used in a `handleof` expression. | ||
- The `handleof` expression cannot be used when there is an existing `handleof` name in scope. For | ||
example a type, namespace, etc ... | ||
|
||
### calli | ||
|
||
The compiler will add support for a new type of `extern` function that efficiently translates into | ||
a `.calli` instruction. The extern attribute will be marked with an attribute of the following | ||
shape: | ||
|
||
``` csharp | ||
[AttributeUsage(AttributeTargets.Method)] | ||
public sealed class CallIndirectAttribute : Attribute | ||
{ | ||
public CallingConvention CallingConvention { get; } | ||
public CallIndirectAttribute(CallingConvention callingConvention) | ||
{ | ||
CallingConvention = callingConvention; | ||
} | ||
} | ||
``` | ||
|
||
This allows developers to define methods in the following form: | ||
|
||
``` csharp | ||
[CallIndirect(CallingConvention.Cdecl)] | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This enum does not have a value for regular managed calling convention. (@mjsabby use case and some other use cases needs the managed calling convention).
An alternative is to add a value to for the managed calling convention to CallingConvention enum. I have rejected it because of the new value would be invalid in all existing APIs that use CallingConvention enum today. All these APIs are for unmanaged calls. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Add a new value in CallingConvention seems better than use two attributes. simpler and easy to understand. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @luqunl I agree in principle that two attributes are more annoying - however, I agree with @jkotas that adding it to the enum is less than ideal. Another approach that would work from the API perspective would be to make the default constructor for The nice aspect about this is that the sentinel value can be anything and doesn't require breaking the API boundary. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Let me add an open issues to track this down. The language design can proceed while we figure this out. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We should dis-allow DllImport method with CallIndirect attribute. |
||
static extern int MapValue(string s, void *ptr); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is the C# compiler going to emit this method into the metadata, or will it disappear at source compilation time? The runtime typically won't load a type that has an What happens if I try to reference this method in a non-call context (try to construct a delegate to this or get a RuntimeHandle of it)? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The current implementation generates method bodies for this reason, and at least for the managed calling convention the calli gets inlined. Or would you like roslyn to prevent the cases you're mentioning and not emit metadata? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I was just wondering if it's going to be actually emitted as There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @MichalStrehovsky the This is still being fleshed out a bit and is more compiler centric vs. language centric. Hence I didn't dive too deep into that here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. For calli, Will we support generic method? also named arguments? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Unsure about supporting a generic method. As long as it's supported in the runtime I think we can make it work fine in the language. As for named arguments: yes. Likely optional as well. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Generic calli is not supported by the runtime. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think @luqunl meant support for this:
Rather than generating a This seems to work fine. Allowing this on generic methods would save quite a bit of typing so I would vote for not blocking this. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I agree that it would nice to allow it - if we do the matching work in CoreCLR and other runtimes to support it.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Yes. Interop would love to support this. Currently MCG and System.Private.Interop will define/generic a lot calli helper methods, similar to the following
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
I do not think that the extra typing is a problem here. This is a low-level feature where saving typing is not important. Also, this would be typically auto-generated, not typed by humans. I would look at the potential support for generic unmanaged calli as second phase of this feature. Generic unmanaged calli would make this feature significantly more complex because of it would dependent on a runtime feature that does not exist yet. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
FWIW: expanding this to include generic support at a later date is basically a bug fix level change. I agree a second phase is a good landing place for this.
It adds complexity yes but not significant complexity to the compiler. The compiler is already setup to look at That being said I think we can move forward for now with generic disabled and revisit once we make a bit more progress. It won't really need language buy off at that point, just keyboard time to get the change in. |
||
|
||
unsafe { | ||
var i = MapValue("42", &int.Parse); | ||
Console.WriteLine(i); | ||
} | ||
``` | ||
|
||
Restrictions on the method which has the `CallIndirect` attribute applied: | ||
|
||
- Cannot have a `DllImport` attribute. | ||
- Cannot be generic. | ||
|
||
## Open Issuess | ||
|
||
### CallingConvention | ||
|
||
The `CallIndirectAttribute` as designed uses the `CallingConvention` enum which lacks an entry for | ||
managed calling conventions. The enum either needs to be extended to include this calling convention | ||
or the attribute needs to take a different approach. | ||
|
||
## Considerations | ||
|
||
### Disambiguating method groups | ||
|
||
There was some discussion around features that would make it easier to disambiguate method groups | ||
passed to an address-of expression. For instance potentially adding signature elements to the | ||
syntax: | ||
|
||
``` csharp | ||
class Util { | ||
public static void Log() { ... } | ||
public static void Log(string) { ... } | ||
} | ||
|
||
unsafe { | ||
// Error: ambiguous Log | ||
void *ptr1 = &Util.Log; | ||
|
||
// Use Util.Log(); | ||
void *ptr2 = &Util.Log(); | ||
} | ||
``` | ||
|
||
This was rejected because a compelling case could not be made nor could a simple syntax be | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Did we consider allowing a "cast"? ie, something like |
||
envisioned here. Also there is a fairly straight forward work around: simple define another | ||
method that is unambiguous and uses C# code to call into the desired function. | ||
|
||
``` csharp | ||
class Workaround { | ||
public static void LocalLog() => Util.Log(); | ||
} | ||
unsafe { | ||
void* ptr = &Workaround.LocalLog; | ||
} | ||
``` | ||
|
||
This becomes even simpler if `static` local functions enter the language. Then the work around | ||
could be defined in the same function that used the ambiguous address-of operation: | ||
|
||
``` csharp | ||
unsafe { | ||
static void LocalLog() => Util.Log(); | ||
void* ptr = &Workaround.LocalLog; | ||
} | ||
``` | ||
|
||
### LoadTypeTokenInt32 | ||
|
||
The original proposal allowed for metadata tokens to be loaded as `int` values at compile time. | ||
Essentially have `tokenof` that has the same arguments as `handleof` but is evaluated at | ||
compile time to an `int` constant. | ||
|
||
This was rejected as it causes significant problem for IL rewrites (of which .NET has many). Such | ||
rewriters often manipulate the metadata tables in a way that could invalidate these values. There | ||
is no reasonable way for such rewriters to update these values when they are stored as simple | ||
`int` values. | ||
|
||
The underlying idea of having an opaque handle for metadata entries will continue to be explored | ||
by the runtime team. | ||
|
||
## Future Considerations | ||
|
||
### static local functions | ||
|
||
This refers to [the proposal](https://github.com/dotnet/csharplang/issues/1565) to allow the | ||
`static` modifier on local functions. Such a function would be guaranteed to be emitted as | ||
`static` and with the exact signature specified in source code. Such a function should be a valid | ||
argument to `&` as it contains none of the problems local functions have today. | ||
|
||
### NativeCallableAttribute | ||
|
||
The CLR has a feature that allows for managed methods to be emitted in such a way that they are | ||
directly callabe from native code. This is done by adding the `NativeCallableAttribute` to | ||
methods. Such a method is only callable from native code and hence must contain only blittable | ||
types in the signature. Calling from managed code results in a runtime error. | ||
|
||
This feature would pattern well with this proposal as it would allow: | ||
|
||
- Passing a funtion defined in managed code to native code as a function pointer (via address-of) | ||
with no overhead in managed or native code. | ||
- Runtime can introduce use site errors for such functions in managed code to prevent them from | ||
being invoked at compile time. | ||
|
||
|
||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
curious: why do we prefer "void *" instead of "IntPtr"? If we use IntPtr, we don't need to use unsafe part?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not a safe operation which is part of the motivation for returning a
void*
here.