xref: /netbsd-src/external/gpl3/gcc/dist/libstdc++-v3/src/c++11/cow-stdexcept.cc (revision b1e838363e3c6fc78a55519254d99869742dd33c)
1 // Methods for Exception Support for -*- C++ -*-
2 
3 // Copyright (C) 2014-2022 Free Software Foundation, Inc.
4 //
5 // This file is part of the GNU ISO C++ Library.  This library is free
6 // software; you can redistribute it and/or modify it under the
7 // terms of the GNU General Public License as published by the
8 // Free Software Foundation; either version 3, or (at your option)
9 // any later version.
10 
11 // This library is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 // GNU General Public License for more details.
15 
16 // Under Section 7 of GPL version 3, you are granted additional
17 // permissions described in the GCC Runtime Library Exception, version
18 // 3.1, as published by the Free Software Foundation.
19 
20 // You should have received a copy of the GNU General Public License and
21 // a copy of the GCC Runtime Library Exception along with this program;
22 // see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
23 // <http://www.gnu.org/licenses/>.
24 
25 //
26 // ISO C++ 14882: 19.1  Exception classes
27 //
28 
29 // Enable hooks for support for the Transactional Memory TS (N4514).
30 #define _GLIBCXX_TM_TS_INTERNAL
31 void
32 _txnal_cow_string_C1_for_exceptions(void* that, const char* s, void* exc);
33 const char*
34 _txnal_cow_string_c_str(const void* that);
35 void
36 _txnal_cow_string_D1(void* that);
37 void
38 _txnal_cow_string_D1_commit(void* that);
39 void*
40 _txnal_logic_error_get_msg(void* e);
41 void*
42 _txnal_runtime_error_get_msg(void* e);
43 
44 // All exception classes still use the classic COW std::string.
45 #define _GLIBCXX_USE_CXX11_ABI 0
46 #define _GLIBCXX_DEFINE_STDEXCEPT_COPY_OPS 1
47 #define __cow_string __cow_stringxxx
48 #include <stdexcept>
49 #include <system_error>
50 #undef __cow_string
51 
52 namespace std _GLIBCXX_VISIBILITY(default)
53 {
54 _GLIBCXX_BEGIN_NAMESPACE_VERSION
55 
56   // Copy/move constructors and assignment operators defined using COW string.
57   // These operations are noexcept even though copying a COW string is not,
58   // but we know that the string member in an exception has not been "leaked"
59   // so copying is a simple reference count increment.
60 
logic_error(const logic_error & e)61   logic_error::logic_error(const logic_error& e) noexcept
62   : exception(e), _M_msg(e._M_msg) { }
63 
operator =(const logic_error & e)64   logic_error& logic_error::operator=(const logic_error& e) noexcept
65   { _M_msg = e._M_msg; return *this; }
66 
67   logic_error::logic_error(logic_error&& e) noexcept = default;
68 
69   logic_error&
70   logic_error::operator=(logic_error&& e) noexcept = default;
71 
runtime_error(const runtime_error & e)72   runtime_error::runtime_error(const runtime_error& e) noexcept
73   : exception(e), _M_msg(e._M_msg) { }
74 
75   runtime_error&
operator =(const runtime_error & e)76   runtime_error::operator=(const runtime_error& e) noexcept
77   { _M_msg = e._M_msg; return *this; }
78 
79   runtime_error::runtime_error(runtime_error&& e) noexcept = default;
80 
81   runtime_error&
82   runtime_error::operator=(runtime_error&& e) noexcept = default;
83 
84   // New C++11 constructors:
85 
logic_error(const char * __arg)86   logic_error::logic_error(const char* __arg)
87   : exception(), _M_msg(__arg) { }
88 
domain_error(const char * __arg)89   domain_error::domain_error(const char* __arg)
90   : logic_error(__arg) { }
91 
invalid_argument(const char * __arg)92   invalid_argument::invalid_argument(const char* __arg)
93   : logic_error(__arg) { }
94 
length_error(const char * __arg)95   length_error::length_error(const char* __arg)
96   : logic_error(__arg) { }
97 
out_of_range(const char * __arg)98   out_of_range::out_of_range(const char* __arg)
99   : logic_error(__arg) { }
100 
runtime_error(const char * __arg)101   runtime_error::runtime_error(const char* __arg)
102   : exception(), _M_msg(__arg) { }
103 
range_error(const char * __arg)104   range_error::range_error(const char* __arg)
105   : runtime_error(__arg) { }
106 
overflow_error(const char * __arg)107   overflow_error::overflow_error(const char* __arg)
108   : runtime_error(__arg) { }
109 
underflow_error(const char * __arg)110   underflow_error::underflow_error(const char* __arg)
111   : runtime_error(__arg) { }
112 
113 #if _GLIBCXX_USE_DUAL_ABI
114   // Converting constructor from COW std::string to SSO string.
__sso_string(const string & s)115   __sso_string::__sso_string(const string& s)
116   : __sso_string(s.c_str(), s.length()) { }
117 
118   // Redefine __cow_string so that we can define and export its members
119   // in terms of the COW std::string.
120   struct __cow_string
121   {
122     union {
123       const char* _M_p;
124       char _M_bytes[sizeof(_M_p)];
125       std::string _M_str;
126     };
127 
128     __cow_string();
129     __cow_string(const std::string& s);
130     __cow_string(const char*, size_t n);
131     __cow_string(const __cow_string&) noexcept;
132     __cow_string& operator=(const __cow_string&) noexcept;
133     ~__cow_string();
134     __cow_string(__cow_string&&) noexcept;
135     __cow_string& operator=(__cow_string&&) noexcept;
136   };
137 
__cow_string()138   __cow_string::__cow_string() : _M_str() { }
139 
__cow_string(const std::string & s)140   __cow_string::__cow_string(const std::string& s) : _M_str(s) { }
141 
__cow_string(const char * s,size_t n)142   __cow_string::__cow_string(const char* s, size_t n) : _M_str(s, n) { }
143 
__cow_string(const __cow_string & s)144   __cow_string::__cow_string(const __cow_string& s) noexcept
145   : _M_str(s._M_str) { }
146 
147   __cow_string&
operator =(const __cow_string & s)148   __cow_string::operator=(const __cow_string& s) noexcept
149   {
150     _M_str = s._M_str;
151     return *this;
152   }
153 
~__cow_string()154   __cow_string::~__cow_string() { _M_str.~basic_string(); }
155 
__cow_string(__cow_string && s)156   __cow_string::__cow_string(__cow_string&& s) noexcept
157   : _M_str(std::move(s._M_str)) { }
158 
159   __cow_string&
operator =(__cow_string && s)160   __cow_string::operator=(__cow_string&& s) noexcept
161   {
162     _M_str = std::move(s._M_str);
163     return *this;
164   }
165 
166   static_assert(sizeof(__cow_string) == sizeof(std::string),
167                 "sizeof(std::string) has changed");
168   static_assert(alignof(__cow_string) == alignof(std::string),
169                 "alignof(std::string) has changed");
170 #endif
171 
172   // Return error_category::message() as an SSO string
173   __sso_string
_M_message(int i) const174   error_category::_M_message(int i) const
175   {
176     string msg = this->message(i);
177     return {msg.c_str(), msg.length()};
178   }
179 
180 _GLIBCXX_END_NAMESPACE_VERSION
181 } // namespace
182 
183 // Support for the Transactional Memory TS (N4514).
184 //
185 // logic_error and runtime_error both carry a message in the form of a COW
186 // string.  This COW string is never made visible to users of the exception
187 // because what() returns a C string.  The COW string can be constructed as
188 // either a copy of a COW string of another logic_error/runtime_error, or
189 // using a C string or SSO string; thus, the COW string's _Rep is only
190 // accessed by logic_error operations.  We control all txnal clones of those
191 // operations and thus can ensure that _Rep is never accessed transactionally.
192 // Furthermore, _Rep will always have been allocated or deallocated via
193 // global new or delete, so nontransactional writes we do to _Rep cannot
194 // interfere with transactional accesses.
195 
196 // We depend on having support for referencing functions declared weak that
197 // are not defined by us.  Without such support, the exceptions will not be
198 // declared transaction-safe, so we just don't provide transactional clones
199 // in this case.
200 #if _GLIBCXX_USE_WEAK_REF
201 #ifdef _GLIBCXX_USE_C99_STDINT_TR1
202 
203 extern "C" {
204 
205 #ifndef _GLIBCXX_MANGLE_SIZE_T
206 #error Mangled name of size_t type not defined.
207 #endif
208 #define CONCAT1(x,y)		x##y
209 #define CONCAT(x,y)		CONCAT1(x,y)
210 #define _ZGTtnaX		CONCAT(_ZGTtna,_GLIBCXX_MANGLE_SIZE_T)
211 
212 #ifdef __i386__
213 /* Only for 32-bit x86.  */
214 # define ITM_REGPARM	__attribute__((regparm(2)))
215 #else
216 # define ITM_REGPARM
217 #endif
218 
219 // Declare all libitm symbols we rely on, but make them weak so that we do
220 // not depend on libitm.
221 extern void* _ZGTtnaX (size_t sz) __attribute__((weak));
222 extern void _ZGTtdlPv (void* ptr) __attribute__((weak));
223 extern uint8_t _ITM_RU1(const uint8_t *p)
224   ITM_REGPARM __attribute__((weak));
225 extern uint16_t _ITM_RU2(const uint16_t *p)
226   ITM_REGPARM __attribute__((weak));
227 extern uint32_t _ITM_RU4(const uint32_t *p)
228   ITM_REGPARM __attribute__((weak));
229 extern uint64_t _ITM_RU8(const uint64_t *p)
230   ITM_REGPARM __attribute__((weak));
231 extern void _ITM_memcpyRtWn(void *, const void *, size_t)
232   ITM_REGPARM __attribute__((weak));
233 extern void _ITM_memcpyRnWt(void *, const void *, size_t)
234   ITM_REGPARM __attribute__((weak));
235 extern void _ITM_addUserCommitAction(void (*)(void *), uint64_t, void *)
236   ITM_REGPARM __attribute__((weak));
237 
238 }
239 
240 // A transactional version of basic_string::basic_string(const char *s)
241 // that also notifies the TM runtime about allocations belonging to this
242 // exception.
243 void
_txnal_cow_string_C1_for_exceptions(void * that,const char * s,void * exc)244 _txnal_cow_string_C1_for_exceptions(void* that, const char* s,
245 				    void *exc __attribute__((unused)))
246 {
247   typedef std::basic_string<char> bs_type;
248   bs_type *bs = (bs_type*) that;
249 
250   // First, do a transactional strlen, but including the trailing zero.
251   bs_type::size_type len = 1;
252   for (const char *ss = s; _ITM_RU1((const uint8_t*) ss) != 0; ss++, len++);
253 
254 
255   // Allocate memory for the string and the refcount.  We use the
256   // transactional clone of global new[]; if this throws, it will do so in a
257   // transaction-compatible way.
258   // The allocation belongs to this exception, so tell the runtime about it.
259   // TODO Once this is supported, link the following allocation to this
260   // exception: void *prev = _ITM_setAssociatedException(exc);
261   bs_type::_Rep *rep;
262   __try
263     {
264       rep = (bs_type::_Rep*) _ZGTtnaX (len + sizeof (bs_type::_Rep));
265     }
266   __catch (...)
267     {
268       // Pop the association with this exception.
269       // TODO Once this is supported, link the following allocation to this
270       // exception: _ITM_setAssociatedException(prev);
271       // We do not need to instrument a rethrow.
272       __throw_exception_again;
273     }
274   // Pop the association with this exception.
275   // TODO Once this is supported, link the following allocation to this
276   // exception: _ITM_setAssociatedException(prev);
277 
278   // Now initialize the rest of the string and copy the C string.  The memory
279   // will be freshly allocated, so nontransactional accesses are sufficient,
280   // including the writes when copying the string (see above).
281   rep->_M_set_sharable();
282   rep->_M_length = rep->_M_capacity = len - 1;
283   _ITM_memcpyRtWn(rep->_M_refdata(), s, len);
284   new (&bs->_M_dataplus) bs_type::_Alloc_hider(rep->_M_refdata(),
285 					       bs_type::allocator_type());
286 }
287 
txnal_read_ptr(void * const * ptr)288 static void* txnal_read_ptr(void* const * ptr)
289 {
290   static_assert(sizeof(uint64_t) == sizeof(void*)
291 		|| sizeof(uint32_t) == sizeof(void*)
292 		|| sizeof(uint16_t) == sizeof(void*),
293 		"Pointers must be 16 bits, 32 bits or 64 bits wide");
294 #if __UINTPTR_MAX__ == __UINT64_MAX__
295   return (void*)_ITM_RU8((const uint64_t*)ptr);
296 #elif __UINTPTR_MAX__ == __UINT32_MAX__
297   return (void*)_ITM_RU4((const uint32_t*)ptr);
298 #else
299   return (void*)_ITM_RU2((const uint16_t*)ptr);
300 #endif
301 }
302 
303 // We must access the data pointer in the COW string transactionally because
304 // another transaction can delete the string and reuse the memory.
305 const char*
_txnal_cow_string_c_str(const void * that)306 _txnal_cow_string_c_str(const void* that)
307 {
308   typedef std::basic_string<char> bs_type;
309   const bs_type *bs = (const bs_type*) that;
310 
311   return (const char*) txnal_read_ptr((void**)&bs->_M_dataplus._M_p);
312 }
313 
314 #if _GLIBCXX_USE_DUAL_ABI
315 const char*
_txnal_sso_string_c_str(const void * that)316 _txnal_sso_string_c_str(const void* that)
317 {
318   return (const char*) txnal_read_ptr(
319       (void* const*)const_cast<char* const*>(
320 	  &((const std::__sso_string*) that)->_M_s._M_p));
321 }
322 #endif
323 
324 void
_txnal_cow_string_D1_commit(void * data)325 _txnal_cow_string_D1_commit(void* data)
326 {
327   typedef std::basic_string<char> bs_type;
328   bs_type::_Rep *rep = (bs_type::_Rep*) data;
329   rep->_M_dispose(bs_type::allocator_type());
330 }
331 
332 void
_txnal_cow_string_D1(void * that)333 _txnal_cow_string_D1(void* that)
334 {
335   typedef std::basic_string<char> bs_type;
336   bs_type::_Rep *rep = reinterpret_cast<bs_type::_Rep*>(
337       const_cast<char*>(_txnal_cow_string_c_str(that))) - 1;
338 
339   // The string can be shared, in which case we would need to decrement the
340   // reference count.  We cannot undo that because we might lose the string
341   // otherwise.  Therefore, we register a commit action that will dispose of
342   // the string's _Rep.
343   enum {_ITM_noTransactionId  = 1};
344   _ITM_addUserCommitAction(_txnal_cow_string_D1_commit, _ITM_noTransactionId,
345 			   rep);
346 }
347 
348 void*
_txnal_logic_error_get_msg(void * e)349 _txnal_logic_error_get_msg(void* e)
350 {
351   std::logic_error* le = (std::logic_error*) e;
352   return &le->_M_msg;
353 }
354 
355 void*
_txnal_runtime_error_get_msg(void * e)356 _txnal_runtime_error_get_msg(void* e)
357 {
358   std::runtime_error* le = (std::runtime_error*) e;
359   return &le->_M_msg;
360 }
361 
362 // The constructors are only declared transaction-safe if the C++11 ABI is
363 // used for std::string and the exception classes use a COW string internally.
364 // A user must not call these constructors otherwise; if they do, it will
365 // result in undefined behavior, which is in this case not initializing this
366 // string.
367 #if _GLIBCXX_USE_DUAL_ABI
368 #define CTORS_FROM_SSOSTRING(NAME, CLASS, BASE)			\
369 void									\
370 _ZGTtNSt##NAME##C1ERKNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE( \
371     CLASS* that, const std::__sso_string& s)				\
372 {									\
373   CLASS e("");								\
374   _ITM_memcpyRnWt(that, &e, sizeof(CLASS));				\
375   /* Get the C string from the SSO string.  */				\
376   _txnal_cow_string_C1_for_exceptions(_txnal_##BASE##_get_msg(that),	\
377 				      _txnal_sso_string_c_str(&s), that); \
378 }									\
379 void									\
380 _ZGTtNSt##NAME##C2ERKNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE( \
381     CLASS*, const std::__sso_string&) __attribute__((alias		\
382 ("_ZGTtNSt" #NAME							\
383   "C1ERKNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE")));
384 #else
385 #define CTORS_FROM_SSOSTRING(NAME, CLASS, BASE)
386 #endif
387 
388 // This macro defines transaction constructors and destructors for a specific
389 // exception class.  NAME is the variable part of the mangled name, CLASS is
390 // the class name, and BASE must be logic_error or runtime_error (which is
391 // then used to call the proper friend function that can return a pointer to
392 // the _M_msg member declared by the given (base) class).
393 #define CTORDTOR(NAME, CLASS, BASE)					\
394 void									\
395 _ZGTtNSt##NAME##C1EPKc (CLASS* that, const char* s)			\
396 {									\
397   /* This will use the singleton _Rep for an empty string and just	\
398      point to it instead of allocating memory.  Thus, we can use it as	\
399      source, copy it into the object we are constructing, and then	\
400      construct the COW string in the latter manually.  Note that the	\
401      exception classes will not be declared transaction_safe if the	\
402      shared empty _Rep is disabled with --enable-fully-dynamic-string	\
403      (in which case _GLIBCXX_FULLY_DYNAMIC_STRING is nonzero).  */	\
404   CLASS e("");								\
405   _ITM_memcpyRnWt(that, &e, sizeof(CLASS));				\
406   _txnal_cow_string_C1_for_exceptions(_txnal_##BASE##_get_msg(that),	\
407 				      s, that);				\
408 }									\
409 void									\
410 _ZGTtNSt##NAME##C2EPKc (CLASS*, const char*)				\
411   __attribute__((alias ("_ZGTtNSt" #NAME "C1EPKc")));			\
412 CTORS_FROM_SSOSTRING(NAME, CLASS, BASE)					\
413 void									\
414 _ZGTtNSt##NAME##D1Ev(CLASS* that)					\
415 { _txnal_cow_string_D1(_txnal_##BASE##_get_msg(that)); }		\
416 void									\
417 _ZGTtNSt##NAME##D2Ev(CLASS*)						\
418 __attribute__((alias ("_ZGTtNSt" #NAME "D1Ev")));			\
419 void									\
420 _ZGTtNSt##NAME##D0Ev(CLASS* that)					\
421 {									\
422   _ZGTtNSt##NAME##D1Ev(that);						\
423   _ZGTtdlPv(that);							\
424 }
425 
426 // Now create all transactional constructors and destructors, as well as the
427 // two virtual what() functions.
428 extern "C" {
429 
430 CTORDTOR(11logic_error, std::logic_error, logic_error)
431 
432 const char*
_ZGTtNKSt11logic_error4whatEv(const std::logic_error * that)433 _ZGTtNKSt11logic_error4whatEv(const std::logic_error* that)
434 {
435   return _txnal_cow_string_c_str(_txnal_logic_error_get_msg(
436       const_cast<std::logic_error*>(that)));
437 }
438 
439 CTORDTOR(12domain_error, std::domain_error, logic_error)
440 CTORDTOR(16invalid_argument, std::invalid_argument, logic_error)
441 CTORDTOR(12length_error, std::length_error, logic_error)
442 CTORDTOR(12out_of_range, std::out_of_range, logic_error)
443 
444 
445 CTORDTOR(13runtime_error, std::runtime_error, runtime_error)
446 
447 const char*
_ZGTtNKSt13runtime_error4whatEv(const std::runtime_error * that)448 _ZGTtNKSt13runtime_error4whatEv(const std::runtime_error* that)
449 {
450   return _txnal_cow_string_c_str(_txnal_runtime_error_get_msg(
451       const_cast<std::runtime_error*>(that)));
452 }
453 
454 CTORDTOR(11range_error, std::range_error, runtime_error)
455 CTORDTOR(14overflow_error, std::overflow_error, runtime_error)
456 CTORDTOR(15underflow_error, std::underflow_error, runtime_error)
457 
458 }
459 
460 #endif  // _GLIBCXX_USE_C99_STDINT_TR1
461 #endif  // _GLIBCXX_USE_WEAK_REF
462