Created on 2009-02-26.00:00:00 last changed 172 months ago
Proposed resolution:
Change [unique.ptr.single.modifiers], p5 (Effects clause for reset), and p6:
-5- Effects:
If get() == nullptr there are no effects. Otherwise get_deleter()(get()).Assigns p to the stored pointer, and then if the old value of the pointer is not equal to nullptr, calls get_deleter()(the old value of the pointer). [Note: The order of these operations is significant because the call to get_deleter() may destroy *this. -- end note]-6- Postconditions: get() == p. [Note: The postcondition does not hold if the call to get_deleter() destroys *this since this->get() is no longer a valid expression. -- end note]
[ Batavia (2009-05): ]
Howard summarizes:
This issue has to do with circular ownership, and affects auto_ptr, too (but we don't really care about that). It is intended to spell out the order in which operations must be performed so as to avoid the possibility of undefined behavior in the self-referential case.
Howard points to message c++std-lib-23175 for another example, requested by Alisdair.
We agree with the issue and with the proposed resolution. Move to Tentatively Ready.
[ Alisdair adds: ]
The example providing the rationale for LWG 998 is poor, as it relies on broken semantics of having two object believing they are unique owners of a single resource. It should not be surprising that UB results from such code, and I feel no need to go out of our way to support such behaviour.
If an example is presented that does not imply multiple ownership of a unique resource, I would be much more ready to accept the proposed resolution.
[ Howard adds: ]
To fix this behavior reset must be specified such that deleting the pointer is the last action to be taken within reset.
[ Thanks to Peter Dimov who recognized the connection to unique_ptr and brought this to the attention of the LWG, and helped with the solution. ]
Consider the following (simplified) implementation of std::auto_ptr<T>::reset():
void reset(T* newptr = 0) { 
   if (this->ptr && this->ptr != newptr) { 
     delete this->ptr; 
   } 
   this->ptr = newptr; 
} 
Now consider the following code which uses the above implementation:
struct foo { 
   std::auto_ptr<foo> ap; 
   foo() : ap(this) {} 
   void reset() { ap.reset(); } 
}; 
int main() { 
   (new foo)->reset(); 
} 
With the above implementation of auto_ptr, this results in U.B. at the point of auto_ptr::reset(). If this isn't obvious yet, let me explain how this goes step by step:
| History | |||
|---|---|---|---|
| Date | User | Action | Args | 
| 2011-08-23 20:07:26 | admin | set | status: wp -> c++11 | 
| 2010-10-21 18:28:33 | admin | set | messages: + msg4737 | 
| 2010-10-21 18:28:33 | admin | set | messages: + msg4736 | 
| 2010-10-21 18:28:33 | admin | set | messages: + msg4735 | 
| 2010-10-21 18:28:33 | admin | set | messages: + msg4734 | 
| 2010-10-21 18:28:33 | admin | set | messages: + msg4733 | 
| 2009-02-26 00:00:00 | admin | create | |