2016-02-26 13:39:01 +11:00
|
|
|
/*
|
|
|
|
* 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:
|
2017-09-06 13:41:01 +10:00
|
|
|
* 2016-2017, Danny Robson <danny@nerdcruft.net>
|
2016-02-26 13:39:01 +11:00
|
|
|
*/
|
|
|
|
|
2017-05-26 16:30:48 +10:00
|
|
|
#ifndef CRUFT_VK_EXCEPT_HPP
|
|
|
|
#define CRUFT_VK_EXCEPT_HPP
|
2016-02-24 11:11:41 +11:00
|
|
|
|
|
|
|
#include "./vk.hpp"
|
|
|
|
|
2017-09-06 13:41:01 +10:00
|
|
|
#include <cruft/util/debug.hpp>
|
|
|
|
#include <cruft/util/types/traits.hpp>
|
|
|
|
|
2016-10-14 18:01:49 +11:00
|
|
|
#include <exception>
|
2017-09-06 12:12:10 +10:00
|
|
|
#include <functional>
|
|
|
|
#include <type_traits>
|
|
|
|
|
2016-02-24 11:11:41 +11:00
|
|
|
|
2017-05-26 16:30:48 +10:00
|
|
|
namespace cruft::vk {
|
2016-02-24 11:11:41 +11:00
|
|
|
class error : public std::exception {
|
|
|
|
public:
|
|
|
|
static void try_code (VkResult);
|
|
|
|
static void throw_code [[noreturn]] (VkResult);
|
2017-09-06 12:12:10 +10:00
|
|
|
|
|
|
|
|
|
|
|
/// invokes a supplied function with the given arguments and tests
|
2017-09-07 12:48:47 +10:00
|
|
|
/// 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)
|
2017-09-06 12:12:10 +10:00
|
|
|
template <
|
|
|
|
typename FuncT,
|
|
|
|
typename ...Args
|
|
|
|
>
|
2017-09-07 12:48:47 +10:00
|
|
|
static auto
|
2017-09-06 12:12:10 +10:00
|
|
|
try_func (FuncT &&func, Args &&...args)
|
|
|
|
{
|
2017-09-07 12:48:47 +10:00
|
|
|
if constexpr (std::is_same_v<
|
|
|
|
typename func_traits<FuncT>::return_type,
|
|
|
|
VkResult
|
|
|
|
>)
|
|
|
|
{
|
|
|
|
try_code (func (std::forward<Args> (args)...));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return func (std::forward<Args> (args)...);
|
2017-09-06 12:12:10 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-07 12:49:58 +10:00
|
|
|
/// returns a ValueT using an invokable FuncT with arguments
|
2017-09-06 12:12:10 +10:00
|
|
|
/// [Args..., &ValueT] and returns the result after testing for
|
|
|
|
/// success.
|
2017-09-07 12:49:58 +10:00
|
|
|
template <typename FuncT, typename ...Args>
|
|
|
|
static auto
|
2017-09-06 12:12:10 +10:00
|
|
|
try_query (FuncT &&func, Args &&...args)
|
|
|
|
{
|
2017-09-07 12:49:58 +10:00
|
|
|
using ValueT = std::remove_pointer_t<
|
|
|
|
std::tuple_element_t<
|
|
|
|
sizeof... (Args),
|
|
|
|
typename func_traits<FuncT>::argument_types
|
|
|
|
>
|
|
|
|
>;
|
|
|
|
|
2017-09-06 12:12:10 +10:00
|
|
|
ValueT value;
|
|
|
|
try_func (func, std::forward<Args> (args)..., &value);
|
|
|
|
return value;
|
|
|
|
}
|
2017-09-06 13:41:01 +10:00
|
|
|
|
|
|
|
|
2017-09-09 13:44:24 +10:00
|
|
|
///--------------------------------------------------------------------
|
|
|
|
/// 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.
|
2017-09-06 13:41:01 +10:00
|
|
|
template <
|
2017-09-06 15:21:44 +10:00
|
|
|
template <typename...> class ContainerT = std::vector,
|
2017-09-06 13:41:01 +10:00
|
|
|
typename FuncT,
|
|
|
|
typename ...Args
|
|
|
|
>
|
|
|
|
static auto
|
2017-09-09 15:23:19 +10:00
|
|
|
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<FuncT>::argument_types
|
|
|
|
>
|
|
|
|
>;
|
|
|
|
|
|
|
|
ContainerT<ValueT> values (expected);
|
|
|
|
uint32_t found = expected;
|
|
|
|
try_func (func, args..., &found, values.data ());
|
|
|
|
CHECK_EQ (expected, found);
|
|
|
|
return values;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
template <
|
|
|
|
typename ReturnT,
|
|
|
|
template <typename,typename...> class ContainerT = std::vector,
|
|
|
|
typename FuncT,
|
|
|
|
typename ...Args
|
|
|
|
>
|
|
|
|
static auto
|
|
|
|
try_handles (FuncT &&func, Args &&...args)
|
2017-09-06 13:41:01 +10:00
|
|
|
{
|
2017-09-09 13:44:24 +10:00
|
|
|
// extract the number of elements available
|
2017-09-06 13:41:01 +10:00
|
|
|
uint32_t expected = 0;
|
|
|
|
try_func (func, args..., &expected, nullptr);
|
|
|
|
|
2017-09-09 13:44:24 +10:00
|
|
|
// find the type of the last argument, ie. the values we're
|
|
|
|
// requesting
|
2017-09-06 15:21:44 +10:00
|
|
|
using ValueT = std::remove_pointer_t<
|
2017-09-06 13:41:01 +10:00
|
|
|
std::tuple_element_t<
|
|
|
|
sizeof...(Args) + 1,
|
|
|
|
typename func_traits<FuncT>::argument_types
|
|
|
|
>
|
|
|
|
>;
|
|
|
|
|
2017-09-09 13:44:24 +10:00
|
|
|
// return the values on the stack temporarily as we may need to
|
|
|
|
// convert them to a seperate result type.
|
2017-09-06 15:21:44 +10:00
|
|
|
ValueT values[expected];
|
2017-09-07 12:50:16 +10:00
|
|
|
uint32_t found = expected;
|
|
|
|
try_func (func, args..., &found, &values[0]);
|
2017-09-06 13:41:01 +10:00
|
|
|
CHECK_EQ (expected, found);
|
2017-09-06 15:21:44 +10:00
|
|
|
|
2017-09-09 13:44:24 +10:00
|
|
|
// convert to the requested type
|
2017-09-09 15:23:19 +10:00
|
|
|
return ContainerT<ReturnT> {
|
|
|
|
values + 0,
|
|
|
|
values + found,
|
2017-09-06 15:21:44 +10:00
|
|
|
};
|
2017-09-06 13:41:01 +10:00
|
|
|
}
|
2016-02-24 11:11:41 +11:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-09-09 13:44:24 +10:00
|
|
|
//-------------------------------------------------------------------------
|
2017-09-05 17:20:17 +10:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-09-09 13:44:24 +10:00
|
|
|
//-------------------------------------------------------------------------
|
2016-02-24 11:11:41 +11:00
|
|
|
template <VkResult>
|
|
|
|
class error_code : public error {
|
|
|
|
public:
|
|
|
|
const char* what (void) const noexcept override;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-09-09 15:23:19 +10:00
|
|
|
|
2016-02-24 11:11:41 +11:00
|
|
|
#endif
|