-
-
Notifications
You must be signed in to change notification settings - Fork 43
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
Introduce support for user-defined, stack allocated value types #750
Comments
An alternative that I'm also strongly considering is to rely entirely on escape analysis similar to e.g. Go, and stack allocate anything we can determine doesn't escape a stack frame. You can then borrow it all you want and it will work. The downside is that values stored in arrays would still need to be heap allocated, since moving the data in/out the array could invalidate borrows. |
Also important to consider: if we want to allow value types to be used in generics, and generate a shape over the size/alignment instead of the class, then we have to include an object header such that the generic code can perform dynamic dispatch. If we don't want an object header then we have to generate a shape for the exact class such that we can perform static dispatch in generic code. |
Per https://www.reddit.com/r/swift/comments/m5zhlf/reference_type_inside_a_struct_automatic/, it seems that Swift allows RC types inside structs and increments them upon copying. We could do something similar to allow |
Another note: if we allow custom value types, we need to handle recursive value types and produce a compile-time error (i.e. a |
A use-case I'm currently already seeing for value types is thin wrappers around platform-specific types. For example, file descriptors are an We could do something like this to work around the need for heap allocating:
However, in this case we're abusing C types as value types, rather than using first-class value types. |
If/when we add this, I think we might also want to consider replacing |
I'm working on this as part of #778 |
An inline type is an immutable value type that's allocated inline/on the stack. Both regular and enum types can be defined as "inline". Inline types don't have object headers, and combined with them being stored inline means they don't incur extra indirection/overhead. For example, consider this type: class inline A { let @A: Int let @b: Int } If this were a regular type, its size would be 32 bytes: 16 bytes for the header, and 16 bytes for the two fields. Because it's an inline type though, it only needs 16 bytes. Inline types are restricted to types that can be trivially copied, such as Int, Float, and other inline types. String isn't allowed in inline types at this point as this could result in an unexpected copy cost due to String using atomic reference counting. Inline types are immutable because supporting mutations introduces significant compiler complexity, especially when dealing with closures that capture generic type parameters, as support for mutations would require rewriting part of the generated code as part of type specialization. This fixes #750. Changelog: added
An inline type is an immutable value type that's allocated inline/on the stack. Both regular and enum types can be defined as "inline". Inline types don't have object headers, and combined with them being stored inline means they don't incur extra indirection/overhead. For example, consider this type: class inline A { let @A: Int let @b: Int } If this were a regular type, its size would be 32 bytes: 16 bytes for the header, and 16 bytes for the two fields. Because it's an inline type though, it only needs 16 bytes. Inline types are restricted to types that can be trivially copied, such as Int, Float, and other inline types. String isn't allowed in inline types at this point as this could result in an unexpected copy cost due to String using atomic reference counting. Inline types are immutable because supporting mutations introduces significant compiler complexity, especially when dealing with closures that capture generic type parameters, as support for mutations would require rewriting part of the generated code as part of type specialization. This fixes #750. Changelog: added
An inline type is an immutable value type that's allocated inline/on the stack. Both regular and enum types can be defined as "inline". Inline types don't have object headers, and combined with them being stored inline means they don't incur extra indirection/overhead. For example, consider this type: class inline A { let @A: Int let @b: Int } If this were a regular type, its size would be 32 bytes: 16 bytes for the header, and 16 bytes for the two fields. Because it's an inline type though, it only needs 16 bytes. Inline types are restricted to types that can be trivially copied, such as Int, Float, and other inline types. String isn't allowed in inline types at this point as this could result in an unexpected copy cost due to String using atomic reference counting. Inline types are immutable because supporting mutations introduces significant compiler complexity, especially when dealing with closures that capture generic type parameters, as support for mutations would require rewriting part of the generated code as part of type specialization. This fixes #750. Changelog: added
Description
Instances of classes are heap allocated by default, with the exception of
Int
,Float
,Bool
, andNil
. This is what allows Inko's borrowing mechanism to work, and crucially allows moving of values while borrows exist, removing the need for lifetime checking.The downside is that heap allocating introduces an allocation cost, and extra indirection. In many instances this isn't warranted/desired, especially for small values.
Today one can sort of work around this by using
class extern
, which defines a class using the C ABI and without an object header. These types are then treated as value types and copied upon being moved or borrowed.We need to extend this with a
class inline
declaration that defines a class to be a value type. Such types behave similar toclass extern
in that they are copied upon moves and borrows, and don't have object headers. For methods this poses a bit of a challenge: they should take their receiver by reference but be able to return a new (copy) if necessary. This means that we have to allow passing aref Whatever
/mut Whatever
to aWhatever
and copy it upon doing so.Dynamic dispatch and casting to traits would be disallowed just as with
Int
andFloat
.The extra restriction is that they can't contain non-value types as that would allow violation of the single ownership rule. This means they an only contain other
class inline
types, or types such asInt
andFloat
.Channel
andString
using reference counting means you can't store these in aclass inline
, because they're not copied by just copying some bits.A difference with
class extern
is that aclass inline
type makes no guarantee about the order or padding of fields, whileclass extern
is meant for cases where you need a guaranteed order (aka be compatible with C code).To allow use of these types in generics, we'd need to generate a distinct shape for the size and alignment of each type. This means different
class inline
types with the same layout use the same shape.A challenge/downside of this setup is that the limitations of
class inline
mean that in general you can only storeInt
,Float
,Bool
andNil
in them. This in turn makes them rather useless.A refinement would be to still subject these values to single ownership and thereby allowing you to store non-value types in them. This however makes borrowing difficult because we can't copy upon borrowing, we can't move (since that makes borrowing impossible), and the lack of a header means we can't do reference counting. Even if we include the object header, the value residing on the stack means borrows are invalidated when the value is moved. The only two ways to prevent that from happening is:
In short, this will need some extra thought.
Related work
The text was updated successfully, but these errors were encountered: