1*00b67f09SDavid van Moolenbroek //
2*00b67f09SDavid van Moolenbroek // Automated Testing Framework (atf)
3*00b67f09SDavid van Moolenbroek //
4*00b67f09SDavid van Moolenbroek // Copyright (c) 2007 The NetBSD Foundation, Inc.
5*00b67f09SDavid van Moolenbroek // All rights reserved.
6*00b67f09SDavid van Moolenbroek //
7*00b67f09SDavid van Moolenbroek // Redistribution and use in source and binary forms, with or without
8*00b67f09SDavid van Moolenbroek // modification, are permitted provided that the following conditions
9*00b67f09SDavid van Moolenbroek // are met:
10*00b67f09SDavid van Moolenbroek // 1. Redistributions of source code must retain the above copyright
11*00b67f09SDavid van Moolenbroek // notice, this list of conditions and the following disclaimer.
12*00b67f09SDavid van Moolenbroek // 2. Redistributions in binary form must reproduce the above copyright
13*00b67f09SDavid van Moolenbroek // notice, this list of conditions and the following disclaimer in the
14*00b67f09SDavid van Moolenbroek // documentation and/or other materials provided with the distribution.
15*00b67f09SDavid van Moolenbroek //
16*00b67f09SDavid van Moolenbroek // THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
17*00b67f09SDavid van Moolenbroek // CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
18*00b67f09SDavid van Moolenbroek // INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19*00b67f09SDavid van Moolenbroek // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20*00b67f09SDavid van Moolenbroek // IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
21*00b67f09SDavid van Moolenbroek // DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*00b67f09SDavid van Moolenbroek // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
23*00b67f09SDavid van Moolenbroek // GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*00b67f09SDavid van Moolenbroek // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25*00b67f09SDavid van Moolenbroek // IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26*00b67f09SDavid van Moolenbroek // OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27*00b67f09SDavid van Moolenbroek // IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*00b67f09SDavid van Moolenbroek //
29*00b67f09SDavid van Moolenbroek
30*00b67f09SDavid van Moolenbroek #if defined(HAVE_CONFIG_H)
31*00b67f09SDavid van Moolenbroek #include "bconfig.h"
32*00b67f09SDavid van Moolenbroek #endif
33*00b67f09SDavid van Moolenbroek
34*00b67f09SDavid van Moolenbroek #include <cstdarg>
35*00b67f09SDavid van Moolenbroek #include <cstdio>
36*00b67f09SDavid van Moolenbroek #include <cstring>
37*00b67f09SDavid van Moolenbroek #include <new>
38*00b67f09SDavid van Moolenbroek
39*00b67f09SDavid van Moolenbroek extern "C" {
40*00b67f09SDavid van Moolenbroek #include "../../atf-c/error.h"
41*00b67f09SDavid van Moolenbroek };
42*00b67f09SDavid van Moolenbroek
43*00b67f09SDavid van Moolenbroek #include "exceptions.hpp"
44*00b67f09SDavid van Moolenbroek #include "sanity.hpp"
45*00b67f09SDavid van Moolenbroek
46*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
47*00b67f09SDavid van Moolenbroek // The "system_error" type.
48*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
49*00b67f09SDavid van Moolenbroek
system_error(const std::string & who,const std::string & message,int sys_err)50*00b67f09SDavid van Moolenbroek atf::system_error::system_error(const std::string& who,
51*00b67f09SDavid van Moolenbroek const std::string& message,
52*00b67f09SDavid van Moolenbroek int sys_err) :
53*00b67f09SDavid van Moolenbroek std::runtime_error(who + ": " + message),
54*00b67f09SDavid van Moolenbroek m_sys_err(sys_err)
55*00b67f09SDavid van Moolenbroek {
56*00b67f09SDavid van Moolenbroek }
57*00b67f09SDavid van Moolenbroek
~system_error(void)58*00b67f09SDavid van Moolenbroek atf::system_error::~system_error(void)
59*00b67f09SDavid van Moolenbroek throw()
60*00b67f09SDavid van Moolenbroek {
61*00b67f09SDavid van Moolenbroek }
62*00b67f09SDavid van Moolenbroek
63*00b67f09SDavid van Moolenbroek int
code(void) const64*00b67f09SDavid van Moolenbroek atf::system_error::code(void)
65*00b67f09SDavid van Moolenbroek const
66*00b67f09SDavid van Moolenbroek throw()
67*00b67f09SDavid van Moolenbroek {
68*00b67f09SDavid van Moolenbroek return m_sys_err;
69*00b67f09SDavid van Moolenbroek }
70*00b67f09SDavid van Moolenbroek
71*00b67f09SDavid van Moolenbroek const char*
what(void) const72*00b67f09SDavid van Moolenbroek atf::system_error::what(void)
73*00b67f09SDavid van Moolenbroek const
74*00b67f09SDavid van Moolenbroek throw()
75*00b67f09SDavid van Moolenbroek {
76*00b67f09SDavid van Moolenbroek try {
77*00b67f09SDavid van Moolenbroek if (m_message.length() == 0) {
78*00b67f09SDavid van Moolenbroek m_message = std::string(std::runtime_error::what()) + ": ";
79*00b67f09SDavid van Moolenbroek m_message += ::strerror(m_sys_err);
80*00b67f09SDavid van Moolenbroek }
81*00b67f09SDavid van Moolenbroek
82*00b67f09SDavid van Moolenbroek return m_message.c_str();
83*00b67f09SDavid van Moolenbroek } catch (...) {
84*00b67f09SDavid van Moolenbroek return "Unable to format system_error message";
85*00b67f09SDavid van Moolenbroek }
86*00b67f09SDavid van Moolenbroek }
87*00b67f09SDavid van Moolenbroek
88*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
89*00b67f09SDavid van Moolenbroek // Free functions.
90*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
91*00b67f09SDavid van Moolenbroek
92*00b67f09SDavid van Moolenbroek static
93*00b67f09SDavid van Moolenbroek void
throw_libc_error(atf_error_t err)94*00b67f09SDavid van Moolenbroek throw_libc_error(atf_error_t err)
95*00b67f09SDavid van Moolenbroek {
96*00b67f09SDavid van Moolenbroek PRE(atf_error_is(err, "libc"));
97*00b67f09SDavid van Moolenbroek
98*00b67f09SDavid van Moolenbroek const int ecode = atf_libc_error_code(err);
99*00b67f09SDavid van Moolenbroek const std::string msg = atf_libc_error_msg(err);
100*00b67f09SDavid van Moolenbroek atf_error_free(err);
101*00b67f09SDavid van Moolenbroek throw atf::system_error("XXX", msg, ecode);
102*00b67f09SDavid van Moolenbroek }
103*00b67f09SDavid van Moolenbroek
104*00b67f09SDavid van Moolenbroek static
105*00b67f09SDavid van Moolenbroek void
throw_no_memory_error(atf_error_t err)106*00b67f09SDavid van Moolenbroek throw_no_memory_error(atf_error_t err)
107*00b67f09SDavid van Moolenbroek {
108*00b67f09SDavid van Moolenbroek PRE(atf_error_is(err, "no_memory"));
109*00b67f09SDavid van Moolenbroek
110*00b67f09SDavid van Moolenbroek atf_error_free(err);
111*00b67f09SDavid van Moolenbroek throw std::bad_alloc();
112*00b67f09SDavid van Moolenbroek }
113*00b67f09SDavid van Moolenbroek
114*00b67f09SDavid van Moolenbroek static
115*00b67f09SDavid van Moolenbroek void
throw_unknown_error(atf_error_t err)116*00b67f09SDavid van Moolenbroek throw_unknown_error(atf_error_t err)
117*00b67f09SDavid van Moolenbroek {
118*00b67f09SDavid van Moolenbroek PRE(atf_is_error(err));
119*00b67f09SDavid van Moolenbroek
120*00b67f09SDavid van Moolenbroek static char buf[4096];
121*00b67f09SDavid van Moolenbroek atf_error_format(err, buf, sizeof(buf));
122*00b67f09SDavid van Moolenbroek atf_error_free(err);
123*00b67f09SDavid van Moolenbroek throw std::runtime_error(buf);
124*00b67f09SDavid van Moolenbroek }
125*00b67f09SDavid van Moolenbroek
126*00b67f09SDavid van Moolenbroek void
throw_atf_error(atf_error_t err)127*00b67f09SDavid van Moolenbroek atf::throw_atf_error(atf_error_t err)
128*00b67f09SDavid van Moolenbroek {
129*00b67f09SDavid van Moolenbroek static struct handler {
130*00b67f09SDavid van Moolenbroek const char* m_name;
131*00b67f09SDavid van Moolenbroek void (*m_func)(atf_error_t);
132*00b67f09SDavid van Moolenbroek } handlers[] = {
133*00b67f09SDavid van Moolenbroek { "libc", throw_libc_error },
134*00b67f09SDavid van Moolenbroek { "no_memory", throw_no_memory_error },
135*00b67f09SDavid van Moolenbroek { NULL, throw_unknown_error },
136*00b67f09SDavid van Moolenbroek };
137*00b67f09SDavid van Moolenbroek
138*00b67f09SDavid van Moolenbroek PRE(atf_is_error(err));
139*00b67f09SDavid van Moolenbroek
140*00b67f09SDavid van Moolenbroek handler* h = handlers;
141*00b67f09SDavid van Moolenbroek while (h->m_name != NULL) {
142*00b67f09SDavid van Moolenbroek if (atf_error_is(err, h->m_name)) {
143*00b67f09SDavid van Moolenbroek h->m_func(err);
144*00b67f09SDavid van Moolenbroek UNREACHABLE;
145*00b67f09SDavid van Moolenbroek } else
146*00b67f09SDavid van Moolenbroek h++;
147*00b67f09SDavid van Moolenbroek }
148*00b67f09SDavid van Moolenbroek // XXX: I'm not sure that raising an "unknown" error is a wise thing
149*00b67f09SDavid van Moolenbroek // to do here. The C++ binding is supposed to have feature parity
150*00b67f09SDavid van Moolenbroek // with the C one, so all possible errors raised by the C library
151*00b67f09SDavid van Moolenbroek // should have their counterpart in the C++ library. Still, removing
152*00b67f09SDavid van Moolenbroek // this will require some code auditing that I can't afford at the
153*00b67f09SDavid van Moolenbroek // moment.
154*00b67f09SDavid van Moolenbroek INV(h->m_name == NULL && h->m_func != NULL);
155*00b67f09SDavid van Moolenbroek h->m_func(err);
156*00b67f09SDavid van Moolenbroek UNREACHABLE;
157*00b67f09SDavid van Moolenbroek }
158