Mark constexpr values needed within non-implicitly-capturing lambdas as static
[folly.git] / folly / ExceptionWrapper.cpp
1 /*
2  * Copyright 2017 Facebook, Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *   http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 #include <folly/ExceptionWrapper.h>
17
18 #include <iostream>
19
20 #include <folly/Logging.h>
21
22 namespace folly {
23
24 exception_wrapper::VTable const exception_wrapper::uninit_{
25     &noop_<void, exception_wrapper const*, exception_wrapper*>,
26     &noop_<void, exception_wrapper*, exception_wrapper*>,
27     &noop_<void, exception_wrapper*>,
28     &noop_<void, exception_wrapper const*>,
29     &uninit_type_,
30     &noop_<std::exception const*, exception_wrapper const*>,
31     &noop_<exception_wrapper, exception_wrapper const*>};
32
33 exception_wrapper::VTable const exception_wrapper::ExceptionPtr::ops_{
34     copy_,
35     move_,
36     delete_,
37     throw_,
38     type_,
39     get_exception_,
40     get_exception_ptr_};
41
42 exception_wrapper::VTable const exception_wrapper::SharedPtr::ops_{
43     copy_,
44     move_,
45     delete_,
46     throw_,
47     type_,
48     get_exception_,
49     get_exception_ptr_};
50
51 namespace {
52 std::exception const* get_std_exception_(std::exception_ptr eptr) noexcept {
53   try {
54     std::rethrow_exception(eptr);
55   } catch (const std::exception& ex) {
56     return &ex;
57   } catch (...) {
58     return nullptr;
59   }
60 }
61 }
62
63 exception_wrapper::exception_wrapper(std::exception_ptr ptr) noexcept
64     : exception_wrapper{} {
65   if (ptr) {
66     if (auto e = get_std_exception_(ptr)) {
67       LOG(DFATAL)
68           << "Performance error: Please construct exception_wrapper with a "
69              "reference to the std::exception along with the "
70              "std::exception_ptr.";
71       *this = exception_wrapper{std::move(ptr), *e};
72     } else {
73       Unknown uk;
74       *this = exception_wrapper{ptr, uk};
75     }
76   }
77 }
78
79 [[noreturn]] void exception_wrapper::onNoExceptionError() {
80   std::ios_base::Init ioinit_; // ensure std::cerr is alive
81   std::cerr
82       << "Cannot use `throw_exception` with an empty folly::exception_wrapper"
83       << std::endl;
84   std::terminate();
85 }
86
87 fbstring exceptionStr(exception_wrapper const& ew) {
88   return ew.what();
89 }
90
91 } // folly