/*
 * Copyright (c) 2018-2020 Atmosphère-NX
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
#pragma once
#include 
#include 
#include 
namespace ams::ddsf {
    #if defined(AMS_BUILD_FOR_DEBUGGING) || defined(AMS_BUILD_FOR_AUDITING)
        #define AMS_DDSF_CASTABLE_TRAITS(__CLASS__, __BASE__)                                                                                           \
            public:                                                                                                                                     \
                static constexpr inline ::ams::ddsf::impl::TypeTag s_ams_ddsf_castable_type_tag{#__CLASS__, __BASE__::s_ams_ddsf_castable_type_tag};    \
                constexpr virtual const ::ams::ddsf::impl::TypeTag &GetTypeTag() const override { return s_ams_ddsf_castable_type_tag; }
    #else
        #define AMS_DDSF_CASTABLE_TRAITS(__CLASS__, __BASE__)                                                                                           \
            public:                                                                                                                                     \
                static constexpr inline ::ams::ddsf::impl::TypeTag s_ams_ddsf_castable_type_tag{__BASE__::s_ams_ddsf_castable_type_tag};                \
                constexpr virtual const ::ams::ddsf::impl::TypeTag &GetTypeTag() const override { return s_ams_ddsf_castable_type_tag; }
    #endif
    class ICastable {
        private:
            constexpr virtual const impl::TypeTag &GetTypeTag() const = 0;
            template
            constexpr ALWAYS_INLINE void AssertCastableTo() const {
                AMS_ASSERT(this->IsCastableTo());
            }
        public:
            template
            constexpr bool IsCastableTo() const {
                return this->GetTypeTag().DerivesFrom(T::s_ams_ddsf_castable_type_tag);
            }
            template
            constexpr T &SafeCastTo() {
                this->AssertCastableTo();
                return static_cast(*this);
            }
            template
            constexpr const T &SafeCastTo() const {
                this->AssertCastableTo();
                return static_cast(*this);
            }
            template
            constexpr T *SafeCastToPointer() {
                this->AssertCastableTo();
                return static_cast(this);
            }
            template
            constexpr const T *SafeCastToPointer() const {
                this->AssertCastableTo();
                return static_cast(this);
            }
            #if defined(AMS_BUILD_FOR_AUDITING) || defined(AMS_BUILD_FOR_DEBUGGING)
                constexpr const char *GetClassName() const {
                    return this->GetTypeTag().GetClassName();
                }
            #endif
    };
}