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

Guarantee the layout of some unions having a single non-zero-sized field #161

Merged
merged 17 commits into from
Aug 29, 2019

Conversation

gnzlbg
Copy link
Contributor

@gnzlbg gnzlbg commented Jul 8, 2019

This PR guarantees that unions having a single non-zero-sized field have the same layout as that field if:

  • that field has no padding,
  • all zero-sized fields have an alignment requirement of 1.

Without the requirement that the field might have no padding, the union would "inherit" the padding of their non-zero-sized field, such that these unions wouldn't be "bags of bits".

We can relax this requirement later if we decide that unions shouldn't just be bags of bits, but for the time being, requiring no padding allows us to make this guarantee.

Right now, we don't consider padding to be a part of layout. We should probably do so, since for unions padding doesn't really relate to field offsets - e.g., for repr(C) unions it results from the "overlapping" padding of their fields and trailing padding, see #160.

An alternative way to word the no padding requirement here would be to guarantee this only for types with a "call ABI" equal to "Scalar" or "Vector", since those types have no padding. cc @eddyb

@RalfJung
Copy link
Member

all zero-sized fields have an alignment requirement of 1.

This is another place where we should use the "ZST1" terminology.

The layout of unions with a single non-zero-sized field is the same as the
layout of that field if:

* that field has no padding bits, and
Copy link
Member

Choose a reason for hiding this comment

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

We have not defined padding (what it is, how it behaves, where it occurs in which types) anywhere, and IMO it is not part of layout. So I feel like we are getting a little ahead of ourselves here.

Padding is IMO part of the value representation of a type; no interpretation of layout (size, align, niche, call ABI, field offsets) really includes "padding". For structs it naturally emerges as the "gaps between" fields, for enums I have no idea and for unions I am confused. ;)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

So we touched this on the meeting today. And it is not clear whether we have the definitions required to word this yet. We do not have a definition for padding, it is unclear whether a definition for padding is needed, but we could try to word this without using the "term" padding with the definitions that we have, and if we can't, we might be able to identify whatever definitions would be required to word something like this (and block this on working on those).

@RalfJung
Copy link
Member

Without the requirement that the field might have no padding, the union would "inherit" the padding of their non-zero-sized field, such that these unions wouldn't be "bags of bits".

We can say they have the same size+align, but niches and callABI might be different. Then we can avoid talking about padding, which I would prefer.

@RalfJung RalfJung added A-layout Topic: Related to data structure layout (`#[repr]`) A-unions Topic: Related to unions labels Aug 14, 2019
@gnzlbg
Copy link
Contributor Author

gnzlbg commented Aug 15, 2019

So i've pushed a couple of commits that add pictures and make some progress.

The issues with talking about "padding" in the guarantee about Rust unions with a single non-zero sized field are unfortunate. I think that the next step for this PR would be to also write down what could we guarantee if the non-zero-sized field were to have padding bits (e.g. only size and alignment) and why (i.e. write down the rationale) - this should help refresh everybody's memory about what the constraints are.

@RalfJung
Copy link
Member

Can we wait with this PR until after landing the other union layout one? They look like they will heavily conflict anyway.

@gnzlbg
Copy link
Contributor Author

gnzlbg commented Aug 15, 2019

Yes I'd prefer to merge the other one first and then rebase this one and see if it still makes sense in its current form.

@RalfJung
Copy link
Member

In fact I'd prefer to stage this: first just add "pictures" without saying more than we did before. That's basically an editorial change. Then try to say more.

The "picture" part should be roughly synced with #186.

@gnzlbg
Copy link
Contributor Author

gnzlbg commented Aug 27, 2019

So I've rebased and solved the open issues.

@gnzlbg gnzlbg force-pushed the union_single_field branch from e8a9c67 to 83e9f3d Compare August 28, 2019 08:08
Copy link
Member

@RalfJung RalfJung left a comment

Choose a reason for hiding this comment

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

LGTM. Thanks!

@RalfJung RalfJung mentioned this pull request Aug 29, 2019
4 tasks
Copy link

@hanna-kruppe hanna-kruppe left a comment

Choose a reason for hiding this comment

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

One nit but otherwise LGTM

@gnzlbg gnzlbg merged commit fcfb77d into rust-lang:master Aug 29, 2019
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-layout Topic: Related to data structure layout (`#[repr]`) A-unions Topic: Related to unions
Projects
None yet
Development

Successfully merging this pull request may close these issues.

None yet

3 participants