util/Exception: add more utility functions

This commit is contained in:
Max Kellermann 2017-07-05 17:08:26 +02:00
parent 8679074be7
commit 19358d1c42

View File

@ -30,9 +30,73 @@
#ifndef EXCEPTION_HXX
#define EXCEPTION_HXX
#include "Compiler.h"
#include <exception>
#include <string>
/**
* Throws the specified exception. There is an overload for
* std::exception_ptr which throws the contained exception instead of
* the std::exception_ptr itself.
*/
template<typename T>
gcc_noreturn
inline void
ThrowException(T &&t)
{
throw t;
}
gcc_noreturn
inline void
ThrowException(std::exception_ptr ep)
{
std::rethrow_exception(ep);
}
/**
* Create a nested exception, wrapping #ep inside (a copy of) #t.
*/
template<typename T>
inline std::exception_ptr
NestException(std::exception_ptr ep, T &&t) noexcept
{
try {
std::rethrow_exception(ep);
} catch (...) {
try {
std::throw_with_nested(std::forward<T>(t));
} catch (...) {
return std::current_exception();
}
}
}
/**
* Find an instance of #T in the nested exception chain, and rethrow
* it. Does nothing of no such instance was found.
*/
template<typename T>
inline void
FindRetrowNested(std::exception_ptr ep)
{
try {
std::rethrow_exception(ep);
} catch (const T &t) {
throw;
} catch (const std::exception &e) {
try {
std::rethrow_if_nested(e);
} catch (...) {
FindRetrowNested<T>(std::current_exception());
}
} catch (const std::nested_exception &ne) {
FindRetrowNested<T>(ne.nested_ptr());
} catch (...) {
}
}
/**
* Obtain the full concatenated message of an exception and its nested
* chain.