Add cmake build system
Signed-off-by: Anthony Fieroni <bvbfan@abv.bg>
This commit is contained in:
parent
ea6e1064ec
commit
be568c40d1
4 changed files with 637 additions and 46 deletions
205
CMakeLists.txt
205
CMakeLists.txt
|
@ -1,56 +1,169 @@
|
||||||
cmake_minimum_required(VERSION 3.7)
|
|
||||||
project(lbrycrd_clion) # Do not use for full compile. This is for CLion syntax checking only.
|
|
||||||
|
|
||||||
|
cmake_minimum_required(VERSION 3.10)
|
||||||
|
|
||||||
|
project(lbrycrd)
|
||||||
set(CMAKE_CXX_STANDARD 11)
|
set(CMAKE_CXX_STANDARD 11)
|
||||||
|
|
||||||
if(EXISTS "build/boost")
|
include(cmake/CPM.cmake)
|
||||||
set(BOOST_ROOT "build/boost" CACHE PATH "Boost library path")
|
include(ExternalProject)
|
||||||
set(Boost_NO_SYSTEM_PATHS on CACHE BOOL "Do not search system for Boost")
|
|
||||||
|
set(OPTIONS "" CACHE STRING "lbrycrdd configure options")
|
||||||
|
set(CPPFLAGS "" CACHE STRING "lbrycrdd compiler options")
|
||||||
|
set(LDFLAGS "" CACHE STRING "lbrycrdd linker options")
|
||||||
|
|
||||||
|
if(NOT ${CPM_USE_LOCAL_PACKAGES})
|
||||||
|
set(OPTIONS "${OPTIONS} --enable-static --disable-shared")
|
||||||
|
else()
|
||||||
|
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_LIST_DIR}/cmake")
|
||||||
endif()
|
endif()
|
||||||
find_package(Boost REQUIRED COMPONENTS filesystem thread chrono locale)
|
|
||||||
|
|
||||||
file(GLOB sources
|
set(OPTIONS "--enable-cxx --without-gui ${OPTIONS} --with-pic")
|
||||||
src/*.h src/*.cpp
|
|
||||||
src/wallet/*.h src/wallet/*.cpp
|
|
||||||
src/support/*.h src/support/*.cpp src/support/allocators/*.h
|
|
||||||
src/script/*.h src/script/*.cpp
|
|
||||||
src/index/*.h src/index/*.cpp
|
|
||||||
src/interfaces/*.h src/interfaces/*.cpp
|
|
||||||
src/primitives/*.h src/primitives/*.cpp
|
|
||||||
src/policy/*.h src/policy/*.cpp
|
|
||||||
src/crypto/*.h src/crypto/*.cpp
|
|
||||||
src/consensus/*.h src/consensus/*.cpp
|
|
||||||
src/compat/*.h src/compat/*.cpp
|
|
||||||
src/rpc/*.h src/rpc/*.cpp
|
|
||||||
)
|
|
||||||
list(FILTER sources EXCLUDE REGEX "src/bitcoin*.cpp$")
|
|
||||||
|
|
||||||
include_directories(${Boost_INCLUDE_DIRS}
|
string(FIND ${OPTIONS} "--disable-tests" TESTS_DISABLED)
|
||||||
build/bdb/include
|
string(FIND ${OPTIONS} "--disable-wallet" WALLET_DISABLED)
|
||||||
build/libevent/include
|
|
||||||
build/openssl/include
|
CPMAddPackage(
|
||||||
src/support/allocators
|
NAME OpenSSL
|
||||||
src/support
|
GITHUB_REPOSITORY openssl/openssl
|
||||||
src/rpc
|
VERSION 1.0.2
|
||||||
src/policy
|
GIT_TAG OpenSSL_1_0_2r
|
||||||
src/wallet src/script
|
DOWNLOAD_ONLY TRUE
|
||||||
src/leveldb/helpers/memenv
|
|
||||||
src/leveldb/include
|
|
||||||
src/config
|
|
||||||
src/crypto
|
|
||||||
src/compat
|
|
||||||
src/obj
|
|
||||||
src/univalue/include
|
|
||||||
src/secp256k1/include
|
|
||||||
src/
|
|
||||||
)
|
)
|
||||||
|
|
||||||
add_compile_definitions(HAVE_CONFIG_H)
|
if(OpenSSL_ADDED)
|
||||||
|
ExternalProject_Add(OpenSSL
|
||||||
|
PREFIX openssl
|
||||||
|
SOURCE_DIR ${OpenSSL_SOURCE_DIR}
|
||||||
|
CONFIGURE_COMMAND ${OpenSSL_SOURCE_DIR}/Configure linux-x86_64 no-shared no-dso no-engines -fPIC --prefix=<INSTALL_DIR>
|
||||||
|
BUILD_IN_SOURCE 1
|
||||||
|
)
|
||||||
|
set(DEPENDS ${DEPENDS} OpenSSL)
|
||||||
|
ExternalProject_Get_Property(OpenSSL INSTALL_DIR)
|
||||||
|
set(LDFLAGS "${LDFLAGS} -L${INSTALL_DIR}/lib")
|
||||||
|
set(CPPFLAGS "${CPPFLAGS} -I${INSTALL_DIR}/include")
|
||||||
|
set(OPENSSL_CPPFLAGS "CPPFLAGS=-I${INSTALL_DIR}/include")
|
||||||
|
set(OPENSSL_LDFLAGS "LDFLAGS=-L${INSTALL_DIR}/lib")
|
||||||
|
endif(OpenSSL_ADDED)
|
||||||
|
|
||||||
add_executable(lbrycrd-cli src/bitcoin-cli.cpp ${sources})
|
CPMAddPackage(
|
||||||
add_executable(lbrycrd-tx src/bitcoin-tx.cpp ${sources})
|
NAME Libevent
|
||||||
add_executable(lbrycrdd src/bitcoind.cpp ${sources})
|
GITHUB_REPOSITORY libevent/libevent
|
||||||
|
VERSION 2.1.8
|
||||||
|
GIT_TAG release-2.1.8-stable
|
||||||
|
DOWNLOAD_ONLY TRUE
|
||||||
|
)
|
||||||
|
|
||||||
file(GLOB tests src/test/*.cpp src/wallet/test/*.cpp)
|
if(Libevent_ADDED)
|
||||||
add_executable(test_lbrycrd ${tests} ${sources})
|
ExternalProject_Add(Libevent
|
||||||
target_include_directories(test_lbrycrd PRIVATE src/test)
|
PREFIX libevent
|
||||||
|
DEPENDS ${DEPENDS}
|
||||||
|
SOURCE_DIR ${Libevent_SOURCE_DIR}
|
||||||
|
CONFIGURE_COMMAND ${Libevent_SOURCE_DIR}/autogen.sh
|
||||||
|
&& ${Libevent_SOURCE_DIR}/configure ${OPENSSL_CPPFLAGS} --enable-cxx --disable-shared --with-pic ${OPENSSL_LDFLAGS} --prefix=<INSTALL_DIR>
|
||||||
|
BUILD_IN_SOURCE 1
|
||||||
|
)
|
||||||
|
set(DEPENDS ${DEPENDS} Libevent)
|
||||||
|
ExternalProject_Get_Property(Libevent INSTALL_DIR)
|
||||||
|
set(LDFLAGS "${LDFLAGS} -L${INSTALL_DIR}/lib")
|
||||||
|
set(CPPFLAGS "${CPPFLAGS} -I${INSTALL_DIR}/include")
|
||||||
|
endif(Libevent_ADDED)
|
||||||
|
|
||||||
|
if(NOT ${WALLET_DISABLED} GREATER -1)
|
||||||
|
CPMAddPackage(
|
||||||
|
NAME BerkeleyDB
|
||||||
|
VERSION 4.8.30
|
||||||
|
URL https://download.oracle.com/berkeley-db/db-4.8.30.NC.zip
|
||||||
|
URL_HASH SHA256=43ecd76886992ea416fdadc54b7f2b83ef249d9a6964bd07708ccae42d0226ce
|
||||||
|
DOWNLOAD_ONLY TRUE
|
||||||
|
)
|
||||||
|
|
||||||
|
if(NOT ${BerkeleyDB_VERSION} VERSION_LESS "5.0")
|
||||||
|
set(OPTIONS "${OPTIONS} --with-incompatible-bdb")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(BerkeleyDB_ADDED)
|
||||||
|
ExternalProject_Add(BerkeleyDB
|
||||||
|
PREFIX bdb
|
||||||
|
SOURCE_DIR ${BerkeleyDB_SOURCE_DIR}
|
||||||
|
PATCH_COMMAND patch -d ${BerkeleyDB_SOURCE_DIR}/dbinc -i ${CMAKE_CURRENT_SOURCE_DIR}/contrib/patches/atomic.patch -f
|
||||||
|
CONFIGURE_COMMAND ${BerkeleyDB_SOURCE_DIR}/dist/configure --enable-cxx --disable-shared --with-pic --prefix=<INSTALL_DIR>
|
||||||
|
)
|
||||||
|
set(DEPENDS ${DEPENDS} BerkeleyDB)
|
||||||
|
ExternalProject_Get_Property(BerkeleyDB INSTALL_DIR)
|
||||||
|
set(LDFLAGS "${LDFLAGS} -L${INSTALL_DIR}/lib")
|
||||||
|
set(CPPFLAGS "${CPPFLAGS} -I${INSTALL_DIR}/include")
|
||||||
|
endif(BerkeleyDB_ADDED)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
CPMAddPackage(
|
||||||
|
NAME ICU
|
||||||
|
GITHUB_REPOSITORY unicode-org/icu
|
||||||
|
VERSION 63.1
|
||||||
|
GIT_TAG release-63-1
|
||||||
|
DOWNLOAD_ONLY TRUE
|
||||||
|
)
|
||||||
|
|
||||||
|
if(ICU_ADDED)
|
||||||
|
ExternalProject_Add(ICU
|
||||||
|
PREFIX icu
|
||||||
|
SOURCE_DIR ${ICU_SOURCE_DIR}
|
||||||
|
CONFIGURE_COMMAND ${ICU_SOURCE_DIR}/icu4c/source/configure --enable-draft --enable-tools
|
||||||
|
--disable-shared --enable-static --disable-extras --disable-icuio --disable-dyload
|
||||||
|
--disable-layout --disable-layoutex --disable-tests --disable-samples CFLAGS=-fPIC CPPFLAGS=-fPIC --prefix=<INSTALL_DIR>
|
||||||
|
)
|
||||||
|
set(DEPENDS ${DEPENDS} ICU)
|
||||||
|
ExternalProject_Get_Property(ICU INSTALL_DIR)
|
||||||
|
set(ICU_PATH ${INSTALL_DIR})
|
||||||
|
set(OPTIONS "${OPTIONS} --with-icu=${ICU_PATH}")
|
||||||
|
set(LDFLAGS "${LDFLAGS} -L${ICU_PATH}/lib")
|
||||||
|
set(CPPFLAGS "${CPPFLAGS} -I${ICU_PATH}/include")
|
||||||
|
endif(ICU_ADDED)
|
||||||
|
|
||||||
|
set(BOOST_LIBS chrono,filesystem,program_options,system,locale,regex,thread)
|
||||||
|
|
||||||
|
if(NOT ${TESTS_DISABLED} GREATER -1)
|
||||||
|
set(BOOST_LIBS ${BOOST_LIBS},unit_test_framework)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
string(REPLACE "," ";" BOOST_COMPONENTS ${BOOST_LIBS})
|
||||||
|
|
||||||
|
CPMAddPackage(
|
||||||
|
NAME Boost
|
||||||
|
GITHUB_REPOSITORY boostorg/boost
|
||||||
|
VERSION 1.64.0
|
||||||
|
COMPONENTS ${BOOST_COMPONENTS}
|
||||||
|
GIT_TAG boost-1.64.0
|
||||||
|
GIT_SUBMODULES libs/* tools/*
|
||||||
|
DOWNLOAD_ONLY TRUE
|
||||||
|
)
|
||||||
|
|
||||||
|
if(Boost_ADDED)
|
||||||
|
ExternalProject_Add(Boost
|
||||||
|
PREFIX boost
|
||||||
|
DEPENDS ${DEPENDS}
|
||||||
|
SOURCE_DIR ${Boost_SOURCE_DIR}
|
||||||
|
CONFIGURE_COMMAND ${Boost_SOURCE_DIR}/bootstrap.sh --with-icu=${ICU_PATH} --with-libraries=${BOOST_LIBS} && ${Boost_SOURCE_DIR}/b2 headers
|
||||||
|
BUILD_COMMAND ${Boost_SOURCE_DIR}/b2 install link=static cxxflags=-fPIC boost.locale.iconv=off boost.locale.posix=off -sICU_PATH=${ICU_PATH} --prefix=<INSTALL_DIR>
|
||||||
|
INSTALL_COMMAND ""
|
||||||
|
BUILD_IN_SOURCE 1
|
||||||
|
)
|
||||||
|
set(DEPENDS ${DEPENDS} Boost)
|
||||||
|
ExternalProject_Get_Property(Boost INSTALL_DIR)
|
||||||
|
set(OPTIONS "${OPTIONS} --with-boost=${INSTALL_DIR}")
|
||||||
|
set(LDFLAGS "${LDFLAGS} -L${INSTALL_DIR}/lib")
|
||||||
|
set(CPPFLAGS "${CPPFLAGS} -I${INSTALL_DIR}/include")
|
||||||
|
endif(Boost_ADDED)
|
||||||
|
|
||||||
|
set(${CPPFLAGS} "${CPPFLAGS} -Wno-unused-local-typedefs -Wno-deprecated -Wno-implicit-fallthrough -Wno-unused-parameter")
|
||||||
|
|
||||||
|
separate_arguments(OPTIONS)
|
||||||
|
|
||||||
|
ExternalProject_Add(lbrycrdd
|
||||||
|
PREFIX lbrycrdd
|
||||||
|
DEPENDS ${DEPENDS}
|
||||||
|
SOURCE_DIR ${CMAKE_CURRENT_SOURCE_DIR}
|
||||||
|
CONFIGURE_COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/autogen.sh
|
||||||
|
&& ${CMAKE_CURRENT_SOURCE_DIR}/configure ${OPTIONS} CPPFLAGS=${CPPFLAGS} LDFLAGS=${LDFLAGS} --prefix=<INSTALL_DIR>
|
||||||
|
BUILD_IN_SOURCE 1
|
||||||
|
BUILD_ALWAYS 1
|
||||||
|
)
|
||||||
|
|
210
cmake/CPM.cmake
Normal file
210
cmake/CPM.cmake
Normal file
|
@ -0,0 +1,210 @@
|
||||||
|
# TheLartians/CPM - A simple Git dependency manager
|
||||||
|
# =================================================
|
||||||
|
# See https://github.com/TheLartians/CPM for usage and update instructions.
|
||||||
|
#
|
||||||
|
# MIT License
|
||||||
|
# -----------
|
||||||
|
#[[
|
||||||
|
Copyright (c) 2019 Lars Melchior
|
||||||
|
|
||||||
|
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.
|
||||||
|
]]
|
||||||
|
|
||||||
|
cmake_minimum_required(VERSION 3.10 FATAL_ERROR)
|
||||||
|
|
||||||
|
set(CURRENT_CPM_VERSION 0.11.1)
|
||||||
|
|
||||||
|
if(CPM_DIRECTORY)
|
||||||
|
if(NOT ${CPM_DIRECTORY} MATCHES ${CMAKE_CURRENT_LIST_DIR})
|
||||||
|
if (${CPM_VERSION} VERSION_LESS ${CURRENT_CPM_VERSION})
|
||||||
|
CPM_HANDLE_OLD_VERSION(${CURRENT_CPM_VERSION})
|
||||||
|
endif()
|
||||||
|
return()
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(CPM_VERSION ${CURRENT_CPM_VERSION} CACHE INTERNAL "")
|
||||||
|
set(CPM_DIRECTORY ${CMAKE_CURRENT_LIST_DIR} CACHE INTERNAL "")
|
||||||
|
set(CPM_PACKAGES "" CACHE INTERNAL "")
|
||||||
|
|
||||||
|
option(CPM_USE_LOCAL_PACKAGES "Use locally installed packages (find_package)" OFF)
|
||||||
|
option(CPM_LOCAL_PACKAGES_ONLY "Use only locally installed packages" OFF)
|
||||||
|
|
||||||
|
include(FetchContent)
|
||||||
|
include(CMakeParseArguments)
|
||||||
|
|
||||||
|
# Initialize logging prefix
|
||||||
|
if(NOT CPM_INDENT)
|
||||||
|
set(CPM_INDENT "CPM:")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# The main workhorse of CPM
|
||||||
|
function(CPMAddPackage)
|
||||||
|
|
||||||
|
set(oneValueArgs
|
||||||
|
NAME
|
||||||
|
VERSION
|
||||||
|
GIT_TAG
|
||||||
|
DOWNLOAD_ONLY
|
||||||
|
GITHUB_REPOSITORY
|
||||||
|
GITLAB_REPOSITORY
|
||||||
|
)
|
||||||
|
|
||||||
|
set(multiValueArgs
|
||||||
|
OPTIONS
|
||||||
|
COMPONENTS
|
||||||
|
)
|
||||||
|
|
||||||
|
cmake_parse_arguments(CPM_ARGS "" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
|
||||||
|
|
||||||
|
if(${CPM_USE_LOCAL_PACKAGES} OR ${CPM_LOCAL_PACKAGES_ONLY})
|
||||||
|
find_package(${CPM_ARGS_NAME} ${CPM_ARGS_VERSION} OPTIONAL_COMPONENTS ${CPM_ARGS_COMPONENTS} QUIET)
|
||||||
|
|
||||||
|
if(${CPM_ARGS_NAME}_FOUND)
|
||||||
|
message(STATUS "CPM: adding local package ${CPM_ARGS_NAME}@${${CPM_ARGS_NAME}_VERSION}")
|
||||||
|
set(${CPM_ARGS_NAME}_VERSION "${${CPM_ARGS_NAME}_VERSION}" PARENT_SCOPE)
|
||||||
|
return()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(${CPM_LOCAL_PACKAGES_ONLY})
|
||||||
|
message(SEND_ERROR "CPM: ${CPM_ARGS_NAME} not found via find_package(${CPM_ARGS_NAME} ${CPM_ARGS_VERSION})")
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (NOT CPM_ARGS_VERSION)
|
||||||
|
set(CPM_ARGS_VERSION 0)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (NOT CPM_ARGS_GIT_TAG)
|
||||||
|
set(CPM_ARGS_GIT_TAG v${CPM_ARGS_VERSION})
|
||||||
|
endif()
|
||||||
|
|
||||||
|
list(APPEND CPM_ARGS_UNPARSED_ARGUMENTS GIT_TAG ${CPM_ARGS_GIT_TAG})
|
||||||
|
|
||||||
|
if(CPM_ARGS_DOWNLOAD_ONLY)
|
||||||
|
set(DOWNLOAD_ONLY ${CPM_ARGS_DOWNLOAD_ONLY})
|
||||||
|
else()
|
||||||
|
set(DOWNLOAD_ONLY NO)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (CPM_ARGS_GITHUB_REPOSITORY)
|
||||||
|
list(APPEND CPM_ARGS_UNPARSED_ARGUMENTS GIT_REPOSITORY "https://github.com/${CPM_ARGS_GITHUB_REPOSITORY}.git")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (CPM_ARGS_GITLAB_REPOSITORY)
|
||||||
|
list(APPEND CPM_ARGS_UNPARSED_ARGUMENTS GIT_REPOSITORY "https://gitlab.com/${CPM_ARGS_GITLAB_REPOSITORY}.git")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (${CPM_ARGS_NAME} IN_LIST CPM_PACKAGES)
|
||||||
|
CPM_GET_PACKAGE_VERSION(${CPM_ARGS_NAME})
|
||||||
|
if(${CPM_PACKAGE_VERSION} VERSION_LESS ${CPM_ARGS_VERSION})
|
||||||
|
message(WARNING "${CPM_INDENT} requires a newer version of ${CPM_ARGS_NAME} (${CPM_ARGS_VERSION}) than currently included (${CPM_PACKAGE_VERSION}).")
|
||||||
|
endif()
|
||||||
|
if (CPM_ARGS_OPTIONS)
|
||||||
|
foreach(OPTION ${CPM_ARGS_OPTIONS})
|
||||||
|
CPM_PARSE_OPTION(${OPTION})
|
||||||
|
if(NOT "${${OPTION_KEY}}" STREQUAL ${OPTION_VALUE})
|
||||||
|
message(WARNING "${CPM_INDENT} ignoring package option for ${CPM_ARGS_NAME}: ${OPTION_KEY} = ${OPTION_VALUE} (${${OPTION_KEY}})")
|
||||||
|
endif()
|
||||||
|
endforeach()
|
||||||
|
endif()
|
||||||
|
CPM_FETCH_PACKAGE(${CPM_ARGS_NAME} ${DOWNLOAD_ONLY})
|
||||||
|
CPMGetProperties(${CPM_ARGS_NAME})
|
||||||
|
set(${CPM_ARGS_NAME}_VERSION ${CPM_ARGS_VERSION} PARENT_SCOPE)
|
||||||
|
set(${CPM_ARGS_NAME}_SOURCE_DIR "${${CPM_ARGS_NAME}_SOURCE_DIR}" PARENT_SCOPE)
|
||||||
|
set(${CPM_ARGS_NAME}_BINARY_DIR "${${CPM_ARGS_NAME}_BINARY_DIR}" PARENT_SCOPE)
|
||||||
|
set(${CPM_ARGS_NAME}_ADDED NO PARENT_SCOPE)
|
||||||
|
return()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
CPMRegisterPackage(${CPM_ARGS_NAME} ${CPM_ARGS_VERSION})
|
||||||
|
|
||||||
|
if (CPM_ARGS_OPTIONS)
|
||||||
|
foreach(OPTION ${CPM_ARGS_OPTIONS})
|
||||||
|
CPM_PARSE_OPTION(${OPTION})
|
||||||
|
set(${OPTION_KEY} ${OPTION_VALUE} CACHE INTERNAL "")
|
||||||
|
endforeach()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
CPM_DECLARE_PACKAGE(${CPM_ARGS_NAME} ${CPM_ARGS_VERSION} ${CPM_ARGS_GIT_TAG} "${CPM_ARGS_UNPARSED_ARGUMENTS}")
|
||||||
|
CPM_FETCH_PACKAGE(${CPM_ARGS_NAME} ${DOWNLOAD_ONLY})
|
||||||
|
CPMGetProperties(${CPM_ARGS_NAME})
|
||||||
|
set(${CPM_ARGS_NAME}_VERSION ${CPM_ARGS_VERSION} PARENT_SCOPE)
|
||||||
|
set(${CPM_ARGS_NAME}_SOURCE_DIR "${${CPM_ARGS_NAME}_SOURCE_DIR}" PARENT_SCOPE)
|
||||||
|
set(${CPM_ARGS_NAME}_BINARY_DIR "${${CPM_ARGS_NAME}_BINARY_DIR}" PARENT_SCOPE)
|
||||||
|
set(${CPM_ARGS_NAME}_ADDED YES PARENT_SCOPE)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function (CPM_DECLARE_PACKAGE PACKAGE VERSION GIT_TAG)
|
||||||
|
message(STATUS "${CPM_INDENT} adding package ${PACKAGE}@${VERSION} (${GIT_TAG})")
|
||||||
|
|
||||||
|
FetchContent_Declare(
|
||||||
|
${PACKAGE}
|
||||||
|
${ARGN}
|
||||||
|
)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function (CPM_FETCH_PACKAGE PACKAGE DOWNLOAD_ONLY)
|
||||||
|
set(CPM_OLD_INDENT "${CPM_INDENT}")
|
||||||
|
set(CPM_INDENT "${CPM_INDENT} ${PACKAGE}:")
|
||||||
|
if(${DOWNLOAD_ONLY})
|
||||||
|
if(NOT "${PACKAGE}_POPULATED")
|
||||||
|
FetchContent_Populate(${PACKAGE})
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
FetchContent_MakeAvailable(${PACKAGE})
|
||||||
|
endif()
|
||||||
|
set(CPM_INDENT "${CPM_OLD_INDENT}")
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function (CPMGetProperties PACKAGE)
|
||||||
|
FetchContent_GetProperties(${PACKAGE})
|
||||||
|
string(TOLOWER ${PACKAGE} lpackage)
|
||||||
|
set(${PACKAGE}_SOURCE_DIR "${${lpackage}_SOURCE_DIR}" PARENT_SCOPE)
|
||||||
|
set(${PACKAGE}_BINARY_DIR "${${lpackage}_BINARY_DIR}" PARENT_SCOPE)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function(CPMRegisterPackage PACKAGE VERSION)
|
||||||
|
list(APPEND CPM_PACKAGES ${PACKAGE})
|
||||||
|
set(CPM_PACKAGES ${CPM_PACKAGES} CACHE INTERNAL "")
|
||||||
|
set("CPM_PACKAGE_${PACKAGE}_VERSION" ${VERSION} CACHE INTERNAL "")
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function(CPM_GET_PACKAGE_VERSION PACKAGE)
|
||||||
|
set(CPM_PACKAGE_VERSION "${CPM_PACKAGE_${PACKAGE}_VERSION}" PARENT_SCOPE)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function(CPM_PARSE_OPTION OPTION)
|
||||||
|
string(REGEX MATCH "^[^ ]+" OPTION_KEY ${OPTION})
|
||||||
|
string(LENGTH ${OPTION_KEY} OPTION_KEY_LENGTH)
|
||||||
|
math(EXPR OPTION_KEY_LENGTH "${OPTION_KEY_LENGTH}+1")
|
||||||
|
string(SUBSTRING ${OPTION} "${OPTION_KEY_LENGTH}" "-1" OPTION_VALUE)
|
||||||
|
set(OPTION_KEY "${OPTION_KEY}" PARENT_SCOPE)
|
||||||
|
set(OPTION_VALUE "${OPTION_VALUE}" PARENT_SCOPE)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
function (CPM_HANDLE_OLD_VERSION NEW_CPM_VERSION)
|
||||||
|
|
||||||
|
message(AUTHOR_WARNING "${CPM_INDENT} \
|
||||||
|
A dependency is using a more recent CPM (${NEW_CPM_VERSION}) than the current project (${CPM_VERSION}). \
|
||||||
|
It is recommended to upgrade CPM to the most recent version. \
|
||||||
|
See https://github.com/TheLartians/CPM for more information."
|
||||||
|
)
|
||||||
|
|
||||||
|
endfunction()
|
171
cmake/FindBerkeleyDB.cmake
Normal file
171
cmake/FindBerkeleyDB.cmake
Normal file
|
@ -0,0 +1,171 @@
|
||||||
|
# Author: sum01 <sum01@protonmail.com>
|
||||||
|
# Git: https://github.com/sum01/FindBerkeleyDB
|
||||||
|
# Read the README.md for the full info.
|
||||||
|
|
||||||
|
# NOTE: If Berkeley DB ever gets a Pkg-config ".pc" file, add pkg_check_modules() here
|
||||||
|
|
||||||
|
# Checks if environment paths are empty, set them if they aren't
|
||||||
|
if(NOT "$ENV{BERKELEYDB_ROOT}" STREQUAL "")
|
||||||
|
set(_BERKELEYDB_HINTS "$ENV{BERKELEYDB_ROOT}")
|
||||||
|
elseif(NOT "$ENV{Berkeleydb_ROOT}" STREQUAL "")
|
||||||
|
set(_BERKELEYDB_HINTS "$ENV{Berkeleydb_ROOT}")
|
||||||
|
elseif(NOT "$ENV{BERKELEYDBROOT}" STREQUAL "")
|
||||||
|
set(_BERKELEYDB_HINTS "$ENV{BERKELEYDBROOT}")
|
||||||
|
else()
|
||||||
|
# Set just in case, as it's used regardless if it's empty or not
|
||||||
|
set(_BERKELEYDB_HINTS "")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Allow user to pass a path instead of guessing
|
||||||
|
if(BerkeleyDB_ROOT_DIR)
|
||||||
|
set(_BERKELEYDB_PATHS "${BerkeleyDB_ROOT_DIR}")
|
||||||
|
elseif(CMAKE_SYSTEM_NAME MATCHES ".*[wW]indows.*")
|
||||||
|
# MATCHES is used to work on any devies with windows in the name
|
||||||
|
# Shameless copy-paste from FindOpenSSL.cmake v3.8
|
||||||
|
file(TO_CMAKE_PATH "$ENV{PROGRAMFILES}" _programfiles)
|
||||||
|
list(APPEND _BERKELEYDB_HINTS "${_programfiles}")
|
||||||
|
|
||||||
|
# There's actually production release and version numbers in the file path.
|
||||||
|
# For example, if they're on v6.2.32: C:/Program Files/Oracle/Berkeley DB 12cR1 6.2.32/
|
||||||
|
# But this still works to find it, so I'm guessing it can accept partial path matches.
|
||||||
|
|
||||||
|
foreach(_TARGET_BERKELEYDB_PATH "Oracle/Berkeley DB" "Berkeley DB")
|
||||||
|
list(APPEND _BERKELEYDB_PATHS
|
||||||
|
"${_programfiles}/${_TARGET_BERKELEYDB_PATH}"
|
||||||
|
"C:/Program Files (x86)/${_TARGET_BERKELEYDB_PATH}"
|
||||||
|
"C:/Program Files/${_TARGET_BERKELEYDB_PATH}"
|
||||||
|
"C:/${_TARGET_BERKELEYDB_PATH}"
|
||||||
|
)
|
||||||
|
endforeach()
|
||||||
|
else()
|
||||||
|
# Paths for anything other than Windows
|
||||||
|
# Cellar/berkeley-db is for macOS from homebrew installation
|
||||||
|
list(APPEND _BERKELEYDB_PATHS
|
||||||
|
"/usr"
|
||||||
|
"/usr/local"
|
||||||
|
"/usr/local/Cellar/berkeley-db"
|
||||||
|
"/opt"
|
||||||
|
"/opt/local"
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Find includes path
|
||||||
|
find_path(BerkeleyDB_INCLUDE_DIRS
|
||||||
|
NAMES "db.h"
|
||||||
|
HINTS ${_BERKELEYDB_HINTS}
|
||||||
|
PATH_SUFFIXES "include" "includes"
|
||||||
|
PATHS ${_BERKELEYDB_PATHS}
|
||||||
|
)
|
||||||
|
|
||||||
|
# Checks if the version file exists, save the version file to a var, and fail if there's no version file
|
||||||
|
if(BerkeleyDB_INCLUDE_DIRS)
|
||||||
|
# Read the version file db.h into a variable
|
||||||
|
file(READ "${BerkeleyDB_INCLUDE_DIRS}/db.h" _BERKELEYDB_DB_HEADER)
|
||||||
|
# Parse the DB version into variables to be used in the lib names
|
||||||
|
string(REGEX REPLACE ".*DB_VERSION_MAJOR ([0-9]+).*" "\\1" BerkeleyDB_VERSION_MAJOR "${_BERKELEYDB_DB_HEADER}")
|
||||||
|
string(REGEX REPLACE ".*DB_VERSION_MINOR ([0-9]+).*" "\\1" BerkeleyDB_VERSION_MINOR "${_BERKELEYDB_DB_HEADER}")
|
||||||
|
# Patch version example on non-crypto installs: x.x.xNC
|
||||||
|
string(REGEX REPLACE ".*DB_VERSION_PATCH ([0-9]+(NC)?).*" "\\1" BerkeleyDB_VERSION_PATCH "${_BERKELEYDB_DB_HEADER}")
|
||||||
|
else()
|
||||||
|
if(BerkeleyDB_FIND_REQUIRED)
|
||||||
|
# If the find_package(BerkeleyDB REQUIRED) was used, fail since we couldn't find the header
|
||||||
|
message(FATAL_ERROR "Failed to find Berkeley DB's header file \"db.h\"! Try setting \"BerkeleyDB_ROOT_DIR\" when initiating Cmake.")
|
||||||
|
elseif(NOT BerkeleyDB_FIND_QUIETLY)
|
||||||
|
message(WARNING "Failed to find Berkeley DB's header file \"db.h\"! Try setting \"BerkeleyDB_ROOT_DIR\" when initiating Cmake.")
|
||||||
|
endif()
|
||||||
|
# Set some garbage values to the versions since we didn't find a file to read
|
||||||
|
set(BerkeleyDB_VERSION_MAJOR "0")
|
||||||
|
set(BerkeleyDB_VERSION_MINOR "0")
|
||||||
|
set(BerkeleyDB_VERSION_PATCH "0")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# The actual returned/output version variable (the others can be used if needed)
|
||||||
|
set(BerkeleyDB_VERSION "${BerkeleyDB_VERSION_MAJOR}.${BerkeleyDB_VERSION_MINOR}.${BerkeleyDB_VERSION_PATCH}")
|
||||||
|
|
||||||
|
# Finds the target library for berkeley db, since they all follow the same naming conventions
|
||||||
|
macro(_berkeleydb_get_lib _BERKELEYDB_OUTPUT_VARNAME _TARGET_BERKELEYDB_LIB)
|
||||||
|
# Different systems sometimes have a version in the lib name...
|
||||||
|
# and some have a dash or underscore before the versions.
|
||||||
|
# CMake recommends to put unversioned names before versioned names
|
||||||
|
find_library(${_BERKELEYDB_OUTPUT_VARNAME}
|
||||||
|
NAMES
|
||||||
|
"${_TARGET_BERKELEYDB_LIB}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}${BerkeleyDB_VERSION_MAJOR}.${BerkeleyDB_VERSION_MINOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}-${BerkeleyDB_VERSION_MAJOR}.${BerkeleyDB_VERSION_MINOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}_${BerkeleyDB_VERSION_MAJOR}.${BerkeleyDB_VERSION_MINOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}${BerkeleyDB_VERSION_MAJOR}${BerkeleyDB_VERSION_MINOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}-${BerkeleyDB_VERSION_MAJOR}${BerkeleyDB_VERSION_MINOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}_${BerkeleyDB_VERSION_MAJOR}${BerkeleyDB_VERSION_MINOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}${BerkeleyDB_VERSION_MAJOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}-${BerkeleyDB_VERSION_MAJOR}"
|
||||||
|
"lib${_TARGET_BERKELEYDB_LIB}_${BerkeleyDB_VERSION_MAJOR}"
|
||||||
|
HINTS ${_BERKELEYDB_HINTS}
|
||||||
|
PATH_SUFFIXES
|
||||||
|
"lib"
|
||||||
|
"lib64"
|
||||||
|
"libs"
|
||||||
|
"libs64"
|
||||||
|
PATHS ${_BERKELEYDB_PATHS}
|
||||||
|
)
|
||||||
|
# If the library was found, add it to our list of libraries
|
||||||
|
if(${_BERKELEYDB_OUTPUT_VARNAME})
|
||||||
|
# If found, append to our libraries variable
|
||||||
|
# The ${{}} is because the first expands to target the real variable, the second expands the variable's contents...
|
||||||
|
# and the real variable's contents is the path to the lib. Thus, it appends the path of the lib to BerkeleyDB_LIBRARIES.
|
||||||
|
list(APPEND BerkeleyDB_LIBRARIES "${${_BERKELEYDB_OUTPUT_VARNAME}}")
|
||||||
|
endif()
|
||||||
|
endmacro()
|
||||||
|
|
||||||
|
# Find and set the paths of the specific library to the variable
|
||||||
|
_berkeleydb_get_lib(BerkeleyDB_LIBRARY "db")
|
||||||
|
# NOTE: Windows doesn't have a db_cxx lib, but instead compiles the cxx code into the "db" lib
|
||||||
|
_berkeleydb_get_lib(BerkeleyDB_Cxx_LIBRARY "db_cxx")
|
||||||
|
# NOTE: I don't think Linux/Unix gets an SQL lib
|
||||||
|
_berkeleydb_get_lib(BerkeleyDB_Sql_LIBRARY "db_sql")
|
||||||
|
_berkeleydb_get_lib(BerkeleyDB_Stl_LIBRARY "db_stl")
|
||||||
|
|
||||||
|
# Needed for find_package_handle_standard_args()
|
||||||
|
include(FindPackageHandleStandardArgs)
|
||||||
|
# Fails if required vars aren't found, or if the version doesn't meet specifications.
|
||||||
|
find_package_handle_standard_args(BerkeleyDB
|
||||||
|
FOUND_VAR BerkeleyDB_FOUND
|
||||||
|
REQUIRED_VARS
|
||||||
|
BerkeleyDB_INCLUDE_DIRS
|
||||||
|
BerkeleyDB_LIBRARY
|
||||||
|
BerkeleyDB_LIBRARIES
|
||||||
|
VERSION_VAR BerkeleyDB_VERSION
|
||||||
|
)
|
||||||
|
|
||||||
|
# Only show the variables in the GUI if they click "advanced".
|
||||||
|
# Does nothing when using the CLI
|
||||||
|
mark_as_advanced(FORCE
|
||||||
|
BerkeleyDB_FOUND
|
||||||
|
BerkeleyDB_INCLUDE_DIRS
|
||||||
|
BerkeleyDB_LIBRARIES
|
||||||
|
BerkeleyDB_VERSION
|
||||||
|
BerkeleyDB_VERSION_MAJOR
|
||||||
|
BerkeleyDB_VERSION_MINOR
|
||||||
|
BerkeleyDB_VERSION_PATCH
|
||||||
|
BerkeleyDB_LIBRARY
|
||||||
|
BerkeleyDB_Cxx_LIBRARY
|
||||||
|
BerkeleyDB_Stl_LIBRARY
|
||||||
|
BerkeleyDB_Sql_LIBRARY
|
||||||
|
)
|
||||||
|
|
||||||
|
# Create an imported lib for easy linking by external projects
|
||||||
|
if(BerkeleyDB_FOUND AND BerkeleyDB_LIBRARIES AND NOT TARGET Oracle::BerkeleyDB)
|
||||||
|
add_library(Oracle::BerkeleyDB UNKNOWN IMPORTED)
|
||||||
|
set_target_properties(Oracle::BerkeleyDB PROPERTIES
|
||||||
|
INTERFACE_INCLUDE_DIRECTORIES "${BerkeleyDB_INCLUDE_DIRS}"
|
||||||
|
IMPORTED_LOCATION "${BerkeleyDB_LIBRARY}"
|
||||||
|
INTERFACE_LINK_LIBRARIES "${BerkeleyDB_LIBRARIES}"
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
include(FindPackageMessage)
|
||||||
|
# A message that tells the user what includes/libs were found, and obeys the QUIET command.
|
||||||
|
find_package_message(BerkeleyDB
|
||||||
|
"Found BerkeleyDB libraries: ${BerkeleyDB_LIBRARIES}"
|
||||||
|
"[${BerkeleyDB_LIBRARIES}[${BerkeleyDB_INCLUDE_DIRS}]]"
|
||||||
|
)
|
97
cmake/FindLibevent.cmake
Normal file
97
cmake/FindLibevent.cmake
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
# - Try to find libevent
|
||||||
|
#.rst
|
||||||
|
# FindLibevent
|
||||||
|
# ------------
|
||||||
|
#
|
||||||
|
# Find Libevent include directories and libraries. Invoke as::
|
||||||
|
#
|
||||||
|
# find_package(Libevent
|
||||||
|
# [version] [EXACT] # Minimum or exact version
|
||||||
|
# [REQUIRED] # Fail if Libevent is not found
|
||||||
|
# [COMPONENT <C>...]) # Libraries to look for
|
||||||
|
#
|
||||||
|
# Valid components are one or more of:: libevent core extra pthreads openssl.
|
||||||
|
# Note that 'libevent' contains both core and extra. You must specify one of
|
||||||
|
# them for the other components.
|
||||||
|
#
|
||||||
|
# This module will define the following variables::
|
||||||
|
#
|
||||||
|
# LIBEVENT_FOUND - True if headers and requested libraries were found
|
||||||
|
# LIBEVENT_INCLUDE_DIRS - Libevent include directories
|
||||||
|
# LIBEVENT_LIBRARIES - Libevent libraries to be linked
|
||||||
|
# LIBEVENT_<C>_FOUND - Component <C> was found (<C> is uppercase)
|
||||||
|
# LIBEVENT_<C>_LIBRARY - Library to be linked for Libevent component <C>.
|
||||||
|
|
||||||
|
find_package(PkgConfig QUIET)
|
||||||
|
pkg_check_modules(PC_LIBEVENT QUIET libevent)
|
||||||
|
|
||||||
|
# Look for the Libevent 2.0 or 1.4 headers
|
||||||
|
find_path(LIBEVENT_INCLUDE_DIR
|
||||||
|
NAMES
|
||||||
|
event2/event-config.h
|
||||||
|
event-config.h
|
||||||
|
HINTS
|
||||||
|
${PC_LIBEVENT_INCLUDE_DIRS}
|
||||||
|
)
|
||||||
|
|
||||||
|
if(LIBEVENT_INCLUDE_DIR)
|
||||||
|
set(_version_regex "^#define[ \t]+_EVENT_VERSION[ \t]+\"([^\"]+)\".*")
|
||||||
|
if(EXISTS "${LIBEVENT_INCLUDE_DIR}/event2/event-config.h")
|
||||||
|
# Libevent 2.0
|
||||||
|
file(STRINGS "${LIBEVENT_INCLUDE_DIR}/event2/event-config.h"
|
||||||
|
LIBEVENT_VERSION REGEX "${_version_regex}")
|
||||||
|
if("${LIBEVENT_VERSION}" STREQUAL "")
|
||||||
|
set(LIBEVENT_VERSION ${PC_LIBEVENT_VERSION})
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
# Libevent 1.4
|
||||||
|
file(STRINGS "${LIBEVENT_INCLUDE_DIR}/event-config.h"
|
||||||
|
LIBEVENT_VERSION REGEX "${_version_regex}")
|
||||||
|
endif()
|
||||||
|
string(REGEX REPLACE "${_version_regex}" "\\1"
|
||||||
|
LIBEVENT_VERSION "${LIBEVENT_VERSION}")
|
||||||
|
unset(_version_regex)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(_LIBEVENT_REQUIRED_VARS)
|
||||||
|
foreach(COMPONENT ${Libevent_FIND_COMPONENTS})
|
||||||
|
set(_LIBEVENT_LIBNAME libevent)
|
||||||
|
# Note: compare two variables to avoid a CMP0054 policy warning
|
||||||
|
if(COMPONENT STREQUAL _LIBEVENT_LIBNAME)
|
||||||
|
set(_LIBEVENT_LIBNAME event)
|
||||||
|
else()
|
||||||
|
set(_LIBEVENT_LIBNAME "event_${COMPONENT}")
|
||||||
|
endif()
|
||||||
|
string(TOUPPER "${COMPONENT}" COMPONENT_UPPER)
|
||||||
|
find_library(LIBEVENT_${COMPONENT_UPPER}_LIBRARY
|
||||||
|
NAMES ${_LIBEVENT_LIBNAME}
|
||||||
|
HINTS ${PC_LIBEVENT_LIBRARY_DIRS}
|
||||||
|
)
|
||||||
|
if(LIBEVENT_${COMPONENT_UPPER}_LIBRARY)
|
||||||
|
set(Libevent_${COMPONENT}_FOUND 1)
|
||||||
|
endif()
|
||||||
|
list(APPEND _LIBEVENT_REQUIRED_VARS LIBEVENT_${COMPONENT_UPPER}_LIBRARY)
|
||||||
|
endforeach()
|
||||||
|
unset(_LIBEVENT_LIBNAME)
|
||||||
|
|
||||||
|
include(FindPackageHandleStandardArgs)
|
||||||
|
# handle the QUIETLY and REQUIRED arguments and set LIBEVENT_FOUND to TRUE
|
||||||
|
# if all listed variables are TRUE and the requested version matches.
|
||||||
|
find_package_handle_standard_args(Libevent REQUIRED_VARS
|
||||||
|
${_LIBEVENT_REQUIRED_VARS}
|
||||||
|
LIBEVENT_INCLUDE_DIR
|
||||||
|
VERSION_VAR LIBEVENT_VERSION
|
||||||
|
HANDLE_COMPONENTS)
|
||||||
|
|
||||||
|
if(LIBEVENT_FOUND)
|
||||||
|
set(LIBEVENT_INCLUDE_DIRS ${LIBEVENT_INCLUDE_DIR})
|
||||||
|
set(LIBEVENT_LIBRARIES)
|
||||||
|
foreach(COMPONENT ${Libevent_FIND_COMPONENTS})
|
||||||
|
string(TOUPPER "${COMPONENT}" COMPONENT_UPPER)
|
||||||
|
list(APPEND LIBEVENT_LIBRARIES ${LIBEVENT_${COMPONENT_UPPER}_LIBRARY})
|
||||||
|
set(LIBEVENT_${COMPONENT_UPPER}_FOUND ${Libevent_${COMPONENT}_FOUND})
|
||||||
|
endforeach()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
mark_as_advanced(LIBEVENT_INCLUDE_DIR ${_LIBEVENT_REQUIRED_VARS})
|
||||||
|
unset(_LIBEVENT_REQUIRED_VARS)
|
Loading…
Add table
Reference in a new issue