/* * Licensed 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. * * Copyright: * 2016-2017, Danny Robson */ #ifndef CRUFT_VK_EXCEPT_HPP #define CRUFT_VK_EXCEPT_HPP #include "./vk.hpp" #include #include #include #include #include namespace cruft::vk { class error : public std::exception { public: static void try_code (VkResult); static void throw_code [[noreturn]] (VkResult); /// invokes a supplied function with the given arguments and tests /// that it indicates success using try_code iff it returns a VkResult, /// else it returns the result of the function. /// /// while it would simplify writing this function to avoid /// non-VkResult functions it simplifies some of the upper layers of /// try_foo functions to allow them anyway (eg, /// vkGetBufferMemoryRequirements is otherwise usable with try_query /// if we ignore the VkResult requirement) template < typename FuncT, typename ...Args > static auto try_func (FuncT &&func, Args &&...args) { if constexpr (std::is_same_v< typename func_traits::return_type, VkResult >) { try_code (func (std::forward (args)...)); return; } else return func (std::forward (args)...); } /// returns a ValueT using an invokable FuncT with arguments /// [Args..., &ValueT] and returns the result after testing for /// success. template static auto try_query (FuncT &&func, Args &&...args) { using ValueT = std::remove_pointer_t< std::tuple_element_t< sizeof... (Args), typename func_traits::argument_types > >; ValueT value; try_func (func, std::forward (args)..., &value); return value; } ///-------------------------------------------------------------------- /// returns a container of values obtained from FuncT /// /// FuncT must take arguments of the form: /// ... FuncT (..., uint32_t *length, ResultT *values) /// and must return the expected length when values is null. /// /// the return type will be checked by try_code if it is a VkResult. /// /// the container will dynamically size so that it can contain all /// available values. template < template class ContainerT = std::vector, typename FuncT, typename ...Args > static auto try_values (FuncT &&func, Args &&...args) { // extract the number of elements available uint32_t expected = 0; try_func (func, args..., &expected, nullptr); // find the type of the last argument, ie. the values we're // requesting using ValueT = std::remove_pointer_t< std::tuple_element_t< sizeof...(Args) + 1, typename func_traits::argument_types > >; ContainerT values (expected); uint32_t found = expected; try_func (func, args..., &found, values.data ()); CHECK_EQ (expected, found); return values; } //--------------------------------------------------------------------- template < typename ReturnT, template class ContainerT = std::vector, typename FuncT, typename ...Args > static auto try_handles (FuncT &&func, Args &&...args) { // extract the number of elements available uint32_t expected = 0; try_func (func, args..., &expected, nullptr); // find the type of the last argument, ie. the values we're // requesting using ValueT = std::remove_pointer_t< std::tuple_element_t< sizeof...(Args) + 1, typename func_traits::argument_types > >; // return the values on the stack temporarily as we may need to // convert them to a seperate result type. ValueT values[expected]; uint32_t found = expected; try_func (func, args..., &found, &values[0]); CHECK_EQ (expected, found); // convert to the requested type return ContainerT { values + 0, values + found, }; } }; //------------------------------------------------------------------------- class invalid_argument : public error { public: invalid_argument (const char *_what) noexcept: m_what (_what) { ; } const char* what (void) const noexcept override { return m_what; } private: const char *m_what; }; //------------------------------------------------------------------------- template class error_code : public error { public: const char* what (void) const noexcept override; }; } #endif