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

Separate methods v.s. "mode" parameters #58

Closed
SimonSapin opened this issue Apr 26, 2020 · 21 comments · Fixed by rust-lang/rust#74850
Closed

Separate methods v.s. "mode" parameters #58

SimonSapin opened this issue Apr 26, 2020 · 21 comments · Fixed by rust-lang/rust#74850
Assignees
Labels
A-Alloc Trait Issues regarding the Alloc trait Proposal

Comments

@SimonSapin
Copy link
Contributor

#41 has split realloc into separate grow and shrink methods. However as of this writing some methods take AllocInit or ReallocPlacement parameters that are effectively boolean mode switches.

Part of the reasoning for separate methods is that most callers are always in one of the two cases, so separating saves a branch in the allocator implementation. The same seems to apply to AllocInit and ReallocPlacement.

The current design with 5 methods seems inconsistent to me. I think the trait should pick one principle among easier/nicer-looking API or single-purpose methods. This would give either 4 methods:

  • alloc
  • dealloc
  • realloc
  • by_ref

Or 10 methods:

  • alloc
  • alloc_zeroed
  • dealloc
  • grow
  • grow_zeroed
  • grow_in_place
  • grow_in_place_zeroed
  • shrink
  • shrink_in_place
  • by_ref

I tend to favor the latter. The reason against it given in #44 is that this "blows up the trait". But I feel 10 methods is not that bad, especially given that they are not 10 completely different behaviors that need to be learned independently, but follow a pattern that’s reflected in their names.

@SimonSapin SimonSapin added A-Alloc Trait Issues regarding the Alloc trait Proposal labels Apr 26, 2020
@TimDiekmann
Copy link
Member

After playing around with AllocRef a lot it's pretty annoying to pass AllocInit::Uninitialized every time I want a simple alloc. I'm up for the change!

@Amanieu
Copy link
Member

Amanieu commented Apr 27, 2020

While I might be convinced to split _in_place into separate methods, I feel that we definitely should not do this for _zeroed. We currently provide convenience methods on AllocInit which allow allocators to easily handle both the uninitialized and zeroed cases.

@TimDiekmann
Copy link
Member

We could add init_zeroed and init_zeroed_offset to MemoryBlock.

@Amanieu
Copy link
Member

Amanieu commented Apr 29, 2020

Eh, I guess I don't feel too strongly either way.

@nnethercote
Copy link

I think the in_place variants are useless. I've spent a lot of time working on things relating to efficient and correct allocation over the years: multiple heap profilers of different kinds, multiple heap allocation checkers, and multiple implementation of growing data structures such as vectors and hash tables. Never have I felt the need to grow or shrink memory while requiring that it stay in place.

Am I overlooking an important use case?

@TimDiekmann
Copy link
Member

TimDiekmann commented May 25, 2020

I feel that we definitely should not do this for _zeroed. We currently provide convenience methods on AllocInit which allow allocators to easily handle both the uninitialized and zeroed cases.

If we decide to use NonNull<[u8]> instead of MemoryBlock, implementing write_bytes on NonNull<T> seems to be convenient enough. It's also the (IMO) expected way to zero a block of memory, especially as *mut T also works this way.

@TimDiekmann
Copy link
Member

TimDiekmann commented May 25, 2020

Am I overlooking an important use case?

Growing and shrinking in-place does not touch the data stored in the memory block (except the tail when shrinking ofc). The pointer returned from a previous call to AllocRef when reallocating in-place remains the same, which is a big deal.

@nnethercote
Copy link

What's a specific use case of InPlace? What problem does it solve? When would someone want to grow or shrink an allocation and have the operation fail if it can't be achieved in place? I can't imagine anything other than extremely contrived use cases.

The fact that all three of the existing implementations just return error on InPlace is suggestive.

@Amanieu
Copy link
Member

Amanieu commented May 26, 2020

After thinking about it for a bit, I agree with @nnethercote: in-place reallocation is a rather niche operation which can be exposed with an extension trait on crates.io if it really becomes necessary. Another strong point in favor of removing in-place reallocation is that none of the standard allocators even support this functionality.

@TimDiekmann
Copy link
Member

TimDiekmann commented May 29, 2020

Another strong point in favor of removing in-place reallocation is that none of the standard allocators even support this functionality.

This is a good point. Then we should just remove the API, use the second approach, replace MemoryBlock with NonNull<[u8]> like proposed in #61, and add more methods from *mut _ to NonNull. This also unblocks rust-lang/rust#72417.

@Amanieu
Copy link
Member

Amanieu commented May 29, 2020

I agree. Shall we go ahead with this and unblock rust-lang/rust#72417?

@SimonSapin
Copy link
Contributor Author

Removing in-place reallocation from AllocRef sounds good to me, as does replacing (the one remaining) mode parameter with separate methods.

MemoryBlock v.s. NonNull<[u8]> seems orthogonal to everything else in this thread, though.

@Lokathor
Copy link

Yeah, I still think NonNull<[u8]> is a bad change even though the rest of these proposals are good.

@TimDiekmann
Copy link
Member

MemoryBlock v.s. NonNull<[u8]> seems orthogonal to everything else in this thread, though.

Not directly:

We currently provide convenience methods on AllocInit which allow allocators to easily handle both the uninitialized and zeroed cases.

With NonNull<T>::write_bytes in mind, which absolutely makes sense, we could remove those methods, too.

@SimonSapin
Copy link
Contributor Author

To replace init and init_offset methods on AllocInit we can have init_zeroed and init_zeroed_from as suggested by… you #58 (comment) :)

These methods can be on MemoryBlock or NonNull<[_]>. Which one is used doesn’t really affect separate methods v.s. mode parameter.

@Wodann
Copy link

Wodann commented Jun 1, 2020

Irrespective of whether we move ahead with the separate decisions, I'd be in favour of not lumping the changes together. Let's just make those two decisions separately and if both are accepted make two PRs. That also makes the "history" cleaner, if both do get accepted.

In general, I am in favor of removing the in-place reallocation and reverting back to separate methods.

@TimDiekmann
Copy link
Member

I will come back to this next week and create a PR upstream, which removes in_place allocations and reverts back to separate methods as suggested. @SimonSapin do you want me to r? you?

@SimonSapin
Copy link
Contributor Author

No need to block on me, but if the auto-assigned reviewer is unresponsive feel free to ping me.

@TimDiekmann TimDiekmann self-assigned this Jul 29, 2020
bors added a commit to rust-lang-ci/rust that referenced this issue Aug 4, 2020
…Amanieu

Remove in-place allocation and revert to separate methods for zeroed allocations

closes rust-lang/wg-allocators#58
@Kixunil
Copy link

Kixunil commented Jan 21, 2022

Removing in_place broke the solution of #40. I have a similar situation: a buffer that can have beginning erased and it'd be great to not copy erased part. However this is theoretical I'm not sure about the real impact. @nnethercote you seem to be experienced, did you ever come across something like this and found out that it doesn't help performance?

@Kixunil
Copy link

Kixunil commented Jan 22, 2022

Also just realized that a simple Vec::reserve() could use this to avoid copying uninitialized part. So maybe not that niche?

@nnethercote
Copy link

@Kixunil I don't recall any situations like that.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-Alloc Trait Issues regarding the Alloc trait Proposal
Projects
None yet
Development

Successfully merging a pull request may close this issue.

7 participants