stdexcept.cpp revision 262801
1//===------------------------ stdexcept.cpp -------------------------------===//
2//
3//                     The LLVM Compiler Infrastructure
4//
5// This file is dual licensed under the MIT and the University of Illinois Open
6// Source Licenses. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "stdexcept"
11#include "new"
12#include "string"
13#include <cstdlib>
14#include <cstring>
15#include <cstdint>
16#include <cstddef>
17#include "system_error"
18
19#ifndef __has_include
20#define __has_include(inc) 0
21#endif
22
23#ifdef __APPLE__
24#include <cxxabi.h>
25#elif defined(LIBCXXRT) || __has_include(<cxxabi.h>)
26#include <cxxabi.h>
27#endif
28
29// Note:  optimize for size
30
31#if ! defined(_LIBCPP_MSVC)
32#pragma GCC visibility push(hidden)
33#endif
34
35namespace
36{
37
38class __libcpp_nmstr
39{
40private:
41    const char* str_;
42
43    typedef std::size_t unused_t;
44    typedef std::ptrdiff_t count_t;
45
46    static const std::ptrdiff_t offset = static_cast<std::ptrdiff_t>(2*sizeof(unused_t) +
47                                                                       sizeof(count_t));
48
49    count_t& count() const _NOEXCEPT {return (count_t&)(*(str_ - sizeof(count_t)));}
50public:
51    explicit __libcpp_nmstr(const char* msg);
52    __libcpp_nmstr(const __libcpp_nmstr& s) _NOEXCEPT;
53    __libcpp_nmstr& operator=(const __libcpp_nmstr& s) _NOEXCEPT;
54    ~__libcpp_nmstr();
55    const char* c_str() const _NOEXCEPT {return str_;}
56};
57
58__libcpp_nmstr::__libcpp_nmstr(const char* msg)
59{
60    std::size_t len = strlen(msg);
61    str_ = new char[len + 1 + offset];
62    unused_t* c = (unused_t*)str_;
63    c[0] = c[1] = len;
64    str_ += offset;
65    count() = 0;
66    std::memcpy(const_cast<char*>(c_str()), msg, len + 1);
67}
68
69inline
70__libcpp_nmstr::__libcpp_nmstr(const __libcpp_nmstr& s) _NOEXCEPT
71    : str_(s.str_)
72{
73    __sync_add_and_fetch(&count(), 1);
74}
75
76__libcpp_nmstr&
77__libcpp_nmstr::operator=(const __libcpp_nmstr& s) _NOEXCEPT
78{
79    const char* p = str_;
80    str_ = s.str_;
81    __sync_add_and_fetch(&count(), 1);
82    if (__sync_add_and_fetch((count_t*)(p-sizeof(count_t)), count_t(-1)) < 0)
83        delete [] (p-offset);
84    return *this;
85}
86
87inline
88__libcpp_nmstr::~__libcpp_nmstr()
89{
90    if (__sync_add_and_fetch(&count(), count_t(-1)) < 0)
91        delete [] (str_ - offset);
92}
93
94}
95
96#if ! defined(_LIBCPP_MSVC)
97#pragma GCC visibility pop
98#endif
99
100namespace std  // purposefully not using versioning namespace
101{
102
103logic_error::logic_error(const string& msg)
104{
105    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
106    ::new(&s) __libcpp_nmstr(msg.c_str());
107}
108
109logic_error::logic_error(const char* msg)
110{
111    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
112    ::new(&s) __libcpp_nmstr(msg);
113}
114
115logic_error::logic_error(const logic_error& le) _NOEXCEPT
116{
117    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
118    ::new(&s) __libcpp_nmstr((const __libcpp_nmstr&)le.__imp_);
119}
120
121logic_error&
122logic_error::operator=(const logic_error& le) _NOEXCEPT
123{
124    __libcpp_nmstr& s1 = (__libcpp_nmstr&)__imp_;
125    const __libcpp_nmstr& s2 = (const __libcpp_nmstr&)le.__imp_;
126    s1 = s2;
127    return *this;
128}
129
130#if !defined(_LIBCPPABI_VERSION) && !defined(LIBSTDCXX)
131
132logic_error::~logic_error() _NOEXCEPT
133{
134    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
135    s.~__libcpp_nmstr();
136}
137
138const char*
139logic_error::what() const _NOEXCEPT
140{
141    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
142    return s.c_str();
143}
144
145#endif
146
147runtime_error::runtime_error(const string& msg)
148{
149    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
150    ::new(&s) __libcpp_nmstr(msg.c_str());
151}
152
153runtime_error::runtime_error(const char* msg)
154{
155    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
156    ::new(&s) __libcpp_nmstr(msg);
157}
158
159runtime_error::runtime_error(const runtime_error& le) _NOEXCEPT
160{
161    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
162    ::new(&s) __libcpp_nmstr((const __libcpp_nmstr&)le.__imp_);
163}
164
165runtime_error&
166runtime_error::operator=(const runtime_error& le) _NOEXCEPT
167{
168    __libcpp_nmstr& s1 = (__libcpp_nmstr&)__imp_;
169    const __libcpp_nmstr& s2 = (const __libcpp_nmstr&)le.__imp_;
170    s1 = s2;
171    return *this;
172}
173
174#if !defined(_LIBCPPABI_VERSION) && !defined(LIBSTDCXX)
175
176runtime_error::~runtime_error() _NOEXCEPT
177{
178    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
179    s.~__libcpp_nmstr();
180}
181
182const char*
183runtime_error::what() const _NOEXCEPT
184{
185    __libcpp_nmstr& s = (__libcpp_nmstr&)__imp_;
186    return s.c_str();
187}
188
189domain_error::~domain_error() _NOEXCEPT {}
190invalid_argument::~invalid_argument() _NOEXCEPT {}
191length_error::~length_error() _NOEXCEPT {}
192out_of_range::~out_of_range() _NOEXCEPT {}
193
194range_error::~range_error() _NOEXCEPT {}
195overflow_error::~overflow_error() _NOEXCEPT {}
196underflow_error::~underflow_error() _NOEXCEPT {}
197
198#endif
199
200}  // std
201