dicej opened PR #4337 from component-type-derive
to main
:
This is a first draft of an implementation of
https://github.com/bytecodealliance/wasmtime/issues/4308, i.e. derive macros for
ComponentType
,Lift
, andLower
for composite types in the component model.It borrows heavily from the work @jameysharp did in
https://github.com/bytecodealliance/wasmtime/pull/4217. Thanks, Jamey!Thanks also to @fibonacci1729 for pairing up on this.
This draft only covers records; I expect the other composite types will follow a
similar pattern. I'm looking to get feedback on the overall approach before I
continue.Known (potential) issues:
Currently, the implementations for
ComponentType
,Lift
, andLower
duplicate a fair amount of code, so I plan to address that.I've made
ComponentTypesBuilder::record_type
public for testing purposes.
Let me know if that's a problem or if there's a better way.Also for testing: I've made
Memory::wasmtime_memory
andStore::inner
crate-public.I've added a
test
module to crates/wasmtime/src/component/func/typed.rs for
testing the derives. Let me know if there's a better place for it.Signed-off-by: Joel Dice <joel.dice@fermyon.com>
<!--
Please ensure that the following steps are all taken care of before submitting
the PR.
[ ] This has been discussed in issue #..., or if not, please tell us why
here.[ ] A short description of what this does, why it is needed; if the
description becomes long, the matter should probably be discussed in an issue
first.[ ] This PR contains test cases, if meaningful.
- [ ] A reviewer from the core maintainer team has been assigned for this PR.
If you don't know who could review this, please indicate so. The list of
suggested reviewers on the right can help you.Please ensure all communication adheres to the code of conduct.
-->
dicej updated PR #4337 from component-type-derive
to main
.
alexcrichton submitted PR review.
alexcrichton submitted PR review.
alexcrichton created PR review comment:
Could this have a
.context(...)
with information about the field name that failed to type-check?
alexcrichton created PR review comment:
I think it's ok to skip the
#[inline]
here since this is pretty beefy internally.
alexcrichton created PR review comment:
Would you be ok switching this to the style of
use
elsewhere in the crate where it's oneuse ..
per line?
alexcrichton created PR review comment:
If you're up for it, could this be moved externally to
tests/all/component_model/*.rs
? I personally prefer to test the public-facing interface wherever possible because the internal ones like the builder used here are often subject to more churn.
alexcrichton created PR review comment:
If possible I'd prefer to keep this hidden, but I think that will be enabled if the test above is moved to an external test.
alexcrichton created PR review comment:
Personally I think it's ok to inline all these at the use-site, otherwise I find myself tabbing back and forth to what exactly this error message is saying
alexcrichton created PR review comment:
I think thes can be reexported outside the
derive
module? (ideally we wouldn't need aderive
module I think)
alexcrichton created PR review comment:
Personally for patterns like this I find it convenient to have "early return" to avoid nested indentation getting too unwieldy. For example:
if attribute.path.leading_colon.is_some() || attribute.path.segments.len() != 1 { continue; } // ... everything else minus one indentation level
alexcrichton created PR review comment:
Instead of cloning could this pass
&T
since&T
isLower
ifT
isLower
?
alexcrichton created PR review comment:
Oh this is clever! I hadn't actually thought of simply delegating to the tuple implementation. That definitely makes this easier since you don't have to do offset calculations here in the derive macro.
That being said though the main limitation of this is that it won't work for structs with > 16 fields. That doesn't seem too common though so otherwise seems like a good way to start?
alexcrichton created PR review comment:
Although note that in moving this test externally I mean doing a whole compile-a-component rigamarole and using
Func::typed::<...>
to run the tests that this is otherwise doing, so the test wouldn't useComponentTypesBuilder
or calltypecheck
at all.
alexcrichton created PR review comment:
One thing you may want to explore though is what the error messages look like if a field of the record doesn't implement the requisite trait. For example what's the compiler error message when you derive
Lift
for a record with aString
in it?
dicej created PR review comment:
I agree -- I only started doing that because rustfmt would refuse to format an expression that contained a long string literal. Perhaps your suggestion to reduce the indentation level will help with that.
dicej submitted PR review.
dicej created PR review comment:
The cleverness is due to @jameysharp -- that's straight from his patch.
dicej submitted PR review.
dicej updated PR #4337 from component-type-derive
to main
.
dicej updated PR #4337 from component-type-derive
to main
.
jameysharp created PR review comment:
I'm certainly enjoying being called clever! But before I got sidetracked, I had intended to drop this trick. :laughing:
In #4217 I factored out
next_field
to make both tuple and record macro expansions relatively straightforward. Whilelower
andlift
were already pretty easy,next_field
wraps up the common math forsize
,store
, andload
. So I hope it should be simple now to almost just copyimpl_component_ty_for_tuples
, except as a proc-macro instead ofmacro_rules
, and with field names instead of indexes.As I recall, this bit of cleverness is also what forced me to call
.clone()
inlower
: I couldn't build a tuple of borrows, because the implementation I delegated to expected the borrow to be on the outside of the tuple instead. Skipping the indirection should let you remove those clones.
jameysharp submitted PR review.
dicej submitted PR review.
dicej created PR review comment:
Thanks for pointing that out, @jameysharp. I was able to change the
.clone()
to a borrow with no apparent ill effects -- the compiler and the test are both happy. I'll admit I haven't thought deeply about _why_ it worked, though :)I do agree that the limit of 16 fields for structs is annoying, so I like your plan. Just to be clear: the plan is to have the proc-macro generate a custom struct for use as the
ComponentType::Lower
associated type for each#[derive(ComponentType)]
#[component(record)]
struct alongside the generatedComponentType
impl, analogous to whatimpl_component_ty_for_tuples
does for tuples -- correct?
dicej edited PR #4337 from component-type-derive
to main
:
This is a first draft of an implementation of
https://github.com/bytecodealliance/wasmtime/issues/4308, i.e. derive macros for
ComponentType
,Lift
, andLower
for composite types in the component model.It borrows heavily from the work @jameysharp did in
https://github.com/bytecodealliance/wasmtime/pull/4217. Thanks, Jamey!Thanks also to @fibonacci1729 for pairing up on this.
This draft only covers records; I expect the other composite types will follow a
similar pattern. I'm looking to get feedback on the overall approach before I
continue.Known (potential) issues:
Currently, the implementations for
ComponentType
,Lift
, andLower
duplicate a fair amount of code, so I plan to address that.~~I've made
ComponentTypesBuilder::record_type
public for testing purposes.
Let me know if that's a problem or if there's a better way.~~~~Also for testing: I've made
Memory::wasmtime_memory
andStore::inner
crate-public.~~~~I've added a
test
module to crates/wasmtime/src/component/func/typed.rs for
testing the derives. Let me know if there's a better place for it.~~Signed-off-by: Joel Dice <joel.dice@fermyon.com>
<!--
Please ensure that the following steps are all taken care of before submitting
the PR.
[ ] This has been discussed in issue #..., or if not, please tell us why
here.[ ] A short description of what this does, why it is needed; if the
description becomes long, the matter should probably be discussed in an issue
first.[ ] This PR contains test cases, if meaningful.
- [ ] A reviewer from the core maintainer team has been assigned for this PR.
If you don't know who could review this, please indicate so. The list of
suggested reviewers on the right can help you.Please ensure all communication adheres to the code of conduct.
-->
alexcrichton created PR review comment:
Yeah that sounds right to me, generating a new type, and it could even be an "anonymous" type along the lines of:
const _: () = { struct Foo; }; // doesn't have access to `Foo` here
alexcrichton submitted PR review.
dicej updated PR #4337 from component-type-derive
to main
.
dicej created PR review comment:
@alexcrichton and @jameysharp please see the commit I just pushed and let me know what you think.
Alex, I wasn't sure how to apply your "anonymous" type trick at the item level that the derive macro operates. Can you elaborate on what that would look like, i.e. how I might use it in
emit_record_type
?
dicej submitted PR review.
alexcrichton created PR review comment:
I'm not actually sure if this is possible, but if we can I think it'd be best to reuse the
map_maybe_uninit!
macro used internally here as well. If that doesn't work out though can you fill in the two type parameters totransmute
here to clarify what's happening?
alexcrichton submitted PR review.
alexcrichton submitted PR review.
dicej submitted PR review.
dicej created PR review comment:
Yeah, I started out reusing
map_maybe_uninit!
, but that required adding a#[macro_export]
and makingMaybeUninitExt
public. If that sounds okay to you, I'll do that.
alexcrichton created PR review comment:
Could this also get prefixed with
dep:
to indicate to Cargo we shouldn't have a feature calledwasmtime-component-macro
?
alexcrichton submitted PR review.
alexcrichton created PR review comment:
That should be fine as long as they're all marked
#[doc(hidden)]
to not actually show up in the public documentation
alexcrichton submitted PR review.
dicej updated PR #4337 from component-type-derive
to main
.
dicej updated PR #4337 from component-type-derive
to main
.
jameysharp created PR review comment:
Random thought: for alignment, you could de-duplicate the field types, e.g. a struct with a bunch of
u32
s has the same alignment as oneu32
. I think it might marginally improve compile times overall. It's certainly not necessary to do though.Overall, this looks good to me! It's up to Alex on final review though.
jameysharp submitted PR review.
dicej updated PR #4337 from component-type-derive
to main
.
dicej updated PR #4337 from component-type-derive
to main
.
dicej updated PR #4337 from component-type-derive
to main
.
dicej edited PR #4337 from component-type-derive
to main
:
This is a first draft of an implementation of
https://github.com/bytecodealliance/wasmtime/issues/4308, i.e. derive macros for
ComponentType
,Lift
, andLower
for composite types in the component model.It borrows heavily from the work @jameysharp did in
https://github.com/bytecodealliance/wasmtime/pull/4217. Thanks, Jamey!Thanks also to @fibonacci1729 for pairing up on this.
This draft only covers records; I expect the other composite types will follow a
similar pattern. I'm looking to get feedback on the overall approach before I
continue.Known (potential) issues:
~~Currently, the implementations for
ComponentType
,Lift
, andLower
duplicate a fair amount of code, so I plan to address that.~~~~I've made
ComponentTypesBuilder::record_type
public for testing purposes.
Let me know if that's a problem or if there's a better way.~~~~Also for testing: I've made
Memory::wasmtime_memory
andStore::inner
crate-public.~~~~I've added a
test
module to crates/wasmtime/src/component/func/typed.rs for
testing the derives. Let me know if there's a better place for it.~~Signed-off-by: Joel Dice <joel.dice@fermyon.com>
<!--
Please ensure that the following steps are all taken care of before submitting
the PR.
[ ] This has been discussed in issue #..., or if not, please tell us why
here.[ ] A short description of what this does, why it is needed; if the
description becomes long, the matter should probably be discussed in an issue
first.[ ] This PR contains test cases, if meaningful.
- [ ] A reviewer from the core maintainer team has been assigned for this PR.
If you don't know who could review this, please indicate so. The list of
suggested reviewers on the right can help you.Please ensure all communication adheres to the code of conduct.
-->
dicej updated PR #4337 from component-type-derive
to main
.
jameysharp created PR review comment:
Maybe one more field-count test, with too many fields? Looks like you have everything you need to do that easily.
jameysharp submitted PR review.
jameysharp submitted PR review.
dicej updated PR #4337 from component-type-derive
to main
.
dicej updated PR #4337 from component-type-derive
to main
.
dicej has marked PR #4337 as ready for review.
Last updated: Jan 24 2025 at 00:11 UTC