/* * Author: Sven Gothel * Copyright (c) 2020 Gothel Software e.K. * Copyright (c) 2020 ZAFENA AB * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #ifndef JAU_ENV_HPP_ #define JAU_ENV_HPP_ #include #include #include #include #include extern "C" { #include } #include namespace jau { /** @defgroup SysUtils System and OS Utilities * System and OS related types and functionality. * * @{ */ /** * Base jau environment class, * merely to tag all environment settings by inheritance and hence documentation. *

* See main environment {@link environment} and * {@link environment::getExplodingProperties(const std::string & prefix_domain)}. *

*/ class root_environment { public: /** * Optional path to signal early termination, i.e. JVM shutdown. */ static void set_terminating() noexcept; /** * Returns true if program is terminating as detected via atexit() callback * or set_terminating() has been called. */ static bool is_terminating() noexcept; }; /** * Main jau environment class, * supporting environment variable access and fetching elapsed time using its stored startup-time. */ class environment : public root_environment { private: const std::string root_prefix_domain; environment(const std::string & root_prefix_domain) noexcept; static bool local_debug; static void envSet(std::string prefix_domain, std::string basepair) noexcept; static void envExplodeProperties(std::string prefix_domain, std::string list) noexcept; static bool getExplodingPropertiesImpl(const std::string & root_prefix_domain, const std::string & prefix_domain) noexcept; public: /** * Module startup time t0 in monotonic time using high precision and range of fraction_timespec. */ static const fraction_timespec startupTimeMonotonic; /** * Module startup time t0 in monotonic time in milliseconds. */ static const uint64_t startupTimeMilliseconds; /** * Returns elapsed monotonic time using fraction_timespec since module startup, * see {@link #startupTimeMonotonic} and getMonotonicTime(). *
             *    return getMonotonicTime() - startupTimeMonotonic;
             * 
*/ static fraction_timespec getElapsedMonotonicTime() noexcept { return getMonotonicTime() - startupTimeMonotonic; } /** * Returns elapsed monotonic time using fraction_timespec since module startup up to the given current_ts, see {@link #startupTimeMonotonic}. *
             *    return current_ts - startupTimeMonotonic;
             * 
*/ static fraction_timespec getElapsedMonotonicTime(const fraction_timespec& current_ts) noexcept { return current_ts - startupTimeMonotonic; } /** * Returns current elapsed monotonic time in milliseconds since module startup, see {@link #startupTimeMilliseconds}. */ static uint64_t getElapsedMillisecond() noexcept { return getCurrentMilliseconds() - startupTimeMilliseconds; } /** * Returns elapsed monotonic time in milliseconds since module startup comparing against the given timestamp, see {@link #startupTimeMilliseconds}. */ static uint64_t getElapsedMillisecond(const uint64_t& current_ts) noexcept { return current_ts - startupTimeMilliseconds; } /** * Returns the value of the environment's variable 'name'. *

* Note that only '[org.]tinyb.*' and 'direct_bt.*' Java JVM properties are passed via 'org.tinyb.BluetoothFactory' *

*

* Implementation attempts to also find a Unix conform environment name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found.
* * Dots are not allowed as valid Unix envrionment variable identifier. * If the property 'name' isn't found and if the 'name' contains a dot ('.'), * all dots ('.') will be replaced y underscore ('_') and looked up again.
* This allows Unix shell user to set the property 'direct_bt_debug' instead of 'direct_bt.debug'. *

*/ static std::string getProperty(const std::string & name) noexcept; /** * Returns the value of the environment's variable 'name', * or the 'default_value' if the environment variable's value is null. *

* Implementation uses {@link #getProperty(const std::string & name)} * and hence attempts to also find a Unix conform name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found. *

*/ static std::string getProperty(const std::string & name, const std::string & default_value) noexcept; /** * Returns the boolean value of the environment's variable 'name', * or the 'default_value' if the environment variable's value is null. *

* If the environment variable is set (value != null), * true is determined if the value equals 'true'. *

*

* Implementation uses {@link #getProperty(const std::string & name)} * and hence attempts to also find a Unix conform name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found. *

*/ static bool getBooleanProperty(const std::string & name, const bool default_value) noexcept; /** * Returns the int32_t value of the environment's variable 'name', * or the 'default_value' if the environment variable's value is null * or not within int32_t value range or within the given value range. *

* Implementation uses {@link #getProperty(const std::string & name)} * and hence attempts to also find a Unix conform name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found. *

*/ static int32_t getInt32Property(const std::string & name, const int32_t default_value, const int32_t min_allowed=INT32_MIN, const int32_t max_allowed=INT32_MAX) noexcept; /** * Returns the uint32_t value of the environment's variable 'name', * or the 'default_value' if the environment variable's value is null * or not within uint32_t value range or within the given value range. *

* Implementation uses {@link #getProperty(const std::string & name)} * and hence attempts to also find a Unix conform name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found. *

*/ static uint32_t getUint32Property(const std::string & name, const uint32_t default_value, const uint32_t min_allowed=0, const uint32_t max_allowed=UINT32_MAX) noexcept; /** * Returns the fraction_i64 value of the environment's variable 'name' in format `/`, * with white space allowed, if within given fraction_i64 value range. * * Otherwise returns the 'default_value' if the environment variable's value is null * or of invalid format or not within given fraction_i64 value range. *

* Implementation uses {@link #getProperty(const std::string & name)} * and hence attempts to also find a Unix conform name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found. *

*/ static fraction_i64 getFractionProperty(const std::string & name, const fraction_i64& default_value, const fraction_i64& min_allowed, const fraction_i64& max_allowed) noexcept; /** * Fetches exploding variable-name (prefix_domain) values. *

* Implementation uses {@link #getProperty(const std::string & name)} * and hence attempts to also find a Unix conform name, * e.g. 'direct_bt_debug' if ''direct_bt.debug' wasn't found. *

*

* If the value of a prefix_domain is neither 'true' or 'false', * it is treated as a list of sub-variable names including their optional value separated by comma ','. *

* If the value is not given for the sub-variable name, a boolean "true" will be used per default. *

*

* Example 1 *

             * Input Environment:
             *   "direct_bt.debug" := "jni,adapter.event,gatt.data=false,hci.event,mgmt.event=true"
             *
             * Result Environment:
             *   "direct_bt.debug.jni"           := "true"
             *   "direct_bt.debug.adapter.event" := "true"
             *   "direct_bt.debug.gatt.data"     := "false"
             *   "direct_bt.debug.hci.event"     := "true"
             *   "direct_bt.debug.mgmt.event"    := "true"
             *   "direct_bt.debug"               := "true" (will be overwritten)
             * 
* Example 2 *
             * Input Environment:
             *   "direct_bt.gatt" := "cmd.read.timeout=20000,cmd.write.timeout=20001,ringsize=256"
             *
             * Result Environment:
             *   "direct_bt.gatt.cmd.read.timeout"  := "20000"
             *   "direct_bt.gatt.cmd.write.timeout" := "20001"
             *   "direct_bt.gatt.ringsize"          := "256"
             *   "direct_bt.gatt"                   := "true" (will be overwritten)
             * 
*

*

* Each sub-variable name/value pair will be trimmed and if not zero-length * appended to the prefix_domain with a dot '.'.
* * Each new variable name will be set in the environment with value 'true'.
* * The prefix_domain will also be set to the new value 'true', hence gets overwritten. *

*

* This is automatically performed for environment::debug root_prefix_domain+".debug", * and environment::verbose root_prefix_domain+'.verbose', * e.g: 'direct_bt.debug' and verbose 'direct_bt.verbose'. *

* * @param prefix_domain the queried prefix domain, e.g. "direct_bt.debug" or "direct_bt.verbose" etc. * @return */ static bool getExplodingProperties(const std::string & prefix_domain) noexcept { return getExplodingPropertiesImpl("", prefix_domain); } /** * Static singleton initialization of this project's environment * with the given global root prefix_domain. *

* The root prefix_domain defines the value for environment::debug, environment::debug_jni and environment::verbose. *

*

* The resulting singleton instance will be constructed only once. *

* @param root_prefix_domain the project's global singleton root prefix_domain, e.g. "direct_bt". * Default value to "jau", only intended for subsequent queries. * Initial call shall utilize the actual project's root_prefix_domain! * @return the static singleton instance. */ static environment& get(const std::string root_prefix_domain="jau") noexcept { /** * Thread safe starting with C++11 6.7: * * If control enters the declaration concurrently while the variable is being initialized, * the concurrent execution shall wait for completion of the initialization. * * (Magic Statics) * * Avoiding non-working double checked locking. */ static environment e(root_prefix_domain); return e; } /** * Returns the project's global singleton root prefix_domain, used at first call of * environment::get(const std::string root_prefix_domain). */ const std::string & getRootPrefixDomain() const noexcept { return root_prefix_domain; } /** * Debug logging enabled or disabled. *

* Environment variable depending on the root prefix_domain, e.g. for "direct_bt" this is 'direct_bt.debug', boolean, default 'false', * see get(const std::string & root_prefix_domain). *

*

* Implementation uses {@link #getProperty(const std::string & name)} *

*

* Exploding variable-name values are implemented here, * see {@link #getExplodingProperties(const std::string & prefix_domain)}. *

*/ const bool debug; /** * JNI Debug logging enabled or disabled. *

* Environment variable depending on the root prefix_domain, e.g. for "direct_bt" this is 'direct_bt.debug.jni', boolean, default 'false', * see get(const std::string & root_prefix_domain). *

*

* Implementation uses getBooleanProperty(). *

*/ const bool debug_jni; /** * Verbose info logging enabled or disabled. *

* Environment variable depending on the root prefix_domain, e.g. for "direct_bt" this is 'direct_bt.verbose', boolean, default 'false', * see get(const std::string & root_prefix_domain). *

*

* Implementation uses {@link #getProperty(const std::string & name)} *

*

* VERBOSE is also enabled if DEBUG is enabled! *

*

* Exploding variable-name values are implemented here, * see {@link #getExplodingProperties(const std::string & prefix_domain)}. *

*/ const bool verbose; }; /**@}*/ } // namespace jau #endif /* JAU_ENV_HPP_ */