2019-08-21 02:28:04 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <type_traits>
|
|
|
|
#include <typeinfo>
|
|
|
|
#include <typeindex>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include <Common/Exception.h>
|
|
|
|
#include <common/demangle.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int BAD_CAST;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** Perform static_cast in release build.
|
|
|
|
* Checks type by comparing typeid and throw an exception in debug build.
|
|
|
|
* The exact match of the type is checked. That is, cast to the ancestor will be unsuccessful.
|
|
|
|
*/
|
|
|
|
template <typename To, typename From>
|
|
|
|
To assert_cast(From && from)
|
|
|
|
{
|
|
|
|
#ifndef NDEBUG
|
|
|
|
try
|
|
|
|
{
|
2019-08-21 02:46:54 +00:00
|
|
|
if constexpr (std::is_pointer_v<To>)
|
|
|
|
{
|
|
|
|
if (typeid(*from) == typeid(std::remove_pointer_t<To>))
|
|
|
|
return static_cast<To>(from);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (typeid(from) == typeid(To))
|
|
|
|
return static_cast<To>(from);
|
|
|
|
}
|
2019-08-21 02:28:04 +00:00
|
|
|
}
|
|
|
|
catch (const std::exception & e)
|
|
|
|
{
|
|
|
|
throw DB::Exception(e.what(), DB::ErrorCodes::BAD_CAST);
|
|
|
|
}
|
|
|
|
|
|
|
|
throw DB::Exception("Bad cast from type " + demangle(typeid(from).name()) + " to " + demangle(typeid(To).name()),
|
|
|
|
DB::ErrorCodes::BAD_CAST);
|
|
|
|
#else
|
|
|
|
return static_cast<To>(from);
|
|
|
|
#endif
|
|
|
|
}
|