summaryrefslogtreecommitdiff
path: root/cpp/src/qpid/sys/ExceptionHolder.h
blob: 4bc934cf75c224ce462f5e915e775f86bf3017de (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
#ifndef QPID_EXCEPTIONHOLDER_H
#define QPID_EXCEPTIONHOLDER_H

/*
 *
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 *
 */

#include <boost/shared_ptr.hpp>


namespace qpid {
namespace sys {

struct Raisable {
    virtual ~Raisable() {};
    virtual void raise() const=0;
    virtual std::string what() const=0;
};

/**
 * Holder for exceptions. Allows the thread that notices an error condition to
 * create an exception and store it to be thrown by another thread.
 */
class ExceptionHolder : public Raisable {
  public:
    ExceptionHolder() {}
    // Use default copy & assign.

    /** Take ownership of ex */
    template <class Ex> ExceptionHolder(Ex* ex) { wrap(ex); }
    template <class Ex> ExceptionHolder& operator=(Ex* ex) { wrap(ex); return *this; }

    void raise() const { if (wrapper.get()) wrapper->raise() ; }
    std::string what() const { return wrapper.get() ? wrapper->what() : std::string(); }
    bool empty() const { return !wrapper.get(); }
    operator bool() const { return !empty(); }
    void reset() { wrapper.reset(); }

  private:
    template <class Ex> struct Wrapper : public Raisable {
        Wrapper(Ex* ptr) : exception(ptr) {}
        void raise() const { throw *exception; }
        std::string what() const { return exception->what(); }
        boost::shared_ptr<Ex> exception;
    };
    template <class Ex> void wrap(Ex* ex) { wrapper.reset(new Wrapper<Ex>(ex)); }
    boost::shared_ptr<Raisable> wrapper;
};


}} // namespace qpid::sys


#endif  /*!QPID_EXCEPTIONHOLDER_H*/