diff --git a/.gitignore b/.gitignore index 4933a2d..70315af 100644 --- a/.gitignore +++ b/.gitignore @@ -35,6 +35,7 @@ .idea /build +/build-ex /cmake-build-* /.build diff --git a/CMakeLists.txt b/CMakeLists.txt index a84f22b..7bc2e53 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -15,6 +15,7 @@ set(CMAKE_CXX_STANDARD 20) set(CMAKE_CXX_STANDARD_REQUIRED ON) set(CMAKE_CXX_EXTENSIONS OFF) set(CMAKE_POSITION_INDEPENDENT_CODE ON) +set(CMAKE_EXPERIMENTAL_CXX_MODULE_CMAKE_API ON) project(cpp-jam VERSION 0.0.1 @@ -40,6 +41,12 @@ find_package(Boost.DI CONFIG REQUIRED) find_package(qtils CONFIG REQUIRED) find_package(prometheus-cpp CONFIG REQUIRED) +if(CMAKE_CXX_COMPILER_ID STREQUAL "GNU") + add_compile_options(-fmodules-ts) +elseif(CMAKE_CXX_COMPILER_ID MATCHES "Clang|AppleClang") + add_compile_options(-fmodules) +endif() + add_library(headers INTERFACE) target_include_directories(headers INTERFACE $ diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 08f0356..87a038b 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -24,3 +24,8 @@ add_subdirectory(metrics) # Clocks and time subsystem add_subdirectory(clock) +# Subscription Engine subsystem +add_subdirectory(se) + +# Modules subsystem +add_subdirectory(modules) diff --git a/src/app/CMakeLists.txt b/src/app/CMakeLists.txt index 61fd884..3fc38cd 100644 --- a/src/app/CMakeLists.txt +++ b/src/app/CMakeLists.txt @@ -28,7 +28,9 @@ add_library(build_version add_library(app_configuration SHARED configuration.cpp) target_link_libraries(app_configuration - Boost::boost) + Boost::boost + fmt::fmt +) add_library(app_configurator SHARED configurator.cpp) target_link_libraries(app_configurator diff --git a/src/app/impl/application_impl.cpp b/src/app/impl/application_impl.cpp index 2f29cb1..b5a8181 100644 --- a/src/app/impl/application_impl.cpp +++ b/src/app/impl/application_impl.cpp @@ -20,12 +20,12 @@ namespace jam::app { ApplicationImpl::ApplicationImpl( - std::shared_ptr logsys, - std::shared_ptr config, - std::shared_ptr state_manager, - std::shared_ptr watchdog, - std::shared_ptr metrics_exposer, - std::shared_ptr system_clock) + qtils::StrictSharedPtr logsys, + qtils::StrictSharedPtr config, + qtils::StrictSharedPtr state_manager, + qtils::StrictSharedPtr watchdog, + qtils::StrictSharedPtr metrics_exposer, + qtils::StrictSharedPtr system_clock) : logger_(logsys->getLogger("Application", "application")), app_config_(std::move(config)), state_manager_(std::move(state_manager)), @@ -34,15 +34,13 @@ namespace jam::app { system_clock_(std::move(system_clock)), metrics_registry_(metrics::createRegistry()) { // Metric for exposing name and version of node - constexpr auto buildInfoMetricName = "jam_build_info"; - metrics_registry_->registerGaugeFamily( - buildInfoMetricName, - "A metric with a constant '1' value labeled by name, version"); - auto metric_build_info = metrics_registry_->registerGaugeMetric( - buildInfoMetricName, - {{"name", app_config_->nodeName()}, - {"version", app_config_->nodeVersion()}}); - metric_build_info->set(1); + metrics::GaugeHelper( + "jam_build_info", + "A metric with a constant '1' value labeled by name, version", + std::map{ + {"name", app_config_->nodeName()}, + {"version", app_config_->nodeVersion()}}) + ->set(1); } void ApplicationImpl::run() { diff --git a/src/app/impl/application_impl.hpp b/src/app/impl/application_impl.hpp index 43a4d38..105a1df 100644 --- a/src/app/impl/application_impl.hpp +++ b/src/app/impl/application_impl.hpp @@ -10,6 +10,8 @@ #include +#include + #include namespace jam { @@ -43,22 +45,22 @@ namespace jam::app { class ApplicationImpl final : public Application { public: - ApplicationImpl(std::shared_ptr logsys, - std::shared_ptr config, - std::shared_ptr state_manager, - std::shared_ptr watchdog, - std::shared_ptr metrics_exposer, - std::shared_ptr system_clock); + ApplicationImpl(qtils::StrictSharedPtr logsys, + qtils::StrictSharedPtr config, + qtils::StrictSharedPtr state_manager, + qtils::StrictSharedPtr watchdog, + qtils::StrictSharedPtr metrics_exposer, + qtils::StrictSharedPtr system_clock); void run() override; private: - std::shared_ptr logger_; - std::shared_ptr app_config_; - std::shared_ptr state_manager_; - std::shared_ptr watchdog_; - std::shared_ptr metrics_exposer_; - std::shared_ptr system_clock_; + qtils::StrictSharedPtr logger_; + qtils::StrictSharedPtr app_config_; + qtils::StrictSharedPtr state_manager_; + qtils::StrictSharedPtr watchdog_; + qtils::StrictSharedPtr metrics_exposer_; + qtils::StrictSharedPtr system_clock_; // Metrics std::unique_ptr metrics_registry_; diff --git a/src/app/impl/state_manager_impl.cpp b/src/app/impl/state_manager_impl.cpp index f5e2b73..0d43e14 100644 --- a/src/app/impl/state_manager_impl.cpp +++ b/src/app/impl/state_manager_impl.cpp @@ -92,7 +92,7 @@ namespace jam::app { } StateManagerImpl::StateManagerImpl( - std::shared_ptr logging_system) + qtils::StrictSharedPtr logging_system) : logger_(logging_system->getLogger("StateManager", "application")), logging_system_(std::move(logging_system)) { shuttingDownSignalsEnable(); diff --git a/src/app/impl/state_manager_impl.hpp b/src/app/impl/state_manager_impl.hpp index 13f5f26..50cbf54 100644 --- a/src/app/impl/state_manager_impl.hpp +++ b/src/app/impl/state_manager_impl.hpp @@ -9,10 +9,11 @@ #include "app/state_manager.hpp" #include -#include #include #include +#include + #include "utils/ctor_limiters.hpp" namespace soralog { @@ -29,7 +30,7 @@ namespace jam::app { public StateManager, public std::enable_shared_from_this { public: - StateManagerImpl(std::shared_ptr logging_system); + StateManagerImpl(qtils::StrictSharedPtr logging_system); ~StateManagerImpl() override; @@ -66,8 +67,8 @@ namespace jam::app { void shutdownRequestWaiting(); - std::shared_ptr logger_; - std::shared_ptr logging_system_; + qtils::StrictSharedPtr logger_; + qtils::StrictSharedPtr logging_system_; std::atomic state_ = State::Init; diff --git a/src/injector/node_injector.cpp b/src/injector/node_injector.cpp index 48ef7d5..0b5ca2f 100644 --- a/src/injector/node_injector.cpp +++ b/src/injector/node_injector.cpp @@ -29,7 +29,7 @@ namespace { template auto useConfig(C c) { - return boost::di::bind >().to( + return boost::di::bind>().to( std::move(c))[boost::di::override]; } @@ -96,6 +96,6 @@ namespace jam::injector { std::shared_ptr NodeInjector::injectApplication() { return pimpl_->injector_ - .template create >(); + .template create>(); } } // namespace jam::injector diff --git a/src/loaders/README b/src/loaders/README new file mode 100644 index 0000000..31be259 --- /dev/null +++ b/src/loaders/README @@ -0,0 +1 @@ +# loaders are locating here \ No newline at end of file diff --git a/src/loaders/loader.hpp b/src/loaders/loader.hpp new file mode 100644 index 0000000..8d75e93 --- /dev/null +++ b/src/loaders/loader.hpp @@ -0,0 +1,34 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "injector/node_injector.hpp" +#include "modules/module.hpp" + +namespace jam::loaders { + + class Loader { + public: + Loader(const Loader &) = delete; + Loader &operator=(const Loader &) = delete; + + virtual ~Loader() = default; + virtual void start() = 0; + + std::optional module_info() { + auto result = module_.getFunctionFromLibrary("module_info"); + if (result) { + return std::string((*result)()); + } + return std::nullopt; + } + + protected: + injector::NodeInjector injector_; + modules::Module module_; + }; +} // namespace jam::loaders diff --git a/src/log/logger.hpp b/src/log/logger.hpp index 5cce5fd..c35952e 100644 --- a/src/log/logger.hpp +++ b/src/log/logger.hpp @@ -9,19 +9,22 @@ #include #include -#include #include +#include +#include #include #include #include #include -#include "utils/ctor_limiters.hpp" #include "injector/dont_inject.hpp" +#include "utils/ctor_limiters.hpp" namespace jam::log { using soralog::Level; + using Logger = qtils::StrictSharedPtr; + enum class Error : uint8_t { WRONG_LEVEL = 1, WRONG_GROUP, WRONG_LOGGER }; outcome::result str2lvl(std::string_view str); @@ -58,23 +61,22 @@ namespace jam::log { return logging_system_->getLogger(logger_name, group_name, level); } - [[nodiscard]] - bool setLevelOfGroup(const std::string &group_name, Level level) const { + [[nodiscard]] bool setLevelOfGroup(const std::string &group_name, + Level level) const { return logging_system_->setLevelOfGroup(group_name, level); } - [[nodiscard]] - bool resetLevelOfGroup(const std::string &group_name) const { + [[nodiscard]] bool resetLevelOfGroup(const std::string &group_name) const { return logging_system_->resetLevelOfGroup(group_name); } - [[nodiscard]] - bool setLevelOfLogger(const std::string &logger_name, Level level) const { + [[nodiscard]] bool setLevelOfLogger(const std::string &logger_name, + Level level) const { return logging_system_->setLevelOfLogger(logger_name, level); } - [[nodiscard]] - bool resetLevelOfLogger(const std::string &logger_name) const { + [[nodiscard]] bool resetLevelOfLogger( + const std::string &logger_name) const { return logging_system_->resetLevelOfLogger(logger_name); } diff --git a/src/metrics/histogram_timer.hpp b/src/metrics/histogram_timer.hpp index e907969..446cc78 100644 --- a/src/metrics/histogram_timer.hpp +++ b/src/metrics/histogram_timer.hpp @@ -27,6 +27,12 @@ namespace jam::metrics { registry_->registerGaugeFamily(name, help); metric_ = registry_->registerGaugeMetric(name); } + GaugeHelper(const std::string &name, + const std::string &help, + const std::map &labels) { + registry_->registerGaugeFamily(name, help); + metric_ = registry_->registerGaugeMetric(name, labels); + } auto *operator->() const { return metric_; diff --git a/src/modules/CMakeLists.txt b/src/modules/CMakeLists.txt new file mode 100644 index 0000000..cd74ca1 --- /dev/null +++ b/src/modules/CMakeLists.txt @@ -0,0 +1,87 @@ +# +# Copyright Quadrivium LLC +# All Rights Reserved +# SPDX-License-Identifier: Apache-2.0 +# + +function(add_jam_module NAME) + set(MODULE_NAME ${NAME}) + + set(MODULE "${MODULE_NAME}_module") + + # Parse named arguments + cmake_parse_arguments( + # Prefix for parsed argument variables + MODULE + # List of flags (boolean arguments without values) + "" + # List of named arguments with a single value + "" + # List of named arguments with multiple values + "SOURCE;INCLUDE_DIRS;LIBRARIES;DEFINITIONS" + # Input arguments + ${ARGN} + ) + + if (NOT EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/module.cpp) + message(FATAL_ERROR "Not found `module.cpp` file (main file of module)") + endif () + if (NOT EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${MODULE_NAME}.hpp" OR NOT EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${MODULE_NAME}.cpp") + message(FATAL_ERROR "Not found `${MODULE_NAME}.hpp` nor `${MODULE_NAME}.cpp` file (class of module)") + endif () + + # Create a shared module library + add_library(${MODULE} MODULE # or SHARED + module.cpp + ${MODULE_NAME}.cpp + ${MODULE_SOURCE} + ) + + # Set exported symbols visibility + set_target_properties(${MODULE} PROPERTIES + CXX_VISIBILITY_PRESET hidden + VISIBILITY_INLINES_HIDDEN ON + ) + + # Set include directories + if (MODULE_INCLUDE_DIRS) + target_include_directories(${MODULE} PRIVATE + ${MODULE_INCLUDE_DIRS} + ) + endif () + + # Set definitions specified for module + if (MODULE_DEFINITIONS) + target_compile_definitions(${MODULE} PRIVATE + ${MODULE_DEFINITIONS} + ) + endif () + + # Link with libs + if (MODULE_LIBRARIES) + target_link_libraries(${MODULE} + ${MODULE_LIBRARIES} + ) + endif () + + # Set C++ standard + target_compile_features(${MODULE} PRIVATE + cxx_std_20 + ) + +endfunction() + +# -------------- Core-part of module subsystem -------------- + +add_library(modules + module_loader.cpp +) + +target_link_libraries(modules + qtils::qtils +) + +# -------------- Modules -------------- + +# Example module +add_subdirectory(example) \ No newline at end of file diff --git a/src/modules/README b/src/modules/README new file mode 100644 index 0000000..a7c5cfe --- /dev/null +++ b/src/modules/README @@ -0,0 +1 @@ +# modules are locating here \ No newline at end of file diff --git a/src/modules/example/CMakeLists.txt b/src/modules/example/CMakeLists.txt new file mode 100644 index 0000000..7ff734f --- /dev/null +++ b/src/modules/example/CMakeLists.txt @@ -0,0 +1,17 @@ +# +# Copyright Quadrivium LLC +# All Rights Reserved +# SPDX-License-Identifier: Apache-2.0 +# + +add_jam_module(example + SOURCE + example.cpp + INCLUDE_DIRS + ${CMAKE_SOURCE_DIR}/src + DEFINITIONS + SOME_FLAG=1 + LIBRARIES + qtils::qtils + soralog::soralog +) \ No newline at end of file diff --git a/src/modules/example/example.cpp b/src/modules/example/example.cpp new file mode 100644 index 0000000..87f7e2f --- /dev/null +++ b/src/modules/example/example.cpp @@ -0,0 +1,20 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "modules/example/example.hpp" + +namespace jam::modules { + std::shared_ptr ExampleModule::instance; + + ExampleModule::ExampleModule( + qtils::StrictSharedPtr loader, + qtils::StrictSharedPtr logging_system) + : loader_(loader), + logger_(logging_system->getLogger("ExampleModule", "example_module")) + + {} + +} // namespace jam::modules diff --git a/src/modules/example/example.hpp b/src/modules/example/example.hpp new file mode 100644 index 0000000..8e2958a --- /dev/null +++ b/src/modules/example/example.hpp @@ -0,0 +1,34 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include +#include + +namespace jam::modules { + class ExampleModule; + class ExampleModuleLoader; +} + +class BlockTree; + +namespace jam::modules { + + class ExampleModule : public Singleton { + public: + static std::shared_ptr instance; + CREATE_SHARED_METHOD(ExampleModule); + + ExampleModule(qtils::StrictSharedPtr loader, + qtils::StrictSharedPtr logging_system); + + qtils::StrictSharedPtr loader_; + log::Logger logger_; + }; + +} // namespace jam::modules diff --git a/src/modules/example/module.cpp b/src/modules/example/module.cpp new file mode 100644 index 0000000..e4afe0a --- /dev/null +++ b/src/modules/example/module.cpp @@ -0,0 +1,31 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#include + +#define MODULE_C_API extern "C" __attribute__((visibility("default"))) +#define MODULE_API __attribute__((visibility("default"))) + +MODULE_C_API const char *loader_id() { + return "ExampleLoader"; +} + +MODULE_C_API const char *module_info() { + return "ExampleModule v1.0"; +} + +MODULE_API std::weak_ptr query_module_instance( + std::shared_ptr loader, + std::shared_ptr block_tree) { + return jam::modules::ExampleModule::instance + ? jam::modules::ExampleModule::instance + : (jam::modules::ExampleModule::instance = jam::modules::ExampleModule::create_shared( + loader, block_tree)); +} + +MODULE_API void release_module_instance() { + jam::modules::ExampleModule::instance.reset(); +} diff --git a/src/modules/module.hpp b/src/modules/module.hpp new file mode 100644 index 0000000..4279464 --- /dev/null +++ b/src/modules/module.hpp @@ -0,0 +1,69 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include +#include +#include +#include +#include + +#include + +namespace jam::modules { + + class Module final { + public: + CREATE_SHARED_METHOD(Module) + + // Static method for Module object creation + static std::shared_ptr create( + const std::string &path, + std::unique_ptr handle, + const std::string &loader_id) { + return std::shared_ptr( + new Module(path, std::move(handle), loader_id)); + } + + // Getter for library path + const std::string &get_path() const { + return path_; + } + + // Getter for loader Id + const std::string &get_loader_id() const { + return loader_id_; + } + + // Get function address from library + template + std::optional getFunctionFromLibrary(const char *funcName) { + void *funcAddr = dlsym(handle_.get(), funcName); + if (!funcAddr) { + return std::nullopt; + } + return reinterpret_cast(funcAddr); + } + + private: + Module(const std::string &path, + std::unique_ptr handle, + const std::string &loader_id) + : path_(path), handle_(std::move(handle)), loader_id_(loader_id) {} + + std::string path_; // Library path + std::unique_ptr handle_; // Library handle + std::string loader_id_; // Loader ID + + Module(const Module &) = delete; + Module &operator=(const Module &) = delete; + Module(Module &&) = delete; + Module &operator=(Module &&) = delete; + }; + +} // namespace jam::modules diff --git a/src/modules/module_loader.cpp b/src/modules/module_loader.cpp new file mode 100644 index 0000000..b7c9da0 --- /dev/null +++ b/src/modules/module_loader.cpp @@ -0,0 +1,80 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "modules/module_loader.hpp" + +#define COMPONENT_NAME "ModuleLoader" + +OUTCOME_CPP_DEFINE_CATEGORY(jam::modules, ModuleLoader::Error, e) { + using E = jam::modules::ModuleLoader::Error; + switch (e) { + case E::PathIsNotADir: + return COMPONENT_NAME ": path is not a directory"; + case E::OpenLibraryFailed: + return COMPONENT_NAME ": open library failed"; + case E::NoLoaderIdExport: + return COMPONENT_NAME ": library doesn't provide loader_id function"; + case E::UnexpectedLoaderId: + return COMPONENT_NAME ": unexpected loader id"; + } + return COMPONENT_NAME ": unknown error"; +} + +namespace jam::modules { + + Result ModuleLoader::recursive_search( + const fs::path &dir_path, std::deque> &modules) { + if (!fs::exists(dir_path) || !fs::is_directory(dir_path)) { + return Error::PathIsNotADir; + } + + for (const auto &entry : fs::directory_iterator(dir_path)) { + const auto &entry_path = entry.path(); + const auto &entry_name = entry.path().filename().string(); + + if (entry_name[0] == '.' || entry_name[0] == '_') { + continue; + } + + if (fs::is_directory(entry)) { + OUTCOME_TRY(recursive_search(entry_path, modules)); + } else if (fs::is_regular_file(entry) + && entry_path.extension() == ".so") { + OUTCOME_TRY(load_module(entry_path.string(), modules)); + } + } + return outcome::success(); + } + + Result ModuleLoader::load_module( + const std::string &module_path, + std::deque> &modules) { + std::unique_ptr handle( + dlopen(module_path.c_str(), RTLD_LAZY), dlclose); + if (!handle) { + return Error::OpenLibraryFailed; + } + + typedef const char *(*LoaderIdFunc)(); + LoaderIdFunc loader_id_func = + (LoaderIdFunc)dlsym(handle.get(), "loader_id"); + + if (!loader_id_func) { + return Error::NoLoaderIdExport; + } + + const char *loader_id = loader_id_func(); + if (!loader_id) { + return Error::UnexpectedLoaderId; + } + + auto module = Module::create_shared(module_path, std::move(handle), loader_id); + modules.push_back(module); + return outcome::success(); + } + + +} // namespace jam::modules diff --git a/src/modules/module_loader.hpp b/src/modules/module_loader.hpp new file mode 100644 index 0000000..daae0d8 --- /dev/null +++ b/src/modules/module_loader.hpp @@ -0,0 +1,56 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#include "modules/module.hpp" + +namespace fs = std::filesystem; +template +using Result = outcome::result; + +namespace jam::modules { + + class ModuleLoader { + public: + enum class Error : uint8_t { + PathIsNotADir, + OpenLibraryFailed, + NoLoaderIdExport, + UnexpectedLoaderId, + }; + + explicit ModuleLoader(const std::string &dir_path) : dir_path_(dir_path) {} + + Result>> get_modules() { + std::deque> modules; + OUTCOME_TRY(recursive_search(fs::path(dir_path_), modules)); + return modules; + } + + private: + std::string dir_path_; + + Result recursive_search(const fs::path &dir_path, + std::deque> &modules); + Result load_module(const std::string &module_path, + std::deque> &modules); + }; + +} // namespace jam::modules + +OUTCOME_HPP_DECLARE_ERROR(jam::modules, ModuleLoader::Error); diff --git a/src/se/CMakeLists.txt b/src/se/CMakeLists.txt new file mode 100644 index 0000000..fe524f6 --- /dev/null +++ b/src/se/CMakeLists.txt @@ -0,0 +1,21 @@ +# +# Copyright Quadrivium LLC +# All Rights Reserved +# SPDX-License-Identifier: Apache-2.0 +# + +add_library(se_async + async_dispatcher.cpp + subscription.cpp + ) + +target_link_libraries(se_async + ) + +add_library(se_sync + sync_dispatcher.cpp + subscription.cpp + ) + +target_link_libraries(se_sync + ) diff --git a/src/se/async_dispatcher.cpp b/src/se/async_dispatcher.cpp new file mode 100644 index 0000000..a742632 --- /dev/null +++ b/src/se/async_dispatcher.cpp @@ -0,0 +1,20 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#include + +#include "impl/async_dispatcher_impl.hpp" +#include "subscription.hpp" + +namespace jam::se { + + std::shared_ptr getDispatcher() { + return std::make_shared< + AsyncDispatcher>(); + } + +} // namespace jam::se diff --git a/src/se/impl/async_dispatcher_impl.hpp b/src/se/impl/async_dispatcher_impl.hpp new file mode 100644 index 0000000..8169cca --- /dev/null +++ b/src/se/impl/async_dispatcher_impl.hpp @@ -0,0 +1,160 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "common.hpp" +#include "dispatcher.hpp" +#include "thread_handler.hpp" + +namespace jam::se { + + template + class AsyncDispatcher final : public IDispatcher, + utils::NoCopy, + utils::NoMove { + public: + static constexpr uint32_t kHandlersCount = kCount; + static constexpr uint32_t kPoolThreadsCount = kPoolSize; + + private: + using Parent = IDispatcher; + + struct SchedulerContext { + /// Scheduler to execute tasks + std::shared_ptr handler; + }; + + SchedulerContext handlers_[kHandlersCount]; + SchedulerContext pool_[kPoolThreadsCount]; + + std::atomic_int64_t temporary_handlers_tasks_counter_; + std::atomic is_disposed_; + + struct BoundContexts { + typename Parent::Tid next_tid_offset = 0u; + std::unordered_map contexts; + }; + utils::ReadWriteObject bound_; + + void uploadToHandler(const typename Parent::Tid tid, + std::chrono::microseconds timeout, + typename Parent::Task &&task, + typename Parent::Predicate &&pred) { + assert(tid != kExecuteInPool || !pred); + if (is_disposed_.load()) { + return; + } + + if (tid < kHandlersCount) { + pred ? handlers_[tid].handler->repeat( + timeout, std::move(task), std::move(pred)) + : handlers_[tid].handler->addDelayed(timeout, std::move(task)); + return; + } + + if (auto context = + bound_.sharedAccess([tid](const BoundContexts &bound) + -> std::optional { + if (auto it = bound.contexts.find(tid); + it != bound.contexts.end()) { + return it->second; + } + return std::nullopt; + })) { + pred ? context->handler->repeat( + timeout, std::move(task), std::move(pred)) + : context->handler->addDelayed(timeout, std::move(task)); + return; + } + + std::optional opt_task = std::move(task); + for (auto &handler : pool_) { + if (opt_task = + handler.handler->uploadIfFree(timeout, std::move(*opt_task)); + !opt_task) { + return; + } + } + + auto h = std::make_shared(); + ++temporary_handlers_tasks_counter_; + h->addDelayed(timeout, [this, h, task{std::move(*opt_task)}]() mutable { + if (!is_disposed_.load()) { + task(); + } + --temporary_handlers_tasks_counter_; + h->dispose(false); + }); + } + + public: + AsyncDispatcher() { + temporary_handlers_tasks_counter_.store(0); + is_disposed_ = false; + for (auto &h : handlers_) { + h.handler = std::make_shared(); + } + for (auto &h : pool_) { + h.handler = std::make_shared(); + } + } + + void dispose() override { + is_disposed_ = true; + for (auto &h : handlers_) { + h.handler->dispose(); + } + for (auto &h : pool_) { + h.handler->dispose(); + } + + while (temporary_handlers_tasks_counter_.load() != 0) { + std::this_thread::sleep_for(std::chrono::microseconds(0ull)); + } + } + + void add(typename Parent::Tid tid, typename Parent::Task &&task) override { + uploadToHandler( + tid, std::chrono::microseconds(0ull), std::move(task), nullptr); + } + + void addDelayed(typename Parent::Tid tid, + std::chrono::microseconds timeout, + typename Parent::Task &&task) override { + uploadToHandler(tid, timeout, std::move(task), nullptr); + } + + void repeat(typename Parent::Tid tid, + std::chrono::microseconds timeout, + typename Parent::Task &&task, + typename Parent::Predicate &&pred) override { + uploadToHandler(tid, timeout, std::move(task), std::move(pred)); + } + + std::optional bind(std::shared_ptr scheduler) override { + if (!scheduler) { + return std::nullopt; + } + + return bound_.exclusiveAccess( + [scheduler(std::move(scheduler))](BoundContexts &bound) { + const auto execution_tid = kHandlersCount + bound.next_tid_offset; + assert(bound.contexts.find(execution_tid) == bound.contexts.end()); + bound.contexts[execution_tid] = SchedulerContext{scheduler}; + ++bound.next_tid_offset; + return execution_tid; + }); + } + + bool unbind(Tid tid) override { + return bound_.exclusiveAccess([tid](BoundContexts &bound) { + return bound.contexts.erase(tid) == 1; + }); + } + }; + +} // namespace jam::se diff --git a/src/se/impl/common.hpp b/src/se/impl/common.hpp new file mode 100644 index 0000000..6e53912 --- /dev/null +++ b/src/se/impl/common.hpp @@ -0,0 +1,108 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include +#include +#include + +namespace jam::se::utils { + + template + inline std::shared_ptr reinterpret_pointer_cast( + const std::shared_ptr &ptr) noexcept { + return std::shared_ptr(ptr, reinterpret_cast(ptr.get())); + } + + template + inline std::weak_ptr make_weak(const std::shared_ptr &ptr) noexcept { + return ptr; + } + + struct NoCopy { + NoCopy(const NoCopy &) = delete; + NoCopy &operator=(const NoCopy &) = delete; + NoCopy() = default; + }; + + struct NoMove { + NoMove(NoMove &&) = delete; + NoMove &operator=(NoMove &&) = delete; + NoMove() = default; + }; + + template + struct SafeObject { + using Type = T; + + template + SafeObject(Args &&...args) : t_(std::forward(args)...) {} + + template + inline auto exclusiveAccess(F &&f) { + std::unique_lock lock(cs_); + return std::forward(f)(t_); + } + + template + inline auto sharedAccess(F &&f) const { + std::shared_lock lock(cs_); + return std::forward(f)(t_); + } + + T &unsafeGet() { + return t_; + } + + const T &unsafeGet() const { + return t_; + } + + private: + T t_; + mutable M cs_; + }; + + template + using ReadWriteObject = SafeObject; + + class WaitForSingleObject final : NoMove, NoCopy { + std::condition_variable wait_cv_; + std::mutex wait_m_; + bool flag_; + + public: + WaitForSingleObject() : flag_{true} {} + + bool wait(std::chrono::microseconds wait_timeout) { + std::unique_lock _lock(wait_m_); + return wait_cv_.wait_for(_lock, wait_timeout, [&]() { + auto prev = !flag_; + flag_ = true; + return prev; + }); + } + + void wait() { + std::unique_lock _lock(wait_m_); + wait_cv_.wait(_lock, [&]() { + auto prev = !flag_; + flag_ = true; + return prev; + }); + } + + void set() { + { + std::unique_lock _lock(wait_m_); + flag_ = false; + } + wait_cv_.notify_one(); + } + }; +} // namespace jam::se::utils diff --git a/src/se/impl/compile-time_murmur2.hpp b/src/se/impl/compile-time_murmur2.hpp new file mode 100644 index 0000000..4aa1beb --- /dev/null +++ b/src/se/impl/compile-time_murmur2.hpp @@ -0,0 +1,102 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include + +namespace jam::se::utils { + + class Hasher { + static constexpr /* h */ uint32_t __init__(uint32_t len) { + return 0 ^ len; + } + + template + static constexpr uint32_t __load__(__T &data, uint32_t offset) { + return data[offset + 0] | (data[offset + 1] << 8) + | (data[offset + 2] << 16) | (data[offset + 3] << 24); + } + + static constexpr uint32_t __mul__(uint32_t val1, uint32_t val2) { + return val1 * val2; + } + + static constexpr uint32_t __sl__(uint32_t value, uint32_t count) { + return (value << count); + } + + static constexpr uint32_t __sr__(uint32_t value, uint32_t count) { + return (value >> count); + } + + static constexpr uint32_t __xor__(uint32_t h, uint32_t k) { + return h ^ k; + } + + static constexpr uint32_t __xor_with_sr__(uint32_t k, uint32_t r) { + return __xor__(k, __sr__(k, r)); + } + + template + static constexpr /* h */ uint32_t __proc__(__Type &data, + uint32_t len, + uint32_t offset, + uint32_t h, + uint32_t m, + uint32_t r) { + return len >= 4 + ? __proc__( + data, + len - 4, + offset + 4, + __xor__(__mul__(h, m), + __mul__(__xor_with_sr__( + __mul__(__load__(data, offset), m), r), + m)), + m, + r) + : len == 3 ? __proc__(data, + len - 1, + offset, + __xor__(h, __sl__(data[offset + 2], 16)), + m, + r) + : len == 2 ? __proc__(data, + len - 1, + offset, + __xor__(h, __sl__(data[offset + 1], 8)), + m, + r) + : len == 1 + ? __proc__( + data, len - 1, offset, __xor__(h, data[offset]) * m, m, r) + : __xor__(__mul__(__xor_with_sr__(h, 13), m), + __sr__(__mul__(__xor_with_sr__(h, 13), m), 15)); + } + + public: + template + static constexpr uint32_t murmur2(__Type &data, uint32_t len) { + return __proc__(data, len, 0, __init__(len), 0x5bd1e995, 24); + } + }; + +} // namespace jam::se::utils + +#ifndef CT_MURMUR2 +#define CT_MURMUR2(x) \ + ::jam::se::utils::Hasher::murmur2(x, (sizeof(x) / sizeof(x[0])) - 1) +#endif // CT_MURMUR2 + +static_assert(CT_MURMUR2("Called the One Ring, or the Ruling Ring.") + == 1333588607); +static_assert( + CT_MURMUR2("Fashioned by Sauron a decade after the making of the Elven " + "rings in the fires of Mount Doom in Mordor and which") + == 1319897327); +static_assert(CT_MURMUR2("could only be destroyed in that same fire.") + == 702138758); diff --git a/src/se/impl/dispatcher.hpp b/src/se/impl/dispatcher.hpp new file mode 100644 index 0000000..5dda15c --- /dev/null +++ b/src/se/impl/dispatcher.hpp @@ -0,0 +1,39 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include + +#include "scheduler.hpp" + +namespace jam::se { + + struct IDispatcher { + using Tid = uint32_t; + using Task = IScheduler::Task; + using Predicate = IScheduler::Predicate; + + static constexpr Tid kExecuteInPool = std::numeric_limits::max(); + + virtual ~IDispatcher() = default; + + virtual std::optional bind(std::shared_ptr scheduler) = 0; + virtual bool unbind(Tid tid) = 0; + + virtual void dispose() = 0; + virtual void add(Tid tid, Task &&task) = 0; + virtual void addDelayed(Tid tid, + std::chrono::microseconds timeout, + Task &&task) = 0; + virtual void repeat(Tid tid, + std::chrono::microseconds timeout, + Task &&task, + Predicate &&pred) = 0; + }; + +} // namespace jam::se diff --git a/src/se/impl/scheduler.hpp b/src/se/impl/scheduler.hpp new file mode 100644 index 0000000..cdc141a --- /dev/null +++ b/src/se/impl/scheduler.hpp @@ -0,0 +1,42 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include +#include + +namespace jam::se { + + class IScheduler { + public: + using Task = std::function; + using Predicate = std::function; + virtual ~IScheduler() {} + + /// Stops sheduler work and tasks execution + virtual void dispose(bool wait_for_release = true) = 0; + + /// Checks if current scheduler executes task + virtual bool isBusy() const = 0; + + /// If scheduller is not busy it takes task for execution. Otherwise it + /// returns it back. + virtual std::optional uploadIfFree(std::chrono::microseconds timeout, + Task &&task) = 0; + + /// Adds delayed task to execution queue + virtual void addDelayed(std::chrono::microseconds timeout, Task &&t) = 0; + + /// Adds task that will be periodicaly called with timeout period after + /// timeout, until predicate return true + virtual void repeat(std::chrono::microseconds timeout, + Task &&t, + Predicate &&pred) = 0; + }; + +} // namespace jam::se diff --git a/src/se/impl/scheduler_impl.hpp b/src/se/impl/scheduler_impl.hpp new file mode 100644 index 0000000..d2c9841 --- /dev/null +++ b/src/se/impl/scheduler_impl.hpp @@ -0,0 +1,185 @@ +/** + * Copyright Quadrivium LLC + * All Rights Reserved + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "common.hpp" +#include "scheduler.hpp" + +namespace jam::se { + + class SchedulerBase : public IScheduler, utils::NoCopy, utils::NoMove { + private: + using Time = std::chrono::high_resolution_clock; + using Timepoint = std::chrono::time_point