2022-12-13 10:39:57 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <BAN/Formatter.h>
|
|
|
|
|
2022-12-13 12:10:50 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#if defined(__is_kernel)
|
2023-01-09 14:56:20 +02:00
|
|
|
#include <kernel/Panic.h>
|
2023-02-01 21:05:44 +02:00
|
|
|
#define MUST(error) ({ auto e = error; if (e.is_error()) Kernel::panic("{}", e.get_error()); e.value(); })
|
|
|
|
#define ASSERT(cond) do { if (!(cond)) Kernel::panic("ASSERT("#cond") failed"); } while(false)
|
2022-12-13 10:39:57 +02:00
|
|
|
#else
|
|
|
|
#error "NOT IMPLEMENTED"
|
|
|
|
#endif
|
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
#define TRY(error) ({ auto e = error; if (e.is_error()) return e.get_error(); e.value(); })
|
2022-12-13 10:39:57 +02:00
|
|
|
|
2023-01-17 11:38:16 +02:00
|
|
|
namespace BAN
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
|
|
|
|
2023-01-17 11:38:16 +02:00
|
|
|
class Error
|
|
|
|
{
|
|
|
|
public:
|
2023-02-01 21:05:44 +02:00
|
|
|
static Error from_string(const char* message)
|
2023-01-17 11:38:16 +02:00
|
|
|
{
|
|
|
|
Error result;
|
|
|
|
strncpy(result.m_message, message, sizeof(m_message));
|
|
|
|
result.m_message[sizeof(result.m_message) - 1] = '\0';
|
|
|
|
result.m_error_code = 0xFF;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
uint8_t get_error_code() const { return m_error_code; }
|
|
|
|
const char* get_message() const { return m_message; }
|
2023-01-17 11:38:16 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
char m_message[128];
|
|
|
|
uint8_t m_error_code;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class ErrorOr
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ErrorOr(const T& value) : m_has_error(false) { m_data = (void*)new T(value); }
|
|
|
|
ErrorOr(const Error& error) : m_has_error(true) { m_data = (void*)new Error(error); }
|
2023-02-01 21:05:44 +02:00
|
|
|
template<typename S> ErrorOr(const ErrorOr<S>& other) : ErrorOr(other.get_error()) {}
|
|
|
|
~ErrorOr() { is_error() ? (delete reinterpret_cast<Error*>(m_data)) : (delete reinterpret_cast<T*>(m_data)); }
|
2023-01-17 11:38:16 +02:00
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
bool is_error() const { return m_has_error; }
|
|
|
|
const Error& get_error() const { return *reinterpret_cast<Error*>(m_data); }
|
|
|
|
T& value() { return *reinterpret_cast<T*>(m_data); }
|
2023-01-17 11:38:16 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
bool m_has_error = false;
|
|
|
|
void* m_data = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
class ErrorOr<void>
|
|
|
|
{
|
|
|
|
public:
|
2023-02-10 01:06:18 +02:00
|
|
|
ErrorOr() { }
|
|
|
|
ErrorOr(const Error& error) : m_error(error), m_has_error(true) { }
|
|
|
|
~ErrorOr() { }
|
2022-12-13 10:39:57 +02:00
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
bool is_error() const { return m_has_error; }
|
|
|
|
const Error& get_error() const { return m_error; }
|
|
|
|
void value() { }
|
2022-12-13 10:39:57 +02:00
|
|
|
|
2023-01-17 11:38:16 +02:00
|
|
|
private:
|
|
|
|
Error m_error;
|
|
|
|
bool m_has_error = false;
|
|
|
|
};
|
2022-12-13 10:39:57 +02:00
|
|
|
|
2023-01-17 11:38:16 +02:00
|
|
|
}
|
2022-12-13 10:39:57 +02:00
|
|
|
|
|
|
|
namespace BAN::Formatter
|
|
|
|
{
|
2022-12-27 19:55:07 +02:00
|
|
|
template<typename F>
|
|
|
|
void print_argument_impl(F putc, const Error& error, const ValueFormat&)
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
if (error.get_error_code() == 0xFF)
|
|
|
|
print(putc, error.get_message());
|
2022-12-13 10:39:57 +02:00
|
|
|
else
|
2023-02-01 21:05:44 +02:00
|
|
|
print(putc, "{} ({})", error.get_message(), error.get_error_code());
|
2022-12-13 10:39:57 +02:00
|
|
|
}
|
2023-01-17 11:38:16 +02:00
|
|
|
}
|