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

Make zero copy safe by default, add account(zero_copy(unsafe)) feature. #2330

Merged
merged 17 commits into from
Jan 26, 2023

Conversation

ckamm
Copy link
Contributor

@ckamm ckamm commented Dec 21, 2022

The default account(zero_copy) feature unsafe impls the bytemuck traits

The new one derives it instead, which runs desired sanity checks like "struct has no padding".

@vercel
Copy link

vercel bot commented Dec 21, 2022

@ckamm is attempting to deploy a commit to the coral-xyz Team on Vercel.

A member of the Team first needs to authorize it.

@Aursen
Copy link
Contributor

Aursen commented Dec 21, 2022

Why not do it by default? It would be interesting to remove all the unsafe code

@ckamm
Copy link
Contributor Author

ckamm commented Dec 22, 2022

Why not do it by default? It would be interesting to remove all the unsafe code

That would break backwards compatibility. It'd prefer making the safe option the default, but it is a choice I'd leave up to the anchor maintainers.

@Henry-E
Copy link

Henry-E commented Dec 22, 2022

Having talked to armani and robert chen about this a while ago, it's probably best to make this the default and anyone that wants to can manually implement their own unsafe types can do that.

  • So we should make this default, that way we can use the existing zero copy tests (assuming they still pass)
  • We probably need some extra documentation to explain what limitations this places on zero copy structs now, I know that we have to use repr(C) and there can't be any padding (unless you use some special option with repr(c) to align stuff).
  • as with all these PRs, please add an entry to the changelog under the breaking change heading
  • If you felt like it, we could be really accommodating to existing programs and make unsafe_bytemuck the optional value instead. While also marking the attribute as deprecated / unsupported

The default account(zero_copy) feature unsafe impls the bytemuck traits

The new one derives it instead, which runs desired sanity checks like
"struct has no padding".
@ckamm ckamm force-pushed the ckamm/safe-bytemuck branch from f5835e0 to 33c9a14 Compare December 26, 2022 15:36
@ckamm ckamm force-pushed the ckamm/safe-bytemuck branch from 33c9a14 to d1fe74e Compare December 26, 2022 15:39
@ckamm
Copy link
Contributor Author

ckamm commented Dec 26, 2022

@Henry-E I've covered some of your suggestions. The existing zero_copy docs already referenced bytemuck::Pod safety restrictions, so I didn't know what to add there.

  • I didn't know where to add the note about deprecated unsafe_bytemuck_derives
  • Fixing the test looked tricky, I made it pass by using unsafe_bytemuck for now

Could you take care of the rest?

@Henry-E
Copy link

Henry-E commented Dec 26, 2022

Sure, thanks very much for all that. I'll look into finishing the rest soon.

@Henry-E
Copy link

Henry-E commented Jan 17, 2023

158 |     pub events: [Event; 25000],
    |                 ^^^^^^^^^^^^^^ the trait `Pod` is not implemented for `[Event; 25000]`

I kind of forgot about this kind of error, where rust wants you to implement traits on arrays of specific lengths. Is this something you've had to deal with @ckamm or is it just the case that you only ever use Vec<T> with zero copy?

I can't quite remember but I thought there was some reason that people didn't use Vec<T> with zero copy.

@ckamm
Copy link
Contributor Author

ckamm commented Jan 18, 2023

@Henry-E Use bytemuck with the features = ["min_const_generics"] feature -- newer rust versions allow generically implementing Pod for static arrays of all sizes.

You can't use Vec<> with zero-copy. Vec is just a struct of a few pointers and all the actual data ends up on the heap.

@Henry-E
Copy link

Henry-E commented Jan 18, 2023

@ckamm that solved it, thanks very much!

Another thing that I find kind of annoying but don't really know what to do about is how even though we use a reference to anchor's privately exported bytemuck crate derive(anchor_lang::__private::bytemuck::Pod), once these macros are expanded they switch to using the local crate's bytemuck, e.g. ::bytemuck::Pod. I'm not sure if it's possible to do anything about this.

@ckamm
Copy link
Contributor Author

ckamm commented Jan 18, 2023

Glad it helped! I don't know of a way to use the privately exported instance, unfortunately.

@Henry-E Henry-E changed the title Add account(zero_copy(safe_bytemuck_derives)) feature. Make zero copy safe by default, add account(zero_copy(unsafe)) feature. Jan 18, 2023
@Henry-E
Copy link

Henry-E commented Jan 18, 2023

No worries, I think I'll do one more review of this tomorrow morning and then I think it's good to merge.

@Henry-E
Copy link

Henry-E commented Jan 19, 2023

One solution is to use a crate alias but i don't know if we really want to overwrite the local crate's bytemuck. Now I'm wondering if we should just make it entirely using the local crate's bytemuck and not worry about whatever version of bytemuck anchor's using.

@Henry-E Henry-E merged commit ed2769e into coral-xyz:master Jan 26, 2023
@vercel
Copy link

vercel bot commented Jan 26, 2023

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Comments Updated
anchor-docs ✅ Ready (Inspect) Visit Preview 💬 Add your feedback Jan 26, 2023 at 2:48PM (UTC)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants