// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #include > 32), uint32_t(cpuid[1] >> 64), uint32_t(cpuid[1] >> 96), uint32_t(cpuid[3]), uint32_t(cpuid[3] >> 32), uint32_t(cpuid[3] >> 64), uint32_t(cpuid[3] >> 96), }; #endif #ifdef __x86_64__ log::debug { "8..01 [%08x|%08x|%08x|%08x] " "8..1c [%08x|%08x|%08x|%08x]", uint32_t(cpuid[5]), uint32_t(cpuid[5] >> 32), uint32_t(cpuid[5] >> 64), uint32_t(cpuid[5] >> 96), uint32_t(cpuid[6]), uint32_t(cpuid[6] >> 32), uint32_t(cpuid[6] >> 64), uint32_t(cpuid[6] >> 96), }; #endif } // // Version registry // template<> decltype(ircd::util::instance_list::allocator) ircd::util::instance_list::allocator {}; template<> decltype(ircd::util::instance_list::list) ircd::util::instance_list::list { allocator }; /// Straightforward construction of versions members; string is copied /// into member buffer with null termination. ircd::info::versions::versions(const string_view &name, const enum type &type, const long &monotonic, const std::array &semantic, const string_view &string) :name{name} ,type{type} ,monotonic{monotonic} ,semantic{semantic} { strlcpy(this->string, string); } /// Construction of versions members with closure for custom string generation. /// The version string must be stored into buffer with null termination. ircd::info::versions::versions(const string_view &name, const enum type &type, const long &monotonic, const std::array &semantic, const std::function &closure) :name{name} ,type{type} ,monotonic{monotonic} ,semantic{semantic} { closure(*this, this->string); } // // Primary information // decltype(ircd::info::server_agent) ircd::info::server_agent { BRANDING_NAME " (IRCd " BRANDING_VERSION ")" }; decltype(ircd::info::user_agent) ircd::info::user_agent { BRANDING_NAME " (IRCd " BRANDING_VERSION ")" }; decltype(ircd::info::version) ircd::info::version { RB_VERSION }; decltype(ircd::info::name) ircd::info::name { PACKAGE_NAME }; extern "C" const char *const ircd_version { RB_VERSION }; extern "C" const char *const ircd_name { PACKAGE_NAME }; // // Third party dependency information // // // Host information // #ifdef HAVE_SYS_UTSNAME_H decltype(ircd::info::utsname) ircd::info::utsname{[] { struct ::utsname utsname; syscall(::uname, &utsname); return utsname; }()}; #endif // kernel decltype(ircd::info::kernel_name) ircd::info::kernel_name { #ifdef HAVE_SYS_UTSNAME_H utsname.sysname #endif }; decltype(ircd::info::kernel_release) ircd::info::kernel_release { #ifdef HAVE_SYS_UTSNAME_H utsname.release #endif }; decltype(ircd::info::kernel_version) ircd::info::kernel_version { "kernel", versions::ABI, 0, { [] // major { const auto str(split(kernel_release, '.').first); return try_lex_cast(str)? lex_cast(str): 0; }(), [] // minor { auto str(split(kernel_release, '.').second); str = split(str, '.').first; return try_lex_cast(str)? lex_cast(str): 0; }(), 0 // patch }, utsname.release }; // // gnuc // #if defined(__GNUC__) && defined(__GLIBC__) decltype(ircd::info::gnuc_version_api) ircd::info::gnuc_version_api { "gnuc", versions::API, 0, { __GNUC__, __GNUC_MINOR__, __GNUC_PATCHLEVEL__, }, // version string #if defined(__VERSION__) __VERSION__ #else "<__VERSION__ undefined>" #endif }; #endif defined(__GNUC__) && defined(__GLIBC__) // // glibc // #if defined(__GNU_LIBRARY__) && defined(__GLIBC__) && defined(__GLIBC_MINOR__) decltype(ircd::info::glibc_version_api) ircd::info::glibc_version_api { "glibc", versions::API, 0, { __GNU_LIBRARY__, __GLIBC__, __GLIBC_MINOR__, }, [](auto &that, const auto &buf) { ::snprintf(data(buf), size(buf), "%ld.%ld.%ld", that.semantic[0], that.semantic[1], that.semantic[2]); } }; #ifdef HAVE_GNU_LIBC_VERSION_H decltype(ircd::info::glibc_version_abi) ircd::info::glibc_version_abi { "glibc", versions::ABI, 0, {0}, ::gnu_get_libc_version() }; #endif HAVE_GNU_LIBC_VERSION_H #endif defined(__GNU_LIBRARY__) && defined(__GLIBC__) && defined(__GLIBC_MINOR__) // // System information // #ifdef HAVE_SYS_RESOURCE_H static uint64_t _get_rlimit(const int &resource) { rlimit rlim; ircd::syscall(getrlimit, resource, &rlim); return rlim.rlim_cur; } #else static uint64_t _get_rlimit(const int &resource) { return 0; } #endif decltype(ircd::info::rlimit_rttime) ircd::info::rlimit_rttime { #ifdef RLIMIT_RTTIME _get_rlimit(RLIMIT_RTTIME) #endif }; decltype(ircd::info::rlimit_nofile) ircd::info::rlimit_nofile { #ifdef RLIMIT_NOFILE _get_rlimit(RLIMIT_NOFILE) #endif }; decltype(ircd::info::rlimit_rss) ircd::info::rlimit_rss { #ifdef RLIMIT_RSS _get_rlimit(RLIMIT_RSS) #endif }; decltype(ircd::info::rlimit_data) ircd::info::rlimit_data { #ifdef RLIMIT_DATA _get_rlimit(RLIMIT_DATA) #endif }; decltype(ircd::info::rlimit_as) ircd::info::rlimit_as { #ifdef RLIMIT_AS _get_rlimit(RLIMIT_AS) #endif }; #ifdef _SC_CLK_TCK decltype(ircd::info::clk_tck) ircd::info::clk_tck { size_t(syscall(::sysconf, _SC_CLK_TCK)) }; #else decltype(ircd::info::clk_tck) ircd::info::clk_tck { 1 // prevent #DE }; #endif decltype(ircd::info::aio_reqprio_max) ircd::info::aio_reqprio_max { #ifdef _SC_AIO_PRIO_DELTA_MAX size_t(syscall(::sysconf, _SC_AIO_PRIO_DELTA_MAX)) #endif }; decltype(ircd::info::aio_max) ircd::info::aio_max { #ifdef _SC_AIO_MAX 0 //size_t(syscall(::sysconf, _SC_AIO_MAX)) #endif }; decltype(ircd::info::iov_max) ircd::info::iov_max { #ifdef _SC_IOV_MAX size_t(syscall(::sysconf, _SC_IOV_MAX)) #endif }; decltype(ircd::info::page_size) ircd::info::page_size { #ifdef _SC_PAGESIZE size_t(syscall(::sysconf, _SC_PAGESIZE)) #endif }; // // Platform information // #ifdef __x86_64__ static ircd::uint128_t get_cpuid(const uint &leaf, const uint &subleaf) { using ircd::uint128_t; uint32_t reg[4]; asm volatile ( "cpuid" "\n\t" : "=a" (reg[0]), "=b" (reg[1]), "=c" (reg[2]), "=d" (reg[3]) : "0" (leaf), "2" (subleaf) ); return uint128_t { (uint128_t(reg[3]) << 96) | // edx (uint128_t(reg[2]) << 64) | // ecx (uint128_t(reg[1]) << 32) | // ebx (uint128_t(reg[0]) << 0) // eax }; } #else static ircd::uint128_t get_cpuid(const uint &leaf, const uint &subleaf) { return 0; } #endif decltype(ircd::info::cpuid) ircd::info::cpuid { get_cpuid(0x00000000U, 0), get_cpuid(0x00000001U, 0), 0UL, get_cpuid(0x00000007U, 0U), get_cpuid(0x80000000U, 0), get_cpuid(0x80000001U, 0), get_cpuid(0x8000001CU, 0), //AMD Vol.2 13.4.3.3 (LWP) 0UL, }; char _cpuvendor_[12]; decltype(ircd::info::cpuvendor) ircd::info::cpuvendor{[&] { const auto b { reinterpret_cast(cpuid + 0) }; _cpuvendor_[0] = b[4]; _cpuvendor_[1] = b[5]; _cpuvendor_[2] = b[6]; _cpuvendor_[3] = b[7]; _cpuvendor_[4] = b[12]; _cpuvendor_[5] = b[13]; _cpuvendor_[6] = b[14]; _cpuvendor_[7] = b[15]; _cpuvendor_[8] = b[8]; _cpuvendor_[9] = b[9]; _cpuvendor_[10] = b[10]; _cpuvendor_[11] = b[11]; return string_view { _cpuvendor_, sizeof(_cpuvendor_) }; }()}; decltype(ircd::info::constructive_interference) ircd::info::constructive_interference { #ifdef __cpp_lib_hardware_interference_size std::hardware_constructive_interference_size #else 0 #endif }; decltype(ircd::info::destructive_interference) ircd::info::destructive_interference { #ifdef __cpp_lib_hardware_interference_size std::hardware_destructive_interference_size #else 0 #endif }; decltype(ircd::info::hardware_concurrency) ircd::info::hardware_concurrency { std::thread::hardware_concurrency() }; decltype(ircd::info::max_align) ircd::info::max_align { alignof(std::max_align_t) }; // // Build information // decltype(ircd::info::startup) ircd::info::startup { rstrip(ctime(&startup_time), '\n') }; decltype(ircd::info::compiled) ircd::info::compiled { __TIMESTAMP__ }; decltype(ircd::info::configured) ircd::info::configured { rstrip(ctime(&configured_time), '\n') }; decltype(ircd::info::startup_time) ircd::info::startup_time { std::time(nullptr) }; decltype(ircd::info::configured_time) ircd::info::configured_time { RB_TIME_CONFIGURED }; decltype(ircd::info::commit) ircd::info::commit { RB_VERSION_COMMIT }; decltype(ircd::info::branch) ircd::info::branch { RB_VERSION_BRANCH }; decltype(ircd::info::tag) ircd::info::tag { RB_VERSION_TAG };