Skip to content
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

Add proposal for value types marked as @frozen #306

Conversation

kotlarmilos
Copy link
Member

Description

In Swift, when value types are annotated with @frozen, the calling convention directly handles them if they can fit within 4 registers. If the runtime supports passing 4-word structs by value and allows designating the self register, the 4-word struct returns, and an exception register, then binding tools should be able to handle most cases without requiring wrappers.

For value types annotated as @Frozen, Swift calling convention directly handles them if they fit within 4 registers
@@ -106,6 +106,18 @@ Rejected Option 5 would have provided a better ".NET" shape than our selected al

In the SwiftAsync calling convention, there is also an Async Context register, similar to the Self and Error registers. Like the error register, the async context must be passed by a pointer value. As a result, similar options apply here, with the same constraints. Additionally, we don't already have an existing `CallConvAsync` calling convention modifier, so going the calling convention route like proposed for the self register is not practical. As a result, we will likely need to use a special type like `SwiftAsyncContext` to represent the async context register, similar to the proposals for the error register and the self register.

##### Value types marked as @frozen
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is covered in https://github.com/dotnet/designs/blob/3768dc57e7d90f2b21e17464bffcb590b157a810/proposed/swift-interop.md#structsenums paragraph already.

Can we use the plan described in that paragraph? (Handle non-frozen structs in projection layer, and assume that all structs are frozen in the low-level calling convention.)

@jkoritzinsky
Copy link
Member

If we want to have both frozen and opaque structs in the signature (and not represent opaque structs as void*), we should take the following approach instead:

  • All @frozen layout types are in the signature directly and require no annotations.
  • All opaque types are passed as a SwiftOpaque<T> type, where SwiftOpaque<T> is a pointer-sized struct that is treated as a pointer at the ABI level.

This ensures that we don't need to look at type attributes as part of the signature parsing, which we would prefer not to do. This also ensures that we can detect that a return of a Swift function is an opaque struct (and we can ensure we give it the same behavior as Clang gives a swift_indirect_result parameter, which represents a struct return of an opaque struct in Swift.

@jkotas
Copy link
Member

jkotas commented Nov 28, 2023

we can ensure we give it the same behavior as Clang gives a swift_indirect_result parameter, which represents a struct return of an opaque struct in Swift.

Would SwiftOpaque<T> be the managed method return type or the managed method parameter in the swift_indirect_result case? If it is the earlier, how does the JIT figure out the size of the return buffer to allocate? If it is the latter, how are we going to tell a difference between return value and parameter value?

@jkoritzinsky
Copy link
Member

we can ensure we give it the same behavior as Clang gives a swift_indirect_result parameter, which represents a struct return of an opaque struct in Swift.

Would SwiftOpaque<T> be the managed method return type or the managed method parameter in the swift_indirect_result case? If it is the earlier, how does the JIT figure out the size of the return buffer to allocate? If it is the latter, how are we going to tell a difference between return value and parameter value?

You're right, I don't think we would have enough information with just a SwiftOpaque<T> type to get any special experience here, either as the managed return type or as a parameter.

If we were to do it as a return type, the JIT would have to know how to find the value witness table, which we don't want to plumb down that deep. If we were to do it as a parameter, then we wouldn't have a way to know if it was supposed to be a return value or not.

In that case, I'd go back to my original suggestion and Jan's suggestion of "all structs in a signature are @frozen, all opaque types should be passed as void*(and struct returns are a void* parameter at the start of the parameter list)" without a SwiftOpaque<T> type.

@AaronRobinsonMSFT
Copy link
Member

"all structs in a signature are @frozen, all opaque types should be passed as void*(and struct returns are a void* parameter at the start of the parameter list)" without a SwiftOpaque type.

It sounds like we are just using void* instead of SwiftOpaque<T>. Why can't we replace the void* in the sentence above with SwiftOpaque<T> and treat it like a pointer sized type? Is the statement here, "SwiftOpaque<T> doesn't give us anymore than void*"?

I simply trying to clarify precisely why void* is any different from a pointer sized value type in this position.

@jkotas
Copy link
Member

jkotas commented Nov 29, 2023

SwiftOpaque doesn't give us anymore than void*"?

Yep.

@kotlarmilos
Copy link
Member Author

Agreed. Since we can't determine the buffer size without the value witness table, we will handle non-frozen structs in projection layer.

@kotlarmilos kotlarmilos closed this Dec 4, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants