Consider this snippet of C++23 code:
struct MyData {
std::int32_t value;
};
std::vector<char> read_from_net();
void send_over_net(std::vector<char>&);
...
std::vector<char> buffer = read_from_net();
std::start_lifetime_as<MyData>(buffer.data())->value = 222; // (1)
send_over_net(buffer); // (2)
Assuming all requirements to start lifetime of MyData
at buffer.data()
are respected:
- Is there any UB in this code? Is starting lifetime like this and modifying field well defined?
- Can I access original buffer as chars at
(2)
without UB considering thatstart_lifetime_as
ended previous object's lifetime and started new one's? - Or does
start_lifetime_as
just start another lifetime but also preserves previous one? - Should I use
std::launder(buffer.data())[0]
to access original buffer as char array after(2)
?
Consider this snippet of C++23 code:
struct MyData {
std::int32_t value;
};
std::vector<char> read_from_net();
void send_over_net(std::vector<char>&);
...
std::vector<char> buffer = read_from_net();
std::start_lifetime_as<MyData>(buffer.data())->value = 222; // (1)
send_over_net(buffer); // (2)
Assuming all requirements to start lifetime of MyData
at buffer.data()
are respected:
- Is there any UB in this code? Is starting lifetime like this and modifying field well defined?
- Can I access original buffer as chars at
(2)
without UB considering thatstart_lifetime_as
ended previous object's lifetime and started new one's? - Or does
start_lifetime_as
just start another lifetime but also preserves previous one? - Should I use
std::launder(buffer.data())[0]
to access original buffer as char array after(2)
?
2 Answers
Reset to default 7TL;DR It's undefined behavior.
Kudos for thinking about this.
The key points are that
reusing the storage to create a new object does end the lifetime of old objects1
https://eel.is/c++draft/basic.life#2.5
aliasing through
char
is allowedhttps://eel.is/c++draft/basic.lval#11.3
- but you have to have a validly derived pointer, which you don't, because
starting lifetime of a new object of different type ends validity of all pointers into any part of that object
https://eel.is/c++draft/basic.life#10
You can get a new, valid pointer through
std::launder
of an existing expired pointer, or throughreinterpret_cast
ing of a valid pointer to the new object (the pointer returned fromstd::start_lifetime_as
).With
std::launder
, the returned pointer is valid, but it doesn't restore validity of other expired pointers (such as the ones insidestd::vector
itself). Ouch.https://eel.is/c++draft/ptr.launder#5
1 The one rule that might be interpreted to permit this shows up here, in the relatively-new language around
reused by an object that is not nested within
If the new object is "nested within" the old objects owned by std::array
, then their lifetime continues making everything ok.
https://eel.is/c++draft/intro.object#4
But that permission applies only to unsigned char
and std::byte
, not plain char
, so it doesn't help here.
https://eel.is/c++draft/intro.object#3
1.Is there any UB in this code? Is starting lifetime like this and modifying field well defined?
Assuming if you fullfilling all the requirements the shouldnt be any undefined behaviour.
ifbuffer.data()
ponts to a memory region that stisfies the size and alignment requirements ofMyData
, thestd::start_lifetime_as
will succesfully start the lifetime of aMyData
object at that location. In your snippet, after you start the lifetime ofMyData
atbuffer.data()
, writing to its member (i.e. setting value = 222) is a defined operation.2. and 3. Yes, you can access the original
buffer
aschar
at (2) without undefined behavior.4.In C++ 17 std::launder can to used to obtain pointer to an object which may have been relocated. if you want to optimize something you can use it. From my POV you dont need it because you already have an valid pointer from
buffer.data()
.
MyData
. Then you need to modify some part of it via a member. It is one of the most common things you can do in C. Your point doesn't stand here, this example is minimal and representable enough – Sergey Kolesnik Commented yesterdaylanguage-lawyer
tag – Sergey Kolesnik Commented yesterday