// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.

#pragma once
#define HAVE_IRCD_CTX_EXCEPTION_HANDLER_H

namespace ircd::ctx
{
	struct exception_handler;
}

/// An instance of exception_handler must be present to allow a context
/// switch inside a catch block. This is due to ABI limitations that stack
/// exceptions with thread-local assumptions and don't expect catch blocks
/// on the same thread to interleave when we switch the stack.
///
/// We first increment the refcount for the caught exception so it remains
/// intuitively accessible for the rest of the catch block. Then the presence
/// of this object makes the ABI believe the catch block has ended.
///
/// The exception cannot then be rethrown. DO NOT RETHROW THE EXCEPTION.
///
struct ircd::ctx::exception_handler
:std::exception_ptr
{
	static uint uncaught_exceptions(const uint &) noexcept;
	static uint uncaught_exceptions() noexcept;
	static void end_catch() noexcept;

  public:
	exception_handler() noexcept;
	exception_handler(exception_handler &&) = delete;
	exception_handler(const exception_handler &) = delete;
	exception_handler &operator=(exception_handler &&) = delete;
	exception_handler &operator=(const exception_handler &) = delete;
};