Compare commits

...

77 Commits

Author SHA1 Message Date
Brett ff2d77a1cd this is a breaking change (logging) 2025-03-02 21:21:19 -05:00
Brett dd030a9b5b silly code 2025-03-01 21:56:57 -05:00
Brett 21f1e66bff logging files 2025-03-01 20:44:25 -05:00
Brett be721a4e52 slight change, quotes around strings 2025-02-28 18:08:52 -05:00
Brett ecd3d1a701 i think argparse is done 2025-02-28 18:08:20 -05:00
Brett 8902e17b40 fix issue with compiling on clang, as_const now works with tuple types 2025-02-28 17:10:47 -05:00
Brett bac63ae815 flatten, and flatten recursive. as_const doesn't work. 2025-02-28 12:39:13 -05:00
Brett e48fdf0c86 flatten partially works 2025-02-28 02:29:30 -05:00
Brett d8f943ba62 chhange 2025-02-27 15:19:50 -05:00
Brett 58ea39be08 i am now working on flatten 2025-02-27 15:17:59 -05:00
Brett 6935ae4785 this doesn't work rn 2025-02-27 01:53:22 -05:00
Brett 17e6b507ea parsing 2025-02-25 16:29:36 -05:00
Brett 0913208e6b help with choices 2025-02-25 12:34:14 -05:00
Brett 1b6d23fad4 more help 2025-02-25 02:14:19 -05:00
Brett fc27d7503a some help 2025-02-24 22:22:21 -05:00
Brett 34184d46a3 need to update add 2025-02-24 18:28:38 -05:00
Brett 7129c929eb some update 2025-02-23 23:41:34 -05:00
Brett 8b03dda1fe some help 2025-02-22 18:39:51 -05:00
Brett f8ed21fda5 more tests 2025-02-22 13:38:36 -05:00
Brett 30f975e165 more tests, i think most things work now? 2025-02-21 16:37:05 -05:00
Brett f403e8a69b minor work 2025-02-20 23:19:49 -05:00
Brett e8e891d4fc a few more tests 2025-02-20 00:50:49 -05:00
Brett 389762e2ae positionals are broken because im silly 2025-02-20 00:47:27 -05:00
Brett 188e9dba88 postionals may work now? 2025-02-19 20:52:55 -05:00
Brett 5f9ea32671 idk what changed 2025-02-19 16:29:17 -05:00
Brett c23759ac6d fix a lot of bugs, current testing works 2025-02-19 13:28:58 -05:00
Brett 56611d5aef begin testing, flag work. something is really broken though 2025-02-19 02:26:31 -05:00
Brett e0d36269bf i made some changes 2025-02-18 01:32:26 -05:00
Brett 96e5343d02 more work on argparse 2025-02-18 00:46:30 -05:00
Brett 7dc19efbaa more metaprogramming fun 2025-02-17 21:43:09 -05:00
Brett 735371b7bd silly works now i need to bed" 2025-02-17 02:20:40 -05:00
Brett fe6ce712e7 partial solution 2025-02-17 01:56:27 -05:00
Brett a78ad58479 gotta think of a way of handling the whole "templates are silly" thing 2025-02-17 01:47:42 -05:00
Brett 6e5caf3ac5 good work on argparse 2025-02-16 23:22:00 -05:00
Brett d7373ac832 more argparse work 2025-02-13 17:47:27 -05:00
Brett 31b28c7787 usages in subparsers 2025-02-13 14:04:39 -05:00
Brett 89d95dfec4 working on subparsers 2025-02-13 13:59:59 -05:00
Brett 44a57e5ec2 i am tired 2025-02-13 01:53:21 -05:00
Brett 0b2dad0dda forgot about . in files 2025-02-12 22:18:15 -05:00
Brett 174b46ae94 path_helper file provides base_name of a path. Argparse working on 2025-02-12 22:17:04 -05:00
Brett 457dd5203b hi 2025-02-12 19:42:50 -05:00
Brett a437935ab0 Argparse v2 breaking change 2025-02-12 15:43:54 -05:00
Brett 3726f6840f starting arge parse 2025-02-12 02:54:22 -05:00
Brett 02b8f54c7e expected should inherit from itself when not copy constructable 2025-01-28 00:35:08 -05:00
Brett eb73a6e189 revert change that breaks graphics lib 2025-01-28 00:00:47 -05:00
Brett baa5952666 decltype(auto) on the select method 2025-01-21 21:10:10 -05:00
Brett 74c1010118 fix missing paren 2025-01-17 14:48:50 -05:00
Brett 28cbc89840 hi 2025-01-17 14:30:55 -05:00
Brett d436f1b93a mem test 2025-01-17 12:19:06 -05:00
Brett 1aa7f12c0f fix ptr stuff 2025-01-16 19:47:13 -05:00
Brett 41c6dea002 ptr stuff seems to wrok 2025-01-16 19:33:12 -05:00
Brett 2e6abb8013 breaking changes to imports, adding pointer helpers 2025-01-16 15:24:08 -05:00
Brett eca09acc01 static function checks 2025-01-15 19:08:02 -05:00
Brett 4c462dff38 minor change 2025-01-13 14:43:32 -05:00
Brett 8133553ed8 make write profile work 2025-01-11 17:58:28 -05:00
Brett 9860845831 add zip for single iterator pairs 2025-01-07 13:51:18 -05:00
Brett c953ef3544 what's different? 2025-01-07 13:49:32 -05:00
Brett 1798980ac6 hash_value on vec 2024-12-10 00:33:04 -05:00
Brett d1a9aab859 make byte functions more concise 2024-12-09 16:00:08 -05:00
Brett d32b5d398a forgot a cast 2024-11-15 12:54:34 -05:00
Brett 0431106b7e vectors now use correct ops + ret is statically calculated now. vec_cast is a thing 2024-11-15 12:53:27 -05:00
Brett 1c931b2515 make this compile 2024-11-11 18:19:32 -05:00
Brett 05e5fcf7f1 binary tree 2024-11-11 16:47:18 -05:00
Brett 7bf24a11a5 initiailizer list now works for dynamic spans 2024-11-08 19:22:06 -05:00
Brett f7ef78f351 ranges now accept container 2024-11-08 19:03:29 -05:00
Tri11Paragon ea16aa3847 fixes for windows 2024-10-31 19:28:43 -07:00
Brett e81f590f5e string changes, add path seperator 2024-10-21 15:02:13 -04:00
Brett 3003e424e1 make enumerate use distance 2024-10-05 14:49:53 -04:00
Brett a9f5b9e97d very silly const 2024-10-05 14:48:04 -04:00
Brett 7656e43e85 data in matrix 2024-10-05 14:40:36 -04:00
Brett ce4e1807de make vectors actually have bipolar function 2024-10-05 01:13:24 -04:00
Brett dafbe6ea8b iterators with begin(), end(). vector / matrix bipolar 2024-10-05 01:11:11 -04:00
Brett ffeab29e5f streamable object to string function in logging 2024-10-04 14:51:27 -04:00
Brett 463e36e48d why 2024-10-04 13:47:10 -04:00
Brett 15e1b263a4 silly 2024-10-04 13:45:35 -04:00
Brett ff5ba4a667 types for compile 2024-10-04 13:41:27 -04:00
Brett 7fa4fd0af2 some ansi codes 2024-10-03 19:55:04 -04:00
45 changed files with 5245 additions and 1589 deletions

View File

@ -1,6 +1,6 @@
cmake_minimum_required(VERSION 3.20)
include(cmake/color.cmake)
set(BLT_VERSION 2.0.4)
set(BLT_VERSION 5.0.0)
set(BLT_TARGET BLT)
@ -17,6 +17,7 @@ option(BUILD_PROFILING "Build the BLT profiler extension" ON)
option(BUILD_FS "Build the BLT FS utilities including the NBT + eNBT extension" ON)
option(BUILD_PARSE "Build the BLT parsers" ON)
option(BUILD_FORMAT "Build the BLT formatters" ON)
option(BUILD_LOGGING "Build the BLT logging utilities" ON)
option(BUILD_TESTS "Build the BLT test set" OFF)
@ -70,8 +71,14 @@ if (${BUILD_FORMAT})
message(STATUS "Building ${Yellow}format${ColourReset} cxx files")
file(GLOB_RECURSE FORMAT_FILES "${CMAKE_CURRENT_SOURCE_DIR}/src/blt/format/*.cpp")
else ()
set(FORMAT_FILES ""
include/blt/std/iterator.h)
set(FORMAT_FILES "")
endif ()
if (${BUILD_LOGGING})
message(STATUS "Building ${Yellow}logging${ColourReset} cxx files")
file(GLOB_RECURSE LOGGING_FILES "${CMAKE_CURRENT_SOURCE_DIR}/src/blt/logging/*.cpp")
else ()
set(PARSE_FILES "")
endif ()
if (EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/libraries/parallel-hashmap)
@ -93,7 +100,7 @@ endif ()
include_directories(include/)
include_directories(${CMAKE_CURRENT_BINARY_DIR}/config/)
add_library(${BLT_TARGET} ${STD_FILES} ${PROFILING_FILES} ${FS_FILES} ${PARSE_FILES} ${FORMAT_FILES})
add_library(${BLT_TARGET} ${STD_FILES} ${PROFILING_FILES} ${FS_FILES} ${PARSE_FILES} ${FORMAT_FILES} ${LOGGING_FILES})
string(REPLACE "+" "\\+" escaped_source ${CMAKE_CURRENT_SOURCE_DIR})
string(APPEND escaped_source "/src/blt/.*/")
@ -148,8 +155,9 @@ install(TARGETS ${BLT_TARGET}
LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
macro(blt_add_project name source type)
macro(blt_add_test name source type)
message("Adding project ${name} of type ${type}" DEBUG)
project(${name}-${type})
add_executable(${name}-${type} ${source})
@ -194,7 +202,9 @@ endmacro()
if (${BUILD_TESTS})
message("Building tests for version ${BLT_VERSION}")
blt_add_project(blt-iterator tests/iterator_tests.cpp test)
blt_add_test(blt_iterator tests/iterator_tests.cpp test)
blt_add_test(blt_argparse tests/argparse_tests.cpp test)
blt_add_test(blt_logging tests/logger_tests.cpp test)
message("Built tests")
endif ()

View File

@ -1,4 +1,4 @@
#!/usr/bin/python3
#!python3
import subprocess
import argparse

View File

@ -1,74 +0,0 @@
#!/usr/bin/python3
import subprocess
#---------------------------------------
# CONFIG
#---------------------------------------
VERSION_BEGIN_STR = "set(BLT_VERSION "
VERSION_END_STR = ")"
#---------------------------------------
# DO NOT TOUCH
#---------------------------------------
type = input("What kind of commit is this ((M)ajor, (m)inor, (p)atch)? ")
def load_cmake():
cmake_file = open("CMakeLists.txt", 'r')
cmake_text = cmake_file.read()
cmake_file.close()
return cmake_text
def write_cmake(cmake_text):
cmake_file = open("CMakeLists.txt", 'w')
cmake_file.write(cmake_text)
cmake_file.close()
def get_version(cmake_text):
begin = cmake_text.find(VERSION_BEGIN_STR) + len(find_text)
end = cmake_text.find(VERSION_END_STR, begin)
return (cmake_text[begin:end], begin, end)
def split_version(cmake_text):
version, begin, end = get_version(cmake_text)
version_parts = version.split('.')
return (version_parts, begin, end)
def recombine(cmake_text, version_parts, begin, end):
constructed_version = version_parts[0] + '.' + version_parts[1] + '.' + version_parts[2]
constructed_text_begin = cmake_text[0:begin]
constrcuted_text_end = cmake_text[end::]
return constructed_text_begin + constructed_version + constrcuted_text_end
def inc_major(cmake_text):
version_parts, begin, end = split_version(cmake_text)
version_parts[0] = str(int(version_parts[0]) + 1)
return recombine(cmake_text, version_parts, begin, end)
def inc_minor(cmake_text):
version_parts, begin, end = split_version(cmake_text)
version_parts[1] = str(int(version_parts[1]) + 1)
return recombine(cmake_text, version_parts, begin, end)
def inc_patch(cmake_text):
version_parts, begin, end = split_version(cmake_text)
version_parts[2] = str(int(version_parts[2]) + 1)
return recombine(cmake_text, version_parts, begin, end)
if type.startswith('M'):
print("Selected major")
write_cmake(inc_major(load_cmake()))
elif type.startswith('m'):
print("Selected minor")
write_cmake(inc_minor(load_cmake()))
elif type.startswith('p') or type.startswith('P') or len(type) == 0:
print("Selected patch")
write_cmake(inc_patch(load_cmake()))
#subprocess.call("./py_commit_helper.sh")
subprocess.call("git", "add", "*")
subprocess.call("git", "commit")
subprocess.call("sh -e 'git remote | xargs -L1 git push --all'")

View File

@ -51,4 +51,12 @@
#error Filesystem ops not supported!\
#endif
#if defined(WIN32) || defined(_WIN32) || defined(__WIN32) && !defined(__CYGWIN__)
#define BLT_OS_WINDOWS
#elif defined(__linux__) || defined(__unix__)
#define BLT_OS_LINUX
#else
#define BLT_OS_UNKNOWN
#endif
#endif //BLT_COMPATIBILITY_H

View File

@ -0,0 +1,33 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_FS_PATH_HELPER_H
#define BLT_FS_PATH_HELPER_H
#include <string>
#include <string_view>
namespace blt::fs
{
std::string base_name(const std::string& str);
std::string_view base_name_sv(std::string_view str);
}
#endif //BLT_FS_PATH_HELPER_H

View File

@ -27,10 +27,43 @@
#include <blt/meta/iterator.h>
#include <functional>
#include <optional>
#include <blt/meta/type_traits.h>
namespace blt::iterator
{
template<typename Derived>
namespace detail
{
template<typename T>
static auto forward_as_tuple(T&& t) -> decltype(auto)
{
using Decay = std::decay_t<T>;
static_assert(!(meta::is_tuple_v<Decay> || meta::is_pair_v<Decay>), "Tuple or pair passed to forward_as_tuple! Must not be a tuple!");
if constexpr (std::is_lvalue_reference_v<T>)
{
return std::forward_as_tuple(std::forward<T>(t));
}
else
{
return std::make_tuple(std::forward<T>(t));
}
}
template <typename Tuple>
static auto ensure_tuple(Tuple&& tuple) -> decltype(auto)
{
using Decay = std::decay_t<Tuple>;
if constexpr (meta::is_tuple_v<Decay> || meta::is_pair_v<Decay>)
{
return std::forward<Tuple>(tuple);
}
else
{
return forward_as_tuple(std::forward<Tuple>(tuple));
}
}
}
template <typename Derived>
struct base_wrapper
{
base_wrapper operator++(int)
@ -39,355 +72,454 @@ namespace blt::iterator
++*this;
return tmp;
}
base_wrapper operator--(int)
{
static_assert(meta::is_bidirectional_or_better_category_v<typename Derived::iterator_category>, "Iterator must allow random access");
static_assert(meta::is_bidirectional_or_better_category_v<typename Derived::iterator_category>,
"Iterator must allow bidirectional access");
auto tmp = *this;
--*this;
return tmp;
}
auto operator[](blt::ptrdiff_t n) const
auto operator[](ptrdiff_t n) const
{
static_assert(meta::is_random_access_iterator_category_v<typename Derived::iterator_category>, "Iterator must allow random access");
static_assert(meta::is_random_access_iterator_category_v<typename Derived::iterator_category>,
"Iterator must allow bidirectional access");
return *(*this + n);
}
friend base_wrapper operator+(blt::ptrdiff_t n, const base_wrapper& a)
friend base_wrapper operator+(ptrdiff_t n, const base_wrapper& a)
{
return a + n;
}
friend bool operator<(const base_wrapper& a, const base_wrapper& b)
{
static_assert(meta::is_random_access_iterator_category_v<typename Derived::iterator_category>, "Iterator must allow random access");
return b - a > 0;
}
friend bool operator>(const base_wrapper& a, const base_wrapper& b)
{
static_assert(meta::is_random_access_iterator_category_v<typename Derived::iterator_category>, "Iterator must allow random access");
return b < a;
}
friend bool operator>=(const base_wrapper& a, base_wrapper& b)
{
static_assert(meta::is_random_access_iterator_category_v<typename Derived::iterator_category>, "Iterator must allow random access");
return !(a < b); // NOLINT
}
friend bool operator<=(const base_wrapper& a, const base_wrapper& b)
{
static_assert(meta::is_random_access_iterator_category_v<typename Derived::iterator_category>, "Iterator must allow random access");
return !(a > b); // NOLINT
}
friend bool operator==(const base_wrapper& a, const base_wrapper& b)
{
return static_cast<const Derived&>(a).base() == static_cast<const Derived&>(b).base();
}
friend bool operator!=(const base_wrapper& a, const base_wrapper& b)
{
return !(static_cast<const Derived&>(a).base() == static_cast<const Derived&>(b).base()); // NOLINT
}
};
template<typename Iter, typename Derived, bool dereference = false>
struct passthrough_wrapper : public base_wrapper<Derived>
template <typename Iter, typename Derived, bool dereference = false>
struct passthrough_wrapper : base_wrapper<Derived>
{
public:
explicit passthrough_wrapper(Iter iter): iter(std::move(iter))
{}
auto base() const
{
return iter;
}
friend blt::ptrdiff_t operator-(const passthrough_wrapper& a, const passthrough_wrapper& b)
{
return a.base() - b.base();
}
protected:
mutable Iter iter;
explicit passthrough_wrapper(Iter iter): iter(std::move(iter))
{
}
auto base() const
{
return iter;
}
friend ptrdiff_t operator-(const passthrough_wrapper& a, const passthrough_wrapper& b)
{
return a.base() - b.base();
}
protected:
mutable Iter iter;
};
template<typename Iter, typename Derived>
struct passthrough_wrapper<Iter, Derived, true> : public passthrough_wrapper<Iter, Derived>
template <typename Iter, typename Derived>
struct passthrough_wrapper<Iter, Derived, true> : passthrough_wrapper<Iter, Derived>
{
using passthrough_wrapper<Iter, Derived>::passthrough_wrapper;
meta::deref_return_t<Iter> operator*() const
{
return *this->iter;
}
};
template<typename Iter, typename Derived>
template <typename Iter, typename Derived>
class deref_only_wrapper : public passthrough_wrapper<Iter, Derived, false>
{
public:
using passthrough_wrapper<Iter, Derived, false>::passthrough_wrapper;
deref_only_wrapper& operator++()
{
++this->iter;
return *this;
}
deref_only_wrapper& operator--()
{
--this->iter;
return *this;
}
deref_only_wrapper& operator+(blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
this->iter = this->iter + n;
return *this;
}
deref_only_wrapper& operator-(blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
this->iter = this->iter - n;
return *this;
}
public:
using passthrough_wrapper<Iter, Derived, false>::passthrough_wrapper;
deref_only_wrapper& operator++()
{
++this->iter;
return *this;
}
deref_only_wrapper& operator--()
{
--this->iter;
return *this;
}
deref_only_wrapper& operator+(blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
this->iter = this->iter + n;
return *this;
}
deref_only_wrapper& operator-(blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
this->iter = this->iter - n;
return *this;
}
};
template<typename Iter, typename Func>
template <typename Iter, typename Func>
class map_wrapper : public deref_only_wrapper<Iter, map_wrapper<Iter, Func>>
{
public:
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = std::invoke_result_t<Func, meta::deref_return_t<Iter>>;
using difference_type = blt::ptrdiff_t;
using pointer = value_type;
using reference = value_type;
map_wrapper(Iter iter, Func func):
deref_only_wrapper<Iter, map_wrapper<Iter, Func>>(std::move(iter)), func(std::move(func))
{}
reference operator*() const
{
return func(*this->iter);
}
private:
Func func;
public:
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = std::invoke_result_t<Func, meta::deref_return_t<Iter>>;
using difference_type = blt::ptrdiff_t;
using pointer = value_type;
using reference = value_type;
map_wrapper(Iter iter, Func func):
deref_only_wrapper<Iter, map_wrapper<Iter, Func>>(std::move(iter)), func(std::move(func))
{
}
reference operator*() const
{
return func(*this->iter);
}
private:
Func func;
};
template<typename Iter, typename Pred>
template <typename Iter, typename Pred>
class filter_wrapper : public deref_only_wrapper<Iter, filter_wrapper<Iter, Pred>>
{
public:
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = std::conditional_t<
std::is_reference_v<meta::deref_return_t<Iter>>,
std::optional<std::reference_wrapper<std::remove_reference_t<meta::deref_return_t<Iter>>>>,
std::optional<meta::deref_return_t<Iter>>>;
using difference_type = blt::ptrdiff_t;
using pointer = value_type;
using reference = value_type;
filter_wrapper(Iter iter, Pred func):
deref_only_wrapper<Iter, filter_wrapper<Iter, Pred>>(std::move(iter)), func(std::move(func))
{}
reference operator*() const
public:
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = std::conditional_t<
std::is_reference_v<meta::deref_return_t<Iter>>,
std::optional<std::reference_wrapper<std::remove_reference_t<meta::deref_return_t<Iter>>>>,
std::optional<meta::deref_return_t<Iter>>>;
using difference_type = blt::ptrdiff_t;
using pointer = value_type;
using reference = value_type;
filter_wrapper(Iter iter, Pred func):
deref_only_wrapper<Iter, filter_wrapper<Iter, Pred>>(std::move(iter)), func(std::move(func))
{
}
reference operator*() const
{
if (!func(*this->iter))
{
if (!func(*this->iter))
return {};
}
return *this->iter;
}
private:
Pred func;
};
template <typename Iter>
class const_wrapper : public deref_only_wrapper<Iter, const_wrapper<Iter>>
{
public:
using ref_return = meta::deref_return_t<Iter>;
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = std::conditional_t<std::is_reference_v<ref_return>, std::remove_reference_t<ref_return>, ref_return>;
using difference_type = ptrdiff_t;
using pointer = const value_type*;
using reference = const value_type&;
explicit const_wrapper(Iter iter): deref_only_wrapper<Iter, const_wrapper>(std::move(iter))
{
}
template<typename T>
static auto make_const(T&& value) -> decltype(auto)
{
using Decay = std::decay_t<T>;
if constexpr (std::is_lvalue_reference_v<T>)
{
return const_cast<const Decay&>(value);
} else
{
return static_cast<const Decay>(value);
}
}
auto operator*() const
{
if constexpr (std::is_reference_v<ref_return>)
{
return const_cast<const value_type&>(*this->iter);
} else if constexpr (meta::is_tuple_v<value_type> || meta::is_pair_v<value_type>)
{
return std::apply([](auto&&... args)
{
return {};
}
return std::tuple<decltype(make_const(std::forward<decltype(args)>(args)))...>{make_const(std::forward<decltype(args)>(args))...};
}, *this->iter);
}
else
{
return *this->iter;
}
private:
Pred func;
}
};
namespace impl
{
template<typename Derived>
template <typename Derived>
class skip_t
{
private:
template<bool check>
auto skip_base(blt::size_t n)
private:
template <bool check>
auto skip_base(blt::size_t n)
{
auto* d = static_cast<Derived*>(this);
auto begin = d->begin();
auto end = d->end();
if constexpr (meta::is_random_access_iterator_category_v<typename Derived::iterator_category>)
{
auto* d = static_cast<Derived*>(this);
auto begin = d->begin();
auto end = d->end();
if constexpr (meta::is_random_access_iterator_category_v<typename Derived::iterator_category>)
// random access iterators can have math directly applied to them.
if constexpr (check)
{
// random access iterators can have math directly applied to them.
if constexpr (check)
{
return Derived{begin + std::min(static_cast<blt::ptrdiff_t>(n), std::distance(begin, end)), end};
} else
{
return Derived{begin + n, end};
}
} else
return Derived{begin + std::min(static_cast<blt::ptrdiff_t>(n), std::distance(begin, end)), end};
}
else
{
for (blt::size_t i = 0; i < n; i++)
{
if constexpr (check)
{
if (begin == end)
break;
}
++begin;
}
return Derived{std::move(begin), std::move(end)};
return Derived{begin + n, end};
}
}
public:
auto skip(blt::size_t n)
{ return skip_base<false>(n); }
auto skip_or(blt::size_t n)
{ return skip_base<true>(n); }
else
{
for (blt::size_t i = 0; i < n; i++)
{
if constexpr (check)
{
if (begin == end)
break;
}
++begin;
}
return Derived{std::move(begin), std::move(end)};
}
}
public:
auto skip(blt::size_t n)
{
return skip_base<false>(n);
}
auto skip_or(blt::size_t n)
{
return skip_base<true>(n);
}
};
template<typename Derived>
template <typename Derived>
class take_t
{
private:
template<bool check>
auto take_base(blt::size_t n)
private:
template <bool check>
auto take_base(blt::size_t n)
{
static_assert(!meta::is_input_iterator_category_v<typename Derived::iterator_category>,
"Cannot .take() on an input iterator!");
auto* d = static_cast<Derived*>(this);
auto begin = d->begin();
auto end = d->end();
// take variant for forward and bidirectional iterators
if constexpr (meta::is_forward_iterator_category_v<typename Derived::iterator_category> ||
meta::is_bidirectional_iterator_category_v<typename Derived::iterator_category>)
{
static_assert(!meta::is_input_iterator_category_v<typename Derived::iterator_category>,
"Cannot .take() on an input iterator!");
auto* d = static_cast<Derived*>(this);
auto begin = d->begin();
auto end = d->end();
// take variant for forward and bidirectional iterators
if constexpr (meta::is_forward_iterator_category_v<typename Derived::iterator_category> ||
meta::is_bidirectional_iterator_category_v<typename Derived::iterator_category>)
// with these guys we have to loop forward to move the iterators. an unfortunate inefficiency
auto new_end = begin;
for (blt::size_t i = 0; i < n; i++)
{
// with these guys we have to loop forward to move the iterators. an unfortunate inefficiency
auto new_end = begin;
for (blt::size_t i = 0; i < n; i++)
{
if constexpr (check)
{
if (new_end == end)
break;
}
++new_end;
}
return Derived{std::move(begin), std::move(new_end)};
} else if constexpr (meta::is_random_access_iterator_category_v<typename Derived::iterator_category>)
{
// random access iterators can have math directly applied to them.
if constexpr (check)
{
return Derived{begin, begin + std::min(static_cast<blt::ptrdiff_t>(n), std::distance(begin, end))};
} else
{
return Derived{begin, begin + n};
if (new_end == end)
break;
}
++new_end;
}
return Derived{std::move(begin), std::move(new_end)};
}
else if constexpr (meta::is_random_access_iterator_category_v<typename Derived::iterator_category>)
{
// random access iterators can have math directly applied to them.
if constexpr (check)
{
return Derived{begin, begin + std::min(static_cast<blt::ptrdiff_t>(n), std::distance(begin, end))};
}
else
{
return Derived{begin, begin + n};
}
}
public:
auto take(blt::size_t n)
{ return take_base<false>(n); }
auto take_or(blt::size_t n)
{ return take_base<true>(n); }
}
public:
auto take(blt::size_t n)
{
return take_base<false>(n);
}
auto take_or(blt::size_t n)
{
return take_base<true>(n);
}
};
}
template<typename IterBase>
template <typename IterBase>
class iterator_container : public impl::take_t<iterator_container<IterBase>>,
public impl::skip_t<iterator_container<IterBase>>
{
public:
using iterator_category = typename std::iterator_traits<IterBase>::iterator_category;
using iterator = IterBase;
iterator_container(IterBase begin, IterBase end): m_begin(std::move(begin)), m_end(std::move(end))
{}
template<typename Iter>
iterator_container(Iter&& begin, Iter&& end): m_begin(std::forward<Iter>(begin)), m_end(std::forward<Iter>(end))
{}
auto rev() const
{
static_assert(meta::is_bidirectional_or_better_category_v<iterator_category>,
".rev() must be used with bidirectional (or better) iterators!");
return iterator_container<std::reverse_iterator<IterBase>>{std::reverse_iterator<IterBase>{end()},
std::reverse_iterator<IterBase>{begin()}};
}
template<typename... Iter>
auto zip(iterator_pair<Iter>... iterator_pairs) const
{
return zip_iterator_container(iterator_pair<decltype(begin())>{begin(), end()}, iterator_pairs...);
}
template<typename... Container>
auto zip(Container& ... containers) const
{
return zip_iterator_container(iterator_pair<decltype(begin())>{begin(), end()},
iterator_pair{containers.begin(), containers.end()}...);
}
template<typename... Container>
auto zip(const Container& ... containers) const
{
return zip_iterator_container(iterator_pair<decltype(begin())>{begin(), end()},
iterator_pair{containers.begin(), containers.end()}...);
}
auto enumerate() const
{
return enumerate_iterator_container{begin(), end(), static_cast<blt::size_t>(std::distance(begin(), end()))};
}
template<typename Func>
auto map(Func func) const
{
return iterator_container<blt::iterator::map_wrapper<IterBase, Func>>{
blt::iterator::map_wrapper<IterBase, Func>{m_begin, func},
blt::iterator::map_wrapper<IterBase, Func>{m_end, func}};
}
template<typename Pred>
auto filter(Pred pred) const
{
return iterator_container<blt::iterator::filter_wrapper<IterBase, Pred>>{
blt::iterator::filter_wrapper<IterBase, Pred>{m_begin, pred},
blt::iterator::filter_wrapper<IterBase, Pred>{m_end, pred}};
}
auto begin() const
{
return m_begin;
}
auto end() const
{
return m_end;
}
protected:
IterBase m_begin;
IterBase m_end;
public:
using iterator_category = typename std::iterator_traits<IterBase>::iterator_category;
using iterator = IterBase;
iterator_container(IterBase begin, IterBase end): m_begin(std::move(begin)), m_end(std::move(end))
{
}
template <typename Iter>
iterator_container(Iter&& begin, Iter&& end): m_begin(std::forward<Iter>(begin)), m_end(std::forward<Iter>(end))
{
}
auto rev() const
{
static_assert(meta::is_bidirectional_or_better_category_v<iterator_category>,
".rev() must be used with bidirectional (or better) iterators!");
return iterator_container<std::reverse_iterator<IterBase>>{
std::reverse_iterator<IterBase>{end()},
std::reverse_iterator<IterBase>{begin()}
};
}
template <typename... Iter>
auto zip(iterator_pair<Iter>... iterator_pairs) const
{
return zip_iterator_container(iterator_pair<decltype(begin())>{begin(), end()}, iterator_pairs...);
}
template <typename... Container>
auto zip(Container&... containers) const
{
return zip_iterator_container(iterator_pair<decltype(begin())>{begin(), end()},
iterator_pair{containers.begin(), containers.end()}...);
}
template <typename... Container>
auto zip(const Container&... containers) const
{
return zip_iterator_container(iterator_pair<decltype(begin())>{begin(), end()},
iterator_pair{containers.begin(), containers.end()}...);
}
template <typename Iter>
auto zip(Iter begin, Iter end)
{
return zip(iterator_pair<Iter>{begin, end});
}
auto enumerate() const
{
return enumerate_iterator_container{begin(), end(), static_cast<blt::size_t>(std::distance(begin(), end()))};
}
auto flatten() const
{
return iterator_container<flatten_wrapper<IterBase, false>>{
blt::iterator::flatten_wrapper<IterBase, false>{m_begin},
blt::iterator::flatten_wrapper<IterBase, false>{m_end}
};
}
auto flatten_all() const
{
return iterator_container<flatten_wrapper<IterBase, true>>{
blt::iterator::flatten_wrapper<IterBase, true>{m_begin},
blt::iterator::flatten_wrapper<IterBase, true>{m_end}
};
}
auto as_const() const
{
return iterator_container<const_wrapper<IterBase>>{
const_wrapper<IterBase>{m_begin},
const_wrapper<IterBase>{m_end}
};
}
template <typename Func>
auto map(Func func) const
{
return iterator_container<blt::iterator::map_wrapper<IterBase, Func>>{
blt::iterator::map_wrapper<IterBase, Func>{m_begin, func},
blt::iterator::map_wrapper<IterBase, Func>{m_end, func}
};
}
template <typename Pred>
auto filter(Pred pred) const
{
return iterator_container<blt::iterator::filter_wrapper<IterBase, Pred>>{
blt::iterator::filter_wrapper<IterBase, Pred>{m_begin, pred},
blt::iterator::filter_wrapper<IterBase, Pred>{m_end, pred}
};
}
auto begin() const
{
return m_begin;
}
auto end() const
{
return m_end;
}
protected:
IterBase m_begin;
IterBase m_end;
};
}
#endif //BLT_ITERATOR_ITER_COMMON

View File

@ -20,118 +20,112 @@
#define BLT_ITERATOR_ENUMERATE_H
#include <blt/iterator/common.h>
#include <tuple>
namespace blt
{
namespace iterator
{
/**
* struct which is returned by the enumerator.
* @tparam T type to store.
*/
template<typename T>
struct enumerate_item
{
blt::size_t index;
T value;
};
template<typename Iter>
template <typename Iter>
class enumerate_wrapper : public passthrough_wrapper<Iter, enumerate_wrapper<Iter>>
{
public:
enumerate_wrapper(blt::size_t index, Iter iter): passthrough_wrapper<Iter, enumerate_wrapper<Iter>>(std::move(iter)), index(index)
{}
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = enumerate_item<meta::deref_return_t<Iter>>;
using difference_type = blt::ptrdiff_t;
using pointer = value_type;
using reference = value_type;
enumerate_item<meta::deref_return_t<Iter>> operator*() const
{
return {index, *this->iter};
}
enumerate_wrapper& operator++()
{
++index;
++this->iter;
return *this;
}
enumerate_wrapper& operator--()
{
--index;
--this->iter;
return *this;
}
friend enumerate_wrapper operator+(const enumerate_wrapper& a, blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
auto copy = a;
copy.index += n;
copy.iter = copy.iter + n;
return copy;
}
friend enumerate_wrapper operator-(const enumerate_wrapper& a, blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
auto copy = a;
copy.index -= n;
copy.iter = copy.iter - n;
return copy;
}
private:
blt::size_t index;
public:
enumerate_wrapper(const size_t index, Iter iter): passthrough_wrapper<Iter, enumerate_wrapper<Iter>>(std::move(iter)), index(index)
{
}
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = enumerate_item<Iter>;
using difference_type = blt::ptrdiff_t;
using pointer = value_type;
using reference = value_type;
enumerate_item<meta::deref_return_t<Iter>> operator*() const
{
return {index, *this->iter};
}
enumerate_wrapper& operator++()
{
++index;
++this->iter;
return *this;
}
enumerate_wrapper& operator--()
{
--index;
--this->iter;
return *this;
}
friend enumerate_wrapper operator+(const enumerate_wrapper& a, blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
auto copy = a;
copy.index += n;
copy.iter = copy.iter + n;
return copy;
}
friend enumerate_wrapper operator-(const enumerate_wrapper& a, blt::ptrdiff_t n)
{
static_assert(meta::is_random_access_iterator_v<Iter>, "Iterator must allow random access");
auto copy = a;
copy.index -= n;
copy.iter = copy.iter - n;
return copy;
}
private:
blt::size_t index;
};
}
template<typename Iter>
template <typename Iter>
class enumerate_iterator_container : public iterator::iterator_container<iterator::enumerate_wrapper<Iter>>
{
public:
using iterator::iterator_container<iterator::enumerate_wrapper<Iter>>::iterator_container;
enumerate_iterator_container(Iter begin, Iter end, blt::size_t size):
iterator::iterator_container<iterator::enumerate_wrapper<Iter>>(
iterator::enumerate_wrapper<Iter>{0, std::move(begin)}, iterator::enumerate_wrapper<Iter>{size, std::move(end)})
{}
public:
using iterator::iterator_container<iterator::enumerate_wrapper<Iter>>::iterator_container;
enumerate_iterator_container(Iter begin, Iter end, blt::size_t size):
iterator::iterator_container<iterator::enumerate_wrapper<Iter>>(
iterator::enumerate_wrapper<Iter>{0, std::move(begin)}, iterator::enumerate_wrapper<Iter>{size, std::move(end)})
{
}
};
template<typename Iter>
template <typename Iter>
enumerate_iterator_container(Iter, Iter, blt::size_t) -> enumerate_iterator_container<Iter>;
template<typename T>
template <typename T>
static inline auto enumerate(T& container)
{
return enumerate_iterator_container{container.begin(), container.end(), container.size()};
return enumerate_iterator_container{
container.begin(), container.end(),
static_cast<blt::size_t>(std::distance(container.begin(), container.end()))
};
}
template<typename T>
template <typename T>
static inline auto enumerate(const T& container)
{
return enumerate_iterator_container{container.begin(), container.end(), container.size()};
return enumerate_iterator_container{
container.begin(), container.end(),
static_cast<blt::size_t>(std::distance(container.begin(), container.end()))
};
}
template<typename T, blt::size_t size>
static inline auto enumerate(const T(& container)[size])
template <typename T, blt::size_t size>
static inline auto enumerate(const T (&container)[size])
{
return enumerate_iterator_container{&container[0], &container[size], size};
}
template<typename T, blt::size_t size>
static inline auto enumerate(T(& container)[size])
template <typename T, blt::size_t size>
static inline auto enumerate(T (&container)[size])
{
return enumerate_iterator_container{&container[0], &container[size], size};
}
}
#endif //BLT_ITERATOR_ENUMERATE_H

View File

@ -0,0 +1,89 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_ITERATOR_FLATTEN_H
#define BLT_ITERATOR_FLATTEN_H
#include <blt/iterator/common.h>
#include <blt/meta/type_traits.h>
#include <tuple>
namespace blt::iterator
{
namespace detail
{
template <typename Tuple>
static auto flatten_recursive(Tuple&& tuple) -> decltype(auto)
{
using Decay = std::decay_t<Tuple>;
if constexpr (meta::is_tuple_v<Decay> || meta::is_pair_v<Decay>)
{
return std::apply([](auto&&... args)
{
return std::tuple_cat(flatten_recursive(std::forward<decltype(args)>(args))...);
}, std::forward<Tuple>(tuple));
}
else
{
return forward_as_tuple(std::forward<Tuple>(tuple));
}
}
template <typename Tuple>
static auto flatten(Tuple&& tuple) -> decltype(auto)
{
return std::apply([](auto&&... args)
{
return std::tuple_cat(ensure_tuple(std::forward<decltype(args)>(args))...);
}, std::forward<Tuple>(tuple));
}
}
template <typename Iter, bool Recursive>
class flatten_wrapper : public deref_only_wrapper<Iter, flatten_wrapper<Iter, Recursive>>
{
public:
using iterator_category = typename std::iterator_traits<Iter>::iterator_category;
using value_type = std::conditional_t<Recursive,
std::remove_reference_t<decltype(detail::flatten_recursive(*std::declval<Iter>()))>,
std::remove_reference_t<decltype(detail::flatten(*std::declval<Iter>()))>>;
using difference_type = ptrdiff_t;
using pointer = value_type*;
using reference = value_type&;
explicit flatten_wrapper(Iter iter):
deref_only_wrapper<Iter, flatten_wrapper>(std::move(iter))
{
}
auto operator*() const -> decltype(auto)
{
if constexpr (Recursive)
{
return detail::flatten_recursive(*this->iter);
}
else
{
return detail::flatten(*this->iter);
}
}
};
}
#endif //BLT_ITERATOR_FLATTEN_H

View File

@ -19,6 +19,8 @@
#ifndef BLT_ITERATOR_FWDDECL_H
#define BLT_ITERATOR_FWDDECL_H
#include <utility>
namespace blt
{
template<typename... Iter>
@ -33,7 +35,7 @@ namespace blt
struct iterator_pair;
template<typename T>
struct enumerate_item;
using enumerate_item = std::pair<size_t, T>;
template<typename Iter>
class enumerate_wrapper;
@ -46,6 +48,12 @@ namespace blt
template<typename Iter, typename Pred>
class filter_wrapper;
template<typename Iter, bool Recursive>
class flatten_wrapper;
template<typename Iter>
class const_wrapper;
namespace impl
{

View File

@ -29,6 +29,12 @@
namespace blt
{
template<typename T>
static inline auto iterate(T begin, T end)
{
return iterator::iterator_container<T>{std::move(begin), std::move(end)};
}
template<typename T>
static inline auto iterate(T& container)
{

View File

@ -0,0 +1,83 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_LOGGING_LOGGING_H
#define BLT_LOGGING_LOGGING_H
#include <iostream>
#include <ostream>
#include <sstream>
#include <string>
#include <blt/std/utility.h>
#include <blt/meta/meta.h>
namespace blt::logging
{
struct logger_t
{
explicit logger_t() = default;
template <typename T>
void print_value(T&& t)
{
static_assert(meta::is_streamable_v<T>, "T must be streamable in order to work with blt::logging!");
m_stream << std::forward<T>(t);
}
template <typename... Args>
std::string log(std::string fmt, Args&&... args)
{
compile(std::move(fmt));
((consume_until_fmt(), print_value(std::forward<Args>(args))), ...);
return to_string();
}
std::string to_string();
private:
void compile(std::string fmt);
void consume_until_fmt();
std::string m_fmt;
std::stringstream m_stream;
size_t m_last_fmt_pos = 0;
};
void print(const std::string& fmt);
void newline();
logger_t& get_global_logger();
template <typename... Args>
void print(std::string fmt, Args&&... args)
{
auto& logger = get_global_logger();
print(logger.log(std::move(fmt), std::forward<Args>(args)...));
}
template <typename... Args>
void println(std::string fmt, Args&&... args)
{
print(std::move(fmt), std::forward<Args>(args)...);
newline();
}
}
#endif // BLT_LOGGING_LOGGING_H

View File

@ -0,0 +1,27 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_LOGGING_STATUS_H
#define BLT_LOGGING_STATUS_H
namespace blt::logging
{
}
#endif //BLT_LOGGING_STATUS_H

View File

@ -12,6 +12,7 @@
#include <type_traits>
#include <array>
#include <initializer_list>
#include "blt/iterator/iterator.h"
#ifndef M_PI
// MSVC does not have M_PI
@ -24,6 +25,10 @@ namespace blt
template<typename T, blt::u32 rows, blt::u32 columns>
class generalized_matrix
{
public:
static constexpr auto data_rows = rows;
static constexpr auto data_columns = columns;
private:
using matrix_t = generalized_matrix<T, rows, columns>;
enum class init_type
{
@ -163,6 +168,14 @@ namespace blt
return copy;
}
[[nodiscard]] constexpr matrix_t bipolar() const
{
matrix_t copy = *this;
for (auto& v : copy.data)
v = v.bipolar();
return copy;
}
constexpr inline const blt::vec<T, rows>& operator[](u32 column) const
{
return data[column];

View File

@ -17,377 +17,401 @@
namespace blt
{
#define MSVC_COMPILER (!defined(__GNUC__) && !defined(__clang__))
constexpr float EPSILON = std::numeric_limits<float>::epsilon();
static inline constexpr bool f_equal(float v1, float v2)
{
return v1 >= v2 - EPSILON && v1 <= v2 + EPSILON;
}
template<typename T, blt::u32 size>
template <typename T, blt::u32 size>
struct vec
{
static_assert(std::is_arithmetic_v<T> && "blt::vec must be created using an arithmetic type!");
private:
std::array<T, size> elements;
public:
constexpr vec()
static_assert(std::is_arithmetic_v<T> && "blt::vec must be created using an arithmetic type!");
private:
std::array<T, size> elements;
public:
constexpr static blt::u32 data_size = size;
constexpr vec()
{
for (auto& v : elements)
v = static_cast<T>(0);
}
/**
* Create a vector with initializer list, if the initializer list doesn't contain enough values to fill this vec, it will use t
* @param t default value to fill with
* @param args list of args
*/
template <typename U, std::enable_if_t<std::is_same_v<T, U> || std::is_convertible_v<U, T>, bool> = true>
constexpr vec(U t, std::initializer_list<U> args): elements()
{
auto b = args.begin();
for (auto& v : elements)
{
for (auto& v : elements)
v = static_cast<T>(0);
}
/**
* Create a vector with initializer list, if the initializer list doesn't contain enough values to fill this vec, it will use t
* @param t default value to fill with
* @param args list of args
*/
template<typename U, std::enable_if_t<std::is_same_v<T, U> || std::is_convertible_v<U, T>, bool> = true>
constexpr vec(U t, std::initializer_list<U> args): elements()
{
auto b = args.begin();
for (auto& v : elements)
if (b == args.end())
{
if (b == args.end())
{
v = t;
continue;
}
v = *b;
++b;
}
}
/**
* Create a vector from an initializer list, if the list doesn't have enough elements it will be filled with the default value (0)
* @param args
*/
template<typename U, std::enable_if_t<std::is_same_v<T, U> || std::is_convertible_v<U, T>, bool> = true>
constexpr vec(std::initializer_list<U> args): vec(U(), args)
{}
template<typename... Args, std::enable_if_t<sizeof...(Args) == size, bool> = true>
constexpr explicit vec(Args... args): vec(std::array<T, size>{static_cast<T>(args)...})
{}
constexpr explicit vec(T t)
{
for (auto& v : elements)
v = t;
}
constexpr explicit vec(const T elem[size])
{
for (size_t i = 0; i < size; i++)
elements[i] = elem[i];
}
constexpr explicit vec(std::array<T, size> elem): elements(elem)
{}
template<typename G, size_t base_size, std::enable_if_t<std::is_convertible_v<G, T>, bool> = true>
constexpr explicit vec(std::array<G, base_size> el): elements()
{
auto b = el.begin();
auto m = elements.begin();
while (b != el.end() && m != elements.end())
{
*m = *b;
++m;
++b;
continue;
}
v = *b;
++b;
}
[[nodiscard]] constexpr inline T x() const
}
/**
* Create a vector from an initializer list, if the list doesn't have enough elements it will be filled with the default value (0)
* @param args
*/
template <typename U, std::enable_if_t<std::is_same_v<T, U> || std::is_convertible_v<U, T>, bool> = true>
constexpr vec(std::initializer_list<U> args): vec(U(), args)
{
}
template <typename... Args, std::enable_if_t<sizeof...(Args) == size, bool> = true>
constexpr explicit vec(Args... args): vec(std::array<T, size>{static_cast<T>(args)...})
{
}
constexpr explicit vec(T t)
{
for (auto& v : elements)
v = t;
}
constexpr explicit vec(const T elem[size])
{
for (size_t i = 0; i < size; i++)
elements[i] = elem[i];
}
constexpr explicit vec(std::array<T, size> elem): elements(elem)
{
}
template <typename G, size_t base_size, std::enable_if_t<std::is_convertible_v<G, T>, bool> = true>
constexpr explicit vec(std::array<G, base_size> el): elements()
{
auto b = el.begin();
auto m = elements.begin();
while (b != el.end() && m != elements.end())
{
return elements[0];
}
[[nodiscard]] constexpr inline T y() const
{
static_assert(size > 1);
return elements[1];
}
[[nodiscard]] constexpr inline T z() const
{
static_assert(size > 2);
return elements[2];
}
[[nodiscard]] constexpr inline T w() const
{
static_assert(size > 3);
return elements[3];
}
[[nodiscard]] constexpr inline vec<T, size> abs() const
{
auto copy = *this;
for (auto& v : copy.elements)
v = std::abs(v);
return copy;
}
[[nodiscard]] constexpr inline T magnitude() const
{
T total = 0;
for (blt::u32 i = 0; i < size; i++)
total += elements[i] * elements[i];
return std::sqrt(total);
}
[[nodiscard]] constexpr inline vec<T, size> normalize() const
{
T mag = this->magnitude();
if (mag == 0)
return vec<T, size>(*this);
return *this / mag;
}
constexpr inline T& operator[](blt::size_t index)
{
return elements[index];
}
constexpr inline T operator[](blt::size_t index) const
{
return elements[index];
}
constexpr inline vec<T, size>& operator=(T v)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] = v;
return *this;
}
constexpr inline vec<T, size> operator-()
{
vec<T, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = -elements[i];
return vec<T, size>{initializer};
}
constexpr inline vec<T, size>& operator+=(const vec<T, size>& other)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] += other[i];
return *this;
}
constexpr inline vec<T, size>& operator*=(const vec<T, size>& other)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] *= other[i];
return *this;
}
constexpr inline vec<T, size>& operator+=(T f)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] += f;
return *this;
}
constexpr inline vec<T, size>& operator*=(T f)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] *= f;
return *this;
}
constexpr inline vec<T, size>& operator-=(const vec<T, size>& other)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] -= other[i];
return *this;
}
constexpr inline vec<T, size>& operator-=(T f)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] -= f;
return *this;
}
/**
* performs the dot product of left * right
*/
constexpr static inline T dot(const vec<T, size>& left, const vec<T, size>& right)
{
T dot = 0;
for (blt::u32 i = 0; i < size; i++)
dot += left[i] * right[i];
return dot;
}
constexpr static inline vec<T, size> cross(
const vec<T, size>& left, const vec<T, size>& right
)
{
// cross is only defined on vectors of size 3. 2D could be implemented, which is a TODO
static_assert(size == 3);
return {left.y() * right.z() - left.z() * right.y(),
left.z() * right.x() - left.x() * right.z(),
left.x() * right.y() - left.y() * right.x()};
}
constexpr static inline vec<T, size> project(
const vec<T, size>& u, const vec<T, size>& v
)
{
T du = dot(u);
T dv = dot(v);
return (du / dv) * v;
}
constexpr inline auto* data()
{
return elements.data();
}
[[nodiscard]] constexpr inline const auto* data() const
{
return elements.data();
}
constexpr auto begin()
{
return elements.begin();
}
constexpr auto end()
{
return elements.end();
}
constexpr auto rbegin()
{
return elements.rbegin();
}
constexpr auto rend()
{
return elements.rend();
}
[[nodiscard]] constexpr auto cbegin() const
{
return elements.cbegin();
}
[[nodiscard]] constexpr auto cend() const
{
return elements.cend();
*m = *b;
++m;
++b;
}
}
[[nodiscard]] constexpr inline T x() const
{
return elements[0];
}
[[nodiscard]] constexpr inline T y() const
{
static_assert(size > 1);
return elements[1];
}
[[nodiscard]] constexpr inline T z() const
{
static_assert(size > 2);
return elements[2];
}
[[nodiscard]] constexpr inline T w() const
{
static_assert(size > 3);
return elements[3];
}
[[nodiscard]] constexpr inline vec<T, size> abs() const
{
auto copy = *this;
for (auto& v : copy.elements)
v = std::abs(v);
return copy;
}
[[nodiscard]] constexpr inline vec<T, size> bipolar() const
{
auto copy = *this;
for (auto& v : copy.elements)
v = v >= 0 ? 1 : -1;
return copy;
}
[[nodiscard]] constexpr inline T magnitude() const
{
T total = 0;
for (blt::u32 i = 0; i < size; i++)
total += elements[i] * elements[i];
return std::sqrt(total);
}
[[nodiscard]] constexpr inline vec<T, size> normalize() const
{
T mag = this->magnitude();
if (mag == 0)
return vec<T, size>(*this);
return *this / mag;
}
constexpr inline T& operator[](blt::size_t index)
{
return elements[index];
}
constexpr inline T operator[](blt::size_t index) const
{
return elements[index];
}
constexpr inline vec<T, size>& operator=(T v)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] = v;
return *this;
}
constexpr inline vec<T, size> operator-()
{
vec<T, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = -elements[i];
return vec<T, size>{initializer};
}
constexpr inline vec<T, size>& operator+=(const vec<T, size>& other)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] += other[i];
return *this;
}
constexpr inline vec<T, size>& operator*=(const vec<T, size>& other)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] *= other[i];
return *this;
}
constexpr inline vec<T, size>& operator+=(T f)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] += f;
return *this;
}
constexpr inline vec<T, size>& operator*=(T f)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] *= f;
return *this;
}
constexpr inline vec<T, size>& operator-=(const vec<T, size>& other)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] -= other[i];
return *this;
}
constexpr inline vec<T, size>& operator-=(T f)
{
for (blt::u32 i = 0; i < size; i++)
elements[i] -= f;
return *this;
}
/**
* performs the dot product of left * right
*/
constexpr static inline T dot(const vec<T, size>& left, const vec<T, size>& right)
{
T dot = 0;
for (blt::u32 i = 0; i < size; i++)
dot += left[i] * right[i];
return dot;
}
constexpr static inline vec<T, size> cross(
const vec<T, size>& left, const vec<T, size>& right
)
{
// cross is only defined on vectors of size 3. 2D could be implemented, which is a TODO
static_assert(size == 3);
return {
left.y() * right.z() - left.z() * right.y(),
left.z() * right.x() - left.x() * right.z(),
left.x() * right.y() - left.y() * right.x()
};
}
constexpr static inline vec<T, size> project(
const vec<T, size>& u, const vec<T, size>& v
)
{
T du = dot(u);
T dv = dot(v);
return (du / dv) * v;
}
constexpr inline auto* data()
{
return elements.data();
}
[[nodiscard]] constexpr inline const auto* data() const
{
return elements.data();
}
[[nodiscard]] constexpr auto begin() const
{
return elements.begin();
}
[[nodiscard]] constexpr auto end() const
{
return elements.end();
}
[[nodiscard]] constexpr auto rbegin() const
{
return elements.rbegin();
}
[[nodiscard]] constexpr auto rend() const
{
return elements.rend();
}
[[nodiscard]] constexpr auto cbegin() const
{
return elements.cbegin();
}
[[nodiscard]] constexpr auto cend() const
{
return elements.cend();
}
friend std::size_t hash_value(const vec& obj)
{
std::size_t seed = 0x5410391E;
for (const auto& v : obj)
seed ^= (seed << 6) + (seed >> 2) + std::hash<T>{}(v);
return seed;
}
};
template<typename T, blt::u32 size>
inline constexpr vec<T, size> operator+(const vec<T, size>& left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() + std::declval<G>())>
inline constexpr vec<R, size> operator+(const vec<T, size>& left, const vec<G, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] + right[i];
initializer[i] = static_cast<R>(left[i]) + static_cast<R>(right[i]);
return initializer;
}
template<typename T, blt::u32 size>
inline constexpr vec<T, size> operator-(const vec<T, size>& left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() - std::declval<G>())>
inline constexpr vec<R, size> operator-(const vec<T, size>& left, const vec<G, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] - right[i];
initializer[i] = static_cast<R>(left[i]) - static_cast<R>(right[i]);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator+(const vec<T, size>& left, G right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() + std::declval<G>())>
inline constexpr vec<R, size> operator+(const vec<T, size>& left, G right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] + static_cast<T>(right);
initializer[i] = static_cast<R>(left[i]) + static_cast<R>(right);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator-(const vec<T, size>& left, G right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() - std::declval<G>())>
inline constexpr vec<R, size> operator-(const vec<T, size>& left, G right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] + static_cast<T>(right);
initializer[i] = static_cast<R>(left[i]) - static_cast<R>(right);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator+(G left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() + std::declval<G>())>
inline constexpr vec<R, size> operator+(G left, const vec<T, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = static_cast<T>(left) + right[i];
initializer[i] = static_cast<R>(left) + static_cast<R>(right[i]);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator-(G left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() - std::declval<G>())>
inline constexpr vec<R, size> operator-(G left, const vec<T, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = static_cast<T>(left) - right[i];
initializer[i] = static_cast<R>(left) - static_cast<R>(right[i]);
return initializer;
}
template<typename T, blt::u32 size>
inline constexpr vec<T, size> operator*(const vec<T, size>& left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() * std::declval<G>())>
inline constexpr vec<R, size> operator*(const vec<T, size>& left, const vec<G, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] * right[i];
initializer[i] = static_cast<R>(left[i]) * static_cast<R>(right[i]);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator*(const vec<T, size>& left, G right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() * std::declval<G>())>
inline constexpr vec<R, size> operator*(const vec<T, size>& left, G right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] * static_cast<T>(right);
initializer[i] = static_cast<R>(left[i]) * static_cast<R>(right);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator*(G left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() * std::declval<G>())>
inline constexpr vec<R, size> operator*(G left, const vec<T, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = static_cast<T>(left) * right[i];
initializer[i] = static_cast<R>(left) * static_cast<R>(right[i]);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator/(const vec<T, size>& left, G right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() / std::declval<G>())>
inline constexpr vec<R, size> operator/(const vec<T, size>& left, G right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = left[i] / static_cast<T>(right);
initializer[i] = static_cast<R>(left[i]) / static_cast<R>(right);
return initializer;
}
template<typename T, typename G, blt::u32 size>
inline constexpr vec<T, size> operator/(G left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size, typename R = decltype(std::declval<T>() / std::declval<G>())>
inline constexpr vec<R, size> operator/(G left, const vec<T, size>& right)
{
vec<T, size> initializer{};
vec<R, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
initializer[i] = static_cast<T>(left) / right[i];
initializer[i] = static_cast<R>(left) / static_cast<R>(right[i]);
return initializer;
}
template<typename T, blt::u32 size>
inline constexpr bool operator==(const vec<T, size>& left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size>
inline constexpr bool operator==(const vec<T, size>& left, const vec<G, size>& right)
{
constexpr double E = std::numeric_limits<T>::epsilon();
for (blt::u32 i = 0; i < size; i++)
@ -398,77 +422,79 @@ namespace blt
}
return true;
}
template<typename T, blt::u32 size>
inline constexpr bool operator!=(const vec<T, size>& left, const vec<T, size>& right)
template <typename T, typename G, blt::u32 size>
inline constexpr bool operator!=(const vec<T, size>& left, const vec<G, size>& right)
{
return !(left == right);
}
template<typename T, blt::u32 size>
inline constexpr bool operator&&(const vec<T, size>& left, const vec<T, size>& right)
template <typename Ret, typename T, blt::u32 size>
inline constexpr vec<Ret, size> vec_cast(const vec<T, size>& conv)
{
vec<Ret, size> initializer{};
for (blt::u32 i = 0; i < size; i++)
if (!f_equal(left[i], right[i]))
return false;
return true;
initializer[i] = static_cast<Ret>(conv[i]);
return initializer;
}
using vec2f = vec<float, 2>;
using vec3f = vec<float, 3>;
using vec4f = vec<float, 4>;
using vec2d = vec<double, 2>;
using vec3d = vec<double, 3>;
using vec4d = vec<double, 4>;
using vec2i = vec<blt::i32, 2>;
using vec3i = vec<blt::i32, 3>;
using vec4i = vec<blt::i32, 4>;
using vec2l = vec<blt::i64, 2>;
using vec3l = vec<blt::i64, 3>;
using vec4l = vec<blt::i64, 4>;
using vec2ui = vec<blt::u32, 2>;
using vec3ui = vec<blt::u32, 3>;
using vec4ui = vec<blt::u32, 4>;
using vec2ul = vec<blt::u64, 2>;
using vec3ul = vec<blt::u64, 3>;
using vec4ul = vec<blt::u64, 4>;
using vec2 = vec2f;
using vec3 = vec3f;
using vec4 = vec4f;
using color4 = vec4;
using color3 = vec3;
inline constexpr color4 make_color(float r, float g, float b)
{
return color4{r, g, b, 1.0f};
}
template<typename ValueType, u32 size>
template <typename ValueType, u32 size>
inline constexpr blt::vec<ValueType, 2> make_vec2(const blt::vec<ValueType, size>& t, size_t fill = 0)
{
if constexpr (size >= 2)
{
return blt::vec<ValueType, 2>(t.x(), t.y());
} else
}
else
{
return blt::vec<ValueType, 2>(t.x(), fill);
}
}
template<typename ValueType, u32 size>
template <typename ValueType, u32 size>
inline constexpr blt::vec<ValueType, 3> make_vec3(const blt::vec<ValueType, size>& t, size_t fill = 0)
{
if constexpr (size >= 3)
{
return blt::vec<ValueType, 3>(t.x(), t.y(), t.z());
} else
}
else
{
blt::vec<ValueType, 3> ret;
for (size_t i = 0; i < size; i++)
@ -478,14 +504,15 @@ namespace blt
return ret;
}
}
template<typename ValueType, u32 size>
template <typename ValueType, u32 size>
inline constexpr blt::vec<ValueType, 4> make_vec4(const blt::vec<ValueType, size>& t, size_t fill = 0)
{
if constexpr (size >= 4)
{
return blt::vec<ValueType, 4>(t.x(), t.y(), t.z(), t.w());
} else
}
else
{
blt::vec<ValueType, 4> ret;
for (size_t i = 0; i < size; i++)
@ -495,33 +522,33 @@ namespace blt
return ret;
}
}
namespace vec_algorithm
{
static inline void findOrthogonalBasis(const vec3& v, vec3& v1, vec3& v2, vec3& v3)
{
v1 = v.normalize();
vec3 arbitraryVector{1, 0, 0};
if (std::abs(vec3::dot(v, arbitraryVector)) > 0.9)
{
arbitraryVector = vec3{0, 1, 0};
}
v2 = vec3::cross(v, arbitraryVector).normalize();
v3 = vec3::cross(v1, v2);
}
// Gram-Schmidt orthonormalization algorithm
static inline void gramSchmidt(std::vector<vec3>& vectors)
{
int n = (int) vectors.size();
int n = (int)vectors.size();
std::vector<vec3> basis;
// normalize first vector
basis.push_back(vectors[0]);
basis[0] = basis[0].normalize();
// iterate over the rest of the vectors
for (int i = 1; i < n; ++i)
{
@ -538,7 +565,7 @@ namespace blt
new_vector = new_vector.normalize();
basis.push_back(new_vector);
}
vectors = basis;
}
}

View File

@ -0,0 +1,55 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_META_CODEGEN_H
#define BLT_META_CODEGEN_H
namespace blt
{
#define BLT_CONST_LVALUE_GETTER(TYPE, NAME) const TYPE& get_##NAME() const { return NAME; }
#define BLT_LVALUE_GETTER(TYPE, NAME) TYPE& get_##NAME() { return NAME; }
#define BLT_PRVALUE_GETTER(TYPE, NAME) TYPE get_##NAME() const { return NAME; }
#define BLT_GLVALUE_GETTER(TYPE, NAME) \
BLT_CONST_LVALUE_GETTER(TYPE, NAME) \
BLT_LVALUE_GETTER(TYPE, NAME)
#define BLT_PRVALUE_SETTER(TYPE, NAME) \
auto& set_##NAME(TYPE new_##NAME) \
{ \
NAME = new_##NAME; \
return *this; \
}
#define BLT_PRVALUE_MOVE_SETTER(TYPE, NAME) \
auto& set_##NAME(TYPE new_##NAME) \
{ \
NAME = std::move(new_##NAME); \
return *this; \
}
#define BLT_LVALUE_SETTER(TYPE, NAME) \
auto& set_##NAME(const TYPE& new_##NAME) \
{ \
NAME = new_##NAME; \
return *this; \
}
}
#endif //BLT_META_CODEGEN_H

View File

@ -114,7 +114,7 @@ namespace blt::meta
template<typename T>
struct deref_return
{
using type = typename std::invoke_result_t<decltype(&T::operator*), T&>;
using type = std::invoke_result_t<decltype(&T::operator*), T&>;
};
template<typename T>
@ -137,6 +137,15 @@ namespace blt::meta
template<typename T> \
inline constexpr bool has_func_##FUNC##_v = has_func_##FUNC<T>::value;
#define BLT_META_MAKE_STATIC_FUNCTION_CHECK(FUNC, ...)\
template<typename T, typename = void> \
class has_static_func_##FUNC : public std::false_type \
{}; \
template<typename T> \
class has_static_func_##FUNC<T, std::void_t<decltype(T::FUNC(,##__VA_ARGS__))>> : public std::true_type \
{}; \
template<typename T> \
inline constexpr bool has_static_func_##FUNC##_v = has_static_func_##FUNC<T>::value;
#define BLT_META_MAKE_MEMBER_CHECK(MEMBER)\
template<typename T, typename = void> \

View File

@ -0,0 +1,73 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_META_TYPE_TRAITS_H
#define BLT_META_TYPE_TRAITS_H
#include <type_traits>
#include <tuple>
namespace blt::meta
{
namespace detail
{
template<typename... Args>
void empty_apply_function(Args...)
{
}
inline auto lambda = [](auto...)
{
};
}
template <typename T>
using remove_cvref_t = std::remove_volatile_t<std::remove_const_t<std::remove_reference_t<T>>>;
template <typename U>
using add_const_ref_t = std::conditional_t<std::is_reference_v<U>, const std::remove_reference_t<U>&, const U>;
template <typename>
struct is_tuple : std::false_type
{
};
template <typename... T>
struct is_tuple<std::tuple<T...>> : std::true_type
{
};
template <typename>
struct is_pair : std::false_type
{
};
template <typename T, typename G>
struct is_pair<std::pair<T, G>> : std::true_type
{
};
template <typename T>
static constexpr bool is_tuple_v = is_tuple<T>::value;
template <typename T>
static constexpr bool is_pair_v = is_pair<T>::value;
}
#endif // BLT_META_TYPE_TRAITS_H

View File

@ -0,0 +1,811 @@
#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#ifndef BLT_PARSE_ARGPARSE_V2_H
#define BLT_PARSE_ARGPARSE_V2_H
#include <complex>
#include <blt/std/types.h>
#include <blt/std/hashmap.h>
#include <blt/meta/meta.h>
#include <string>
#include <string_view>
#include <vector>
#include <variant>
#include <optional>
#include <memory>
#include <functional>
#include <type_traits>
#include <blt/iterator/iterator.h>
#include <blt/meta/type_traits.h>
#include <blt/std/assert.h>
#include <blt/std/expected.h>
#include <blt/std/ranges.h>
#include <blt/std/utility.h>
namespace blt::argparse
{
class argument_string_t;
class argument_consumer_t;
class argument_parser_t;
class argument_subparser_t;
class argument_builder_t;
class argument_storage_t;
class argument_positional_storage_t;
enum class action_t
{
STORE,
STORE_CONST,
STORE_TRUE,
STORE_FALSE,
APPEND,
APPEND_CONST,
EXTEND,
COUNT,
HELP,
VERSION
};
enum class nargs_t
{
IF_POSSIBLE,
ALL,
ALL_AT_LEAST_ONE
};
using nargs_v = std::variant<nargs_t, i32>;
namespace detail
{
class bad_flag final : public std::runtime_error
{
public:
explicit bad_flag(const std::string& message): std::runtime_error(message)
{
}
};
class bad_positional final : public std::runtime_error
{
public:
explicit bad_positional(const std::string& message): std::runtime_error(message)
{
}
};
class missing_argument_error final : public std::runtime_error
{
public:
explicit missing_argument_error(const std::string& message): std::runtime_error(message)
{
}
};
class missing_value_error final : public std::runtime_error
{
public:
explicit missing_value_error(const std::string& message): std::runtime_error(message)
{
}
};
class type_error final : public std::runtime_error
{
public:
explicit type_error(const std::string& message): std::runtime_error(message)
{
}
};
class unexpected_argument_error final : public std::runtime_error
{
public:
explicit unexpected_argument_error(const std::string& message): std::runtime_error(message)
{
}
};
class bad_choice_error final : public std::runtime_error
{
public:
explicit bad_choice_error(const std::string& message): std::runtime_error(message)
{
}
};
class subparse_error final : public std::exception
{
public:
explicit subparse_error(const std::string_view found_string, std::vector<std::vector<std::string_view>> allowed_strings):
m_found_string(found_string),
m_allowed_strings(std::move(allowed_strings))
{
}
[[nodiscard]] const std::vector<std::vector<std::string_view>>& get_allowed_strings() const
{
return m_allowed_strings;
}
[[nodiscard]] std::string_view get_found_string() const
{
return m_found_string;
}
[[nodiscard]] std::string error_string() const;
[[nodiscard]] const char* what() const noexcept override
{
m_error_string = error_string();
return m_error_string.c_str();
}
private:
mutable std::string m_error_string;
std::string_view m_found_string;
std::vector<std::vector<std::string_view>> m_allowed_strings;
};
template <typename... Args>
struct arg_data_helper_t
{
using variant_t = std::variant<Args..., std::vector<Args>...>;
using arg_t = meta::arg_helper<Args...>;
using arg_vec_t = meta::arg_helper<std::vector<Args>...>;
template <typename T>
constexpr static bool is_type_stored_v = std::disjunction_v<std::is_same<T, Args>...>;
template <typename DefaultPrimitiveAction, typename DefaultListAction>
static auto make_visitor(const DefaultPrimitiveAction& primitive_action, const DefaultListAction& list_action)
{
return lambda_visitor{
([&primitive_action](Args& arg)
{
return primitive_action(arg);
})...,
([&list_action](std::vector<Args>& arg_vec)
{
return list_action(arg_vec);
})...
};
}
};
using arg_meta_type_helper_t = arg_data_helper_t<bool, i8, i16, i32, i64, u8, u16, u32, u64, float, double, std::string>;
using arg_data_t = arg_meta_type_helper_t::variant_t;
template <typename T>
struct arg_string_converter_t
{
static T convert(const std::string_view value)
{
static_assert(std::is_arithmetic_v<T> || std::is_same_v<T, std::string_view> || std::is_same_v<T, std::string>,
"Type must be arithmetic, string_view or string!");
const std::string temp{value};
if constexpr (std::is_same_v<T, float>)
{
return std::stof(temp);
}
else if constexpr (std::is_same_v<T, double>)
{
return std::stod(temp);
}
else if constexpr (std::is_unsigned_v<T>)
{
return static_cast<T>(std::stoull(temp));
}
else if constexpr (std::is_signed_v<T>)
{
return static_cast<T>(std::stoll(temp));
}
else if constexpr (std::is_same_v<T, std::string_view>)
{
return value;
}
else if constexpr (std::is_same_v<T, std::string>)
{
return std::string(value);
}
BLT_UNREACHABLE;
}
};
template <typename T>
auto ensure_is_string(T&& t)
{
using NO_REF = meta::remove_cvref_t<T>;
if constexpr (std::is_same_v<NO_REF, bool>)
return t ? "true" : "false";
else if constexpr (std::is_arithmetic_v<NO_REF> && !(std::is_same_v<NO_REF, char>
|| std::is_same_v<NO_REF, unsigned char> || std::is_same_v<NO_REF, signed char>))
return std::to_string(std::forward<T>(t));
else
return std::forward<T>(t);
}
void test();
}
class argument_string_t
{
public:
explicit argument_string_t(const std::string_view input, const hashset_t<char>& allowed_flag_prefix): m_argument(input),
m_allowed_flag_prefix(&allowed_flag_prefix)
{
process_argument();
}
[[nodiscard]] std::string_view get_flag() const
{
return m_flag_section;
}
[[nodiscard]] std::string_view get_name() const
{
return m_name_section;
}
[[nodiscard]] std::string_view value() const
{
return get_name();
}
[[nodiscard]] bool is_flag() const
{
return !m_flag_section.empty();
}
[[nodiscard]] std::string_view get_argument() const
{
return m_argument;
}
private:
/**
* This function takes the command line argument represented by this class,
* stored in m_argument and converts into flag side and name side arguments.
*
* What this means is in the case of a flag provided, for example passing --foo or --bar, this function will split the argument into
*
* m_flag_section = "--"
* m_name_section = "foo" || "bar"
*
* If the argument is purely positional, meaning there is no flag prefix,
* this function will do nothing and m_flag_section will be true for .empty()
*
* For example, if you provide res/some/folder/or/file as a command line positional argument,
* this function will create the following internal state:
*
* m_flag_section = ""
* m_flag_section.empty() == true
* m_name_section = "res/some/folder/or/file"
*
* m_argument is not modified by this function
*/
void process_argument()
{
// user provides a list of allowed prefix characters to argument_parser_t, which is then provided to this class at construction time
// it is not the job of this class to validate flag prefixes beyond this. // TODO: requiring this pointer is a code smell.
size_t start = 0;
for (; start < m_argument.size() && m_allowed_flag_prefix->contains(m_argument[start]); start++)
{
}
m_flag_section = {m_argument.data(), start};
m_name_section = {m_argument.data() + start, m_argument.size() - start};
}
std::string_view m_argument;
std::string_view m_flag_section;
std::string_view m_name_section;
const hashset_t<char>* m_allowed_flag_prefix;
};
class argument_consumer_t
{
public:
explicit argument_consumer_t(const span<argument_string_t>& args): m_absolute_begin(args.data()), m_begin(args.data() + 1),
m_end(args.data() + args.size())
{
BLT_ASSERT(!args.empty() &&
"Argument consumer must have at least one argument allocated to it. First argument is always assumed to be program");
}
[[nodiscard]] const argument_string_t& absolute_first() const
{
return *m_absolute_begin;
}
[[nodiscard]] argument_string_t peek(const i32 offset = 0) const
{
return *(m_begin + offset);
}
[[nodiscard]] argument_string_t r_peek(const i32 offset = 0) const
{
return *(m_end - 1 - offset);
}
argument_string_t consume()
{
return *(m_begin++);
}
argument_string_t r_consume()
{
return *(--m_end);
}
[[nodiscard]] i32 remaining() const
{
return static_cast<i32>(size());
}
[[nodiscard]] bool can_consume(const i32 amount = 0) const
{
return amount < remaining();
}
private:
[[nodiscard]] ptrdiff_t size() const
{
return m_end - m_begin;
}
argument_string_t* m_absolute_begin;
argument_string_t* m_begin;
argument_string_t* m_end;
};
class argument_storage_t
{
friend argument_parser_t;
friend argument_subparser_t;
friend argument_builder_t;
public:
template <typename T>
[[nodiscard]] const T& get(const std::string_view key) const
{
return std::get<T>(m_data.at(key));
}
[[nodiscard]] const std::string& get(const std::string_view key) const
{
return std::get<std::string>(m_data.at(key));
}
bool contains(const std::string_view key)
{
return m_data.find(key) != m_data.end();
}
[[nodiscard]] size_t size() const
{
return m_data.size();
}
private:
void add(const argument_storage_t& values)
{
for (const auto& value : values.m_data)
m_data.insert(value);
}
hashmap_t<std::string, detail::arg_data_t> m_data;
};
class argument_builder_t
{
friend argument_parser_t;
public:
argument_builder_t()
{
m_dest_func = [](const std::string_view dest, argument_storage_t& storage, std::string_view value)
{
storage.m_data.emplace(std::string{dest}, std::string{value});
};
m_dest_vec_func = [](const std::string_view dest, argument_storage_t& storage, const std::vector<std::string>& values)
{
storage.m_data.emplace(std::string{dest}, values);
};
}
template <typename T>
argument_builder_t& as_type()
{
static_assert(detail::arg_data_helper_t<T>::template is_type_stored_v<T>, "Type is not valid to be stored/converted as an argument");
m_dest_func = [](const std::string_view dest, argument_storage_t& storage, std::string_view value)
{
storage.m_data.emplace(std::string{dest}, detail::arg_string_converter_t<T>::convert(value));
};
m_dest_vec_func = [](const std::string_view dest, argument_storage_t& storage, const std::vector<std::string>& values)
{
if (storage.m_data.contains(dest))
{
auto& data = storage.m_data[dest];
if (!std::holds_alternative<std::vector<T>>(data))
throw detail::type_error("Invalid type conversion. Trying to add type " + blt::type_string<T>() +
" but this does not match existing type index '" + std::to_string(data.index()) + "'!");
auto& converted_values = std::get<std::vector<T>>(data);
for (const auto& value : values)
converted_values.push_back(detail::arg_string_converter_t<T>::convert(value));
}
else
{
std::vector<T> converted_values;
for (const auto& value : values)
converted_values.push_back(detail::arg_string_converter_t<T>::convert(value));
storage.m_data.emplace(std::string{dest}, std::move(converted_values));
}
};
return *this;
}
argument_builder_t& make_flag()
{
return set_action(action_t::STORE_TRUE);
}
argument_builder_t& set_action(action_t action);
argument_builder_t& set_required(const bool required)
{
m_required = required;
return *this;
}
argument_builder_t& set_nargs(const nargs_v nargs)
{
m_nargs = nargs;
return *this;
}
argument_builder_t& set_metavar(const std::string& metavar)
{
m_metavar = metavar;
return *this;
}
argument_builder_t& set_help(const std::string& help)
{
m_help = help;
return *this;
}
argument_builder_t& set_choices(const std::vector<std::string>& choices)
{
m_choices = hashset_t<std::string>{};
for (const auto& choice : choices)
m_choices->emplace(choice);
return *this;
}
argument_builder_t& set_choices(const hashset_t<std::string>& choices)
{
m_choices = choices;
return *this;
}
template <typename... Args>
argument_builder_t& set_choices(Args&&... args)
{
m_choices = hashset_t<std::string>{};
((m_choices->emplace(detail::ensure_is_string(std::forward<Args>(args)))), ...);
return *this;
}
argument_builder_t& set_default(const detail::arg_data_t& default_value)
{
m_default_value = default_value;
return *this;
}
argument_builder_t& set_const(const detail::arg_data_t& const_value)
{
m_const_value = const_value;
return *this;
}
argument_builder_t& set_dest(const std::string_view& dest)
{
m_dest = dest;
return *this;
}
private:
action_t m_action = action_t::STORE;
bool m_required = false; // do we require this argument to be provided as an argument?
nargs_v m_nargs = 1; // number of arguments to consume
std::optional<std::string> m_metavar; // variable name to be used in the help string
std::optional<std::string> m_help; // help string to be used in the help string
std::optional<hashset_t<std::string>> m_choices; // optional allowed choices for this argument
std::optional<detail::arg_data_t> m_default_value;
std::optional<detail::arg_data_t> m_const_value;
std::optional<std::string> m_dest;
// dest, storage, value input
std::function<void(std::string_view, argument_storage_t&, std::string_view)> m_dest_func;
// dest, storage, value input
std::function<void(std::string_view, argument_storage_t&, const std::vector<std::string>& values)> m_dest_vec_func;
};
class argument_positional_storage_t
{
public:
explicit argument_positional_storage_t(std::vector<std::pair<std::string, argument_builder_t>> storage): positional_arguments(
std::move(storage))
{
}
argument_builder_t& peek()
{
return positional_arguments[current_positional].second;
}
argument_builder_t& next()
{
return positional_arguments[current_positional++].second;
}
[[nodiscard]] bool has_positional() const
{
return current_positional < positional_arguments.size();
}
[[nodiscard]] auto remaining() const
{
return iterate(positional_arguments).skip(current_positional);
}
private:
std::vector<std::pair<std::string, argument_builder_t>> positional_arguments;
size_t current_positional = 0;
};
class argument_parser_t
{
friend argument_subparser_t;
explicit argument_parser_t(const argument_subparser_t* parent): m_parent(parent)
{
}
public:
explicit argument_parser_t(const std::optional<std::string_view> description = {}, const std::optional<std::string_view> epilogue = {},
const std::optional<std::string_view> version = {},
const std::optional<std::string_view> usage = {}, const std::optional<std::string_view> name = {}):
m_name(name), m_usage(usage), m_description(description), m_epilogue(epilogue), m_version(version)
{
}
template <typename... Aliases>
argument_builder_t& add_flag(const std::string_view arg, Aliases... aliases)
{
static_assert(
std::conjunction_v<std::disjunction<std::is_convertible<Aliases, std::string>, std::is_constructible<
std::string, Aliases>>...>,
"Arguments must be of type string_view, convertible to string_view or be string_view constructable");
m_argument_builders.emplace_back(std::make_unique<argument_builder_t>());
m_argument_builders.back()->set_dest(arg);
m_flag_arguments.emplace(arg, m_argument_builders.back().get());
(m_flag_arguments.emplace(aliases, m_argument_builders.back().get()), ...);
return *m_argument_builders.back().get();
}
argument_builder_t& add_positional(const std::string_view arg)
{
auto& b = m_positional_arguments.emplace_back(arg, argument_builder_t{}).second;
b.set_dest(std::string{arg});
b.set_required(true);
b.set_nargs(1);
return b;
}
argument_subparser_t* add_subparser(std::string_view dest);
argument_parser_t& with_help()
{
add_flag("--help", "-h").set_action(action_t::HELP).set_help("Show this help menu and exit");
return *this;
}
argument_parser_t& with_version()
{
add_flag("--version").set_action(action_t::VERSION);
return *this;
}
argument_storage_t parse(argument_consumer_t& consumer); // NOLINT
argument_storage_t parse(const std::vector<std::string_view>& args)
{
std::vector<argument_string_t> arg_strings;
arg_strings.reserve(args.size());
for (const auto& arg : args)
arg_strings.emplace_back(arg, allowed_flag_prefixes);
argument_consumer_t consumer{arg_strings};
return parse(consumer);
}
argument_storage_t parse(const std::vector<std::string>& args)
{
std::vector<argument_string_t> arg_strings;
arg_strings.reserve(args.size());
for (const auto& arg : args)
arg_strings.emplace_back(arg, allowed_flag_prefixes);
argument_consumer_t consumer{arg_strings};
return parse(consumer);
}
argument_storage_t parse(const int argc, const char** argv)
{
std::vector<argument_string_t> arg_strings;
arg_strings.reserve(argc);
for (int i = 0; i < argc; ++i)
arg_strings.emplace_back(argv[i], allowed_flag_prefixes);
argument_consumer_t consumer{arg_strings};
return parse(consumer);
}
void print_help();
void print_usage();
void print_version() const;
argument_parser_t& set_name(const std::optional<std::string>& name)
{
m_name = name;
return *this;
}
argument_parser_t& set_usage(const std::optional<std::string>& usage)
{
m_usage = usage;
return *this;
}
[[nodiscard]] const std::optional<std::string>& get_usage() const
{
return m_usage;
}
argument_parser_t& set_description(const std::optional<std::string>& description)
{
m_description = description;
return *this;
}
[[nodiscard]] const std::optional<std::string>& get_description() const
{
return m_description;
}
argument_parser_t& set_epilogue(const std::optional<std::string>& epilogue)
{
m_epilogue = epilogue;
return *this;
}
[[nodiscard]] const std::optional<std::string>& get_epilogue() const
{
return m_epilogue;
}
argument_parser_t& set_version(const std::optional<std::string>& version)
{
m_epilogue = version;
return *this;
}
[[nodiscard]] const std::optional<std::string>& get_version() const
{
return m_version;
}
[[nodiscard]] const hashset_t<char>& get_allowed_flag_prefixes() const
{
return allowed_flag_prefixes;
}
private:
void handle_compound_flags(hashset_t<std::string>& found_flags, argument_storage_t& parsed_args, argument_consumer_t& consumer,
const argument_string_t& arg);
void parse_flag(argument_storage_t& parsed_args, argument_consumer_t& consumer, std::string_view arg);
void parse_positional(argument_storage_t& parsed_args, argument_consumer_t& consumer, argument_positional_storage_t& storage,
std::string_view arg);
static void handle_missing_and_default_args(hashmap_t<std::string_view, argument_builder_t*>& arguments,
const hashset_t<std::string>& found, argument_storage_t& parsed_args, std::string_view type);
static expected<std::vector<std::string>, std::string> consume_until_flag_or_end(argument_consumer_t& consumer,
hashset_t<std::string>* allowed_choices);
static std::vector<std::string> consume_argc(i32 argc, argument_consumer_t& consumer, hashset_t<std::string>* allowed_choices,
std::string_view arg);
std::optional<std::string> m_name;
std::optional<std::string> m_usage;
std::optional<std::string> m_description;
std::optional<std::string> m_epilogue;
std::optional<std::string> m_version;
const argument_subparser_t* m_parent = nullptr;
std::vector<std::pair<std::string_view, argument_subparser_t>> m_subparsers;
std::vector<std::unique_ptr<argument_builder_t>> m_argument_builders;
hashmap_t<std::string_view, argument_builder_t*> m_flag_arguments;
std::vector<std::pair<std::string, argument_builder_t>> m_positional_arguments;
hashset_t<char> allowed_flag_prefixes = {'-', '+', '/'};
};
class argument_subparser_t
{
friend argument_parser_t;
public:
explicit argument_subparser_t(const argument_parser_t& parent): m_parent(&parent)
{
}
template <typename... Aliases>
argument_parser_t* add_parser(const std::string_view name, Aliases... aliases)
{
static_assert(
std::conjunction_v<std::disjunction<std::is_convertible<Aliases, std::string_view>, std::is_constructible<
std::string_view, Aliases>>...>,
"Arguments must be of type string_view, convertible to string_view or be string_view constructable");
m_parsers.emplace_back(new argument_parser_t{this});
m_aliases[name] = m_parsers.back().get();
((m_aliases[std::string_view{aliases}] = m_parsers.back().get()), ...);
return m_parsers.back().get();
}
argument_subparser_t* set_help(const std::optional<std::string>& help)
{
m_help = help;
return this;
}
/**
* Parses the next argument using the provided argument consumer.
*
* This function uses an argument consumer to extract and process the next argument.
* If the argument is a flag or if it cannot be matched against the available parsers,
* an exception is thrown.
*
* @param consumer Reference to an argument_consumer_t object, which handles argument parsing.
* The consumer provides the next argument to be parsed.
*
* @throws detail::subparse_error If the argument is a flag or does not match any known parser.
*/
std::pair<argument_string_t, argument_storage_t> parse(argument_consumer_t& consumer); // NOLINT
private:
[[nodiscard]] hashmap_t<argument_parser_t*, std::vector<std::string_view>> get_allowed_strings() const;
// annoying compatability because im lazy
static std::vector<std::vector<std::string_view>> to_vec(const hashmap_t<argument_parser_t*, std::vector<std::string_view>>& map);
const argument_parser_t* m_parent;
std::optional<std::string> m_last_parsed_parser; // bad hack
std::optional<std::string> m_help;
std::vector<std::unique_ptr<argument_parser_t>> m_parsers;
hashmap_t<std::string_view, argument_parser_t*> m_aliases;
};
}
#endif //BLT_PARSE_ARGPARSE_V2_H

View File

@ -23,41 +23,41 @@ namespace blt
static inline constexpr std::uint32_t PRINT_WALL = 0x4;
// print out the thread CPU time
static inline constexpr std::uint32_t PRINT_THREAD = 0x8;
enum class sort_by
{
CYCLES,
WALL,
THREAD
};
// 32 bit currently not supported
typedef std::int64_t pf_time_t;
typedef std::uint64_t pf_cycle_t;
struct interval_t
{
pf_time_t wall_start = 0;
pf_time_t wall_end = 0;
pf_time_t wall_total = 0;
pf_time_t thread_start = 0;
pf_time_t thread_end = 0;
pf_time_t thread_total = 0;
pf_cycle_t cycles_start = 0;
pf_cycle_t cycles_end = 0;
pf_cycle_t cycles_total = 0;
std::uint64_t count = 0;
std::string interval_name;
interval_t() = default;
interval_t(pf_time_t wallStart, pf_time_t wallEnd, pf_time_t wallTotal, pf_time_t threadStart, pf_time_t threadEnd, pf_time_t threadTotal,
pf_cycle_t cyclesStart, pf_cycle_t cyclesEnd, pf_cycle_t cyclesTotal, uint64_t count, std::string intervalName);
};
struct cycle_interval_t
{
pf_cycle_t cycles_start = 0;
@ -66,81 +66,86 @@ namespace blt
std::uint64_t count = 0;
std::string interval_name;
};
struct profile_t
{
std::vector<interval_t*> intervals;
std::vector<cycle_interval_t*> cycle_intervals;
std::string name;
explicit profile_t(std::string name): name(std::move(name))
{}
{
}
profile_t(const profile_t& p) = delete;
profile_t& operator=(const profile_t& p) = delete;
~profile_t();
};
interval_t* createInterval(profile_t& profiler, std::string interval_name);
void startInterval(interval_t* interval);
inline interval_t* startInterval(profile_t& profiler, std::string interval_name)
{
auto* p = createInterval(profiler, std::move(interval_name));
startInterval(p);
return p;
}
void endInterval(interval_t* interval);
void printProfile(profile_t& profiler, std::uint32_t flags = AVERAGE_HISTORY | PRINT_CYCLES | PRINT_THREAD | PRINT_WALL,
sort_by sort = sort_by::CYCLES, blt::logging::log_level log_level = blt::logging::log_level::NONE);
void writeProfile(std::ifstream& stream, const profile_t& profiler);
void writeProfile(std::ostream& stream, profile_t& profiler,
std::uint32_t flags = AVERAGE_HISTORY | PRINT_CYCLES | PRINT_THREAD | PRINT_WALL,
sort_by sort = sort_by::CYCLES);
void clearProfile(profile_t& profiler);
namespace _internal
{
void startInterval(const std::string& profile_name, const std::string& interval_name);
void endInterval(const std::string& profile_name, const std::string& interval_name);
void printProfile(const std::string& profile_name, std::uint32_t flags = AVERAGE_HISTORY | PRINT_CYCLES | PRINT_THREAD | PRINT_WALL,
sort_by sort = sort_by::CYCLES, blt::logging::log_level log_level = blt::logging::log_level::NONE);
void writeProfile(std::ifstream& stream, const std::string& profile_name);
void writeProfile(std::ostream& stream, const std::string& profile_name,
std::uint32_t flags = AVERAGE_HISTORY | PRINT_CYCLES | PRINT_THREAD | PRINT_WALL,
sort_by sort = sort_by::CYCLES);
}
class auto_interval
{
private:
interval_t* iv;
public:
auto_interval(std::string interval_name, profile_t& profiler)
{
iv = blt::createInterval(profiler, std::move(interval_name));
blt::startInterval(iv);
}
explicit auto_interval(interval_t* iv): iv(iv)
{
blt::startInterval(iv);
}
auto_interval(const auto_interval& i) = delete;
auto_interval& operator=(const auto_interval& i) = delete;
~auto_interval()
{
blt::endInterval(iv);
}
private:
interval_t* iv;
public:
auto_interval(std::string interval_name, profile_t& profiler)
{
iv = blt::createInterval(profiler, std::move(interval_name));
blt::startInterval(iv);
}
explicit auto_interval(interval_t* iv): iv(iv)
{
blt::startInterval(iv);
}
auto_interval(const auto_interval& i) = delete;
auto_interval& operator=(const auto_interval& i) = delete;
~auto_interval()
{
blt::endInterval(iv);
}
};
}
#ifdef BLT_DISABLE_PROFILING
@ -152,22 +157,22 @@ namespace blt
/**
* Starts an interval to be measured, when ended the row will be added to the specified profile.
*/
#define BLT_START_INTERVAL(profileName, intervalName) blt::_internal::startInterval(profileName, intervalName)
#define BLT_START_INTERVAL(profileName, intervalName) blt::_internal::startInterval(profileName, intervalName)
/**
* Ends an interval, adds the interval to the profile.
*/
#define BLT_END_INTERVAL(profileName, intervalName) blt::_internal::endInterval(profileName, intervalName)
#define BLT_END_INTERVAL(profileName, intervalName) blt::_internal::endInterval(profileName, intervalName)
/**
* Prints the profile order from least time to most time.
* @param profileName the profile to print
* @param loggingLevel blt::logging::LOG_LEVEL to log with (default: BLT_NONE)
* @param averageHistory use the historical collection of interval rows in an average or just the latest? (default: false)
*/
#define BLT_PRINT_PROFILE(profileName, ...) blt::_internal::printProfile(profileName, ##__VA_ARGS__)
#define BLT_PRINT_PROFILE(profileName, ...) blt::_internal::printProfile(profileName, ##__VA_ARGS__)
/**
* writes the profile to an output stream, ordered from least time to most time, in CSV format.
*/
#define BLT_WRITE_PROFILE(stream, profileName) blt::_internal::writeProfile(stream, profileName)
#define BLT_WRITE_PROFILE(stream, profileName) blt::_internal::writeProfile(stream, profileName)
#endif
#endif //BLT_PROFILER_V2_H

View File

@ -546,6 +546,7 @@ namespace blt
template<typename T, bool WARN_ON_FAIL = false>
static inline T* allocate_huge_page(blt::size_t BLOCK_SIZE, blt::size_t HUGE_PAGE_SIZE = BLT_2MB_SIZE)
{
#ifdef __unix__
BLT_ASSERT((BLOCK_SIZE & (HUGE_PAGE_SIZE - 1)) == 0 && "Must be multiple of the huge page size!");
T* buffer = static_cast<T*>(mmap(nullptr, BLOCK_SIZE, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB | MAP_POPULATE, -1, 0));
@ -578,6 +579,8 @@ namespace blt
BLT_ERROR("Offset by %ld pages, resulting: %p", (reinterpret_cast<blt::size_t>(buffer) - ptr_size) / 4096, buffer);
}
return buffer;
#endif
return malloc(BLOCK_SIZE);
}
/**
@ -701,7 +704,7 @@ namespace blt
} else
buffer = reinterpret_cast<block*>(std::aligned_alloc(BLOCK_SIZE, BLOCK_SIZE));
#else
buffer = reinterpret_cast<block*>(std::aligned_alloc(BLOCK_SIZE, BLOCK_SIZE));
buffer = static_cast<block*>(_aligned_malloc(BLOCK_SIZE, BLOCK_SIZE));
#endif
construct(buffer);
#ifndef BLT_DISABLE_STATS

View File

@ -10,10 +10,12 @@
#include <stdexcept>
#include <vector>
#include <blt/std/allocator.h>
#include <blt/std/types.h>
#include <iostream>
#include <memory>
// TODO: blt::queue
#include <queue>
#include <stack>
namespace blt
{
@ -142,6 +144,185 @@ namespace blt
}
};
template<typename K, typename V>
class range_tree_t
{
public:
struct node_t
{
K k;
V v;
blt::i64 children = 0;
node_t(K k, V v): k(std::move(k)), v(std::move(v))
{}
};
void insert(K k, V v)
{
auto insert_point = nodes.begin();
auto insert_parent = insert_point;
while (insert_point != nodes.end())
{
// no children
if (insert_point->children == 0)
{
++insert_point->children;
++insert_point;
break;
} else if (insert_point->children == 1)
{
// 1 child case
insert_parent = insert_point;
// find if child is min & move to it
++insert_point;
bool min = insert_point->k < insert_parent->k;
if (k < insert_parent->k)
{
// if the parent's child is a min value, then we can safely move towards it
if (min)
continue;
else
{
// otherwise we can break and this will insert the new node as the new min.
++insert_parent->children;
break;
}
} else
{
// parents child is min, so we move past it
if (min)
{
insert_point = skip_children(insert_point);
// can break as we insert here
++insert_parent->children;
break;
} else
{
// parents child is max, we can safely move towards it
continue;
}
}
} else
{
insert_parent = insert_point;
++insert_point;
if (k < insert_parent->k)
continue;
else
insert_point = skip_children(insert_point);
}
}
nodes.insert(insert_point, {std::move(k), std::move(v)});
}
void print(std::ostream& out, bool pretty_print)
{
std::stack<blt::size_t> left;
blt::size_t indent = 0;
for (auto& v : nodes)
{
if (v.children > 0)
{
create_indent(out, indent, pretty_print) << "(";
indent++;
left.emplace(v.children);
out << v.k << ": " << v.v << end_indent(pretty_print);
} else
create_indent(out, indent, pretty_print) << v.k << ": " << v.v << end_indent(pretty_print);
while (!left.empty())
{
auto top = left.top();
left.pop();
if (top == 0)
{
indent--;
create_indent(out, indent, pretty_print) << ")" << end_indent(pretty_print);
continue;
} else
{
if (!pretty_print)
out << " ";
left.push(top - 1);
break;
}
}
}
while (!left.empty())
{
auto top = left.top();
left.pop();
if (top == 0)
{
indent--;
create_indent(out, indent, pretty_print) << ")" << end_indent(pretty_print);
continue;
} else
{
out << "TREE MISMATCH";
break;
}
}
out << '\n';
}
std::optional<V> search(const K& k)
{
auto point = nodes.begin();
while (point != nodes.end())
{
if (k == point->k)
return point->v;
if (point->children == 0)
return {};
auto parent = point;
++point;
auto min = point->k < parent->k;
if (k >= parent->k)
{
if (min)
point = skip_children(point);
}
}
return {};
}
private:
auto skip_children(typename std::vector<node_t>::iterator begin)
{
blt::i64 children_left = 0;
do
{
if (children_left != 0)
children_left--;
if (begin->children > 0)
children_left += begin->children;
++begin;
} while (children_left > 0);
return begin;
}
std::ostream& create_indent(std::ostream& out, blt::size_t amount, bool pretty_print)
{
if (!pretty_print)
return out;
for (blt::size_t i = 0; i < amount; i++)
out << '\t';
return out;
}
std::string_view end_indent(bool pretty_print)
{
return pretty_print ? "\n" : "";
}
std::vector<node_t> nodes;
};
}
#endif //BLT_BINARY_TREE_H

View File

@ -375,12 +375,12 @@ namespace blt
};
template<typename T, typename E>
class expected<T, E, false>
class expected<T, E, false> : public expected<T, E, true>
{
public:
using expected<T, E, true>::expected;
constexpr expected(const expected<T, E, false>& copy) = delete;
constexpr expected(const expected& copy) = delete;
expected& operator=(const expected& copy) = delete;
};

File diff suppressed because it is too large Load Diff

View File

@ -11,13 +11,9 @@
#include <initializer_list>
#include <iterator>
#include <cstring>
#include "queue.h"
#include "utility.h"
#include <blt/std/assert.h>
#include <cstdint>
#include <type_traits>
#include <utility>
#include <cstring>
namespace blt
{
@ -42,6 +38,7 @@ namespace blt
using const_pointer = const T*;
using reference = T&;
using const_reference = const T&;
using iterator = ptr_iterator<T>;
using const_iterator = ptr_iterator<const T>;
using reverse_iterator = std::reverse_iterator<iterator>;

View File

@ -19,38 +19,41 @@
#ifndef BLT_MEMORY_UTIL_H
#define BLT_MEMORY_UTIL_H
#include <blt/std/types.h>
#include <type_traits>
#include <array>
#include <string>
#include <cstring>
#include <algorithm>
#include <cstdint>
#include <climits>
#include <cmath>
#include <iostream>
#if defined(__clang__) || defined(__llvm__) || defined(__GNUC__) || defined(__GNUG__)
#if defined(__GNUC__) || defined(__GNUG__)
#include <byteswap.h>
#define SWAP16(val) bswap_16(val)
#define SWAP32(val) bswap_32(val)
#define SWAP64(val) bswap_64(val)
#else
#if (defined(__GNUC__) || defined(__GNUG__)) && !defined(WIN32)
#include <byteswap.h>
#define SWAP16(val) bswap_16(val)
#define SWAP32(val) bswap_32(val)
#define SWAP64(val) bswap_64(val)
#else
#define SWAP16(val) __builtin_bswap16(val)
#define SWAP32(val) __builtin_bswap32(val)
#define SWAP64(val) __builtin_bswap64(val)
#endif
#if __cplusplus >= 202002L
#endif
#if __cplusplus >= 202002L
#include <bit>
#define ENDIAN_LOOKUP(little_endian) (std::endian::native == std::endian::little && !little_endian) || \
(std::endian::native == std::endian::big && little_endian)
#else
#define ENDIAN_LOOKUP(little_endian) !little_endian
#endif
#else
#define ENDIAN_LOOKUP(little_endian) !little_endian
#endif
#elif defined(_MSC_VER)
#include <intrin.h>
#define SWAP16(val) _byteswap_ushort(val)
@ -61,60 +64,77 @@
namespace blt::mem
{
template <typename R, typename T>
static R type_cast(T type)
{
static_assert(std::is_trivially_copyable_v<T>, "Type must be trivially copyable to be type casted!");
static_assert(sizeof(T) == sizeof(R));
R r;
std::memcpy(&r, &type, sizeof(type));
return r;
}
template <typename T>
void reverse(T& out)
{
static_assert(std::is_trivially_copyable_v<T>, "Type must be trivially copyable to be reversible!");
// if we need to swap find the best way to do so
if constexpr (std::is_same_v<T, std::int16_t> || std::is_same_v<T, std::uint16_t>)
out = type_cast<T>(SWAP16(type_cast<std::uint16_t>(out)));
else if constexpr (std::is_same_v<T, std::int32_t> || std::is_same_v<T, std::uint32_t> || std::is_same_v<T, float>)
out = type_cast<T>(SWAP32(type_cast<std::uint32_t>(out)));
else if constexpr (std::is_same_v<T, std::int64_t> || std::is_same_v<T, std::uint64_t> || std::is_same_v<T, double>)
out = type_cast<T>(SWAP64(type_cast<std::uint64_t>(out)));
else
{
std::array<std::byte, sizeof(T)> data;
std::memcpy(data.data(), &out, sizeof(T));
std::reverse(data.begin(), data.end());
std::memcpy(&out, data.data(), sizeof(T));
}
}
// Used to grab the byte-data of any T element. Defaults to Big Endian, however can be configured to use little endian
template<bool little_endian = false, typename BYTE_TYPE, typename T>
inline static int toBytes(const T& in, BYTE_TYPE* out)
template <bool little_endian = false, typename BYTE_TYPE, typename T>
int toBytes(const T& in, BYTE_TYPE* out)
{
if constexpr (!(std::is_same_v<BYTE_TYPE, std::int8_t> || std::is_same_v<BYTE_TYPE, std::uint8_t>))
static_assert("Must provide a signed/unsigned int8 type");
std::memcpy(out, (void*) &in, sizeof(T));
std::memcpy(out, &in, sizeof(T));
if constexpr (ENDIAN_LOOKUP(little_endian))
{
// TODO: this but better.
for (size_t i = 0; i < sizeof(T) / 2; i++)
std::swap(out[i], out[sizeof(T) - 1 - i]);
}
return 0;
}
// Used to cast the binary data of any T object, into a T object. Assumes data is in big ending (configurable)
template<bool little_endian = false, typename BYTE_TYPE, typename T>
inline static int fromBytes(const BYTE_TYPE* in, T& out)
template <bool little_endian = false, typename BYTE_TYPE, typename T>
int fromBytes(const BYTE_TYPE* in, T& out)
{
if constexpr (!(std::is_same_v<BYTE_TYPE, std::int8_t> || std::is_same_v<BYTE_TYPE, std::uint8_t>))
static_assert("Must provide a signed/unsigned int8 type");
std::array<BYTE_TYPE, sizeof(T)> data;
std::memcpy(data.data(), in, sizeof(T));
std::memcpy(&out, in, sizeof(T));
if constexpr (ENDIAN_LOOKUP(little_endian))
{
// if we need to swap find the best way to do so
if constexpr (std::is_same_v<T, int16_t> || std::is_same_v<T, uint16_t>)
out = SWAP16(*reinterpret_cast<T*>(data.data()));
else if constexpr (std::is_same_v<T, int32_t> || std::is_same_v<T, uint32_t>)
out = SWAP32(*reinterpret_cast<T*>(data.data()));
else if constexpr (std::is_same_v<T, int64_t> || std::is_same_v<T, uint64_t>)
out = SWAP64(*reinterpret_cast<T*>(data.data()));
else
{
std::reverse(data.begin(), data.end());
out = *reinterpret_cast<T*>(data.data());
}
reverse(out);
}
return 0;
}
template<bool little_endian = false, typename BYTE_TYPE, typename T>
inline static int fromBytes(const BYTE_TYPE* in, T* out)
template <bool little_endian = false, typename BYTE_TYPE, typename T>
static int fromBytes(const BYTE_TYPE* in, T* out)
{
return fromBytes(in, *out);
return fromBytes<little_endian>(in, *out);
}
inline static size_t next_byte_allocation(size_t prev_size, size_t default_allocation_block = 8192, size_t default_size = 16)
inline std::size_t next_byte_allocation(std::size_t prev_size, std::size_t default_allocation_block = 8192, std::size_t default_size = 16)
{
if (prev_size < default_size)
return default_size;
@ -122,139 +142,333 @@ namespace blt::mem
return prev_size * 2;
return prev_size + default_allocation_block;
}
template<typename R, typename T>
inline static R type_cast(T type)
template <bool bits = true, typename OStream, typename Value>
void print_bytes(OStream& stream, const Value& value)
{
static_assert(sizeof(T) == sizeof(R));
R r;
std::memcpy(&r, &type, sizeof(type));
return r;
constexpr auto size = sizeof(Value);
std::string line;
for (std::size_t i = 0; i < size; i++)
{
std::uint8_t byte;
std::memcpy(&byte, reinterpret_cast<const char*>(&value) + i, 1);
if constexpr (bits)
{
for (std::ptrdiff_t j = CHAR_BIT - 1; j >= 0; j--)
{
const auto bit = (byte >> j) & 1;
line += std::to_string(bit);
}
}
else
{
const auto byte_str = std::to_string(byte);
const auto amount = CHAR_BIT - byte_str.size();
for (std::size_t j = 0; j < static_cast<std::size_t>(std::ceil(static_cast<double>(amount) / 2.0)); j++)
line += ' ';
line += byte_str;
for (std::size_t j = 0; j < static_cast<std::size_t>(std::floor(static_cast<double>(amount) / 2.0)); j++)
line += ' ';
}
if (i != size - 1)
line += " : ";
}
for (std::size_t i = 0; i < size; i++)
{
auto index = std::to_string(i);
const auto amount = CHAR_BIT - index.size();
for (std::size_t j = 0; j < static_cast<std::size_t>(std::ceil(static_cast<double>(amount) / 2.0)); j++)
stream << ' ';
stream << index;
for (std::size_t j = 0; j < static_cast<std::size_t>(std::floor(static_cast<double>(amount) / 2.0)); j++)
stream << ' ';
if (i != size - 1)
stream << " | ";
}
stream << '\n';
stream << line;
stream << '\n';
}
// TODO: check if the platform actually has enough room in their pointers for this. plus endianness
struct bit_storage
{
static constexpr std::size_t START_BIT = 48;
static constexpr std::size_t END_BIT = sizeof(std::uintptr_t) * CHAR_BIT;
static constexpr std::size_t AVAILABLE_BITS = END_BIT - START_BIT;
static constexpr std::size_t make_storage_ones()
{
std::size_t result = 0;
for (std::size_t i = START_BIT; i < END_BIT; i++)
result |= 1ul << i;
return result;
}
static constexpr std::size_t make_ptr_ones()
{
std::size_t result = 0;
for (std::size_t i = 0; i < START_BIT; i++)
result |= 1ul << i;
return result;
}
bit_storage(): bits(0)
{
}
explicit bit_storage(const std::uint16_t bits): bits(bits)
{
}
std::uint16_t bits : AVAILABLE_BITS;
};
template <typename Ptr>
struct pointer_storage
{
static constexpr std::size_t STORAGE_ALL_ONES = bit_storage::make_storage_ones();
static constexpr std::size_t PTR_ALL_ONES = bit_storage::make_ptr_ones();
explicit pointer_storage(Ptr* ptr): ptr_bits(reinterpret_cast<std::uintptr_t>(ptr))
{
}
explicit pointer_storage(Ptr* ptr, const bit_storage bits): ptr_bits(reinterpret_cast<std::uintptr_t>(ptr))
{
storage(bits);
}
template <typename T, std::enable_if_t<!std::is_same_v<T, bit_storage>, bool> = false>
explicit pointer_storage(Ptr* ptr, const T& type): ptr_bits(reinterpret_cast<std::uintptr_t>(ptr))
{
storage(type);
}
[[nodiscard]] bit_storage storage() const noexcept
{
bit_storage storage{};
storage.bits = (ptr_bits & STORAGE_ALL_ONES) >> bit_storage::START_BIT;
return storage;
}
[[nodiscard]] bool bit(const std::size_t index) const noexcept
{
if (index >= bit_storage::END_BIT)
return false;
return (ptr_bits >> (bit_storage::START_BIT + index)) & 1;
}
pointer_storage& bit(const std::size_t index, const bool b) noexcept
{
if (index >= bit_storage::END_BIT)
return *this;
ptr_bits &= ~(1ul << (bit_storage::START_BIT + index));
ptr_bits |= (static_cast<std::uintptr_t>(b) << (bit_storage::START_BIT + index));
return *this;
}
template<typename T, std::enable_if_t<!std::is_same_v<T, bit_storage>, bool> = false>
pointer_storage& storage(const T& type)
{
static_assert(sizeof(T) <= sizeof(std::uintptr_t), "Type takes too many bits to be stored!");
static constexpr std::uintptr_t store_bits = (2 << (bit_storage::AVAILABLE_BITS - 1)) - 1;
std::uintptr_t bit_store = 0;
bit_storage store{};
std::memcpy(&bit_store, &type, sizeof(T));
store.bits |= bit_store & store_bits;
storage(store);
return *this;
}
pointer_storage& storage(const bit_storage bits) noexcept
{
ptr_bits = ((ptr_bits & PTR_ALL_ONES) | (static_cast<std::uintptr_t>(bits.bits) << bit_storage::START_BIT));
return *this;
}
pointer_storage& operator=(const bit_storage bits) noexcept
{
storage(bits);
return *this;
}
template<typename T, std::enable_if_t<!std::is_same_v<T, bit_storage>, bool> = false>
pointer_storage& operator=(const T& type)
{
storage(type);
return *this;
}
pointer_storage& clear_storage() noexcept
{
ptr_bits &= PTR_ALL_ONES;
return *this;
}
// changes pointer without changing the tag bits
pointer_storage& pointer(Ptr* ptr) noexcept
{
const bit_storage old_storage = storage();
ptr_bits = (reinterpret_cast<std::uintptr_t>(ptr) & PTR_ALL_ONES) | old_storage.bits;
return *this;
}
pointer_storage& operator=(Ptr* ptr) noexcept
{
pointer(ptr);
return *this;
}
pointer_storage& clear_pointer() noexcept
{
ptr_bits &= STORAGE_ALL_ONES;
return *this;
}
Ptr* get() const noexcept
{
return reinterpret_cast<Ptr*>(ptr_bits & PTR_ALL_ONES);
}
Ptr& operator*() const noexcept
{
return *get();
}
Ptr* operator->() const noexcept
{
return get();
}
private:
std::uintptr_t ptr_bits;
};
}
namespace blt
{
template<typename V>
template <typename V>
struct ptr_iterator
{
public:
using iterator_category = std::random_access_iterator_tag;
using difference_type = blt::ptrdiff_t;
using value_type = V;
using pointer = value_type*;
using reference = value_type&;
using iter_reference = ptr_iterator&;
explicit ptr_iterator(V* v): _v(v)
{}
reference operator*() const
{
return *_v;
}
pointer operator->()
{
return _v;
}
ptr_iterator& operator++()
{
_v++;
return *this;
}
ptr_iterator& operator--()
{
_v--;
return *this;
}
ptr_iterator operator++(int)
{
auto tmp = *this;
++(*this);
return tmp;
}
ptr_iterator operator--(int)
{
auto tmp = *this;
--(*this);
return tmp;
}
iter_reference operator+=(difference_type amount)
{
_v += amount;
return *this;
}
iter_reference operator-=(difference_type amount)
{
_v -= amount;
return *this;
}
reference operator[](difference_type index)
{
return *(_v + index);
}
reference operator[](blt::size_t index)
{
return *(_v + index);
}
friend bool operator<(const ptr_iterator& a, const ptr_iterator& b)
{
return b._v - a._v > 0;
}
friend bool operator>(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v - b._v > 0;
}
friend bool operator<=(const ptr_iterator& a, const ptr_iterator& b)
{
return b._v - a._v >= 0;
}
friend bool operator>=(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v - b._v >= 0;
}
friend difference_type operator-(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v - b._v;
}
friend ptr_iterator operator+(const ptr_iterator& a, difference_type n)
{
return ptr_iterator(a._v + n);
}
friend ptr_iterator operator+(difference_type n, const ptr_iterator& a)
{
return ptr_iterator(a._v + n);
}
friend bool operator==(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v == b._v;
}
friend bool operator!=(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v != b._v;
}
private:
V* _v;
using iterator_category = std::random_access_iterator_tag;
using difference_type = std::ptrdiff_t;
using value_type = V;
using pointer = value_type*;
using reference = value_type&;
using iter_reference = ptr_iterator&;
explicit ptr_iterator(V* v): _v(v)
{
}
reference operator*() const
{
return *_v;
}
pointer operator->()
{
return _v;
}
ptr_iterator& operator++()
{
_v++;
return *this;
}
ptr_iterator& operator--()
{
_v--;
return *this;
}
ptr_iterator operator++(int)
{
auto tmp = *this;
++(*this);
return tmp;
}
ptr_iterator operator--(int)
{
auto tmp = *this;
--(*this);
return tmp;
}
iter_reference operator+=(difference_type amount)
{
_v += amount;
return *this;
}
iter_reference operator-=(difference_type amount)
{
_v -= amount;
return *this;
}
reference operator[](difference_type index)
{
return *(_v + index);
}
reference operator[](std::size_t index)
{
return *(_v + index);
}
friend bool operator<(const ptr_iterator& a, const ptr_iterator& b)
{
return b._v - a._v > 0;
}
friend bool operator>(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v - b._v > 0;
}
friend bool operator<=(const ptr_iterator& a, const ptr_iterator& b)
{
return b._v - a._v >= 0;
}
friend bool operator>=(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v - b._v >= 0;
}
friend difference_type operator-(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v - b._v;
}
friend ptr_iterator operator+(const ptr_iterator& a, difference_type n)
{
return ptr_iterator(a._v + n);
}
friend ptr_iterator operator+(difference_type n, const ptr_iterator& a)
{
return ptr_iterator(a._v + n);
}
friend bool operator==(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v == b._v;
}
friend bool operator!=(const ptr_iterator& a, const ptr_iterator& b)
{
return a._v != b._v;
}
private:
V* _v;
};
}

View File

@ -95,7 +95,11 @@ namespace blt
public:
void* allocate(blt::size_t bytes) // NOLINT
{
#ifdef WIN32
return _aligned_malloc(bytes, BLT_2MB_SIZE);
#else
return std::aligned_alloc(BLT_2MB_SIZE, bytes);
#endif
}
void deallocate(void* ptr, blt::size_t) // NOLINT

View File

@ -191,13 +191,7 @@ namespace blt::random
}
template<typename Container>
constexpr auto& select(Container& container)
{
return container[get_u64(0, container.size())];
}
template<typename Container>
constexpr const auto& select(const Container& container)
constexpr decltype(auto) select(Container& container)
{
return container[get_u64(0, container.size())];
}

View File

@ -213,25 +213,28 @@ namespace blt
{}
template<class R, class RCV = std::remove_cv_t<std::remove_reference_t<R>>, typename std::enable_if_t<
extent != dynamic_extent && span_detail::is_cont_v<RCV> &&
std::is_convertible_v<std::remove_pointer_t<decltype(std::data(std::declval<R>()))>(*)[], T(*)[]>, bool> = true>
extent != dynamic_extent && span_detail::is_cont_v<RCV>, bool> = true>
explicit constexpr span(R&& range): size_(std::size(range)), data_(std::data(range))
{}
template<class R, class RCV = std::remove_cv_t<std::remove_reference_t<R>>, typename std::enable_if_t<
extent == dynamic_extent && span_detail::is_cont_v<RCV> &&
std::is_convertible_v<std::remove_pointer_t<decltype(std::data(std::declval<R>()))>(*)[], T(*)[]>, bool> = true>
constexpr span(R&& range): size_(std::size(range)), data_(std::data(range)) // NOLINT
extent == dynamic_extent && span_detail::is_cont_v<RCV>, bool> = true>
constexpr span(R& range): size_(std::size(range)), data_(range.data()) // NOLINT
{}
template<size_type SIZE, typename std::enable_if_t<
template<class R, class RCV = std::remove_cv_t<std::remove_reference_t<R>>, typename std::enable_if_t<
extent == dynamic_extent && span_detail::is_cont_v<RCV>, bool> = true>
constexpr span(const R& range): size_(std::size(range)), data_(range.data()) // NOLINT
{}
template<blt::size_t SIZE, typename std::enable_if_t<
extent != dynamic_extent && SIZE == extent && std::is_const_v<element_type>, bool> = true>
explicit constexpr span(std::initializer_list<value_type> il) noexcept: size_(il.size()), data_(&il.begin()) // NOLINT
{}
template<size_type SIZE, typename std::enable_if_t<
extent == dynamic_extent && SIZE == extent && std::is_const_v<element_type>, bool> = true>
explicit span(std::initializer_list<value_type> il) noexcept: size_(il.size()), data_(&il.begin()) // NOLINT
// template<blt::size_t SIZE, typename std::enable_if_t<
// extent == dynamic_extent && SIZE == extent && std::is_const_v<pointer>, bool> = true>
span(std::initializer_list<T> il) noexcept: size_(il.size()), data_(std::data(il)) // NOLINT
{}
template<class U, std::size_t N, typename std::enable_if_t<
@ -355,20 +358,27 @@ namespace blt
};
template<class T, std::size_t N>
template<typename T, std::size_t N>
span(T (&)[N]) -> span<T, N>;
template<class T, std::size_t N>
template<typename T, std::size_t N>
span(const T (&)[N]) -> span<T, N>;
template<typename T, std::size_t N>
span(std::array<T, N>&) -> span<T, N>;
template<class T, std::size_t N>
template<typename T, std::size_t N>
span(const std::array<T, N>&) -> span<const T, N>;
template<class Cont>
template<typename Cont>
span(Cont&) -> span<typename Cont::value_type>;
template<class Cont>
template<typename Cont>
span(const Cont&) -> span<const typename Cont::value_type>;
template<typename T>
span(std::initializer_list<T>) -> span<const T>;
}
#endif //BLT_RANGES_H

View File

@ -252,6 +252,22 @@ namespace blt::string
#endif
}
inline std::string ensure_ends_with_path_separator(std::string_view string)
{
if (ends_with(string, '/'))
return std::string(string);
else
return std::string(string) += '/';
}
inline std::string ensure_ends_with_path_separator(std::string&& string)
{
if (ends_with(string, '/'))
return string;
else
return (std::move(string) + '/');
}
class match
{
private:

View File

@ -146,7 +146,7 @@ namespace blt::system
std::uint64_t dt;
};
#ifdef _MSC_VER
#if defined(_MSC_VER) || defined(WIN32)
using suseconds_t = std::size_t;
#endif

View File

@ -62,11 +62,11 @@ namespace blt
// (unique_lock acquires lock)
std::unique_lock lock(count_mutex);
std::size_t current_uses = use_count;
if (++threads_waiting == thread_count)
{
threads_waiting = 0;
use_count++;
++use_count;
cv.notify_all();
} else
{

View File

@ -19,9 +19,6 @@
#ifndef BLT_UTILITY_H
#define BLT_UTILITY_H
#include <blt/compatibility.h>
#include <blt/std/ranges.h>
#include <blt/std/expected.h>
#include <string>
#include <utility>
@ -59,13 +56,13 @@ namespace blt
namespace blt
{
template<typename T>
static BLT_CPP20_CONSTEXPR inline std::string type_string()
static std::string type_string()
{
return demangle(typeid(T).name());
}
template<typename T>
static BLT_CPP20_CONSTEXPR inline std::string type_string_raw()
static std::string type_string_raw()
{
return typeid(T).name();
}

View File

@ -1,71 +0,0 @@
/*
* Created by Brett on 16/01/23.
* Licensed under GNU General Public License V3.0
* See LICENSE file for license detail
*/
#ifndef BLT_WINDOW_H
#define BLT_WINDOW_H
#include <functional>
#include <vector>
#ifndef BLT_MAP_FUNC
#include <unordered_map>
#define BLT_MAP_FUNC std::unordered_map
#endif
#define KEY_MAP BLT_MAP_FUNC<int, bool>
namespace blt {
class window {
protected:
bool m_windowOpen = true;
int m_width = 800, m_height = 600;
std::vector<std::function<void(window*)>> renderFunctions{};
std::vector<std::function<void(window*, int, bool)>> keyListeners{};
std::vector<std::function<void(window*, int, bool)>> mouseListeners{};
KEY_MAP keysDown{};
KEY_MAP mouseDown{};
public:
window() = default;
window(int width_, int height_) {
m_width = width_;
m_height = height_;
}
virtual void createWindow() = 0;
virtual void startMainLoop() = 0;
virtual void destroyWindow() = 0;
virtual ~window() = 0;
virtual inline bool setResizeable(bool resizeEnabled) = 0;
virtual inline bool setWindowSize(int width_, int height_) = 0;
[[nodiscard]] inline int getWidth() const {return m_width;};
[[nodiscard]] inline int getHeight() const {return m_height;};
[[nodiscard]] virtual inline bool isWindowOpen() const {return m_windowOpen;};
virtual inline void closeWindow(){
m_windowOpen = false;
}
virtual inline void registerLoopFunction(const std::function<void(window*)>& func) {
renderFunctions.push_back(func);
}
virtual inline bool isKeyDown(int key) const { return keysDown.at(key); }
virtual inline bool isMouseDown(int button) const {return mouseDown.at(button);};
// Function signature is window pointer to this, key press, pressed/released (true/false)
virtual inline void registerKeyListener(const std::function<void(window*, int, bool)>& listener) {
keyListeners.push_back(listener);
}
// Function signature is window pointer to this, mouse button press, pressed/released (true/false)
virtual inline void registerMouseListener(const std::function<void(window*, int, bool)>& listener) {
mouseListeners.push_back(listener);
}
};
}
#endif

0
py_commit_helper.sh Executable file → Normal file
View File

View File

@ -0,0 +1,41 @@
/*
* <Short Description>
* Copyright (C) 2025 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#include <blt/fs/path_helper.h>
#include <blt/std/string.h>
#include <blt/compatibility.h>
namespace blt::fs
{
#ifdef BLT_WINDOWS
constexpr static char delim = '\\';
#else
constexpr static char delim = '/';
#endif
std::string base_name(const std::string& str)
{
return std::string(base_name_sv(str));
}
std::string_view base_name_sv(const std::string_view str)
{
const auto parts = string::split_sv(str, delim);
const auto file_parts = string::split_sv(parts.back(), '.');
return file_parts.front();
}
}

View File

@ -0,0 +1,65 @@
/*
* <Short Description>
* Copyright (C) 2025 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#include <blt/logging/logging.h>
#include <blt/std/types.h>
namespace blt::logging
{
struct logging_thread_context_t
{
logger_t logger;
};
std::string logger_t::to_string()
{
auto str = m_stream.str();
m_stream.str("");
m_stream.clear();
return str;
}
void logger_t::compile(std::string fmt)
{
m_fmt = std::move(fmt);
m_last_fmt_pos = 0;
}
void logger_t::consume_until_fmt()
{
const auto begin = m_fmt.find('{', m_last_fmt_pos);
const auto end = m_fmt.find('}', begin);
m_stream << std::string_view(m_fmt.data() + static_cast<ptrdiff_t>(m_last_fmt_pos), begin - m_last_fmt_pos);
m_last_fmt_pos = end;
}
logger_t& get_global_logger()
{
thread_local logging_thread_context_t context;
return context.logger;
}
void print(const std::string& fmt)
{
std::cout << fmt;
}
void newline()
{
std::cout << std::endl;
}
}

View File

@ -0,0 +1,23 @@
/*
* <Short Description>
* Copyright (C) 2025 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#include <blt/logging/status.h>
namespace blt::logging
{
}

File diff suppressed because it is too large Load Diff

View File

@ -89,11 +89,6 @@ namespace blt
profiler.cycle_intervals.clear();
}
void writeProfile(std::ifstream&, const profile_t&)
{
BLT_WARN("Write profile for V2 is currently a TODO");
}
void sort_intervals(std::vector<interval_t*>& intervals, sort_by sort, bool)
{
std::function<bool(const interval_t* a, const interval_t* b)> sort_func;
@ -136,27 +131,22 @@ namespace blt
container.thread = unit::S;
return container;
}
inline void println(const std::vector<std::string>&& lines, logging::log_level level) {
for (const auto& line : lines)
BLT_LOG_STREAM(level) << line << "\n";
}
void printProfile(profile_t& profiler, std::uint32_t flags, sort_by sort, blt::logging::log_level log_level)
void writeProfile(std::ostream& stream, profile_t& profiler, std::uint32_t flags, sort_by sort)
{
bool printHistory = flags & AVERAGE_HISTORY;
bool printCycles = flags & PRINT_CYCLES;
bool printThread = flags & PRINT_THREAD;
bool printWall = flags & PRINT_WALL;
sort_intervals(profiler.intervals, sort, printHistory);
auto units = determine_max_unit(profiler.intervals, printHistory);
std::string thread_unit_string = units.thread == unit::MS ? "ms" : units.thread == unit::NS ? "ns" : "s";
std::string wall_unit_string = units.wall == unit::MS ? "ms" : units.wall == unit::NS ? "ns" : "s";
auto thread_unit_divide = units.thread == unit::MS ? 1e6 : units.thread == unit::NS ? 1 : 1e9;
auto wall_unit_divide = units.wall == unit::MS ? 1e6 : units.wall == unit::NS ? 1 : 1e9;
string::TableFormatter formatter{profiler.name};
formatter.addColumn("Order");
if (printHistory)
@ -168,17 +158,17 @@ namespace blt
formatter.addColumn("CPU Time (" + thread_unit_string += ")");
if (printWall)
formatter.addColumn("Wall Time (" + wall_unit_string += ")");
for (size_t i = 0; i < profiler.intervals.size(); i++)
{
blt::string::TableRow row;
auto interval = profiler.intervals[i];
if (interval->count == 0)
continue;
INTERVAL_DIFFERENCE_MACRO(printHistory, interval);
row.rowValues.push_back(std::to_string(i + 1));
if (printHistory)
row.rowValues.push_back(std::to_string(interval->count));
@ -191,8 +181,17 @@ namespace blt
row.rowValues.push_back(std::to_string(wall / static_cast<double>(wall_unit_divide)));
formatter.addRow(row);
}
println(formatter.createTable(true, true), log_level);
auto lines = formatter.createTable(true, true);
for (const auto& line : lines)
stream << line << "\n";
}
void printProfile(profile_t& profiler, const std::uint32_t flags, sort_by sort, blt::logging::log_level log_level)
{
std::stringstream stream;
writeProfile(stream, profiler, flags, sort);
BLT_LOG_STREAM(log_level) << stream.str();
}
profile_t::~profile_t()
@ -226,7 +225,7 @@ namespace blt
blt::endInterval(profiles[profile_name].at(interval_name));
}
void _internal::writeProfile(std::ifstream& stream, const std::string& profile_name)
void _internal::writeProfile(std::ostream& stream, const std::string& profile_name, std::uint32_t flags, sort_by sort)
{
if (profiles.find(profile_name) == profiles.end())
return;
@ -234,7 +233,7 @@ namespace blt
profile_t profile{profile_name};
for (const auto& i : pref)
profile.intervals.push_back(i.second);
blt::writeProfile(stream, profile);
blt::writeProfile(stream, profile, flags, sort);
profiles.erase(profile_name);
}

View File

@ -13,7 +13,7 @@
#include <exception>
#include <cstring>
struct abort_exception : public std::exception
struct abort_exception final : public std::exception
{
public:
explicit abort_exception(const char* what)
@ -42,14 +42,18 @@ struct abort_exception : public std::exception
char* error{nullptr};
};
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__) && !defined(WIN32)
#define IS_GNU_BACKTRACE
#endif
#ifdef IS_GNU_BACKTRACE
#include <execinfo.h>
#include <cstdlib>
#endif
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#ifdef IS_GNU_BACKTRACE
#define BLT_STACK_TRACE(number) void* ptrs[number]; \
int size = backtrace(ptrs, number); \
char** messages = backtrace_symbols(ptrs, size);
@ -64,7 +68,7 @@ struct abort_exception : public std::exception
namespace blt
{
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#ifdef IS_GNU_BACKTRACE
static inline std::string _macro_filename(const std::string& path)
{
@ -79,7 +83,7 @@ namespace blt
void b_throw(const char* what, const char* path, int line)
{
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#ifdef IS_GNU_BACKTRACE
BLT_STACK_TRACE(50);
BLT_ERROR("An exception '%s' has occurred in file '%s:%d'", what, path, line);
@ -96,7 +100,7 @@ namespace blt
void b_assert_failed(const char* expression, const char* msg, const char* path, int line)
{
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#ifdef IS_GNU_BACKTRACE
BLT_STACK_TRACE(50);
BLT_ERROR("The assertion '%s' has failed in file '%s:%d'", expression, path, line);
@ -122,7 +126,7 @@ namespace blt
{
if (messages == nullptr)
return;
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#ifdef IS_GNU_BACKTRACE
for (int i = 1; i < size; i++)
{
int tabs = i - 1;
@ -167,13 +171,13 @@ namespace blt
void b_abort(const char* what, const char* path, int line)
{
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
#ifdef IS_GNU_BACKTRACE
BLT_STACK_TRACE(50);
#endif
BLT_FATAL("----{BLT ABORT}----");
BLT_FATAL("\tWhat: %s", what);
BLT_FATAL("\tcalled from %s:%d", path, line);
#if defined(__GNUC__) && !defined(__EMSCRIPTEN__)
BLT_FATAL("\tCalled from %s:%d", path, line);
#ifdef IS_GNU_BACKTRACE
printStacktrace(messages, size, path, line);
BLT_FREE_STACK_TRACE();

View File

@ -21,6 +21,8 @@
#include <sys/mman.h>
#else
#include <blt/std/assert.h>
#endif
namespace blt
@ -99,16 +101,23 @@ namespace blt
}
return buffer;
#else
(void)page_type;
(void)bytes;
BLT_ABORT("Platform not supported for huge page allocation!");
#endif
}
void mmap_free(void* ptr, blt::size_t bytes)
{
#ifdef __unix__
if (munmap(ptr, bytes))
{
BLT_ERROR_STREAM << "Failed to deallocate\n";
throw bad_alloc_t(handle_mmap_error());
}
#else
(void)ptr;
(void)bytes;
#endif
}
}

View File

@ -130,10 +130,13 @@ namespace blt
{
auto size = pos - from;
auto token = s.substr(from, size);
tokens.emplace_back(token);
if (!token.empty())
tokens.emplace_back(token);
from += size + delim.length();
}
tokens.emplace_back(s.substr(from));
auto str = s.substr(from);
if (!str.empty())
tokens.emplace_back(str);
return tokens;
}
@ -146,10 +149,13 @@ namespace blt
{
auto size = pos - from;
auto token = s.substr(from, size);
tokens.emplace_back(token);
if (!token.empty())
tokens.emplace_back(token);
from += size + 1;
}
tokens.emplace_back(s.substr(from));
auto str = s.substr(from);
if (!str.empty())
tokens.emplace_back(str);
return tokens;
}
@ -162,10 +168,13 @@ namespace blt
{
auto size = pos - from;
auto token = s.substr(from, size);
tokens.push_back(token);
if (!token.empty())
tokens.push_back(token);
from += size + delim.length();
}
tokens.push_back(s.substr(from));
auto str = s.substr(from);
if (!str.empty())
tokens.push_back(str);
return tokens;
}
@ -178,10 +187,13 @@ namespace blt
{
auto size = pos - from;
auto token = s.substr(from, size);
tokens.push_back(token);
if (!token.empty())
tokens.push_back(token);
from += size + 1;
}
tokens.push_back(s.substr(from));
auto str = s.substr(from);
if (!str.empty())
tokens.push_back(str);
return tokens;
}

View File

@ -6,7 +6,7 @@
#include <blt/std/system.h>
#include <blt/std/logging.h>
#ifndef _MSC_VER
#if !defined(_MSC_VER) && !defined(WIN32)
#include <sys/time.h> /* for struct timeval */
#include <sys/resource.h>
#else

23
tests/argparse_tests.cpp Normal file
View File

@ -0,0 +1,23 @@
/*
* <Short Description>
* Copyright (C) 2025 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#include <blt/parse/argparse_v2.h>
int main(){
blt::argparse::detail::test();
return 0;
}

25
tests/logger_tests.cpp Normal file
View File

@ -0,0 +1,25 @@
/*
* <Short Description>
* Copyright (C) 2025 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that 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 <https://www.gnu.org/licenses/>.
*/
#include <blt/logging/logging.h>
int main()
{
blt::logging::println("This is a println!");
blt::logging::println("This is a println with args '{}'", 42);
blt::logging::println("This is a println with multiple args '{}' '{}' '{}'", 42, 32.34231233, "Hello World!");
}