diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md index db61d59d9e..c05b871b16 100644 --- a/.github/ISSUE_TEMPLATE/bug_report.md +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -8,7 +8,7 @@ labels: bug +* Discord https://discord.gg/UMWUnMjN4x --> diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml index 68151bb779..e32839fe8f 100644 --- a/.github/ISSUE_TEMPLATE/config.yml +++ b/.github/ISSUE_TEMPLATE/config.yml @@ -1,7 +1,7 @@ blank_issues_enabled: false contact_links: - name: Gridcoin Discord - url: https://discord.gg/jf9XX4a + url: https://discord.gg/UMWUnMjN4x about: Please go here if you have any general issues that are not bug reports. We can assist you much faster there. - name: Gridcoin Subreddit url: https://reddit.com/r/gridcoin diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md index a1b7ccd993..1e9ba30806 100644 --- a/.github/ISSUE_TEMPLATE/feature_request.md +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -8,7 +8,7 @@ labels: enhancement +* Discord https://discord.gg/UMWUnMjN4x --> # Feature Request diff --git a/.github/workflows/cmake-ci.yml b/.github/workflows/cmake-ci.yml new file mode 100644 index 0000000000..dd34e2d624 --- /dev/null +++ b/.github/workflows/cmake-ci.yml @@ -0,0 +1,194 @@ +name: CMake CI +on: + - push + - pull_request + - workflow_dispatch + +jobs: + test-linux: + runs-on: ubuntu-latest + env: + CCACHE_DIR: ${{github.workspace}}/ccache + CCACHE_MAXSIZE: 400M + CCACHE_COMPILERCHECK: content + strategy: + matrix: + tag: + - minimal + - no-asm + - gui-full + - system-libs + include: + - tag: no-asm + deps: null + options: -DENABLE_PIE=ON -DUSE_ASM=OFF + - tag: gui-full + deps: >- + libminiupnpc-dev + libqrencode-dev + qtbase5-dev + qttools5-dev + options: >- + -DENABLE_GUI=ON + -DENABLE_QRENCODE=ON + -DENABLE_UPNP=ON + -DUSE_DBUS=ON + - tag: system-libs + deps: >- + libdb5.3++-dev + libleveldb-dev + libsnappy-dev + libsecp256k1-dev + libunivalue-dev + xxd + options: >- + -DSYSTEM_BDB=ON + -DSYSTEM_LEVELDB=ON + -DSYSTEM_UNIVALUE=ON + steps: + - name: Checkout + uses: actions/checkout@v3 + - name: Install dependencies + uses: awalsh128/cache-apt-pkgs-action@latest + with: + packages: | + ${{matrix.deps}} + ccache + libcurl4-openssl-dev + libzip-dev + ninja-build + zipcmp + zipmerge + ziptool + version: ${{matrix.tag}} + - name: Install Boost dependencies + run: | + sudo apt-get install -y --no-install-recommends \ + libboost-dev \ + libboost-date-time-dev \ + libboost-exception-dev \ + libboost-filesystem-dev \ + libboost-iostreams-dev \ + libboost-serialization-dev \ + libboost-test-dev \ + libboost-thread-dev + - name: Configure + run: | + cmake -B ${{github.workspace}}/build -G Ninja \ + -DCMAKE_C_COMPILER_LAUNCHER=ccache \ + -DCMAKE_CXX_COMPILER_LAUNCHER=ccache \ + ${{matrix.options}} \ + -DENABLE_TESTS=ON + - name: Restore cache + uses: actions/cache/restore@v3 + if: always() + with: + path: ${{env.CCACHE_DIR}} + key: ccache-linux-${{matrix.tag}}-${{github.run_id}} + restore-keys: | + ccache-linux-${{matrix.tag}}- + - name: Build + run: | + cmake --build ${{github.workspace}}/build -v -j $(nproc) + - name: Save cache + uses: actions/cache/save@v3 + if: always() + with: + path: ${{env.CCACHE_DIR}} + key: ccache-linux-${{matrix.tag}}-${{github.run_id}} + - name: Run tests + run: | + ctest --test-dir ${{github.workspace}}/build -j $(nproc) + - name: Upload test logs + uses: actions/upload-artifact@v3 + if: always() + with: + name: testlog-linux-${{matrix.tag}} + path: ${{github.workspace}}/build/Testing/Temporary/LastTest.log + retention-days: 7 + + test-macos: + runs-on: macos-latest + env: + CCACHE_DIR: ${{github.workspace}}/ccache + CCACHE_MAXSIZE: 400M + CCACHE_COMPILERCHECK: content + strategy: + matrix: + tag: + - minimal + - no-asm + - gui-full + - system-libs + include: + - tag: no-asm + deps: null + options: -DENABLE_PIE=ON -DUSE_ASM=OFF + - tag: gui-full + deps: >- + miniupnpc + qrencode + qt@5 + options: >- + -DENABLE_GUI=ON + -DQt5_DIR=/usr/local/opt/qt5/lib/cmake/Qt5 + -DENABLE_QRENCODE=ON + -DENABLE_UPNP=ON + - tag: system-libs + deps: >- + berkeley-db@5 + secp256k1 + vim + options: >- + -DSYSTEM_BDB=ON + -DBerkeleyDB_INCLUDE_DIR=/usr/local/opt/berkeley-db@5/include + -DBerkeleyDB_CXX_LIBRARY=/usr/local/opt/berkeley-db@5/lib/libdb_cxx.dylib + -DSYSTEM_SECP256K1=ON + -DSYSTEM_XXD=ON + steps: + - name: Checkout + uses: actions/checkout@v3 + - name: Install dependencies + run: | + brew install boost ccache ninja ${{matrix.deps}} + - name: Configure + run: | + PKG_CONFIG_PATH="/usr/local/opt/openssl@3/lib/pkgconfig:${PKG_CONFIG_PATH}" + export PKG_CONFIG_PATH + + pushd src + ../contrib/nomacro.pl + popd + + cmake -B ${{github.workspace}}/build -G Ninja \ + -DCMAKE_C_COMPILER_LAUNCHER=ccache \ + -DCMAKE_CXX_COMPILER_LAUNCHER=ccache \ + ${{matrix.options}} \ + -DENABLE_TESTS=ON + - name: Restore cache + uses: actions/cache/restore@v3 + if: always() + with: + path: ${{env.CCACHE_DIR}} + key: ccache-macos-${{matrix.tag}}-${{github.run_id}} + restore-keys: | + ccache-macos-${{matrix.tag}}- + - name: Build + run: | + cmake --build ${{github.workspace}}/build -v -j $(sysctl -n hw.logicalcpu) + - name: Save cache + uses: actions/cache/save@v3 + if: always() + with: + path: ${{env.CCACHE_DIR}} + key: ccache-macos-${{matrix.tag}}-${{github.run_id}} + - name: Run tests + run: | + ctest --test-dir ${{github.workspace}}/build -j $(sysctl -n hw.logicalcpu) + - name: Upload test logs + uses: actions/upload-artifact@v3 + if: always() + with: + name: testlog-macos-${{matrix.tag}} + path: ${{github.workspace}}/build/Testing/Temporary/LastTest.log + retention-days: 7 diff --git a/.gitignore b/.gitignore index aed5378031..f2d2617bf1 100644 --- a/.gitignore +++ b/.gitignore @@ -12,7 +12,7 @@ src/test/test_gridcoin.log src/test/test_gridcoin.trs src/build.h -src/obj +src/obj/build.h src/qt/forms/*.h src/qt/forms/voting/*.h src/qt/moc_*.cpp @@ -99,7 +99,6 @@ config.status configure libtool src/config/gridcoin-config.h -src/config/gridcoin-config.h.in src/config/stamp-h1 share/setup.nsi share/qt/Info.plist diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000000..f458556791 --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,262 @@ +# CMake build system is intended to be used by maintainers to package Gridcoin +# for software repositories. For that reason, only external dependencies are +# supported. +# +# Use Autotools build system for all other needs. +# +# CMake support is experimental. Use with caution and report any bugs. + +cmake_minimum_required(VERSION 3.18) + +project("Gridcoin" + VERSION 5.4.5.4 + DESCRIPTION "POS-based cryptocurrency that rewards BOINC computation" + HOMEPAGE_URL "https://gridcoin.us" + LANGUAGES ASM C CXX +) + +set(CLIENT_VERSION_IS_RELEASE "false") +set(COPYRIGHT_YEAR "2023") +set(COPYRIGHT_HOLDERS_FINAL "The Gridcoin developers") + + +# Toolchain configuration +# ======================= + +set(CMAKE_CXX_STANDARD 17) + +set(CMAKE_C_VISIBILITY_PRESET hidden) +set(CMAKE_CXX_VISIBILITY_PRESET hidden) + +set(CMAKE_MSVC_RUNTIME_LIBRARY MultiThreaded) + +set(CMAKE_INCLUDE_CURRENT_DIR ON) + +# Remove '-DNDEBUG' from flags because we need asserts +string(REPLACE "NDEBUG" "_NDEBUG" CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE}") +string(REPLACE "NDEBUG" "_NDEBUG" CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO}") + + +# Load modules from the source tree +# ================================= + +set(CMAKE_MODULE_PATH "${CMAKE_MODULE_PATH}" "${CMAKE_CURRENT_SOURCE_DIR}/build-aux/cmake") + +include(CheckCXXSymbolExists) +include(CheckFunctionExists) +include(CheckIncludeFile) +include(CheckPIESupported) +include(CheckSSE) +include(CheckStrerrorR) +include(CheckSymbolExists) +include(VersionFromGit) + +# Define options +# ============== + +# Build configuration +option(ENABLE_DAEMON "Enable daemon" ON) +option(ENABLE_GUI "Enable Qt-based GUI" OFF) +option(ENABLE_DOCS "Build Doxygen documentation" OFF) +option(ENABLE_TESTS "Build tests" OFF) +option(LUPDATE "Update translation files" OFF) + +# CPU-dependent options +option(ENABLE_SSE41 "Build code that uses SSE4.1 intrinsics" ${HAS_SSE41}) +option(ENABLE_AVX2 "Build code that uses AVX2 intrinsics" ${HAS_AVX2}) +option(ENABLE_X86_SHANI "Build code that uses x86 SHA-NI intrinsics" ${HAS_X86_SHANI}) +option(ENABLE_ARM_SHANI "Build code that uses ARM SHA-NI intrinsics" ${HAS_ARM_SHANI}) +option(USE_ASM "Enable assembly routines" ON) + +# Optional functionality +option(ENABLE_PIE "Build position-independent executables" OFF) +option(ENABLE_QRENCODE "Enable generation of QR Codes for receiving payments" OFF) +option(ENABLE_UPNP "Enable UPnP port mapping support" OFF) +option(DEFAULT_UPNP "Turn UPnP on startup" OFF) +option(USE_DBUS "Enable DBus support" OFF) +option(SYSTEM_BDB "Find system installation of Berkeley DB CXX 5.3" OFF) +option(SYSTEM_LEVELDB "Find system installation of leveldb" OFF) +option(SYSTEM_SECP256K1 "Find system installation of libsecp256k1 with pkg-config" OFF) +option(SYSTEM_UNIVALUE "Find system installation of Univalue with pkg-config" OFF) +option(SYSTEM_XXD "Find system xxd binary" OFF) + + +# Find dependencies +# ================= + +set(BOOST_MINIMUM_VERSION 1.63.0) +set(QT5_MINIMUM_VERSION 5.15.0) + +find_package(Atomics REQUIRED) +find_package(Boost ${BOOST_MINIMUM_VERSION} COMPONENTS filesystem iostreams thread REQUIRED) +find_package(CURL COMPONENTS HTTP HTTPS SSL REQUIRED) +find_package(OpenSSL REQUIRED) +find_package(Threads REQUIRED) +find_package(libzip REQUIRED) + +if(SYSTEM_BDB) + find_package(BerkeleyDB 5.3...<5.4 COMPONENTS CXX REQUIRED) +else() + find_program(MAKE_EXE NAMES gmake nmake make) +endif() + +if(SYSTEM_LEVELDB) + find_package(leveldb REQUIRED) +endif() + +if(SYSTEM_SECP256K1) + find_package(PkgConfig) + pkg_check_modules(SECP256K1 REQUIRED IMPORTED_TARGET "libsecp256k1 >= 0.2.0") +endif() + +if(SYSTEM_UNIVALUE) + find_package(PkgConfig) + pkg_check_modules(UNIVALUE REQUIRED IMPORTED_TARGET libunivalue) +endif() + +if(ENABLE_GUI) + find_package(Qt5 ${QT5_MINIMUM_VERSION} REQUIRED COMPONENTS + Concurrent + Core + Gui + LinguistTools + Network + Widgets + ) + + if(USE_DBUS) + find_package(Qt5 ${QT5_MINIMUM_VERSION} COMPONENTS DBus REQUIRED) + endif() + + if(ENABLE_TESTS) + find_package(Qt5 ${QT5_MINIMUM_VERSION} COMPONENTS Test REQUIRED) + endif() + + if(ENABLE_QRENCODE) + pkg_check_modules(QRENCODE REQUIRED IMPORTED_TARGET libqrencode) + endif() +endif() + +if(ENABLE_UPNP) + pkg_check_modules(MINIUPNPC REQUIRED IMPORTED_TARGET miniupnpc>=1.9) +endif() + +if(ENABLE_TESTS) + find_package(Boost ${BOOST_MINIMUM_VERSION} COMPONENTS unit_test_framework REQUIRED) + enable_testing() + + if(SYSTEM_XXD) + find_program(XXD xxd REQUIRED) + endif() +endif() + +if(UNIX) + find_package(Rt REQUIRED) +endif() + +if(WIN32) + enable_language(RC) + find_program(MAKENSIS makensis) +elseif(APPLE) + enable_language(OBJCXX) +endif() + + +# Run probes +# ========== + +if(ENABLE_PIE) + check_pie_supported() + if(NOT CMAKE_CXX_LINK_PIE_SUPPORTED) + message(FATAL_ERROR "PIE is not supported by the current linker") + endif() +endif() + +set(CMAKE_POSITION_INDEPENDENT_CODE ${ENABLE_PIE}) + +# Set compiler flags +if (APPLE) + add_compile_options(-Wno-error=deprecated-declarations) + add_compile_options(-Wno-error=thread-safety-analysis) + add_compile_options(-Wno-error=thread-safety-reference) +endif() + +# Set endianness +if(CMAKE_CXX_BYTE_ORDER EQUAL BIG_ENDIAN) + set(WORDS_BIGENDIAN 1) +endif() + +# Check headers +check_include_file("byteswap.h" HAVE_BYTESWAP_H) +check_include_file("endian.h" HAVE_ENDIAN_H) +check_include_file("sys/endian.h" HAVE_SYS_ENDIAN_H) +check_include_file("sys/prctl.h" HAVE_SYS_PRCTL_H) + +if(HAVE_ENDIAN_H) + set(ENDIAN_INCLUDES "endian.h") +else() + set(ENDIAN_INCLUDES "sys/endian.h") +endif() + +if(HAVE_BYTESWAP_H) + set(BYTESWAP_INCLUDES "byteswap.h") +endif() + +# Check symbols +check_symbol_exists(fork "unistd.h" HAVE_DECL_FORK) +check_symbol_exists(pipe2 "unistd.h" HAVE_DECL_PIPE2) +check_symbol_exists(setsid "unistd.h" HAVE_DECL_SETSID) + +check_symbol_exists(le16toh "${ENDIAN_INCLUDES}" HAVE_DECL_LE16TOH) +check_symbol_exists(le32toh "${ENDIAN_INCLUDES}" HAVE_DECL_LE32TOH) +check_symbol_exists(le64toh "${ENDIAN_INCLUDES}" HAVE_DECL_LE64TOH) + +check_symbol_exists(htole16 "${ENDIAN_INCLUDES}" HAVE_DECL_HTOLE16) +check_symbol_exists(htole32 "${ENDIAN_INCLUDES}" HAVE_DECL_HTOLE32) +check_symbol_exists(htole64 "${ENDIAN_INCLUDES}" HAVE_DECL_HTOLE64) + +check_symbol_exists(be16toh "${ENDIAN_INCLUDES}" HAVE_DECL_BE16TOH) +check_symbol_exists(be32toh "${ENDIAN_INCLUDES}" HAVE_DECL_BE32TOH) +check_symbol_exists(be64toh "${ENDIAN_INCLUDES}" HAVE_DECL_BE64TOH) + +check_symbol_exists(htobe16 "${ENDIAN_INCLUDES}" HAVE_DECL_HTOBE16) +check_symbol_exists(htobe32 "${ENDIAN_INCLUDES}" HAVE_DECL_HTOBE32) +check_symbol_exists(htobe64 "${ENDIAN_INCLUDES}" HAVE_DECL_HTOBE64) + +check_symbol_exists(bswap_16 "${BYTESWAP_INCLUDES}" HAVE_DECL_BSWAP_16) +check_symbol_exists(bswap_32 "${BYTESWAP_INCLUDES}" HAVE_DECL_BSWAP_32) +check_symbol_exists(bswap_64 "${BYTESWAP_INCLUDES}" HAVE_DECL_BSWAP_64) + +check_function_exists(__builtin_clzl HAVE_BUILTIN_CLZL) +check_function_exists(__builtin_clzll HAVE_BUILTIN_CLZLL) + +check_symbol_exists(MSG_NOSIGNAL "sys/socket.h" HAVE_MSG_NOSIGNAL) +check_symbol_exists(MSG_DONTWAIT "sys/socket.h" HAVE_MSG_DONTWAIT) + +check_symbol_exists(malloc_info "malloc.h" HAVE_MALLOC_INFO) +check_symbol_exists(M_ARENA_MAX "malloc.h" HAVE_MALLOPT_ARENA_MAX) + +check_cxx_symbol_exists(std::system "cstdlib" HAVE_SYSTEM) +check_cxx_symbol_exists(gmtime_r "ctime" HAVE_GMTIME_R) + +if(NOT HAVE_GMTIME_R) + check_cxx_symbol_exists(gmtime_s "ctime" HAVE_GMTIME_S) + if(NOT HAVE_GMTIME_S) + message(FATAL_ERROR "Both gmtime_r and gmtime_s are unavailable") + endif() +endif() + +check_symbol_exists(SYS_getrandom "sys/syscall.h" HAVE_SYS_GETRANDOM) +check_symbol_exists(getentropy "unistd.h" HAVE_GETENTROPY) +check_symbol_exists(KERN_ARND "sys/sysctl.h" HAVE_SYSCTL_ARND) + +check_symbol_exists(O_CLOEXEC "fcntl.h" HAVE_O_CLOEXEC) +check_symbol_exists(getauxval "sys/auxv.h" HAVE_STRONG_GETAUXVAL) + +# Descend into subdirectories +# =========================== + +add_subdirectory(src) +if(ENABLE_DOCS) + add_subdirectory(doc) +endif() diff --git a/README.md b/README.md index 9e088ec989..66b481e5d0 100644 --- a/README.md +++ b/README.md @@ -11,18 +11,28 @@ Building Gridcoin These dependencies are required: - Library | Purpose | Description - ------------|------------------|---------------------------------------------------------------- - pkg-config | Build | Learn library inter-dependencies - libssl | Crypto | Random Number Generation, Elliptic Curve Cryptography - libboost | Utility | Library for threading, data structures, etc - libevent | Networking | OS independent asynchronous networking - miniupnpc | UPnP Support | Firewall-jumping support - qt | GUI | GUI toolkit (only needed when GUI enabled) - libqrencode | QR codes in GUI | Optional for generating QR codes (only needed when GUI enabled) - -To build, run -```./autogen.sh && ./configure && make```. + Library | Purpose | Description + -------------|------------------|---------------------------------------------------------------- + cmake | Build | Build system (optional) + pkgconf | Build | Learn library inter-dependencies + openssl | Crypto | Random Number Generation, Elliptic Curve Cryptography + libboost | Utility | Library for threading, data structures, etc + libcurl | Utility | URL client library + libzip | Utility | Library for manipulating zip archives + miniupnpc | UPnP Support | Firewall-jumping support (optional) + qt5 | GUI | GUI toolkit (optional) + libqrencode | QR codes in GUI | Library for encoding data in a QR Code symbol (optional, depends on GUI) + +To build, run: + +* With CMake: + + `mkdir build && cmake build && cmake .. && cmake --build .` + +* With Autotools: + + `./autogen.sh && ./configure && make` + For more detailed and platform-specific instructions, see [the doc folder.](doc/) Development process @@ -71,7 +81,7 @@ master if the staging branch is busy. Community ========= -For general questions, please visit our Discord server at https://discord.gg/jf9XX4a, or Freenode IRC in #gridcoin-help. We also have a Slack channel at [teamgridcoin.slack.com](https://join.slack.com/t/teamgridcoin/shared_invite/zt-3s81akww-GHt~_KvtxfhxUgi3yW3~Bg). +For general questions, please visit our Discord server at https://discord.gg/UMWUnMjN4x, or Libera Chat in #gridcoin-help. We also have a Slack channel at [teamgridcoin.slack.com](https://join.slack.com/t/teamgridcoin/shared_invite/zt-3s81akww-GHt~_KvtxfhxUgi3yW3~Bg). License ------- diff --git a/build-aux/cmake/CheckSSE.cmake b/build-aux/cmake/CheckSSE.cmake new file mode 100644 index 0000000000..56984403e3 --- /dev/null +++ b/build-aux/cmake/CheckSSE.cmake @@ -0,0 +1,50 @@ +include(CheckCSourceRuns) + +check_c_source_runs(" + #include + + int main() { + __m128i l = _mm_set1_epi32(0); + return _mm_extract_epi32(l, 3); + }" + HAS_SSE41 +) + +check_c_source_runs(" + #include + #include + + int main() { + __m256i l = _mm256_set1_epi32(0); + return _mm256_extract_epi32(l, 7); + }" + HAS_AVX2 +) + +check_c_source_runs(" + #include + #include + + int main() { + __m128i i = _mm_set1_epi32(0); + __m128i j = _mm_set1_epi32(1); + __m128i k = _mm_set1_epi32(2); + return _mm_extract_epi32(_mm_sha256rnds2_epu32(i, i, k), 0); + }" + HAS_X86_SHANI +) + +check_c_source_runs(" + #include + #include + + int main() { + uint32x4_t a, b, c; + vsha256h2q_u32(a, b, c); + vsha256hq_u32(a, b, c); + vsha256su0q_u32(a, b); + vsha256su1q_u32(a, b, c); + return 0; + }" + HAS_ARM_SHANI +) diff --git a/build-aux/cmake/CheckStrerrorR.cmake b/build-aux/cmake/CheckStrerrorR.cmake new file mode 100644 index 0000000000..e74840b7c8 --- /dev/null +++ b/build-aux/cmake/CheckStrerrorR.cmake @@ -0,0 +1,26 @@ +include(CheckCCompilerFlag) +include(CheckCSourceRuns) +include(CheckSymbolExists) + +check_symbol_exists(strerror_r "string.h" HAVE_STRERROR_R) + +if(HAVE_STRERROR_R) + check_c_compiler_flag(-Werror HAS_WERROR) + if(HAS_WERROR) + set(CMAKE_REQUIRED_FLAGS "-Werror") + endif() + check_c_source_runs(" + #include + #include + #include + + int main() { + char buf[280]; + char* s = strerror_r(ENOENT, buf, 280); + printf(\"%s\", s); + return 0; + }" + STRERROR_R_CHAR_P + ) + set(CMAKE_REQUIRED_FLAGS "") +endif() diff --git a/build-aux/cmake/ExternalLibraryHelper.cmake b/build-aux/cmake/ExternalLibraryHelper.cmake new file mode 100644 index 0000000000..2aeee7280e --- /dev/null +++ b/build-aux/cmake/ExternalLibraryHelper.cmake @@ -0,0 +1,62 @@ +# Copyright (c) 2017-2020 The Bitcoin developers +# Distributed under the MIT software license, see the accompanying +# file COPYING or https://www.opensource.org/licenses/mit-license.php. + +include(FindPackageMessage) + +# Find a library component, set the variables and create an imported target. +# Variable names are compliant with cmake standards. +function(find_component LIB COMPONENT) + cmake_parse_arguments(ARG + "" + "" + "HINTS;INCLUDE_DIRS;INTERFACE_LINK_LIBRARIES;NAMES;PATHS;PATH_SUFFIXES" + ${ARGN} + ) + + # If the component is not requested, skip the search. + if(${LIB}_FIND_COMPONENTS AND NOT ${COMPONENT} IN_LIST ${LIB}_FIND_COMPONENTS) + return() + endif() + + find_library(${LIB}_${COMPONENT}_LIBRARY + NAMES ${ARG_NAMES} + PATHS "" ${ARG_PATHS} + HINTS "" ${ARG_HINTS} + PATH_SUFFIXES "lib" ${ARG_PATH_SUFFIXES} + ) + mark_as_advanced(${LIB}_${COMPONENT}_LIBRARY) + + if(${LIB}_${COMPONENT}_LIBRARY) + # On success, set the standard FOUND variable... + set(${LIB}_${COMPONENT}_FOUND TRUE PARENT_SCOPE) + + # ... and append the library path to the LIBRARIES variable ... + list(APPEND ${LIB}_LIBRARIES + "${${LIB}_${COMPONENT}_LIBRARY}" + ${ARG_INTERFACE_LINK_LIBRARIES} + ) + list(REMOVE_DUPLICATES ${LIB}_LIBRARIES) + set(${LIB}_LIBRARIES ${${LIB}_LIBRARIES} PARENT_SCOPE) + + # ... and create an imported target for the component, if not already + # done. + if(NOT TARGET ${LIB}::${COMPONENT}) + add_library(${LIB}::${COMPONENT} UNKNOWN IMPORTED) + set_target_properties(${LIB}::${COMPONENT} PROPERTIES + IMPORTED_LOCATION "${${LIB}_${COMPONENT}_LIBRARY}" + ) + set_property(TARGET ${LIB}::${COMPONENT} PROPERTY + INTERFACE_INCLUDE_DIRECTORIES ${ARG_INCLUDE_DIRS} + ) + set_property(TARGET ${LIB}::${COMPONENT} PROPERTY + INTERFACE_LINK_LIBRARIES ${ARG_INTERFACE_LINK_LIBRARIES} + ) + endif() + + find_package_message("${LIB}_${COMPONENT}" + "Found ${LIB} component ${COMPONENT}: ${${LIB}_${COMPONENT}_LIBRARY}" + "[${${LIB}_${COMPONENT}_LIBRARY}][${ARG_INCLUDE_DIRS}]" + ) + endif() +endfunction() diff --git a/build-aux/cmake/FindAtomics.cmake b/build-aux/cmake/FindAtomics.cmake new file mode 100644 index 0000000000..5954308834 --- /dev/null +++ b/build-aux/cmake/FindAtomics.cmake @@ -0,0 +1,53 @@ +# Original issue: +# * https://gitlab.kitware.com/cmake/cmake/-/issues/23021#note_1098733 +# +# For reference: +# * https://gcc.gnu.org/wiki/Atomic/GCCMM +# +# riscv64 specific: +# * https://lists.debian.org/debian-riscv/2022/01/msg00009.html +# +# ATOMICS_FOUND - system has c++ atomics +# ATOMICS_LIBRARIES - libraries needed to use c++ atomics + +include(CheckCXXSourceCompiles) + +# RISC-V only has 32-bit and 64-bit atomic instructions. GCC is supposed +# to convert smaller atomics to those larger ones via masking and +# shifting like LLVM, but it’s a known bug that it does not. This means +# anything that wants to use atomics on 1-byte or 2-byte types needs +# -latomic, but not 4-byte or 8-byte (though it does no harm). +set(atomic_code " + #include + #include + + std::atomic n8 (0); // riscv64 + std::atomic n64 (0); // armel, mipsel, powerpc + + int main() { + ++n8; + ++n64; + return 0; + }" +) +check_cxx_source_compiles("${atomic_code}" ATOMICS_LOCK_FREE_INSTRUCTIONS) + +if(ATOMICS_LOCK_FREE_INSTRUCTIONS) + set(ATOMICS_FOUND TRUE) + set(ATOMICS_LIBRARIES) +else() + set(CMAKE_REQUIRED_LIBRARIES "-latomic") + check_cxx_source_compiles("${atomic_code}" ATOMICS_IN_LIBRARY) + set(CMAKE_REQUIRED_LIBRARIES) + if(ATOMICS_IN_LIBRARY) + set(ATOMICS_LIBRARY atomic) + include(FindPackageHandleStandardArgs) + find_package_handle_standard_args(Atomics DEFAULT_MSG ATOMICS_LIBRARY) + set(ATOMICS_LIBRARIES ${ATOMICS_LIBRARY}) + unset(ATOMICS_LIBRARY) + else() + if(Atomics_FIND_REQUIRED) + message(FATAL_ERROR "Neither lock free instructions nor -latomic found.") + endif() + endif() +endif() diff --git a/build-aux/cmake/FindBerkeleyDB.cmake b/build-aux/cmake/FindBerkeleyDB.cmake new file mode 100644 index 0000000000..f9b4405ca0 --- /dev/null +++ b/build-aux/cmake/FindBerkeleyDB.cmake @@ -0,0 +1,171 @@ +# Copyright (c) 2017-2020 The Bitcoin developers +# Distributed under the MIT software license, see the accompanying +# file COPYING or https://www.opensource.org/licenses/mit-license.php. + +#.rst +# FindBerkeleyDB +# ------------- +# +# This is inspired by https://github.com/sum01/FindBerkeleyDB. +# +# Find the Berkeley database (versions >= 5.x) libraries The following +# components are available:: +# C +# CXX +# +# This will define the following variables:: +# +# BerkeleyDB_FOUND - system has Berkeley DB lib +# BerkeleyDB_INCLUDE_DIRS - the Berkeley DB include directories +# BerkeleyDB_LIBRARIES - Libraries needed to use Berkeley DB +# BerkeleyDB_VERSION - The library version MAJOR.MINOR.PATCH +# BerkeleyDB_VERSION_MAJOR - Major version number +# BerkeleyDB_VERSION_MINOR - Minor version number +# BerkeleyDB_VERSION_PATCH - Patch version number +# +# And the following imported target:: +# +# BerkeleyDB::C +# BerkeleyDB::CXX + +# Generate a list of all the possible versioned library name variants given a +# list of separators. +function(generate_versions_variants VARIANTS LIB MAJOR MINOR) + set(SEPARATORS + "" "." "-" "_" + ) + + set(${VARIANTS} "${LIB}") + foreach(_separator1 IN LISTS SEPARATORS) + list(APPEND ${VARIANTS} "${LIB}${_separator1}${MAJOR}") + foreach(_separator2 IN LISTS SEPARATORS) + list(APPEND ${VARIANTS} "${LIB}${_separator1}${MAJOR}${_separator2}${MINOR}") + endforeach() + endforeach() + + # We need to search from the most specific to the least specific to prevent + # mismatches, e.g. if the include dir is /usr/include/db5.3 we want to link + # /usr/lib/libdb5.3.so and not libdb.so, which could very well be another + # version. Note that this is not only theoretical and actually happened on + # Archlinux with both db5.3 and db6.2 installed. + list(REVERSE ${VARIANTS}) + + set(${VARIANTS} ${${VARIANTS}} PARENT_SCOPE) +endfunction() + +# If the include directory is user supplied, skip the search +if(NOT BerkeleyDB_INCLUDE_DIR) + # Berkeley DB 5 including latest minor release. + generate_versions_variants(_BerkeleyDB_PATH_SUFFIXES_5_3 db 5 3) + + set(_BerkeleyDB_PATH_SUFFIXES + include + ${_BerkeleyDB_PATH_SUFFIXES_5_3} + ) + list(REMOVE_DUPLICATES _BerkeleyDB_PATH_SUFFIXES) + + # Try to find the db.h header. + # If the header is not found the user can supply the correct path by passing + # the `BerkeleyDB_ROOT` variable to cmake. + find_path(BerkeleyDB_INCLUDE_DIR + NAMES db.h + PATH_SUFFIXES ${_BerkeleyDB_PATH_SUFFIXES} + ) +endif() + +# There is a single common include directory. +# Set the BerkeleyDB_INCLUDE_DIRS variable which is the expected standard output +# variable name for the include directories. +set(BerkeleyDB_INCLUDE_DIRS "${BerkeleyDB_INCLUDE_DIR}") +mark_as_advanced(BerkeleyDB_INCLUDE_DIR) + +if(BerkeleyDB_INCLUDE_DIR) + # Extract version information from the db.h header. + if(NOT DEFINED BerkeleyDB_VERSION) + set(db_version_test_program " + #include + #include \"${BerkeleyDB_INCLUDE_DIR}/db.h\" + + int main() { + printf(\"%d;%d;%d\", DB_VERSION_MAJOR, DB_VERSION_MINOR, DB_VERSION_PATCH); + } + ") + + if(CMAKE_VERSION VERSION_LESS 3.25) + file(WRITE ${CMAKE_BINARY_DIR}/db-version.c "${db_version_test_program}") + try_run( + _run_result + _compile_result + ${CMAKE_BINARY_DIR} ${CMAKE_BINARY_DIR}/db-version.c + RUN_OUTPUT_STDOUT_VARIABLE BerkeleyDB_VERSION_LIST + ) + else() + try_run( + _run_result + _compile_result + SOURCE_FROM_CONTENT db-version.c "${db_version_test_program}" + RUN_OUTPUT_STDOUT_VARIABLE BerkeleyDB_VERSION_LIST + ) + endif() + + list(GET BerkeleyDB_VERSION_LIST 0 BerkeleyDB_VERSION_MAJOR) + list(GET BerkeleyDB_VERSION_LIST 1 BerkeleyDB_VERSION_MINOR) + list(GET BerkeleyDB_VERSION_LIST 2 BerkeleyDB_VERSION_PATCH) + + # Cache the result. + set(BerkeleyDB_VERSION_MAJOR ${BerkeleyDB_VERSION_MAJOR} + CACHE INTERNAL "BerkeleyDB major version number" + ) + set(BerkeleyDB_VERSION_MINOR ${BerkeleyDB_VERSION_MINOR} + CACHE INTERNAL "BerkeleyDB minor version number" + ) + set(BerkeleyDB_VERSION_PATCH ${BerkeleyDB_VERSION_PATCH} + CACHE INTERNAL "BerkeleyDB patch version number" + ) + # 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}" + CACHE INTERNAL "BerkeleyDB full version" + ) + endif() + + include(ExternalLibraryHelper) + + # Different systems sometimes have a version in the lib name... + # and some have a dash or underscore before the versions. + # Generate all combinations from the separators "" (none), ".", "-" and "_". + generate_versions_variants( + _db_variants + db + "${BerkeleyDB_VERSION_MAJOR}" + "${BerkeleyDB_VERSION_MINOR}" + ) + + find_component(BerkeleyDB C + NAMES ${_db_variants} + PATH_SUFFIXES ${_db_variants} + INCLUDE_DIRS ${BerkeleyDB_INCLUDE_DIRS} + ) + + generate_versions_variants( + _db_cxx_variants + db_cxx + "${BerkeleyDB_VERSION_MAJOR}" + "${BerkeleyDB_VERSION_MINOR}" + ) + + find_component(BerkeleyDB CXX + NAMES ${_db_cxx_variants} + PATH_SUFFIXES ${_db_variants} + INCLUDE_DIRS ${BerkeleyDB_INCLUDE_DIRS} + ) +endif() + +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(BerkeleyDB + REQUIRED_VARS BerkeleyDB_INCLUDE_DIR + VERSION_VAR BerkeleyDB_VERSION + HANDLE_VERSION_RANGE + HANDLE_COMPONENTS +) diff --git a/build-aux/cmake/FindRt.cmake b/build-aux/cmake/FindRt.cmake new file mode 100644 index 0000000000..4e5b76c82a --- /dev/null +++ b/build-aux/cmake/FindRt.cmake @@ -0,0 +1,32 @@ +# RT_FOUND - system has shm_open +# RT_LIBRARIES - libraries needed to use shm_open + +include(CheckCSourceCompiles) + +set(rt_code " + #include + #include + + int main() { + shm_open(0, 0, 0); + return 0; + }" +) + +check_c_source_compiles("${rt_code}" RT_BUILT_IN) + +if(RT_BUILT_IN) + set(RT_FOUND TRUE) + set(RT_LIBRARIES) +else() + set(CMAKE_REQUIRED_LIBRARIES "-lrt") + check_c_source_compiles("${rt_code}" RT_IN_LIBRARY) + set(CMAKE_REQUIRED_LIBRARIES) + if(RT_IN_LIBRARY) + set(RT_LIBRARY rt) + include(FindPackageHandleStandardArgs) + find_package_handle_standard_args(Rt DEFAULT_MSG RT_LIBRARY) + set(RT_LIBRARIES ${RT_LIBRARY}) + unset(RT_LIBRARY) + endif() +endif() diff --git a/build-aux/cmake/VersionFromGit.cmake b/build-aux/cmake/VersionFromGit.cmake new file mode 100644 index 0000000000..7d0e500b6f --- /dev/null +++ b/build-aux/cmake/VersionFromGit.cmake @@ -0,0 +1,48 @@ +find_package(Git) + +set(BUILD_GIT_TAG "" CACHE STRING "Current Git tag") +set(BUILD_GIT_COMMIT "" CACHE STRING "Current Git commit") + +if(Git_FOUND AND EXISTS "${CMAKE_SOURCE_DIR}/.git") + execute_process(COMMAND "${GIT_EXECUTABLE}" diff-index --quiet HEAD -- + WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}" + RESULT_VARIABLE BUILD_GIT_IS_DIRTY + ) + + execute_process(COMMAND "${GIT_EXECUTABLE}" rev-parse --short=12 HEAD + WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}" + OUTPUT_VARIABLE BUILD_GIT_COMMIT + ERROR_QUIET + OUTPUT_STRIP_TRAILING_WHITESPACE + ) + + execute_process(COMMAND "${GIT_EXECUTABLE}" describe --abbrev=0 + WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}" + OUTPUT_VARIABLE BUILD_GIT_LATEST_TAG + ERROR_QUIET + OUTPUT_STRIP_TRAILING_WHITESPACE + ) + + execute_process(COMMAND "${GIT_EXECUTABLE}" rev-list "${BUILD_GIT_LATEST_TAG}" -1 --abbrev-commit --abbrev=12 + WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}" + OUTPUT_VARIABLE BUILD_GIT_LATEST_TAG_COMMIT + ERROR_QUIET + OUTPUT_STRIP_TRAILING_WHITESPACE + ) + + if(BUILD_GIT_IS_DIRTY EQUAL 0) # not dirty + if(BUILD_GIT_LATEST_TAG_COMMIT EQUAL BUILD_GIT_COMMIT) + set(BUILD_GIT_TAG "${BUILD_GIT_LATEST_TAG}") + endif() + else() + set(BUILD_GIT_COMMIT "${BUILD_GIT_COMMIT}-dirty") + endif() +endif() + +if(BUILD_GIT_TAG EQUAL "") + unset(BUILD_GIT_TAG) +endif() + +if(BUILD_GIT_COMMIT EQUAL "") + unset(BUILD_GIT_COMMIT) +endif() diff --git a/build-aux/cmake/json2header.cmake b/build-aux/cmake/json2header.cmake new file mode 100644 index 0000000000..da1c68711a --- /dev/null +++ b/build-aux/cmake/json2header.cmake @@ -0,0 +1,19 @@ +cmake_minimum_required(VERSION 3.19) + +if(CMAKE_ARGC LESS 6) + message(FATAL_ERROR "Usage: cmake -P json2header.cmake xxd in_file out_file") +endif() + +set(XXD ${CMAKE_ARGV3}) +set(IN_FILE ${CMAKE_ARGV4}) +set(OUT_FILE ${CMAKE_ARGV5}) + +get_filename_component(var_name "${IN_FILE}" NAME_WE) +execute_process(COMMAND "${XXD}" -i -n ${var_name} "${IN_FILE}" + OUTPUT_VARIABLE command_out + COMMAND_ERROR_IS_FATAL ANY +) + +file(WRITE ${OUT_FILE} "namespace json_tests {\n") +file(APPEND "${OUT_FILE}" "${command_out}") +file(APPEND ${OUT_FILE} "};\n") diff --git a/build-aux/cmake/txt2header.cmake b/build-aux/cmake/txt2header.cmake new file mode 100644 index 0000000000..9d9da453c5 --- /dev/null +++ b/build-aux/cmake/txt2header.cmake @@ -0,0 +1,15 @@ +cmake_minimum_required(VERSION 3.19) + +if(CMAKE_ARGC LESS 5) + message(FATAL_ERROR "Usage: cmake -P txt2header.cmake in_file out_file") +endif() + +set(IN_FILE ${CMAKE_ARGV3}) +set(OUT_FILE ${CMAKE_ARGV4}) + +get_filename_component(var_name "${IN_FILE}" NAME_WE) + +file(READ ${IN_FILE} text) +file(WRITE "${OUT_FILE}" " + static const std::string ${var_name}_text = R\"(${text})\"; +") diff --git a/configure.ac b/configure.ac index fa68acecad..57c58f678c 100755 --- a/configure.ac +++ b/configure.ac @@ -3,7 +3,7 @@ AC_PREREQ([2.60]) define(_CLIENT_VERSION_MAJOR, 5) define(_CLIENT_VERSION_MINOR, 4) define(_CLIENT_VERSION_REVISION, 5) -define(_CLIENT_VERSION_BUILD, 2) +define(_CLIENT_VERSION_BUILD, 4) define(_CLIENT_VERSION_IS_RELEASE, false) define(_COPYRIGHT_YEAR, 2023) define(_COPYRIGHT_HOLDERS,[The %s developers]) @@ -877,13 +877,6 @@ AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[#include [ AC_MSG_RESULT(no)] ) -AC_MSG_CHECKING(for getentropy) -AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[#include ]], - [[ getentropy(nullptr, 32) ]])], - [ AC_MSG_RESULT(yes); AC_DEFINE(HAVE_GETENTROPY, 1,[Define this symbol if the BSD getentropy system call is available]) ], - [ AC_MSG_RESULT(no)] -) - AC_MSG_CHECKING(for sysctl KERN_ARND) AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[#include #include ]], diff --git a/contrib/default.nix b/contrib/default.nix index ae6cc1d043..6efefa74b1 100644 --- a/contrib/default.nix +++ b/contrib/default.nix @@ -4,7 +4,7 @@ with pkgs.stdenv; with pkgs.stdenv.lib; pkgs.mkShell { - nativeBuildInputs = with pkgs.buildPackages; [ autoreconfHook libtool pkg-config qt5.wrapQtAppsHook ]; + nativeBuildInputs = with pkgs.buildPackages; [ autoreconfHook cmake libtool pkg-config qt5.wrapQtAppsHook ]; buildInputs = with pkgs; [ boost openssl libevent curl qt5.qttools libzip qrencode ]; shellHook = '' diff --git a/depends/packages/curl.mk b/depends/packages/curl.mk index 48f0ec17ad..f492df5c03 100644 --- a/depends/packages/curl.mk +++ b/depends/packages/curl.mk @@ -1,9 +1,9 @@ package=curl GCCFLAGS?= -$(package)_version=7.78.0 +$(package)_version=7.88.1 $(package)_download_path=https://curl.haxx.se/download/ $(package)_file_name=$(package)-$($(package)_version).tar.gz -$(package)_sha256_hash=ed936c0b02c06d42cf84b39dd12bb14b62d77c7c4e875ade022280df5dcc81d7 +$(package)_sha256_hash=cdb38b72e36bc5d33d5b8810f8018ece1baa29a8f215b4495e495ded82bbf3c7 $(package)_dependencies=openssl define $(package)_set_vars diff --git a/depends/packages/expat.mk b/depends/packages/expat.mk index 16e94b514a..eca30acdd2 100644 --- a/depends/packages/expat.mk +++ b/depends/packages/expat.mk @@ -1,9 +1,9 @@ package=expat GCCFLAGS?= -$(package)_version=2.4.1 +$(package)_version=2.4.8 $(package)_download_path=https://github.com/libexpat/libexpat/releases/download/R_$(subst .,_,$($(package)_version))/ $(package)_file_name=$(package)-$($(package)_version).tar.xz -$(package)_sha256_hash=cf032d0dba9b928636548e32b327a2d66b1aab63c4f4a13dd132c2d1d2f2fb6a +$(package)_sha256_hash=f79b8f904b749e3e0d20afeadecf8249c55b2e32d4ebb089ae378df479dcaf25 define $(package)_set_vars $(package)_config_opts=--disable-static --without-docbook --without-xmlwf diff --git a/depends/packages/qrencode.mk b/depends/packages/qrencode.mk index 8fe419f2be..fe3e6e377d 100644 --- a/depends/packages/qrencode.mk +++ b/depends/packages/qrencode.mk @@ -1,9 +1,9 @@ package=qrencode GCCFLAGS?= -$(package)_version=3.4.4 +$(package)_version=4.1.1 $(package)_download_path=https://fukuchi.org/works/qrencode/ $(package)_file_name=$(package)-$($(package)_version).tar.bz2 -$(package)_sha256_hash=efe5188b1ddbcbf98763b819b146be6a90481aac30cfc8d858ab78a19cde1fa5 +$(package)_sha256_hash=e455d9732f8041cf5b9c388e345a641fd15707860f928e94507b1961256a6923 define $(package)_set_vars $(package)_config_opts=--disable-shared --without-tools --without-tests --disable-sdltest --libdir="$($($(package)_type)_prefix)/lib" diff --git a/depends/packages/qt.mk b/depends/packages/qt.mk index 7e12066f22..3f721988a1 100644 --- a/depends/packages/qt.mk +++ b/depends/packages/qt.mk @@ -1,9 +1,9 @@ package=qt -$(package)_version=5.15.3 +$(package)_version=5.15.5 $(package)_download_path=https://download.qt.io/official_releases/qt/5.15/$($(package)_version)/submodules $(package)_suffix=everywhere-opensource-src-$($(package)_version).tar.xz $(package)_file_name=qtbase-$($(package)_suffix) -$(package)_sha256_hash=26394ec9375d52c1592bd7b689b1619c6b8dbe9b6f91fdd5c355589787f3a0b6 +$(package)_sha256_hash=0c42c799aa7c89e479a07c451bf5a301e291266ba789e81afc18f95049524edc $(package)_linux_dependencies=freetype fontconfig libxcb libxkbcommon libxcb_util libxcb_util_render libxcb_util_keysyms libxcb_util_image libxcb_util_wm $(package)_qt_libs=corelib network widgets gui plugins testlib concurrent $(package)_patches = fix_qt_pkgconfig.patch @@ -14,20 +14,19 @@ $(package)_patches += fix_montery_include.patch $(package)_patches += fix_android_jni_static.patch $(package)_patches += dont_hardcode_pwd.patch $(package)_patches += qtbase-moc-ignore-gcc-macro.patch -$(package)_patches += fix_limits_header.patch $(package)_patches += no_qrhi.patch $(package)_patches += drop_lrelease_dependency.patch $(package)_patches += subdirs.pro $(package)_qttranslations_file_name=qttranslations-$($(package)_suffix) -$(package)_qttranslations_sha256_hash=5d7869f670a135ad0986e266813b9dd5bbae2b09577338f9cdf8904d4af52db0 +$(package)_qttranslations_sha256_hash=c92af4171397a0ed272330b4fa0669790fcac8d050b07c8b8cc565ebeba6735e $(package)_qttools_file_name=qttools-$($(package)_suffix) -$(package)_qttools_sha256_hash=463b2fe71a085e7ab4e39333ae360ab0ec857b966d7a08f752c427e5df55f90d +$(package)_qttools_sha256_hash=6d0778b71b2742cb527561791d1d3d255366163d54a10f78c683a398f09ffc6c # Gridcoin displays SVG images in the GUI: $(package)_qtsvg_file_name=qtsvg-$($(package)_suffix) -$(package)_qtsvg_sha256_hash=3adc41dfcc67bbe3b8ff553bdac30ee75e270745536a58e54cdb741fa0505d89 +$(package)_qtsvg_sha256_hash=c4cf9e640ad43f157c6b14ee7624047f5945288991ad5de83c9eec673bacb031 $(package)_extra_sources = $($(package)_qttranslations_file_name) $(package)_extra_sources += $($(package)_qttools_file_name) @@ -57,6 +56,7 @@ $(package)_config_opts += -no-linuxfb $(package)_config_opts += -no-libjpeg $(package)_config_opts += -no-libproxy $(package)_config_opts += -no-libudev +$(package)_config_opts += -no-mimetype-database $(package)_config_opts += -no-mtdev $(package)_config_opts += -no-openssl $(package)_config_opts += -no-openvg @@ -262,7 +262,6 @@ define $(package)_preprocess_cmds patch -p1 -i $($(package)_patch_dir)/fix_android_jni_static.patch && \ patch -p1 -i $($(package)_patch_dir)/no-xlib.patch && \ patch -p1 -i $($(package)_patch_dir)/qtbase-moc-ignore-gcc-macro.patch && \ - patch -p1 -i $($(package)_patch_dir)/fix_limits_header.patch && \ patch -p1 -i $($(package)_patch_dir)/fix_montery_include.patch && \ patch -p1 -i $($(package)_patch_dir)/no_qrhi.patch && \ cp $($(package)_patch_dir)/subdirs.pro subdirs.pro && \ diff --git a/depends/packages/zlib.mk b/depends/packages/zlib.mk index acb02020a8..8a2432b8ff 100644 --- a/depends/packages/zlib.mk +++ b/depends/packages/zlib.mk @@ -1,8 +1,8 @@ package=zlib -$(package)_version=1.2.11 +$(package)_version=1.3 $(package)_download_path=https://www.zlib.net $(package)_file_name=$(package)-$($(package)_version).tar.gz -$(package)_sha256_hash=c3e5e9fdd5004dcb542feda5ee4f0ff0744628baf8ed2dd5d66f8ca1197cb1a1 +$(package)_sha256_hash=ff0ba4c292013dbc27530b3a81e1f9a813cd39de01ca5e0f8bf355702efa593e define $(package)_set_vars $(package)_config_opts= CC="$($(package)_cc)" diff --git a/depends/patches/qt/dont_hardcode_x86_64.patch b/depends/patches/qt/dont_hardcode_x86_64.patch index 5c1e030fa4..a66426877a 100644 --- a/depends/patches/qt/dont_hardcode_x86_64.patch +++ b/depends/patches/qt/dont_hardcode_x86_64.patch @@ -73,7 +73,7 @@ diff --git a/mkspecs/features/mac/default_post.prf b/mkspecs/features/mac/defaul index 92a9112bca6..d888731ec8d 100644 --- old/qtbase/mkspecs/features/mac/default_post.prf +++ new/qtbase/mkspecs/features/mac/default_post.prf -@@ -90,6 +90,11 @@ app_extension_api_only { +@@ -95,6 +95,11 @@ app_extension_api_only { QMAKE_LFLAGS += $$QMAKE_CFLAGS_APPLICATION_EXTENSION } @@ -85,7 +85,7 @@ index 92a9112bca6..d888731ec8d 100644 macx-xcode { qmake_pkginfo_typeinfo.name = QMAKE_PKGINFO_TYPEINFO !isEmpty(QMAKE_PKGINFO_TYPEINFO): \ -@@ -145,9 +150,6 @@ macx-xcode { +@@ -150,9 +155,6 @@ macx-xcode { simulator: VALID_SIMULATOR_ARCHS = $$QMAKE_APPLE_SIMULATOR_ARCHS VALID_ARCHS = $$VALID_DEVICE_ARCHS $$VALID_SIMULATOR_ARCHS diff --git a/depends/patches/qt/fix_android_jni_static.patch b/depends/patches/qt/fix_android_jni_static.patch index 22a4d5ab0e..936b82e152 100644 --- a/depends/patches/qt/fix_android_jni_static.patch +++ b/depends/patches/qt/fix_android_jni_static.patch @@ -1,6 +1,6 @@ --- old/qtbase/src/plugins/platforms/android/androidjnimain.cpp +++ new/qtbase/src/plugins/platforms/android/androidjnimain.cpp -@@ -934,6 +934,14 @@ Q_DECL_EXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void */*reserved*/) +@@ -943,6 +943,14 @@ Q_DECL_EXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void */*reserved*/) __android_log_print(ANDROID_LOG_FATAL, "Qt", "registerNatives failed"); return -1; } diff --git a/depends/patches/qt/fix_limits_header.patch b/depends/patches/qt/fix_limits_header.patch deleted file mode 100644 index 32fac6911d..0000000000 --- a/depends/patches/qt/fix_limits_header.patch +++ /dev/null @@ -1,33 +0,0 @@ -Fix compiling with GCC 11 - -Upstream: - - bug report: https://bugreports.qt.io/browse/QTBUG-89977 - - fix in Qt 6.1: 813a928c7c3cf98670b6043149880ed5c955efb9 - ---- old/qtbase/src/corelib/text/qbytearraymatcher.h -+++ new/qtbase/src/corelib/text/qbytearraymatcher.h -@@ -42,6 +42,8 @@ - - #include - -+#include -+ - QT_BEGIN_NAMESPACE - - - -Upstream fix and backports: - - Qt 6.1: 3eab20ad382569cb2c9e6ccec2322c3d08c0f716 - - Qt 6.2: 380294a5971da85010a708dc23b0edec192cbf27 - - Qt 6.3: 2b2b3155d9f6ba1e4f859741468fbc47db09292b - ---- old/qtbase/src/corelib/tools/qoffsetstringarray_p.h -+++ new/qtbase/src/corelib/tools/qoffsetstringarray_p.h -@@ -55,6 +55,7 @@ - - #include - #include -+#include - - QT_BEGIN_NAMESPACE - \ No newline at end of file diff --git a/doc/CMakeLists.txt b/doc/CMakeLists.txt new file mode 100644 index 0000000000..d5ef97c1e9 --- /dev/null +++ b/doc/CMakeLists.txt @@ -0,0 +1,25 @@ +find_package(Doxygen REQUIRED dot) + +set(DOXYGEN_PROJECT_BRIEF "P2P Digital Currency") +set(DOXYGEN_PROJECT_LOGO "${CMAKE_SOURCE_DIR}/share/icons/hicolor/48x48/apps/gridcoinresearch.png") +set(DOXYGEN_JAVADOC_AUTOBRIEF "YES") +set(DOXYGEN_TAB_SIZE 8) +set(DOXYGEN_EXTRACT_ALL "YES") +set(DOXYGEN_EXTRACT_PRIVATE "YES") +set(DOXYGEN_EXTRACT_STATIC "YES") +set(DOXYGEN_RECURSIVE "YES") +set(DOXYGEN_USE_MDFILE_AS_MAINPAGE "${CMAKE_CURRENT_SOURCE_DIR}/README_doxygen.md") +set(DOXYGEN_SOURCE_BROWSER "YES") + +set(DOXYGEN_EXCLUDE + "${CMAKE_SOURCE_DIR}/src/bdb53" + "${CMAKE_SOURCE_DIR}/src/crc32c" + "${CMAKE_SOURCE_DIR}/src/leveldb" + "${CMAKE_SOURCE_DIR}/src/test" + "${CMAKE_SOURCE_DIR}/src/qt/test" +) + +doxygen_add_docs(docs + "${CMAKE_SOURCE_DIR}/src" + ALL +) diff --git a/doc/README.md b/doc/README.md index bff735ff6e..eaa6ad72bd 100644 --- a/doc/README.md +++ b/doc/README.md @@ -2,7 +2,7 @@ Gridcoin ============= Setup ---------------------- +----- Gridcoin is a free open source project derived from Bitcoin, with the goal of providing a long-term energy-efficient cryptocurrency, rewarding BOINC work. Built on the foundation of Bitcoin, PPCoin and NovaCoin, innovations such as proof-of-stake @@ -15,17 +15,19 @@ To download Gridcoin, visit [gridcoin.us](https://gridcoin.us). * See the documentation at the [Gridcoin Wiki](https://wiki.gridcoin.us/Main_Page) for help and more information. * A lot of features core features are based on Bitcoin and have been documented on the [Bitcoin Wiki](https://en.bitcoin.it/wiki/Main_Page) -* For general questions, please visit our Discord server at https://discord.gg/jf9XX4a -* Ask for help or discuss on [#gridcoin](https://webchat.freenode.net?channels=gridcoin) on Freenode -* You can also join us on [Slack](https://join.slack.com/t/teamgridcoin/shared_invite/enQtMjk2NTI4MzAwMzg0LTE4N2I3ZWZjYWJlZGM1Zjg3MTUyMDhiN2M5NmRmZTA2NDA0ZmY1ZTFmOGM3ZGU2YTBkOTdhNTk2ZjkzMGZkODY/)``` +* For general questions, please visit our Discord server at https://discord.gg/UMWUnMjN4x +* Ask for help or discuss on [#gridcoin](https://web.libera.chat/?channels=#gridcoin) on Libera Chat +* You can also join us on [Slack](https://join.slack.com/t/teamgridcoin/shared_invite/enQtMjk2NTI4MzAwMzg0LTE4N2I3ZWZjYWJlZGM1Zjg3MTUyMDhiN2M5NmRmZTA2NDA0ZmY1ZTFmOGM3ZGU2YTBkOTdhNTk2ZjkzMGZkODY/) Building ---------------------- +-------- The following are developer notes on how to build Gridcoin on your native platform. They are not complete guides, but include notes on the necessary libraries, compile flags, etc. +- [CMake Build Options](cmake-options.md) - [OS X Build Notes](build-macos.md) - [Unix Build Notes](build-unix.md) - [Windows Build Notes](build-windows.md) +- [FreeBSD Build Notes](build-freebsd.md) - [OpenBSD Build Notes](build-openbsd.md) Running @@ -35,15 +37,14 @@ To create a secure environment for running Gridcoin see: - [Running Gridcoin](running.md) Development ---------------------- +----------- The Gridcoin repo's [root README](/README.md) contains relevant information on the development process and automated testing. - [Developer Notes](coding.txt) - [Release Process](release-process.md) -- [Travis CI](travis-ci.md) License ---------------------- +------- Distributed under the [MIT software license](/COPYING). This product includes software developed by the OpenSSL Project for use in the [OpenSSL Toolkit](https://www.openssl.org/). This product includes cryptographic software written by Eric Young ([eay@cryptsoft.com](mailto:eay@cryptsoft.com)), and UPnP software written by Thomas Bernard. diff --git a/doc/build-freebsd.md b/doc/build-freebsd.md index d1fe8baae2..0d436cefb8 100644 --- a/doc/build-freebsd.md +++ b/doc/build-freebsd.md @@ -9,25 +9,41 @@ Preparing the Build -------------------- Install the required dependencies the usual way you [install software on FreeBSD](https://www.freebsd.org/doc/en/books/handbook/ports.html) - either with `pkg` or via the Ports collection. The example commands below use `pkg` which is usually run as `root` or via `sudo`. If you want to use `sudo`, and you haven't set it up: [use this guide](http://www.freebsdwiki.net/index.php/Sudo%2C_configuring) to setup `sudo` access on FreeBSD. + #### General Dependencies + ```bash -pkg install autoconf automake boost-libs git gmake libevent libtool pkgconf openssl libzip +pkg install cmake +# or +pkg install autoconf automake gmake libtool +pkg install boost-libs curl db5 leveldb libzip openssl pkgconf secp256k1 ``` + --- #### GUI Dependencies + ```bash pkg install qt5 libqrencode ``` --- #### Test Suite Dependencies + There is an included test suite that is useful for testing code changes when developing. -To run the test suite (recommended), you will need to have Python 3 installed: +To run the test suite (recommended), you will need to have the following packages installed: -```bash -pkg install python3 -``` +* With CMake: + + ```bash + pkg install vim + ``` + +* With Autotools: + + ```bash + pkg install python3 + ``` Clone the repository and cd into it: @@ -42,19 +58,52 @@ To Build ### 1. Configuration There are many ways to configure Gridcoin, here are a few common examples: + ##### Wallet Support, No GUI: -This explicitly enables wallet support and disables the GUI. -```bash -./autogen.sh -./configure --with-gui=no \ - MAKE=gmake -``` +This configuration does not enable the GUI. +* With CMake: + + ```bash + mkdir build && cd build + cmake .. + ``` + +* With Autotools: + + ```bash + ./autogen.sh + ./configure --with-gui=no \ + MAKE=gmake + ``` ### 2. Compile -**Important**: Use `gmake` (the non-GNU `make` will exit with an error). -```bash -gmake # use "-j N" for N parallel jobs -gmake check # Run tests if Python 3 is available -``` +* With CMake: + + ```bash + cmake --build . # use "-j N" for N parallel jobs + ctest . # Run tests + ``` + +* With Autotools: + + ```bash + gmake # use "-j N" for N parallel jobs + ``` + +### 3. Test + +* With CMake: + + ```bash + cmake .. -DENABLE_TESTS=ON + cmake --build . + ctest . + ``` + +* With Autotools: + + ```bash + gmake check + ``` diff --git a/doc/build-macos.md b/doc/build-macos.md index 51becf43a7..2b773e4348 100644 --- a/doc/build-macos.md +++ b/doc/build-macos.md @@ -14,6 +14,7 @@ macOS comes with a built-in Terminal located in: /Applications/Utilities/Terminal.app ``` ### 1. Xcode Command Line Tools + The Xcode Command Line Tools are a collection of build tools for macOS. These tools must be installed in order to build Gridcoin from source. @@ -27,6 +28,7 @@ Upon running the command, you should see a popup appear. Click on `Install` to continue the installation process. ### 2. Homebrew Package Manager + Homebrew is a package manager for macOS that allows one to install packages from the command line easily. While several package managers are available for macOS, this guide will focus on Homebrew as it is the most popular. Since the examples in this guide which walk through the installation of a package will use Homebrew, it is recommended that you install it to follow along. @@ -35,13 +37,20 @@ Otherwise, you can adapt the commands to your package manager of choice. To install the Homebrew package manager, see: https://brew.sh Note: If you run into issues while installing Homebrew or pulling packages, refer to [Homebrew's troubleshooting page](https://docs.brew.sh/Troubleshooting). + ### 3. Install Required Dependencies + The first step is to download the required dependencies. These dependencies represent the packages required to get a barebones installation up and running. To install, run the following from your terminal: ```shell -brew install automake libtool boost openssl pkg-config libzip berkeley-db@4 +brew install cmake +# or +brew install automake libtool + +brew install berkeley-db@5 boost curl leveldb libzip openssl pkgconf secp256k1 +export PKG_CONFIG_PATH="${PKG_CONFIG_PATH}:/usr/local/opt/openssl@3/lib/pkgconfig" ``` ### 4. Clone Gridcoin repository @@ -107,11 +116,19 @@ brew install miniupnpc #### Test Suite Dependencies There is an included test suite that is useful for testing code changes when developing. -To run the test suite (recommended), you will need to have Python 3 installed: +To run the test suite (recommended), you will need to have the following packages installed: -``` bash -brew install python -``` +* With CMake: + + ```bash + brew install vim + ``` + +* With Autotools: + + ```bash + brew install python + ``` --- @@ -132,32 +149,61 @@ pip3 install ds_store mac_alias ## Build Gridcoin -1. Build Gridcoin: - - Prepare the assembly code (requires Perl): - ```shell - cd src/ - ../contrib/nomacro.pl - cd .. - ``` - - Configure and build the headless Gridcoin binaries as well as the GUI (if Qt is found). - ```shell - ./autogen.sh - ./configure - make - ``` - You can disable the GUI build by passing `--without-gui` to configure. - -2. It is recommended to build and run the unit tests: - ```shell - make check - ``` - -3. You can also create a `.dmg` that contains the `.app` bundle (optional): - ```shell - make deploy - ``` +1. Prepare the assembly code (requires Perl): + + ```shell + pushd src + ../contrib/nomacro.pl + popd + ``` + +2. Configure and build the Gridcoin binaries: + + You can enable the GUI build by passing `-DENABLE_GUI=ON` to CMake or + `--with-gui=qt5` to the configure script. + + * With CMake: + + ```shell + mkdir build && cd build + + cmake .. + # or, to configure with GUI + cmake .. -DENABLE_GUI=ON -DQt5_DIR=/usr/local/opt/qt5/lib/cmake/Qt5 + + cmake --build . + ``` + * With Autotools: + + ```shell + ./autogen.sh + ./configure + make + ``` + +3. It is recommended to build and run the unit tests: + + * With CMake: + + ```shell + cmake .. -DENABLE_TESTS=ON + cmake --build . + ctest . + ``` + + * With Autotools: + + ```shell + make check + ``` + +4. You can also create a `.dmg` that contains the `.app` bundle (optional): + + * With Autotools: + + ```shell + make deploy + ``` ## Running diff --git a/doc/build-openbsd.md b/doc/build-openbsd.md index b841e05f36..b40f07b37a 100644 --- a/doc/build-openbsd.md +++ b/doc/build-openbsd.md @@ -1,24 +1,25 @@ OpenBSD build guide -====================== +=================== (updated for OpenBSD 7.0) This guide describes how to build gridcoinresearchd, command-line utilities, and GUI on OpenBSD. Preparation -------------- +----------- Run the following as root to install the base dependencies for building: ```bash -pkg_add git gmake libtool libevent boost libzip -pkg_add qt5 libqrencode # (optional for enabling the GUI) -pkg_add autoconf # (select highest version, e.g. 2.71) -pkg_add automake # (select highest version, e.g. 1.16) -pkg_add python # (select highest version, e.g. 3.10) +pkg_add cmake vim +# or +pkg_add autoconf automake gmake libtool python + +pkg_add boost curl libzip leveldb pkgconf +pkg_add qt5 libqrencode # optional for the GUI ``` Resource limits -------------------- +--------------- If the build runs into out-of-memory errors, the instructions in this section might help. @@ -41,34 +42,52 @@ make the change system-wide, change `datasize-cur` and `datasize-max` in **Important**: use `gmake`, not `make`. The non-GNU `make` will exit with a horrible error. -Preparation: -```bash +To configure with gridcoinresearchd: -# Replace this with the autoconf version that you installed. Include only -# the major and minor parts of the version: use "2.71" for "autoconf-2.71p1". -export AUTOCONF_VERSION=2.71 +* With CMake: -# Replace this with the automake version that you installed. Include only -# the major and minor parts of the version: use "1.16" for "automake-1.16.3". -export AUTOMAKE_VERSION=1.16 + ```bash + mkdir build && cd build + cmake .. + ``` -./autogen.sh -``` +* With Autotools: -To configure with gridcoinresearchd: -```bash -./configure --with-gui=no \ - MAKE=gmake -``` + ```bash + ./autogen.sh + ./configure --with-gui=no \ + MAKE=gmake + ``` To configure with GUI: -```bash -./configure --with-gui=yes \ - MAKE=gmake -``` + +* With CMake: + + ```bash + mkdir build && cd build + cmake -DENABLE_GUI=ON + ``` + +* With Autotools: + + ```bash + ./autogen.sh + ./configure --with-gui=yes \ + MAKE=gmake + ``` Build and run the tests: -```bash -gmake # use "-j N" here for N parallel jobs -gmake check -``` + +* With CMake: + + ```bash + cmake --build . # use "-j N" here for N parallel jobs + ctest . + ``` + +* With Autotools: + + ```bash + gmake # use "-j N" here for N parallel jobs + gmake check + ``` diff --git a/doc/build-unix.md b/doc/build-unix.md index da9c6f7faa..67ab662292 100644 --- a/doc/build-unix.md +++ b/doc/build-unix.md @@ -2,24 +2,15 @@ UNIX BUILD NOTES ==================== Some notes on how to build Gridcoin in Unix. -(For BSD specific instructions, see build-*bsd.md in this directory.) - -Note ---------------------- -Always use absolute paths to configure and compile Gridcoin and the dependencies, -for example, when specifying the path of the dependency: - - ../dist/configure --enable-cxx --disable-shared --with-pic --prefix=$BDB_PREFIX - -Here BDB_PREFIX must be an absolute path - it is defined using $(pwd) which ensures -the usage of the absolute path. +(For BSD specific instructions, see build-\*bsd.md in this directory.) Preparing the Build --------------------- +------------------- Install git: -Ubuntu & Debian: `sudo apt-get install git` -openSUSE: `sudo zypper install git` +* Ubuntu & Debian: `sudo apt install git` +* openSUSE: `sudo zypper install git` + Clone the repository and cd into it: ```bash @@ -27,111 +18,94 @@ git clone https://github.com/gridcoin-community/Gridcoin-Research cd Gridcoin-Research git checkout master ``` + Go to platform specific instructions for the required dependencies below. To Build ---------------------- +-------- -```bash -./autogen.sh -./configure -make # use "-j N" for N parallel jobs -make install # optional -``` +* With CMake: -Or, to keep the source directory clean: -```bash -./autogen.sh -mkdir build -cd build -../configure -make # use "-j N" for N parallel jobs -``` - -This will build gridcoinresearch (Qt client) as well if the dependencies are met. - -Dependencies ---------------------- + ```bash + mkdir build && cd build + # minimal configuration + cmake .. + # full GUI build with ccache + cmake .. -DCMAKE_C_COMPILER_LAUNCHER=ccache -DCMAKE_CXX_COMPILER_LAUNCHER=ccache -DENABLE_GUI=ON -DENABLE_QRENCODE=ON -DENABLE_UPNP=ON -DUSE_DBUS=ON + ``` -These dependencies are required: + ```bash + cmake --build . # use "-j N" for N parallel jobs + cmake --install . # optional + ``` - Library | Purpose | Description - ------------|------------------|---------------------- - libssl | Crypto | Random Number Generation, Elliptic Curve Cryptography - libboost | Utility | Library for threading, data structures, etc - libevent | Networking | OS independent asynchronous networking - miniupnpc | UPnP Support | Firewall-jumping support - qt | GUI | GUI toolkit (only needed when GUI enabled) - libqrencode | QR codes in GUI | Optional for generating QR codes (only needed when GUI enabled) - libzip | Zip Compression | For Zip Compression and Decompression for snapshot and scraper related functions +* With Autotools: -For the versions used in the release, see [release-process.md](release-process.md) under *Fetch and build inputs*. + ```bash + ./autogen.sh + ./configure + make # use "-j N" for N parallel jobs + make install # optional + ``` Memory Requirements --------------------- +------------------- C++ compilers are memory-hungry. It is recommended to have at least 1.5 GB of memory available when compiling Gridcoin. On systems with less, gcc can be tuned to conserve memory with additional CXXFLAGS: + export CXXFLAGS="--param ggc-min-expand=1 --param ggc-min-heapsize=32768" - ./configure CXXFLAGS="--param ggc-min-expand=1 --param ggc-min-heapsize=32768" - -Alternatively, clang (often less resource hungry) can be used instead of gcc, which is used by default: - - ./configure CXX=clang++ CC=clang +Clang (often less resource hungry) can be used instead of gcc, which is used by default: + export CXX=clang++ + export CC=clang Dependency Build Instructions: Ubuntu & Debian ---------------------------------------------- -Build requirements: - sudo apt-get install build-essential libtool autotools-dev automake pkg-config libssl-dev libevent-dev bsdmainutils libzip-dev libfreetype-dev - -**For Ubuntu 18.04 gcc8 is also required** - - sudo apt-get install gcc-8 g++-8 - -Now, you can either build from self-compiled [depends](/depends/README.md) or install the required dependencies: +Build requirements: - sudo apt-get install libboost-system-dev libboost-filesystem-dev libboost-test-dev libboost-thread-dev libboost-iostreams-dev libcurl4-gnutls-dev + sudo apt install build-essential cmake libboost-all-dev libcurl4-gnutls-dev libdb5.3++-dev libleveldb-dev libsecp256k1-dev libssl-dev libzip-dev pkgconf -Optional (see --with-miniupnpc and --enable-upnp-default): +Optional (see `ENABLE_UPNP / DEFAULT_UPNP` options for CMake and +`--with-miniupnpc / --enable-upnp-default` for Autotools): - sudo apt-get install libminiupnpc-dev + sudo apt install libminiupnpc-dev Dependencies for the GUI: Ubuntu & Debian ----------------------------------------- -If you want to build Gridcoin with UI, make sure that the required packages for Qt development -are installed. Qt 5 is necessary to build the GUI. -To build without GUI pass `--without-gui` to configure. +If you want to build Gridcoin with UI, make sure that the required packages for +Qt development are installed. Qt 5 is necessary to build the GUI. -To build with Qt 5 you need the following: +To build with GUI pass `-DENABLE_GUI=ON` to CMake or `--with-gui=qt5` to the +configure script. - sudo apt-get install libqt5gui5 libqt5core5a libqt5dbus5 qttools5-dev qttools5-dev-tools libprotobuf-dev protobuf-compiler +To build with Qt 5 you need the following: -libqrencode (enabled by default, switch off by passing `--without-qrencode` to configure) can be installed with: + sudo apt install qtbase5-dev qttools5-dev - sudo apt-get install libqrencode-dev +libqrencode (switch on by passing `-DENABLE_QRENCODE=ON` to CMake or +`--with-qrencode` to the configure script) can be installed with: -Once these are installed, they will be found by configure and a gridcoinresearch executable will be -built by default. + sudo apt install libqrencode-dev Dependency Build Instructions: OpenSUSE ----------------------------------------------- +--------------------------------------- Build requirements: sudo zypper install -t pattern devel_basis - sudo zypper install libtool automake autoconf pkg-config libopenssl-devel libevent-devel + sudo zypper install cmake libtool automake autoconf pkg-config libopenssl-devel libevent-devel Tumbleweed: - sudo zypper install libboost_system1_*_0-devel libboost_filesystem1_*_0-devel libboost_test1_*_0-devel libboost_thread1_*_0-devel + sudo zypper install libboost_system1_*_0-devel libboost_filesystem1_*_0-devel libboost_test1_*_0-devel libboost_thread1_*_0-devel Leap: - sudo zypper install libboost_system1_61_0-devel libboost_filesystem1_61_0-devel libboost_test1_61_0-devel libboost_thread1_61_0-devel + sudo zypper install libboost_system1_61_0-devel libboost_filesystem1_61_0-devel libboost_test1_61_0-devel libboost_thread1_61_0-devel If that doesn't work, you can install all boost development packages with: @@ -139,58 +113,61 @@ Leap: sudo zypper install boost_1_61-devel -Optional (see --with-miniupnpc and --enable-upnp-default): +Optional (see `ENABLE_UPNP / DEFAULT_UPNP` options for CMake and +`--with-miniupnpc / --enable-upnp-default` for Autotools): sudo zypper install libminiupnpc-devel Dependencies for the GUI: openSUSE ------------------------------------------ +---------------------------------- + +If you want to build Gridcoin with UI, make sure that the required packages for +Qt development are installed. Qt 5 is necessary to build the GUI. -If you want to build gridcoinresearch, make sure that the required packages for Qt development -are installed. Qt 5 is necessary to build the GUI. -To build without GUI pass `--without-gui` to configure. +To build with GUI pass `-DENABLE_GUI=ON` to CMake or `--with-gui=qt5` to the +configure script. To build with Qt 5 you need the following: sudo zypper install libQt5Gui5 libQt5Core5 libQt5DBus5 libQt5Network-devel libqt5-qttools-devel libqt5-qttools -libqrencode (enabled by default, switch off by passing `--without-qrencode` to configure) can be installed with: +libqrencode (switch on by passing `-DENABLE_QRENCODE=ON` to CMake or +`--with-qrencode` to the configure script) can be installed with: sudo zypper install qrencode-devel -Once these are installed, they will be found by configure and a gridcoinresearch executable will be -built by default. - - Dependency Build Instructions: Alpine Linux ----------------------------------------------- +------------------------------------------- Build requirements: - apk add autoconf automake boost-dev build-base curl-dev libtool libzip-dev miniupnpc-dev openssl-dev pkgconfig + apk add autoconf automake boost-dev build-base cmake curl-dev db-dev leveldb-dev libtool libsecp256k1-dev libzip-dev miniupnpc-dev openssl-dev pkgconf Dependencies for the GUI: Alpine Linux ------------------------------------------ +-------------------------------------- To build the Qt GUI on Alpine Linux, we need these dependencies: - apk add libqrencode-dev protobuf-dev qt5-qtbase-dev qt5-qtsvg-dev qt5-qttools-dev + apk add libqrencode-dev qt5-qtbase-dev qt5-qttools-dev Setup and Build Example: Arch Linux ----------------------------------- This example lists the steps necessary to setup and build a command line only of the latest changes on Arch Linux: - pacman -S git base-devel boost libevent python + pacman -S git base-devel boost cmake db5.3 leveldb curl libsecp256k1 + git clone https://github.com/gridcoin/Gridcoin-Research.git git checkout master cd Gridcoin-Research/ - ./autogen.sh - ./configure --without-gui --without-miniupnpc - make check + + mkdir build && cd build + cmake .. + cmake --build . ARM Cross-compilation -------------------- +--------------------- + These steps can be performed on, for example, an Ubuntu VM. The depends system will also work on other Linux distributions, however the commands for installing the toolchain will be different. diff --git a/doc/build-windows.md b/doc/build-windows.md index fa1d14f59d..5f58bf1603 100644 --- a/doc/build-windows.md +++ b/doc/build-windows.md @@ -1,6 +1,8 @@ WINDOWS BUILD NOTES ==================== +> This document is outdated. + Below are some notes on how to build Gridcoin for Windows. The options known to work for building Gridcoin on Windows are: @@ -13,7 +15,7 @@ Other options which may work, but which have not been extensively tested are (pl * On Windows, using a POSIX compatibility layer application such as [cygwin](https://www.cygwin.com/) or [msys2](https://www.msys2.org/). Installing Windows Subsystem for Linux ---------------------------------------- +-------------------------------------- Follow the upstream installation instructions, available [here](https://docs.microsoft.com/windows/wsl/install-win10). @@ -115,7 +117,7 @@ Then build using: For further documentation on the depends system see [README.md](../depends/README.md) in the depends directory. Installation -------------- +------------ After building using the Windows subsystem it can be useful to copy the compiled executables to a directory on the Windows drive in the same directory structure diff --git a/doc/cmake-options.md b/doc/cmake-options.md new file mode 100644 index 0000000000..95cd8c4163 --- /dev/null +++ b/doc/cmake-options.md @@ -0,0 +1,32 @@ +## CMake Build Options + +You can use GUI (`cmake-gui`) or TUI (`ccmake`) to browse and toggle all +available options: + +```bash +mkdir build && cd build +cmake .. +ccmake . +``` + +### Common configurations + +* Build with GUI, QR code support and DBus support: + + `cmake .. -DENABLE_GUI=ON -DENABLE_QRENCODE=ON -DUSE_DBUS=ON` + +* Build with UPnP: + + `cmake .. -DENABLE_UPNP=ON -DDEFAULT_UPNP=ON` + +* Enable PIE and disable assembler routines: + + `cmake .. -DENABLE_PIE=ON -DUSE_ASM=OFF` + +* Build tests and docs, run `lupdate`: + + `cmake .. -DENABLE_DOCS=ON -DENABLE_TESTS=ON -DLUPDATE=ON` + +* Build with system libraries: + + `cmake .. -DSYSTEM_BDB=ON -DSYSTEM_LEVELDB=ON -DSYSTEM_SECP256K1=ON -DSYSTEM_UNIVALUE=ON -DSYSTEM_XXD=ON` diff --git a/doc/readme-qt.rst b/doc/readme-qt.rst deleted file mode 100644 index d59ccd6a88..0000000000 --- a/doc/readme-qt.rst +++ /dev/null @@ -1,141 +0,0 @@ -Gridcoin-qt: Qt5 GUI for Gridcoin -================================= - -Build instructions -================== - -Debian ------- - -First, make sure that the required packages for Qt5 development of your -distribution are installed, for Debian and Ubuntu these are: - -:: - - apt-get install qt5-default qt5-qmake qtbase5-dev-tools qttools5-dev-tools \ - build-essential libboost-dev libboost-system-dev \ - libboost-filesystem-dev libboost-thread-dev \ - libssl-dev libminiupnpc-dev libzip-dev - -then execute the following: - -:: - - qmake - make - -Alternatively, install Qt Creator and open the `gridcoin-qt.pro` file. - -An executable named `gridcoinresearch` will be built. - - -Windows -------- - -Windows build instructions: - -- Download the `QT Windows SDK`_ and install it. You don't need the Symbian stuff, just the desktop Qt. - -- Compile openssl and boost. - -- Open the .pro file in QT creator and build as normal (ctrl-B) - -.. _`QT Windows SDK`: https://qt-project.org/downloads - - -MacOS ------ - -- Download and install the `Qt Mac OS X SDK`_. It is recommended to also install Apple's Xcode with UNIX tools. - -- Execute the following commands in a terminal to get the dependencies: - -:: - - sudo port selfupdate - sudo port install boost miniupnpc - -- Open the .pro file in Qt Creator and build as normal (cmd-B) - -.. _`Qt Mac OS X SDK`: https://qt-project.org/downloads - -Alternatively -------------- - -- Install Qt Creator and open the makefile.am file for Auto-Tools to build for your OS. - -- An executable named gridcoinresearch will be built in the /src/Qt directory. - - -Build configuration options -=========================== - -UPNnP port forwarding ---------------------- - -To use UPnP for port forwarding behind a NAT router (recommended, as more connections overall allow for a faster and more stable Gridcoin experience), pass the following argument to qmake: - -:: - - qmake "USE_UPNP=1" - -(in **Qt Creator**, you can find the setting for additional qmake arguments under "Projects" -> "Build Settings" -> "Build Steps", then click "Details" next to **qmake**) - -This requires miniupnpc for UPnP port mapping. It can be downloaded from -http://miniupnp.tuxfamily.org/files/. UPnP support is not compiled in by default. - -Set USE_UPNP to a different value to control this: - -+------------+--------------------------------------------------------------------------+ -| USE_UPNP=- | no UPnP support, miniupnpc not required; | -+------------+--------------------------------------------------------------------------+ -| USE_UPNP=0 | (the default) built with UPnP, support turned off by default at runtime; | -+------------+--------------------------------------------------------------------------+ -| USE_UPNP=1 | build with UPnP support turned on by default at runtime. | -+------------+--------------------------------------------------------------------------+ - -Notification support for recent (k)ubuntu versions --------------------------------------------------- - -To see desktop notifications on (k)ubuntu versions starting from 10.04, enable usage of the -FreeDesktop notification interface through DBUS using the following qmake option: - -:: - - qmake "USE_DBUS=1" - -Generation of QR codes ----------------------- - -libqrencode may be used to generate QRCode images for payment requests. -It can be downloaded from https://fukuchi.org/works/qrencode/index.html.en, or installed via your package manager. Pass the USE_QRCODE -flag to qmake to control this: - -+--------------+--------------------------------------------------------------------------+ -| USE_QRCODE=0 | (the default) No QRCode support - libarcode not required | -+--------------+--------------------------------------------------------------------------+ -| USE_QRCODE=1 | QRCode support enabled | -+--------------+--------------------------------------------------------------------------+ - -Ubuntu 11.10 warning -==================== - -Ubuntu 11.10 has a package called 'qt-at-spi' installed by default. At the time of writing, having that package -installed causes gridcoin-qt to crash intermittently. The issue has been reported as `launchpad bug 857790`_, but -isn't yet fixed. - -Until the bug is fixed, you can remove the qt-at-spi package to work around the problem, though this will presumably -disable screen reader functionality for Qt apps: - -:: - - sudo apt-get remove qt-at-spi - -.. _`launchpad bug 857790`: https://bugs.launchpad.net/ubuntu/+source/qt-at-spi/+bug/857790 - -Disabling the built-in upgrader -=============================== - -If the Gridcoin binary is managed via a distributions package management the built-in upgrader will do more harm than good. Disable it through the following qmake option:: - - qmake NO_UPGRADE=1 diff --git a/doc/release-process.md b/doc/release-process.md old mode 100755 new mode 100644 index d91775513e..4fec277bf3 --- a/doc/release-process.md +++ b/doc/release-process.md @@ -5,7 +5,7 @@ - git pull - git merge testnet - Update changelog and commit - - Change version in configure.ac to [version] and set CLIENT_VERSION_IS_RELEASE to 'true' and commit + - Change version in configure.ac and CMakeLists.txt to [version] and set CLIENT_VERSION_IS_RELEASE to 'true' and commit - git push ### Merge to master diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt new file mode 100644 index 0000000000..012281546c --- /dev/null +++ b/src/CMakeLists.txt @@ -0,0 +1,290 @@ +configure_file( + config/gridcoin-config.h.cmake.in + config/gridcoin-config.h +) +configure_file( + obj/build.h.in + obj/build.h +) + +add_compile_definitions( + BOOST_SPIRIT_THREADSAFE + __STDC_FORMAT_MACROS +) + +if(WIN32) + add_compile_definitions(_MT _WINDOWS _WIN32_WINNT=0x0601) + add_compile_definitions(WIN32 BOOST_THREAD_USE_LIB UNICODE) +elseif(UNIX AND NOT APPLE) + add_compile_definitions(_FILE_OFFSET_BITS=64) +endif() + + +# Dependencies +# ============ + +if(SYSTEM_BDB) + set(LIBBDB_CXX BerkeleyDB::CXX) +else() + add_subdirectory(bdb53) + set(LIBBDB_CXX libdb_cxx) +endif() + +if(SYSTEM_LEVELDB) + set(LIBLEVELDB leveldb::leveldb) +else() + set(CMAKE_POLICY_DEFAULT_CMP0077 NEW) + + set(LEVELDB_BUILD_TESTS ${ENABLE_TESTS}) + set(LEVELDB_BUILD_BENCHMARKS OFF) + set(LEVELDB_INSTALL OFF) + + set(HAVE_CRC32C OFF) + set(HAVE_SNAPPY OFF) + set(HAVE_TCMALLOC OFF) + set(HAVE_CLANG_THREAD_SAFETY OFF) + + set(BUILD_SHARED_LIBS_ORIG ${BUILD_SHARED_LIBS}) + set(BUILD_SHARED_LIBS OFF) + + add_subdirectory(leveldb) + set(LIBLEVELDB leveldb) + + set(BUILD_SHARED_LIBS ${BUILD_SHARED_LIBS_ORIG}) +endif() + +if(SYSTEM_SECP256K1) + set(LIBSECP256K1 PkgConfig::SECP256K1) +else() + set(CMAKE_POLICY_DEFAULT_CMP0077 NEW) + + set(SECP256K1_BUILD_SHARED OFF) + set(SECP256K1_BUILD_STATIC ON) + set(SECP256K1_ENABLE_MODULE_ECDH OFF) + set(SECP256K1_ENABLE_MODULE_RECOVERY ON) + set(SECP256K1_ENABLE_MODULE_EXTRAKEYS OFF) + set(SECP256K1_ENABLE_MODULE_SCHNORRSIG OFF) + set(SECP256K1_VALGRIND OFF) + + set(SECP256K1_BUILD_BENCHMARK OFF) + set(SECP256K1_BUILD_TESTS OFF) + set(SECP256K1_BUILD_EXHAUSTIVE_TESTS OFF) + set(SECP256K1_BUILD_CTIME_TESTS OFF) + + if(USE_ASM) + set(SECP256K1_ASM "AUTO") + else() + set(SECP256K1_ASM "OFF") + endif() + + add_subdirectory(secp256k1 EXCLUDE_FROM_ALL) + target_include_directories(secp256k1 PUBLIC + "${CMAKE_CURRENT_SOURCE_DIR}/secp256k1/include" + ) + set(LIBSECP256K1 secp256k1) +endif() + +if(SYSTEM_UNIVALUE) + set(LIBUNIVALUE PkgConfig::UNIVALUE) +else() + add_subdirectory(univalue) + set(LIBUNIVALUE univalue) +endif() + + +# libgridcoin_crypto +# ================== + +add_subdirectory(crypto) + + +# libgridcoin_util +# ================ + +add_library(gridcoin_util STATIC + addrdb.cpp + addrman.cpp + alert.cpp + arith_uint256.cpp + banman.cpp + base58.cpp + chainparams.cpp + chainparamsbase.cpp + checkpoints.cpp + clientversion.cpp + consensus/merkle.cpp + consensus/tx_verify.cpp + crypter.cpp + dbwrapper.cpp + fs.cpp + gridcoin/backup.cpp + gridcoin/beacon.cpp + gridcoin/boinc.cpp + gridcoin/claim.cpp + gridcoin/contract/contract.cpp + gridcoin/contract/message.cpp + gridcoin/contract/registry.cpp + gridcoin/cpid.cpp + gridcoin/gridcoin.cpp + gridcoin/mrc.cpp + gridcoin/project.cpp + gridcoin/protocol.cpp + gridcoin/quorum.cpp + gridcoin/researcher.cpp + gridcoin/scraper/http.cpp + gridcoin/scraper/scraper.cpp + gridcoin/scraper/scraper_net.cpp + gridcoin/scraper/scraper_registry.cpp + gridcoin/staking/difficulty.cpp + gridcoin/staking/exceptions.cpp + gridcoin/staking/kernel.cpp + gridcoin/staking/reward.cpp + gridcoin/staking/status.cpp + gridcoin/superblock.cpp + gridcoin/support/block_finder.cpp + gridcoin/tally.cpp + gridcoin/tx_message.cpp + gridcoin/upgrade.cpp + gridcoin/voting/builders.cpp + gridcoin/voting/claims.cpp + gridcoin/voting/poll.cpp + gridcoin/voting/registry.cpp + gridcoin/voting/result.cpp + gridcoin/voting/vote.cpp + hash.cpp + init.cpp + key.cpp + key_io.cpp + keystore.cpp + logging.cpp + main.cpp + miner.cpp + net.cpp + netaddress.cpp + netbase.cpp + node/blockstorage.cpp + node/ui_interface.cpp + noui.cpp + pbkdf2.cpp + policy/policy.cpp + primitives/transaction.cpp + protocol.cpp + pubkey.cpp + random.cpp + randomenv.cpp + rpc/blockchain.cpp + rpc/client.cpp + rpc/dataacq.cpp + rpc/mining.cpp + rpc/misc.cpp + rpc/net.cpp + rpc/protocol.cpp + rpc/rawtransaction.cpp + rpc/server.cpp + rpc/voting.cpp + scheduler.cpp + script.cpp + scrypt-x86.S + scrypt-x86_64.S + scrypt.cpp + support/cleanse.cpp + support/lockedpool.cpp + sync.cpp + uint256.cpp + util.cpp + util/bip32.cpp + util/settings.cpp + util/strencodings.cpp + util/string.cpp + util/syserror.cpp + util/system.cpp + util/threadinterrupt.cpp + util/threadnames.cpp + util/time.cpp + util/tokenpipe.cpp + validation.cpp + wallet/db.cpp + wallet/diagnose.cpp + wallet/rpcdump.cpp + wallet/rpcwallet.cpp + wallet/wallet.cpp + wallet/walletdb.cpp + wallet/walletutil.cpp +) + +target_include_directories(gridcoin_util PUBLIC + ${CMAKE_SOURCE_DIR}/src + ${CMAKE_BINARY_DIR}/src +) + +target_link_libraries(gridcoin_util PUBLIC + -lm ${ATOMICS_LIBRARIES} ${RT_LIBRARIES} + ${LIBBDB_CXX} ${LIBLEVELDB} ${LIBSECP256K1} ${LIBUNIVALUE} + Boost::filesystem Boost::headers Boost::iostreams Boost::thread + CURL::libcurl + OpenSSL::Crypto OpenSSL::SSL + Threads::Threads + libzip::zip +) +target_link_libraries(gridcoin_util PUBLIC ${LIBGRIDCOIN_CRYPTO}) + +target_compile_definitions(gridcoin_util PUBLIC + HAVE_CONFIG_H + HAVE_BUILD_INFO +) + +if(ENABLE_UPNP) + if(DEFAULT_UPNP) + target_compile_definitions(gridcoin_util PRIVATE USE_UPNP=1) + else() + target_compile_definitions(gridcoin_util PRIVATE USE_UPNP=0) + endif() + + if(WIN32) + target_compile_definitions(gridcoin_util PRIVATE MINIUPNP_STATICLIB) + endif() + + target_link_libraries(gridcoin_util PUBLIC PkgConfig::MINIUPNPC) +endif() + + +# Daemon +# ====== + +if(ENABLE_DAEMON) + add_executable(gridcoinresearchd gridcoinresearchd.cpp) + target_link_libraries(gridcoinresearchd PUBLIC gridcoin_util) + + if(WIN32) + target_sources(gridcoinresearchd PRIVATE gridcoinresearchd-res.rc) + endif() + + if(UNIX AND NOT APPLE) + include(GNUInstallDirs) + install(TARGETS gridcoinresearchd + DESTINATION "${CMAKE_INSTALL_BINDIR}" + ) + install(FILES "${CMAKE_SOURCE_DIR}/doc/gridcoinresearchd.1" + DESTINATION "${CMAKE_INSTALL_MANDIR}" + ) + install(FILES "${CMAKE_SOURCE_DIR}/contrib/init/gridcoinresearchd.service" + DESTINATION /lib/systemd/system + ) + endif() +endif() + + +# Qt GUI +# ====== + +if(ENABLE_GUI) + add_subdirectory(qt) +endif() + + +# Tests +# ===== + +if(ENABLE_TESTS) + add_subdirectory(test) +endif() diff --git a/src/addrdb.cpp b/src/addrdb.cpp index c16078c41a..135ebf9176 100644 --- a/src/addrdb.cpp +++ b/src/addrdb.cpp @@ -40,7 +40,8 @@ template bool SerializeFileDB(const std::string& prefix, const fs::path& path, const Data& data) { // Generate random temporary filename - std::string tmpfn = strprintf("%s.%" PRIx64, prefix, GetPerformanceCounter()); + const uint16_t randv{GetRand()}; + std::string tmpfn = strprintf("%s.%04x", prefix, randv); // open temp output file, and associate with CAutoFile fs::path pathTmp = GetDataDir() / tmpfn; diff --git a/src/addrman.cpp b/src/addrman.cpp index fb896a629a..7d62aef2c9 100644 --- a/src/addrman.cpp +++ b/src/addrman.cpp @@ -221,7 +221,7 @@ void CAddrMan::Good_(const CService& addr, int64_t nTime) return; // find a bucket it is in now - int nRnd = GetRandInt(ADDRMAN_NEW_BUCKET_COUNT); + int nRnd = GetRand(ADDRMAN_NEW_BUCKET_COUNT); int nUBucket = -1; for (unsigned int n = 0; n < ADDRMAN_NEW_BUCKET_COUNT; n++) { int nB = (n + nRnd) % ADDRMAN_NEW_BUCKET_COUNT; @@ -278,7 +278,7 @@ bool CAddrMan::Add_(const CAddress& addr, const CNetAddr& source, int64_t nTimeP int nFactor = 1; for (int n = 0; n < pinfo->nRefCount; n++) nFactor *= 2; - if (nFactor > 1 && (GetRandInt(nFactor) != 0)) + if (nFactor > 1 && (GetRand(nFactor) != 0)) return false; } else { pinfo = Create(addr, source, &nId); @@ -467,7 +467,7 @@ void CAddrMan::GetAddr_(std::vector &vAddr) if (vAddr.size() >= nNodes) break; - int nRndPos = GetRandInt(vRandom.size() - n) + n; + int nRndPos = GetRand(vRandom.size() - n) + n; SwapRandom(n, nRndPos); assert(mapInfo.count(vRandom[n]) == 1); @@ -516,5 +516,5 @@ void CAddrMan::SetServices_(const CService& addr, ServiceFlags nServices) } int CAddrMan::RandomInt(int nMax){ - return GetRandInt(nMax); -} \ No newline at end of file + return GetRand(nMax); +} diff --git a/src/addrman.h b/src/addrman.h index 7b37f9a9bc..47a590054e 100644 --- a/src/addrman.h +++ b/src/addrman.h @@ -234,7 +234,7 @@ class CAddrMan //! Select an address to connect to, if newOnly is set to true, only the new table is selected from. CAddrInfo Select_(bool newOnly); - //! Wraps GetRandInt to allow tests to override RandomInt and make it determinismistic. + //! Wraps GetRand to allow tests to override RandomInt and make it determinismistic. virtual int RandomInt(int nMax); #ifdef DEBUG_ADDRMAN diff --git a/src/bdb53/CMakeLists.txt b/src/bdb53/CMakeLists.txt new file mode 100644 index 0000000000..1c98273b73 --- /dev/null +++ b/src/bdb53/CMakeLists.txt @@ -0,0 +1,67 @@ +include(ExternalProject) + +# Configure flags +# =============== +set(BDB_FLAGS + --disable-java + --disable-jdbc + --disable-replication + --enable-cxx +) +if(ENABLE_PIE) + list(APPEND BDB_FLAGS + --with-pic + ) +endif() +if(MINGW) + list(APPEND BDB_FLAGS + --enable-mingw + ) +endif() +if(CMAKE_CXX_COMPILER_ID MATCHES "Clang") + list(APPEND BDB_FLAGS + CFLAGS=-Wno-implicit-function-declaration + ) +endif() + + +# Make flags +# ========== + +include(ProcessorCount) +ProcessorCount(N) +if(N EQUAL 0) + set(N 1) +endif() + +set(MAKEOPTS "-j${N}" CACHE STRING "Options for the 'make' program") + + +# External project +# ================ + +set(LIBDB_BUILD_DIR ${CMAKE_CURRENT_BINARY_DIR}/libdb_build) +set(libdb_cxx_library ${LIBDB_BUILD_DIR}/libdb_cxx.a) +set_directory_properties(PROPERTIES CLEAN_NO_CUSTOM 1) + +ExternalProject_Add(BerkeleyDB_Project + SOURCE_DIR ${CMAKE_CURRENT_SOURCE_DIR} + BINARY_DIR ${LIBDB_BUILD_DIR} + CONFIGURE_COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/dist/configure ${BDB_FLAGS} + BUILD_COMMAND + COMMAND ${MAKE_EXE} ${MAKEOPTS} clean + COMMAND ${MAKE_EXE} ${MAKEOPTS} libdb_cxx.a + INSTALL_COMMAND "" + BUILD_BYPRODUCTS ${libdb_cxx_library} + LOG_CONFIGURE TRUE + LOG_BUILD TRUE + LOG_OUTPUT_ON_FAILURE TRUE +) + +add_library(libdb_cxx STATIC IMPORTED GLOBAL) +add_dependencies(libdb_cxx BerkeleyDB_Project) + +set_target_properties(libdb_cxx PROPERTIES + IMPORTED_LOCATION ${libdb_cxx_library} +) +target_include_directories(libdb_cxx INTERFACE ${LIBDB_BUILD_DIR}) diff --git a/src/config/.empty b/src/config/.empty deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/src/config/gridcoin-config.h.cmake.in b/src/config/gridcoin-config.h.cmake.in new file mode 100644 index 0000000000..94a13f41a3 --- /dev/null +++ b/src/config/gridcoin-config.h.cmake.in @@ -0,0 +1,79 @@ +#ifndef GRIDCOIN_CONFIG_H +#define GRIDCOIN_CONFIG_H + +// Define if thread_local is supported +#define HAVE_THREAD_LOCAL + +#define PACKAGE_NAME "@CMAKE_PROJECT_NAME@" +#define PACKAGE_VERSION "@CMAKE_PROJECT_VERSION@" +#define CLIENT_VERSION_MAJOR @PROJECT_VERSION_MAJOR@ +#define CLIENT_VERSION_MINOR @PROJECT_VERSION_MINOR@ +#define CLIENT_VERSION_REVISION @PROJECT_VERSION_PATCH@ +#define CLIENT_VERSION_BUILD @PROJECT_VERSION_TWEAK@ +#define CLIENT_VERSION_IS_RELEASE @CLIENT_VERSION_IS_RELEASE@ +#define COPYRIGHT_YEAR "@COPYRIGHT_YEAR@" +#define COPYRIGHT_HOLDERS_FINAL "@COPYRIGHT_HOLDERS_FINAL@" + +#cmakedefine ENABLE_SSE41 +#cmakedefine ENABLE_AVX2 +#cmakedefine ENABLE_X86_SHANI +#cmakedefine ENABLE_ARM_CRC +#cmakedefine ENABLE_ARM_SHANI +#cmakedefine USE_ASM + +#cmakedefine USE_DBUS +#cmakedefine USE_QRCODE + +#cmakedefine HAVE_STRERROR_R +#cmakedefine STRERROR_R_CHAR_P + +#cmakedefine WORDS_BIGENDIAN + +#cmakedefine HAVE_BYTESWAP_H +#cmakedefine HAVE_ENDIAN_H +#cmakedefine HAVE_SYS_ENDIAN_H +#cmakedefine HAVE_SYS_PRCTL_H + +#cmakedefine01 HAVE_DECL_LE16TOH +#cmakedefine01 HAVE_DECL_LE32TOH +#cmakedefine01 HAVE_DECL_LE64TOH + +#cmakedefine01 HAVE_DECL_HTOLE16 +#cmakedefine01 HAVE_DECL_HTOLE32 +#cmakedefine01 HAVE_DECL_HTOLE64 + +#cmakedefine01 HAVE_DECL_BE16TOH +#cmakedefine01 HAVE_DECL_BE32TOH +#cmakedefine01 HAVE_DECL_BE64TOH + +#cmakedefine01 HAVE_DECL_HTOBE16 +#cmakedefine01 HAVE_DECL_HTOBE32 +#cmakedefine01 HAVE_DECL_HTOBE64 + +#cmakedefine01 HAVE_DECL_BSWAP_16 +#cmakedefine01 HAVE_DECL_BSWAP_32 +#cmakedefine01 HAVE_DECL_BSWAP_64 + +#cmakedefine HAVE_BUILTIN_CLZL +#cmakedefine HAVE_BUILTIN_CLZLL + +#cmakedefine HAVE_MSG_NOSIGNAL +#cmakedefine HAVE_MSG_DONTWAIT + +#cmakedefine HAVE_MALLOC_INFO +#cmakedefine HAVE_MALLOPT_ARENA_MAX + +#cmakedefine01 HAVE_SYSTEM +#cmakedefine HAVE_GMTIME_R + +// Define if the Linux getrandom system call is available +#cmakedefine HAVE_SYS_GETRANDOM +// Define if the BSD getentropy system call is available +#cmakedefine HAVE_GETENTROPY +// Define if the BSD sysctl(KERN_ARND) is available +#cmakedefine HAVE_SYSCTL_ARND + +#cmakedefine01 HAVE_O_CLOEXEC +#cmakedefine HAVE_STRONG_GETAUXVAL + +#endif //GRIDCOIN_CONFIG_H diff --git a/src/crypto/CMakeLists.txt b/src/crypto/CMakeLists.txt new file mode 100644 index 0000000000..93396debd5 --- /dev/null +++ b/src/crypto/CMakeLists.txt @@ -0,0 +1,57 @@ +set(LIBGRIDCOIN_CRYPTO gridcoin_crypto_base) +add_library(gridcoin_crypto_base STATIC + aes.cpp + chacha20.cpp + hmac_sha256.cpp + hmac_sha512.cpp + poly1305.cpp + ripemd160.cpp + sha1.cpp + sha256.cpp + sha3.cpp + sha512.cpp + siphash.cpp +) + +if(USE_ASM) + target_sources(gridcoin_crypto_base PRIVATE sha256_sse4.cpp) +endif() + +if(ENABLE_SSE41) + list(APPEND LIBGRIDCOIN_CRYPTO gridcoin_crypto_sse41) + add_library(gridcoin_crypto_sse41 STATIC sha256_sse41.cpp) + target_compile_definitions(gridcoin_crypto_sse41 PRIVATE ENABLE_SSE41) + target_compile_options(gridcoin_crypto_sse41 PRIVATE -msse4.1) +endif() + +if(ENABLE_AVX2) + list(APPEND LIBGRIDCOIN_CRYPTO gridcoin_crypto_avx2) + add_library(gridcoin_crypto_avx2 STATIC sha256_avx2.cpp) + target_compile_definitions(gridcoin_crypto_avx2 PRIVATE ENABLE_AVX2) + target_compile_options(gridcoin_crypto_avx2 PRIVATE -mavx -mavx2) +endif() + +if(ENABLE_X86_SHANI) + list(APPEND LIBGRIDCOIN_CRYPTO gridcoin_crypto_x86_shani) + add_library(gridcoin_crypto_x86_shani STATIC sha256_x86_shani.cpp) + target_compile_definitions(gridcoin_crypto_x86_shani PRIVATE ENABLE_X86_SHANI) + target_compile_options(gridcoin_crypto_x86_shani PRIVATE -msse4 -msha) +endif() + +if(ENABLE_ARM_SHANI) + list(APPEND LIBGRIDCOIN_CRYPTO gridcoin_crypto_arm_shani) + add_library(gridcoin_crypto_arm_shani STATIC sha256_arm_shani.cpp) + target_compile_definitions(gridcoin_crypto_arm_shani PRIVATE ENABLE_ARM_SHANI) +endif() + +foreach(library IN LISTS LIBGRIDCOIN_CRYPTO) + target_include_directories(${library} PRIVATE + "${CMAKE_SOURCE_DIR}/src" + "${CMAKE_BINARY_DIR}/src" + ) + target_compile_definitions(${library} PRIVATE HAVE_CONFIG_H) +endforeach() + +set(LIBGRIDCOIN_CRYPTO ${LIBGRIDCOIN_CRYPTO} + CACHE INTERNAL "Gridcoin crypto libraries" +) diff --git a/src/gridcoin/scraper/scraper.cpp b/src/gridcoin/scraper/scraper.cpp index 5ccb008ddb..95788b33e6 100755 --- a/src/gridcoin/scraper/scraper.cpp +++ b/src/gridcoin/scraper/scraper.cpp @@ -6175,7 +6175,7 @@ UniValue testnewsb(const UniValue& params, bool fHelp) if (PastConvergencesSize > 1) { - int i = GetRandInt(PastConvergencesSize - 1); + int i = GetRand(PastConvergencesSize - 1); _log(logattribute::INFO, "testnewsb", "ValidateSuperblock random past RandomPastConvergedManifest index " + ToString(i) + " selected."); diff --git a/src/gridcoin/staking/kernel.cpp b/src/gridcoin/staking/kernel.cpp index 1eb8e195ab..47e436643b 100644 --- a/src/gridcoin/staking/kernel.cpp +++ b/src/gridcoin/staking/kernel.cpp @@ -12,6 +12,8 @@ #include "streams.h" #include "util.h" +#include + using namespace std; using namespace GRC; @@ -385,12 +387,28 @@ bool GRC::ReadStakedInput( CTxDB& txdb, const uint256 prevout_hash, CBlockHeader& out_header, - CTransaction& out_txprev) + CTransaction& out_txprev, + CBlockIndex* pindexPrev) { CTxIndex tx_index; // Get transaction index for the previous transaction if (!txdb.ReadTxIndex(prevout_hash, tx_index)) { + if (pindexPrev != nullptr) { + for (CBlockIndex* pindex = pindexPrev; pindex->pprev != nullptr; pindex = pindex->pprev) { + CBlock block; + ReadBlockFromDisk(block, pindex, Params().GetConsensus()); + + for (const auto& tx : block.vtx) { + if (tx.GetHash() == prevout_hash) { + error("Found tx %s in block %s", tx.GetHash().ToString(), pindex->GetBlockHash().ToString()); + out_txprev = tx; + out_header = pindex->GetBlockHeader(); + return true; + } + } + } + } // Previous transaction not in main chain, may occur during initial download return error("%s: tx index not found for input tx %s", __func__, prevout_hash.GetHex()); } @@ -429,7 +447,7 @@ bool GRC::CalculateLegacyV3HashProof( CTransaction input_tx; CBlockHeader input_block; - if (!ReadStakedInput(txdb, prevout.hash, input_block, input_tx)) { + if (!ReadStakedInput(txdb, prevout.hash, input_block, input_tx, nullptr)) { return coinstake.DoS(1, error("Read staked input failed.")); } @@ -584,7 +602,7 @@ bool GRC::CheckProofOfStakeV8( CBlockHeader header; CTransaction txPrev; - if (!ReadStakedInput(txdb, prevout.hash, header, txPrev)) + if (!ReadStakedInput(txdb, prevout.hash, header, txPrev, pindexPrev)) return tx.DoS(1, error("%s: read staked input failed", __func__)); if (!VerifySignature(txPrev, tx, 0, 0)) diff --git a/src/gridcoin/staking/kernel.h b/src/gridcoin/staking/kernel.h index 237ea7b4b7..380355daad 100644 --- a/src/gridcoin/staking/kernel.h +++ b/src/gridcoin/staking/kernel.h @@ -57,7 +57,8 @@ bool ReadStakedInput( CTxDB& txdb, const uint256 prevout_hash, CBlockHeader& out_header, - CTransaction& out_txprev); + CTransaction& out_txprev, + CBlockIndex* pindexPrev = nullptr); //! //! \brief Calculate the provided block's proof hash with the version 3 staking diff --git a/src/gridcoin/staking/spam.h b/src/gridcoin/staking/spam.h index 41a27eaa1a..84918e8eeb 100644 --- a/src/gridcoin/staking/spam.h +++ b/src/gridcoin/staking/spam.h @@ -201,12 +201,11 @@ class SeenStakes // ...to produce a distribution for values of x with a minimal rate of // collision. // - using limit_t = std::numeric_limits; static const size_t w = sizeof(size_t) * 8; static const size_t M = std::log2(m_proofs_seen.size()); - static const size_t a = (GetRand(limit_t::max()) * 2) + 1; - static const size_t b = GetRand(std::pow(2, w - M) - 1); + static const size_t a = (GetRand() * 2) + 1; + static const size_t b = GetRand(std::pow(2, w - M) - 1); size_t x = 0; diff --git a/src/gridcoin/voting/registry.cpp b/src/gridcoin/voting/registry.cpp index 6b2d1b8213..9c3a7468e9 100644 --- a/src/gridcoin/voting/registry.cpp +++ b/src/gridcoin/voting/registry.cpp @@ -734,7 +734,11 @@ const PollRegistry::Sequence PollRegistry::Polls() const { LOCK(GetPollRegistry().cs_poll_registry); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wthread-safety-reference" return Sequence(m_polls); +#pragma clang diagnostic pop + } const PollReference* PollRegistry::TryLatestActive() const EXCLUSIVE_LOCKS_REQUIRED(PollRegistry::cs_poll_registry) diff --git a/src/gridcoin/voting/result.cpp b/src/gridcoin/voting/result.cpp index 1f5c5cb0d5..a578b4fbf4 100644 --- a/src/gridcoin/voting/result.cpp +++ b/src/gridcoin/voting/result.cpp @@ -920,7 +920,10 @@ class VoteCounter throw InvalidVoteError(); } +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wthread-safety-analysis" ProcessLegacyVote(candidate.LegacyVote()); +#pragma clang diagnostic pop } //! diff --git a/src/key.cpp b/src/key.cpp index 1106f73100..2afde1763c 100644 --- a/src/key.cpp +++ b/src/key.cpp @@ -165,7 +165,7 @@ bool CKey::Check(const unsigned char *vch) { void CKey::MakeNewKey(bool fCompressedIn) { do { - GetStrongRandBytes(keydata.data(), keydata.size()); + GetStrongRandBytes(keydata); } while (!Check(keydata.data())); fValid = true; fCompressed = fCompressedIn; @@ -250,7 +250,7 @@ bool CKey::VerifyPubKey(const CPubKey& pubkey) const { } unsigned char rnd[8]; std::string str = "Bitcoin key verification\n"; - GetRandBytes(rnd, sizeof(rnd)); + GetRandBytes(rnd); uint256 hash; CHash256().Write(MakeUCharSpan(str)).Write(rnd).Finalize(hash); std::vector vchSig; @@ -378,7 +378,7 @@ void ECC_Start() { { // Pass in a random blinding seed to the secp256k1 context. std::vector> vseed(32); - GetRandBytes(vseed.data(), 32); + GetRandBytes(vseed); bool ret = secp256k1_context_randomize(ctx, vseed.data()); assert(ret); } diff --git a/src/leveldb/CMakeLists.txt b/src/leveldb/CMakeLists.txt index 1cb46256c2..42d67d7f63 100644 --- a/src/leveldb/CMakeLists.txt +++ b/src/leveldb/CMakeLists.txt @@ -69,8 +69,8 @@ else(CMAKE_CXX_COMPILER_ID STREQUAL "MSVC") set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fno-exceptions") # Disable RTTI. - string(REGEX REPLACE "-frtti" "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fno-rtti") + #string(REGEX REPLACE "-frtti" "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") + #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fno-rtti") endif(CMAKE_CXX_COMPILER_ID STREQUAL "MSVC") # Test whether -Wthread-safety is available. See diff --git a/src/main.cpp b/src/main.cpp index 30c8f1d1be..32aad9f3aa 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -2929,7 +2929,7 @@ bool SendMessages(CNode* pto, bool fSendTrickle) { uint64_t nonce = 0; while (nonce == 0) { - GetRandBytes((unsigned char*)&nonce, sizeof(nonce)); + GetRandBytes({(unsigned char*)&nonce, sizeof(nonce)}); } pto->fPingQueued = false; pto->nPingUsecStart = GetTimeMicros(); diff --git a/src/net.cpp b/src/net.cpp index b7d0b699d6..e6e26e5d90 100644 --- a/src/net.cpp +++ b/src/net.cpp @@ -212,7 +212,7 @@ void AdvertiseLocal(CNode *pnode) // If discovery is enabled, sometimes give our peer the address it // tells us that it sees us as in case it has a better idea of our // address than we do. - const int randomNumber = GetRandInt((GetnScore(addrLocal) > LOCAL_MANUAL) ? 3+1 : 1+1); + const int randomNumber = GetRand((GetnScore(addrLocal) > LOCAL_MANUAL) ? 3+1 : 1+1); if (IsPeerAddrLocalGood(pnode) && (!addrLocal.IsRoutable() || randomNumber == 0)) { @@ -486,7 +486,7 @@ void CNode::PushVersion() int64_t nTime = GetAdjustedTime(); CAddress addrYou = (addr.IsRoutable() && !IsProxy(addr) ? addr : CAddress(LookupNumeric("0.0.0.0", 0))); CAddress addrMe = CAddress(CService(), nLocalServices); - GetRandBytes((unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)); + GetRandBytes({(unsigned char*)&nLocalHostNonce, sizeof(nLocalHostNonce)}); LogPrint(BCLog::LogFlags::NET, "send version message: version %d, blocks=%d, us=%s, them=%s, peer=%s", PROTOCOL_VERSION, nBestHeight, addrMe.ToString(), addrYou.ToString(), addr.ToString()); diff --git a/src/netaddress.h b/src/netaddress.h index 4831edf300..cbaf02f7a7 100644 --- a/src/netaddress.h +++ b/src/netaddress.h @@ -16,8 +16,6 @@ #include #include -#endif // BITCOIN_NETADDRESS_H - enum Network { NET_UNROUTABLE, @@ -181,3 +179,5 @@ class CService : public CNetAddr } }; +#endif // BITCOIN_NETADDRESS_H + diff --git a/src/node/blockstorage.cpp b/src/node/blockstorage.cpp index f491b723a8..502686fa9f 100644 --- a/src/node/blockstorage.cpp +++ b/src/node/blockstorage.cpp @@ -15,7 +15,10 @@ bool WriteBlockToDisk(const CBlock& block, unsigned int& nFileRet, unsigned int& nBlockPosRet, const CMessageHeader::MessageStartChars& messageStart) + EXCLUSIVE_LOCKS_REQUIRED(cs_main) { + AssertLockHeld(cs_main); + // Open history file to append CAutoFile fileout(AppendBlockFile(nFileRet), SER_DISK, CLIENT_VERSION); if (fileout.IsNull()) diff --git a/src/obj/build.h.in b/src/obj/build.h.in new file mode 100644 index 0000000000..9cbb8aa9c1 --- /dev/null +++ b/src/obj/build.h.in @@ -0,0 +1,7 @@ +#ifndef GRIDCOIN_BUILD_H +#define GRIDCOIN_BUILD_H + +#cmakedefine BUILD_GIT_TAG "@BUILD_GIT_TAG@" +#cmakedefine BUILD_GIT_COMMIT "@BUILD_GIT_COMMIT@" + +#endif // GRIDCOIN_BUILD_H diff --git a/src/qt/CMakeLists.txt b/src/qt/CMakeLists.txt new file mode 100644 index 0000000000..3302a30599 --- /dev/null +++ b/src/qt/CMakeLists.txt @@ -0,0 +1,203 @@ +# Translations +# ============ +add_subdirectory(locale) + + +# libgridcoinqt +# ============= + +add_library(gridcoinqt STATIC + aboutdialog.cpp + addressbookpage.cpp + addresstablemodel.cpp + askpassphrasedialog.cpp + bantablemodel.cpp + bitcoinaddressvalidator.cpp + bitcoinamountfield.cpp + bitcoingui.cpp + bitcoinunits.cpp + clicklabel.cpp + clientmodel.cpp + coincontroldialog.cpp + coincontroltreewidget.cpp + consolidateunspentdialog.cpp + consolidateunspentwizard.cpp + consolidateunspentwizardselectdestinationpage.cpp + consolidateunspentwizardselectinputspage.cpp + consolidateunspentwizardsendpage.cpp + csvmodelwriter.cpp + decoration.cpp + diagnosticsdialog.cpp + editaddressdialog.cpp + favoritespage.cpp + guiutil.cpp + intro.cpp + monitoreddatamapper.cpp + mrcmodel.cpp + mrcrequestpage.cpp + noresult.cpp + notificator.cpp + optionsdialog.cpp + optionsmodel.cpp + overviewpage.cpp + peertablemodel.cpp + qtipcserver.cpp + qvalidatedlineedit.cpp + qvaluecombobox.cpp + receivecoinspage.cpp + researcher/projecttablemodel.cpp + researcher/researchermodel.cpp + researcher/researcherwizard.cpp + researcher/researcherwizardauthpage.cpp + researcher/researcherwizardbeaconpage.cpp + researcher/researcherwizardemailpage.cpp + researcher/researcherwizardinvestorpage.cpp + researcher/researcherwizardmodedetailpage.cpp + researcher/researcherwizardmodepage.cpp + researcher/researcherwizardpoolpage.cpp + researcher/researcherwizardpoolsummarypage.cpp + researcher/researcherwizardprojectspage.cpp + researcher/researcherwizardsummarypage.cpp + rpcconsole.cpp + sendcoinsdialog.cpp + sendcoinsentry.cpp + signverifymessagedialog.cpp + trafficgraphwidget.cpp + transactiondesc.cpp + transactiondescdialog.cpp + transactionfilterproxy.cpp + transactionrecord.cpp + transactiontablemodel.cpp + transactionview.cpp + upgradeqt.cpp + voting/additionalfieldstableview.cpp + voting/additionalfieldstablemodel.cpp + voting/poll_types.cpp + voting/pollcard.cpp + voting/pollcardview.cpp + voting/polldetails.cpp + voting/pollresultchoiceitem.cpp + voting/pollresultdialog.cpp + voting/polltab.cpp + voting/polltablemodel.cpp + voting/pollwizard.cpp + voting/pollwizarddetailspage.cpp + voting/pollwizardprojectpage.cpp + voting/pollwizardsummarypage.cpp + voting/pollwizardtypepage.cpp + voting/votewizard.cpp + voting/votewizardballotpage.cpp + voting/votewizardsummarypage.cpp + voting/votingmodel.cpp + voting/votingpage.cpp + walletmodel.cpp + winshutdownmonitor.cpp + + bitcoin.qrc + bitcoin_locale.qrc +) + +if(WIN32) + target_sources(gridcoinqt PRIVATE res/gridcoinresearch.rc) +elseif(APPLE) + target_sources(gridcoinqt PRIVATE + macdockiconhandler.mm + macnotificationhandler.mm + macos_appnap.mm + ) +endif() + +set_target_properties(gridcoinqt PROPERTIES + AUTOMOC ON + AUTORCC ON + AUTOUIC ON + + AUTOUIC_SEARCH_PATHS "${CMAKE_SOURCE_DIR}/src;${CMAKE_SOURCE_DIR}/src/qt/forms" +) + +# These files include 'node/ui_interface.cpp', which AutoUIC tries to process +set_source_files_properties( + bitcoin.cpp + clientmodel.cpp + mrcmodel.cpp + qtipcserver.cpp + researcher/researchermodel.cpp + transactiondesc.cpp + transactiontablemodel.cpp + voting/votingmodel.cpp + walletmodel.cpp + PROPERTIES + SKIP_AUTOUIC ON +) + +# Libraries to link +# ================= + +target_link_libraries(gridcoinqt PUBLIC + Qt5::Concurrent + Qt5::Core + Qt5::Gui + Qt5::Network + Qt5::Widgets + gridcoin_util +) + +if(APPLE) + target_link_libraries(gridcoinqt PUBLIC + "-framework Foundation" + "-framework ApplicationServices" + "-framework AppKit" + ) +endif() + +target_compile_definitions(gridcoinqt PUBLIC HAVE_CONFIG_H) + +if(USE_DBUS) + target_link_libraries(gridcoinqt PUBLIC Qt5::DBus) +endif() + +if(ENABLE_UPNP) + if(DEFAULT_UPNP) + target_compile_definitions(gridcoinqt PRIVATE USE_UPNP=1) + else() + target_compile_definitions(gridcoinqt PRIVATE USE_UPNP=0) + endif() +endif() + +add_dependencies(gridcoinqt gridcoinqt_l10n) + + +# Application +# =========== + +add_executable(gridcoinresearch WIN32 MACOSX_BUNDLE bitcoin.cpp) + +target_link_libraries(gridcoinresearch PRIVATE + Qt5::Widgets + gridcoin_util + gridcoinqt +) + +if(UNIX AND NOT APPLE) + include(GNUInstallDirs) + install(TARGETS gridcoinresearch + DESTINATION "${CMAKE_INSTALL_BINDIR}" + ) + install(DIRECTORY "${CMAKE_SOURCE_DIR}/share/icons" + DESTINATION "${CMAKE_INSTALL_DATADIR}" + ) + install(FILES "${CMAKE_SOURCE_DIR}/contrib/gridcoinresearch.desktop" + DESTINATION "${CMAKE_INSTALL_DATADIR}/applications" + ) + install(FILES "${CMAKE_SOURCE_DIR}/doc/gridcoinresearch.1" + DESTINATION "${CMAKE_INSTALL_MANDIR}" + ) +endif() + + +# Tests +# ===== + +if(ENABLE_TESTS) + add_subdirectory(test) +endif() diff --git a/src/qt/bitcoingui.cpp b/src/qt/bitcoingui.cpp index d3bf86fb9a..60444b4fe7 100644 --- a/src/qt/bitcoingui.cpp +++ b/src/qt/bitcoingui.cpp @@ -553,7 +553,8 @@ void BitcoinGUI::createMenuBar() file->addAction(verifyMessageAction); file->addSeparator(); - if (!gArgs.GetBoolArg("-testnet", false)) + // Snapshot GUI menu action disabled due to snapshot CDN abuse in 202308. + if (/* !gArgs.GetBoolArg("-testnet", false) */ false) { file->addAction(snapshotAction); } @@ -1884,7 +1885,7 @@ void BitcoinGUI::updateBeaconIcon() if (researcherModel->hasPendingBeacon()) { labelBeaconIcon->setToolTip(tr("CPID: %1\n" - "Time left to activate: %2" + "Time left to activate: %2\n" "%3") .arg(researcherModel->formatCpid(), researcherModel->formatTimeToPendingBeaconExpiration(), diff --git a/src/qt/locale/CMakeLists.txt b/src/qt/locale/CMakeLists.txt new file mode 100644 index 0000000000..696f09dcdf --- /dev/null +++ b/src/qt/locale/CMakeLists.txt @@ -0,0 +1,76 @@ +set(TS_FILES + bitcoin_af_ZA.ts + bitcoin_ar.ts + bitcoin_be_BY.ts + bitcoin_bg.ts + bitcoin_ca.ts + bitcoin_ca@valencia.ts + bitcoin_ca_ES.ts + bitcoin_cs.ts + bitcoin_cy.ts + bitcoin_da.ts + bitcoin_de.ts + bitcoin_el_GR.ts + bitcoin_en.ts + bitcoin_eo.ts + bitcoin_es.ts + bitcoin_es_CL.ts + bitcoin_es_DO.ts + bitcoin_es_MX.ts + bitcoin_es_UY.ts + bitcoin_et.ts + bitcoin_eu_ES.ts + bitcoin_fa.ts + bitcoin_fa_IR.ts + bitcoin_fi.ts + bitcoin_fr.ts + bitcoin_fr_CA.ts + bitcoin_gl.ts + bitcoin_he.ts + bitcoin_hi_IN.ts + bitcoin_hr.ts + bitcoin_hu.ts + bitcoin_id_ID.ts + bitcoin_it.ts + bitcoin_ja.ts + bitcoin_ka.ts + bitcoin_kk_KZ.ts + bitcoin_ko_KR.ts + bitcoin_ky.ts + bitcoin_la.ts + bitcoin_lt.ts + bitcoin_lv_LV.ts + bitcoin_ms_MY.ts + bitcoin_nb.ts + bitcoin_nl.ts + bitcoin_pam.ts + bitcoin_pl.ts + bitcoin_pt_BR.ts + bitcoin_pt_PT.ts + bitcoin_ro_RO.ts + bitcoin_ru.ts + bitcoin_sk.ts + bitcoin_sl_SI.ts + bitcoin_sq.ts + bitcoin_sr.ts + bitcoin_sv.ts + bitcoin_th_TH.ts + bitcoin_tr.ts + bitcoin_uk.ts + bitcoin_ur_PK.ts + bitcoin_vi.ts + bitcoin_zh_CN.ts + bitcoin_zh_TW.ts +) + +set_source_files_properties(${TS_FILES} PROPERTIES + OUTPUT_LOCATION "${CMAKE_CURRENT_SOURCE_DIR}" +) + +if(LUPDATE) + qt_create_translation(QM_FILES ${CMAKE_SOURCE_DIR} ${TS_FILES}) +else() + qt_add_translation(QM_FILES ${TS_FILES}) +endif() + +add_custom_target(gridcoinqt_l10n DEPENDS ${QM_FILES}) diff --git a/src/qt/researcher/researchermodel.cpp b/src/qt/researcher/researchermodel.cpp index b75b24dfa9..a6062c50ba 100644 --- a/src/qt/researcher/researchermodel.cpp +++ b/src/qt/researcher/researchermodel.cpp @@ -276,7 +276,17 @@ bool ResearcherModel::hasActiveBeacon() const bool ResearcherModel::hasPendingBeacon() const { - return m_pending_beacon.operator bool(); + if (!m_pending_beacon.operator bool()) { + return false; + } + + // If here, a pending beacon is present. Determine if expired + // while pending. No need to actually clean the pending entry + // up. It will be eventually cleaned by the contract handler via + // the ActivatePending call. + GRC::PendingBeacon pending_beacon(*m_pending_beacon); + + return !pending_beacon.PendingExpired(GetAdjustedTime()); } bool ResearcherModel::hasRenewableBeacon() const diff --git a/src/qt/researcher/researchermodel.h b/src/qt/researcher/researchermodel.h index dd353a9807..8d6b6df341 100644 --- a/src/qt/researcher/researchermodel.h +++ b/src/qt/researcher/researchermodel.h @@ -93,6 +93,11 @@ class ResearcherModel : public QObject bool hasEligibleProjects() const; bool hasPoolProjects() const; bool hasActiveBeacon() const; + + //! + //! \brief hasPendingBeacon returns true if m_pending_beacon is not null and also not expired while pending. + //! \return boolean + //! bool hasPendingBeacon() const; bool hasRenewableBeacon() const; bool beaconExpired() const; diff --git a/src/qt/test/CMakeLists.txt b/src/qt/test/CMakeLists.txt new file mode 100644 index 0000000000..4717e74597 --- /dev/null +++ b/src/qt/test/CMakeLists.txt @@ -0,0 +1,17 @@ +add_executable(test_gridcoin-qt + test_main.cpp + uritests.cpp +) + +set_target_properties(test_gridcoin-qt PROPERTIES + AUTOMOC ON +) + +target_link_libraries(test_gridcoin-qt PRIVATE + Qt5::Test + Qt5::Widgets + gridcoin_util + gridcoinqt +) + +add_test(NAME gridcoin_qt_tests COMMAND test_gridcoin-qt) diff --git a/src/random.cpp b/src/random.cpp index 2ebc5bd86f..368fa1f9bc 100644 --- a/src/random.cpp +++ b/src/random.cpp @@ -16,9 +16,11 @@ #include // for LogPrintf() #include #include +#include #include // for Mutex #include // for GetTimeMicros() +#include #include #include @@ -31,10 +33,8 @@ #include #include #endif -#if defined(HAVE_GETENTROPY) || (defined(HAVE_GETENTROPY_RAND) && defined(MAC_OSX)) -#include -#endif #if defined(HAVE_GETENTROPY_RAND) && defined(MAC_OSX) +#include #include #endif #ifdef HAVE_SYSCTL_ARND @@ -304,16 +304,14 @@ void GetOSRand(unsigned char *ent32) RandFailure(); } } -#elif defined(HAVE_GETENTROPY) && defined(__OpenBSD__) - /* On OpenBSD this can return up to 256 bytes of entropy, will return an - * error if more are requested. - * The call cannot return less than the requested number of bytes. - getentropy is explicitly limited to openbsd here, as a similar (but not - the same) function may exist on other platforms via glibc. +#elif defined(__OpenBSD__) + /* OpenBSD. From the arc4random(3) man page: + "Use of these functions is encouraged for almost all random number + consumption because the other interfaces are deficient in either + quality, portability, standardization, or availability." + The function call is always successful. */ - if (getentropy(ent32, NUM_OS_RANDOM_BYTES) != 0) { - RandFailure(); - } + arc4random_buf(ent32, NUM_OS_RANDOM_BYTES); // Silence a compiler warning about unused function. (void)GetDevURandom; #elif defined(HAVE_GETENTROPY_RAND) && defined(MAC_OSX) @@ -581,27 +579,22 @@ static void ProcRand(unsigned char* out, int num, RNGLevel level) noexcept } } -void GetRandBytes(unsigned char* buf, int num) noexcept { ProcRand(buf, num, RNGLevel::FAST); } -void GetStrongRandBytes(unsigned char* buf, int num) noexcept { ProcRand(buf, num, RNGLevel::SLOW); } +void GetRandBytes(Span bytes) noexcept { ProcRand(bytes.data(), bytes.size(), RNGLevel::FAST); } +void GetStrongRandBytes(Span bytes) noexcept { ProcRand(bytes.data(), bytes.size(), RNGLevel::SLOW); } void RandAddPeriodic() noexcept { ProcRand(nullptr, 0, RNGLevel::PERIODIC); } void RandAddEvent(const uint32_t event_info) noexcept { GetRNGState().AddEvent(event_info); } bool g_mock_deterministic_tests{false}; -uint64_t GetRand(uint64_t nMax) noexcept +uint64_t GetRandInternal(uint64_t nMax) noexcept { return FastRandomContext(g_mock_deterministic_tests).randrange(nMax); } -int GetRandInt(int nMax) noexcept -{ - return GetRand(nMax); -} - uint256 GetRandHash() noexcept { uint256 hash; - GetRandBytes((unsigned char*)&hash, sizeof(hash)); + GetRandBytes(hash); return hash; } @@ -714,3 +707,9 @@ void RandomInit() ReportHardwareRand(); } + +std::chrono::microseconds GetExponentialRand(std::chrono::microseconds now, std::chrono::seconds average_interval) +{ + double unscaled = -std::log1p(GetRand(uint64_t{1} << 48) * -0.0000000000000035527136788 /* -1/2^48 */); + return now + std::chrono::duration_cast(unscaled * average_interval + 0.5us); +} diff --git a/src/random.h b/src/random.h index 95362b421b..821927af92 100644 --- a/src/random.h +++ b/src/random.h @@ -8,9 +8,10 @@ #include #include +#include #include -#include // For std::chrono::microseconds +#include #include #include @@ -66,9 +67,19 @@ * * Thread-safe. */ -void GetRandBytes(unsigned char* buf, int num) noexcept; +void GetRandBytes(Span bytes) noexcept; /** Generate a uniform random integer in the range [0..range). Precondition: range > 0 */ -uint64_t GetRand(uint64_t nMax) noexcept; +uint64_t GetRandInternal(uint64_t nMax) noexcept; +/** Generate a uniform random integer of type T in the range [0..nMax) + * nMax defaults to std::numeric_limits::max() + * Precondition: nMax > 0, T is an integral type, no larger than uint64_t + */ +template +T GetRand(T nMax=std::numeric_limits::max()) noexcept { + static_assert(std::is_integral(), "T must be integral"); + static_assert(std::numeric_limits::max() <= std::numeric_limits::max(), "GetRand only supports up to uint64_t"); + return T(GetRandInternal(nMax)); +} /** Generate a uniform random duration in the range [0..max). Precondition: max.count() > 0 */ template D GetRandomDuration(typename std::common_type::type max) noexcept @@ -82,6 +93,18 @@ D GetRandomDuration(typename std::common_type::type max) noexcept }; constexpr auto GetRandMicros = GetRandomDuration; constexpr auto GetRandMillis = GetRandomDuration; + +/** + * Return a timestamp in the future sampled from an exponential distribution + * (https://en.wikipedia.org/wiki/Exponential_distribution). This distribution + * is memoryless and should be used for repeated network events (e.g. sending a + * certain type of message) to minimize leaking information to observers. + * + * The probability of an event occuring before time x is 1 - e^-(x/a) where a + * is the average interval between events. + * */ +std::chrono::microseconds GetExponentialRand(std::chrono::microseconds now, std::chrono::seconds average_interval); + int GetRandInt(int nMax) noexcept; uint256 GetRandHash() noexcept; @@ -93,7 +116,7 @@ uint256 GetRandHash() noexcept; * * Thread-safe. */ -void GetStrongRandBytes(unsigned char* buf, int num) noexcept; +void GetStrongRandBytes(Span bytes) noexcept; /** * Gather entropy from various expensive sources, and feed them to the PRNG state. diff --git a/src/rpc/server.cpp b/src/rpc/server.cpp index 6e9d215864..5bf4a69b26 100644 --- a/src/rpc/server.cpp +++ b/src/rpc/server.cpp @@ -594,7 +594,7 @@ void StartRPCThreads() (gArgs.GetArg("-rpcuser", "") == gArgs.GetArg("-rpcpassword", "")))) { unsigned char rand_pwd[32]; - GetRandBytes(rand_pwd, sizeof(rand_pwd)); + GetRandBytes({rand_pwd, sizeof(rand_pwd)}); string strWhatAmI = "To use gridcoind"; if (gArgs.IsArgSet("-server")) strWhatAmI = strprintf(_("To use the %s option"), "\"-server\""); diff --git a/src/rpc/voting.cpp b/src/rpc/voting.cpp index 67d7ab4cfb..02e720f518 100644 --- a/src/rpc/voting.cpp +++ b/src/rpc/voting.cpp @@ -569,9 +569,13 @@ UniValue getpollresults(const UniValue& params, bool fHelp) // We only need to lock the registry to retrieve the reference. If there is a reorg during the PollResultToJson, it will // throw. - if (const PollReference* ref = WITH_LOCK(GetPollRegistry().cs_poll_registry, return TryPollByTitleOrId(title_or_id))) { + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wthread-safety-analysis" + if (const PollReference* ref = WITH_LOCK(GetPollRegistry().cs_poll_registry, return TryPollByTitleOrId(title_or_id))) { return PollResultToJson(*ref); } +#pragma clang diagnostic pop throw JSONRPCError(RPC_MISC_ERROR, "No matching poll found"); } @@ -722,9 +726,12 @@ UniValue votedetails(const UniValue& params, bool fHelp) // We only need to lock the registry to retrieve the reference. If there is a reorg during the PollResultToJson, it will // throw. - if (const PollReference* ref = WITH_LOCK(GetPollRegistry().cs_poll_registry, return TryPollByTitleOrId(title_or_id))) { +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wthread-safety-analysis" + if (const PollReference* ref = WITH_LOCK(GetPollRegistry().cs_poll_registry, return TryPollByTitleOrId(title_or_id))) { return VoteDetailsToJson(*ref); } +#pragma clang diagnostic pop throw JSONRPCError(RPC_MISC_ERROR, "No matching poll found"); } diff --git a/src/script.cpp b/src/script.cpp index 81218332b5..bd451e3a1f 100644 --- a/src/script.cpp +++ b/src/script.cpp @@ -3,8 +3,6 @@ // Distributed under the MIT/X11 software license, see the accompanying // file COPYING or https://opensource.org/licenses/mit-license.php. -using namespace std; - #include "script.h" #include #include "keystore.h" @@ -18,6 +16,8 @@ using namespace std; #include +using namespace std; + CScriptID::CScriptID(const CScript& in) : BaseHash(Hash160(in)) {} //CScriptID::CScriptID(const ScriptHash& in) : BaseHash(static_cast(in)) {} diff --git a/src/serialize.h b/src/serialize.h index cad91085b2..5d6ff5cacf 100644 --- a/src/serialize.h +++ b/src/serialize.h @@ -732,7 +732,10 @@ template void Unserialize(Stream& os, std::unique_p template inline void Serialize(Stream& os, const T& a) { +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wthread-safety-analysis" a.Serialize(os); +#pragma clang diagnostic pop } template diff --git a/src/test/CMakeLists.txt b/src/test/CMakeLists.txt new file mode 100644 index 0000000000..7c4c42d638 --- /dev/null +++ b/src/test/CMakeLists.txt @@ -0,0 +1,64 @@ +if(NOT SYSTEM_XXD) + add_executable(xxd xxd/xxd.c) + set(XXD $) +endif() + +add_executable(test_gridcoin + checkpoints_tests.cpp + dos_tests.cpp + accounting_tests.cpp + addrman_tests.cpp + allocator_tests.cpp + base32_tests.cpp + base58_tests.cpp + base64_tests.cpp + bignum_tests.cpp + bip32_tests.cpp + #compilerbug_tests.cpp + crypto_tests.cpp + fs_tests.cpp + getarg_tests.cpp + gridcoin_tests.cpp + gridcoin/block_finder_tests.cpp + gridcoin/beacon_tests.cpp + gridcoin/claim_tests.cpp + gridcoin/contract_tests.cpp + gridcoin/cpid_tests.cpp + gridcoin/enumbytes_tests.cpp + gridcoin/magnitude_tests.cpp + gridcoin/mrc_tests.cpp + gridcoin/project_tests.cpp + gridcoin/protocol_tests.cpp + gridcoin/researcher_tests.cpp + gridcoin/scraper_registry_tests.cpp + gridcoin/superblock_tests.cpp + key_tests.cpp + merkle_tests.cpp + mruset_tests.cpp + multisig_tests.cpp + netbase_tests.cpp + net_tests.cpp + random_tests.cpp + rpc_tests.cpp + sanity_tests.cpp + scheduler_tests.cpp + script_p2sh_tests.cpp + script_tests.cpp + serialize_tests.cpp + sigopcount_tests.cpp + sync_tests.cpp + test_gridcoin.cpp + transaction_tests.cpp + uint256_tests.cpp + util_tests.cpp + wallet_tests.cpp +) + +add_subdirectory(data) + +target_link_libraries(test_gridcoin PRIVATE + Boost::unit_test_framework + gridcoin_util +) + +add_test(NAME gridcoin_tests COMMAND test_gridcoin) diff --git a/src/test/data/CMakeLists.txt b/src/test/data/CMakeLists.txt new file mode 100644 index 0000000000..784efde82a --- /dev/null +++ b/src/test/data/CMakeLists.txt @@ -0,0 +1,50 @@ +set(JSON_TEST_FILES + base58_keys_valid.json + base58_encode_decode.json + base58_keys_invalid.json + script_valid.json + script_invalid.json + tx_invalid.json + tx_valid.json +) + +set(BINARY_TEST_FILES + mainnet_beacon.bin + superblock_packed.bin + testnet_beacon.bin +) + +set(TEXT_TEST_FILES + superblock.txt + superblock_unpacked.txt +) + +foreach(file_in IN LISTS JSON_TEST_FILES) + set(file_out ${CMAKE_CURRENT_BINARY_DIR}/${file_in}.h) + add_custom_command(OUTPUT ${file_in}.h + COMMAND ${CMAKE_COMMAND} -P ${CMAKE_SOURCE_DIR}/build-aux/cmake/json2header.cmake ${XXD} ${CMAKE_CURRENT_SOURCE_DIR}/${file_in} ${file_out} + DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/${file_in} + ) + add_custom_target(generate_${file_in}.h DEPENDS ${file_in}.h) + add_dependencies(test_gridcoin generate_${file_in}.h) +endforeach() + +foreach(file_in IN LISTS BINARY_TEST_FILES) + get_filename_component(var_name ${file_in} NAME_WE) + add_custom_command(OUTPUT ${file_in}.h + COMMAND ${XXD} -i -n ${var_name}_bin ${CMAKE_CURRENT_SOURCE_DIR}/${file_in} ${CMAKE_CURRENT_BINARY_DIR}/${file_in}.h + DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/${file_in} + ) + add_custom_target(generate_${file_in}.h DEPENDS ${file_in}.h) + add_dependencies(test_gridcoin generate_${file_in}.h) +endforeach() + +foreach(file_in IN LISTS TEXT_TEST_FILES) + set(file_out ${CMAKE_CURRENT_BINARY_DIR}/${file_in}.h) + add_custom_command(OUTPUT ${file_in}.h + COMMAND ${CMAKE_COMMAND} -P ${CMAKE_SOURCE_DIR}/build-aux/cmake/txt2header.cmake ${CMAKE_CURRENT_SOURCE_DIR}/${file_in} ${file_out} + DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/${file_in} + ) + add_custom_target(generate_${file_in}.h DEPENDS ${file_in}.h) + add_dependencies(test_gridcoin generate_${file_in}.h) +endforeach() diff --git a/src/test/random_tests.cpp b/src/test/random_tests.cpp index 452a6b7202..289b063445 100644 --- a/src/test/random_tests.cpp +++ b/src/test/random_tests.cpp @@ -28,8 +28,8 @@ BOOST_AUTO_TEST_CASE(fastrandom_tests) FastRandomContext ctx2(true); for (int i = 10; i > 0; --i) { - BOOST_CHECK_EQUAL(GetRand(std::numeric_limits::max()), uint64_t{10393729187455219830U}); - BOOST_CHECK_EQUAL(GetRandInt(std::numeric_limits::max()), int{769702006}); + BOOST_CHECK_EQUAL(GetRand(), uint64_t{10393729187455219830U}); + BOOST_CHECK_EQUAL(GetRand(), int{769702006}); BOOST_CHECK_EQUAL(GetRandMicros(std::chrono::hours{1}).count(), 2917185654); BOOST_CHECK_EQUAL(GetRandMillis(std::chrono::hours{1}).count(), 2144374); } @@ -49,8 +49,8 @@ BOOST_AUTO_TEST_CASE(fastrandom_tests) // Check that a nondeterministic ones are not g_mock_deterministic_tests = false; for (int i = 10; i > 0; --i) { - BOOST_CHECK(GetRand(std::numeric_limits::max()) != uint64_t{10393729187455219830U}); - BOOST_CHECK(GetRandInt(std::numeric_limits::max()) != int{769702006}); + BOOST_CHECK(GetRand() != uint64_t{10393729187455219830U}); + BOOST_CHECK(GetRand() != int{769702006}); BOOST_CHECK(GetRandMicros(std::chrono::hours{1}) != std::chrono::microseconds{2917185654}); BOOST_CHECK(GetRandMillis(std::chrono::hours{1}) != std::chrono::milliseconds{2144374}); } diff --git a/src/test/xxd/xxd.c b/src/test/xxd/xxd.c new file mode 100644 index 0000000000..e2fc9dff6a --- /dev/null +++ b/src/test/xxd/xxd.c @@ -0,0 +1,897 @@ +/* xxd: my hexdump facility. jw + * + * 2.10.90 changed to word output + * 3.03.93 new indent style, dumb bug inserted and fixed. + * -c option, mls + * 26.04.94 better option parser, -ps, -l, -s added. + * 1.07.94 -r badly needs - as input file. Per default autoskip over + * consecutive lines of zeroes, as unix od does. + * -a shows them too. + * -i dump as c-style #include "file.h" + * 1.11.95 if "xxd -i" knows the filename, an 'unsigned char filename_bits[]' + * array is written in correct c-syntax. + * -s improved, now defaults to absolute seek, relative requires a '+'. + * -r improved, now -r -s -0x... is supported. + * change/suppress leading '\0' bytes. + * -l n improved: stops exactly after n bytes. + * -r improved, better handling of partial lines with trailing garbage. + * -r improved, now -r -p works again! + * -r improved, less flushing, much faster now! (that was silly) + * 3.04.96 Per repeated request of a single person: autoskip defaults to off. + * 15.05.96 -v added. They want to know the version. + * -a fixed, to show last line inf file ends in all zeros. + * -u added: Print upper case hex-letters, as preferred by unix bc. + * -h added to usage message. Usage message extended. + * Now using outfile if specified even in normal mode, aehem. + * No longer mixing of ints and longs. May help doze people. + * Added binify ioctl for same reason. (Enough Doze stress for 1996!) + * 16.05.96 -p improved, removed occasional superfluous linefeed. + * 20.05.96 -l 0 fixed. tried to read anyway. + * 21.05.96 -i fixed. now honours -u, and prepends __ to numeric filenames. + * compile -DWIN32 for NT or W95. George V. Reilly, * -v improved :-) + * support --gnuish-longhorn-options + * 25.05.96 MAC support added: CodeWarrior already uses ``outline'' in Types.h + * which is included by MacHeaders (Axel Kielhorn). Renamed to + * xxdline(). + * 7.06.96 -i printed 'int' instead of 'char'. *blush* + * added Bram's OS2 ifdefs... + * 18.07.96 gcc -Wall @ SunOS4 is now silent. + * Added osver for MSDOS/DJGPP/WIN32. + * 29.08.96 Added size_t to strncmp() for Amiga. + * 24.03.97 Windows NT support (Phil Hanna). Clean exit for Amiga WB (Bram) + * 02.04.97 Added -E option, to have EBCDIC translation instead of ASCII + * (azc10@yahoo.com) + * 22.05.97 added -g (group octets) option (jcook@namerica.kla.com). + * 23.09.98 nasty -p -r misfeature fixed: slightly wrong output, when -c was + * missing or wrong. + * 26.09.98 Fixed: 'xxd -i infile outfile' did not truncate outfile. + * 27.10.98 Fixed: -g option parser required blank. + * option -b added: 01000101 binary output in normal format. + * 16.05.00 Added VAXC changes by Stephen P. Wall + * 16.05.00 Improved MMS file and merge for VMS by Zoltan Arpadffy + * 2011 March Better error handling by Florian Zumbiehl. + * 2011 April Formatting by Bram Moolenaar + * 08.06.2013 Little-endian hexdump (-e) and offset (-o) by Vadim Vygonets. + * 11.01.2019 Add full 64/32 bit range to -o and output by Christer Jensen. + * 04.02.2020 Add -d for decimal offsets by Aapo Rantalainen + * 14.01.2022 Disable extra newlines with -c0 -p by Erik Auerswald. + * 20.06.2022 Permit setting the variable names used by -i by David Gow + * + * (c) 1990-1998 by Juergen Weigert (jnweiger@gmail.com) + * + * I hereby grant permission to distribute and use xxd + * under X11-MIT or GPL-2.0 (at the user's choice). + * + * Contributions by Bram Moolenaar et al. + */ + +/* Visual Studio 2005 has 'deprecated' many of the standard CRT functions */ +#if _MSC_VER >= 1400 +# define _CRT_SECURE_NO_DEPRECATE +# define _CRT_NONSTDC_NO_DEPRECATE +#endif +#if !defined(CYGWIN) && defined(__CYGWIN__) +# define CYGWIN +#endif + +#if (defined(__linux__) && !defined(__ANDROID__)) || defined(__CYGWIN__) +# define _XOPEN_SOURCE 700 /* for fdopen() */ +#endif + +#include +#ifdef VAXC +# include +#else +# include +#endif +#if defined(WIN32) || defined(CYGWIN) +# include /* for setmode() */ +#else +# ifdef UNIX +# include +# endif +#endif +#include +#include /* for strncmp() */ +#include /* for isalnum() */ +#include +#if __MWERKS__ && !defined(BEBOX) +# include /* for fdopen() on MAC */ +#endif + + +/* This corrects the problem of missing prototypes for certain functions + * in some GNU installations (e.g. SunOS 4.1.x). + * Darren Hiebert (sparc-sun-sunos4.1.3_U1/2.7.2.2) + */ +#if defined(__GNUC__) && defined(__STDC__) +# ifndef __USE_FIXED_PROTOTYPES__ +# define __USE_FIXED_PROTOTYPES__ +# endif +#endif + +#ifndef __USE_FIXED_PROTOTYPES__ +/* + * This is historic and works only if the compiler really has no prototypes: + * + * Include prototypes for Sun OS 4.x, when using an ANSI compiler. + * FILE is defined on OS 4.x, not on 5.x (Solaris). + * if __SVR4 is defined (some Solaris versions), don't include this. + */ +#if defined(sun) && defined(FILE) && !defined(__SVR4) && defined(__STDC__) +# define __P(a) a +/* excerpt from my sun_stdlib.h */ +extern int fprintf __P((FILE *, char *, ...)); +extern int fputs __P((char *, FILE *)); +extern int _flsbuf __P((unsigned char, FILE *)); +extern int _filbuf __P((FILE *)); +extern int fflush __P((FILE *)); +extern int fclose __P((FILE *)); +extern int fseek __P((FILE *, long, int)); +extern int rewind __P((FILE *)); + +extern void perror __P((char *)); +# endif +#endif + +char version[] = "xxd 2022-01-14 by Juergen Weigert et al."; +#ifdef WIN32 +char osver[] = " (Win32)"; +#else +char osver[] = ""; +#endif + +#if defined(WIN32) +# define BIN_READ(yes) ((yes) ? "rb" : "rt") +# define BIN_WRITE(yes) ((yes) ? "wb" : "wt") +# define BIN_CREAT(yes) ((yes) ? (O_CREAT|O_BINARY) : O_CREAT) +# define BIN_ASSIGN(fp, yes) setmode(fileno(fp), (yes) ? O_BINARY : O_TEXT) +# define PATH_SEP '\\' +#elif defined(CYGWIN) +# define BIN_READ(yes) ((yes) ? "rb" : "rt") +# define BIN_WRITE(yes) ((yes) ? "wb" : "w") +# define BIN_CREAT(yes) ((yes) ? (O_CREAT|O_BINARY) : O_CREAT) +# define BIN_ASSIGN(fp, yes) ((yes) ? (void) setmode(fileno(fp), O_BINARY) : (void) (fp)) +# define PATH_SEP '/' +#else +# ifdef VMS +# define BIN_READ(dummy) "r" +# define BIN_WRITE(dummy) "w" +# define BIN_CREAT(dummy) O_CREAT +# define BIN_ASSIGN(fp, dummy) fp +# define PATH_SEP ']' +# define FILE_SEP '.' +# else +# define BIN_READ(dummy) "r" +# define BIN_WRITE(dummy) "w" +# define BIN_CREAT(dummy) O_CREAT +# define BIN_ASSIGN(fp, dummy) fp +# define PATH_SEP '/' +# endif +#endif + +/* open has only to arguments on the Mac */ +#if __MWERKS__ +# define OPEN(name, mode, umask) open(name, mode) +#else +# define OPEN(name, mode, umask) open(name, mode, umask) +#endif + +#ifdef AMIGA +# define STRNCMP(s1, s2, l) strncmp(s1, s2, (size_t)l) +#else +# define STRNCMP(s1, s2, l) strncmp(s1, s2, l) +#endif + +#ifndef __P +# if defined(__STDC__) || defined(WIN32) +# define __P(a) a +# else +# define __P(a) () +# endif +#endif + +#define TRY_SEEK /* attempt to use lseek, or skip forward by reading */ +#define COLS 256 /* change here, if you ever need more columns */ +#define LLEN ((2*(int)sizeof(unsigned long)) + 4 + (9*COLS-1) + COLS + 2) + +char hexxa[] = "0123456789abcdef0123456789ABCDEF", *hexx = hexxa; + +/* the different hextypes known by this program: */ +#define HEX_NORMAL 0 +#define HEX_POSTSCRIPT 1 +#define HEX_CINCLUDE 2 +#define HEX_BITS 3 /* not hex a dump, but bits: 01111001 */ +#define HEX_LITTLEENDIAN 4 + +#define CONDITIONAL_CAPITALIZE(c) (capitalize ? toupper((int)c) : c) + +static char *pname; + + static void +exit_with_usage(void) +{ + fprintf(stderr, "Usage:\n %s [options] [infile [outfile]]\n", pname); + fprintf(stderr, " or\n %s -r [-s [-]offset] [-c cols] [-ps] [infile [outfile]]\n", pname); + fprintf(stderr, "Options:\n"); + fprintf(stderr, " -a toggle autoskip: A single '*' replaces nul-lines. Default off.\n"); + fprintf(stderr, " -b binary digit dump (incompatible with -ps,-i,-r). Default hex.\n"); + fprintf(stderr, " -C capitalize variable names in C include file style (-i).\n"); + fprintf(stderr, " -c cols format octets per line. Default 16 (-i: 12, -ps: 30).\n"); + fprintf(stderr, " -E show characters in EBCDIC. Default ASCII.\n"); + fprintf(stderr, " -e little-endian dump (incompatible with -ps,-i,-r).\n"); + fprintf(stderr, " -g bytes number of octets per group in normal output. Default 2 (-e: 4).\n"); + fprintf(stderr, " -h print this summary.\n"); + fprintf(stderr, " -i output in C include file style.\n"); + fprintf(stderr, " -l len stop after octets.\n"); + fprintf(stderr, " -n name set the variable name used in C include output (-i).\n"); + fprintf(stderr, " -o off add to the displayed file position.\n"); + fprintf(stderr, " -ps output in postscript plain hexdump style.\n"); + fprintf(stderr, " -r reverse operation: convert (or patch) hexdump into binary.\n"); + fprintf(stderr, " -r -s off revert with added to file positions found in hexdump.\n"); + fprintf(stderr, " -d show offset in decimal instead of hex.\n"); + fprintf(stderr, " -s %sseek start at bytes abs. %sinfile offset.\n", +#ifdef TRY_SEEK + "[+][-]", "(or +: rel.) "); +#else + "", ""); +#endif + fprintf(stderr, " -u use upper case hex letters.\n"); + fprintf(stderr, " -v show version: \"%s%s\".\n", version, osver); + exit(1); +} + + static void +perror_exit(int ret) +{ + fprintf(stderr, "%s: ", pname); + perror(NULL); + exit(ret); +} + + static void +error_exit(int ret, char *msg) +{ + fprintf(stderr, "%s: %s\n", pname, msg); + exit(ret); +} + + static int +getc_or_die(FILE *fpi) +{ + int c = getc(fpi); + if (c == EOF && ferror(fpi)) + perror_exit(2); + return c; +} + + static void +putc_or_die(int c, FILE *fpo) +{ + if (putc(c, fpo) == EOF) + perror_exit(3); +} + + static void +fputs_or_die(char *s, FILE *fpo) +{ + if (fputs(s, fpo) == EOF) + perror_exit(3); +} + +/* Use a macro to allow for different arguments. */ +#define FPRINTF_OR_DIE(args) if (fprintf args < 0) perror_exit(3) + + static void +fclose_or_die(FILE *fpi, FILE *fpo) +{ + if (fclose(fpo) != 0) + perror_exit(3); + if (fclose(fpi) != 0) + perror_exit(2); +} + +/* + * If "c" is a hex digit, return the value. + * Otherwise return -1. + */ + static int +parse_hex_digit(int c) +{ + return (c >= '0' && c <= '9') ? c - '0' + : (c >= 'a' && c <= 'f') ? c - 'a' + 10 + : (c >= 'A' && c <= 'F') ? c - 'A' + 10 + : -1; +} + +/* + * Ignore text on "fpi" until end-of-line or end-of-file. + * Return the '\n' or EOF character. + * When an error is encountered exit with an error message. + */ + static int +skip_to_eol(FILE *fpi, int c) +{ + while (c != '\n' && c != EOF) + c = getc_or_die(fpi); + return c; +} + +/* + * Max. cols binary characters are decoded from the input stream per line. + * Two adjacent garbage characters after evaluated data delimit valid data. + * Everything up to the next newline is discarded. + * + * The name is historic and came from 'undo type opt h'. + */ + static int +huntype( + FILE *fpi, + FILE *fpo, + int cols, + int hextype, + long base_off) +{ + int c, ign_garb = 1, n1 = -1, n2 = 0, n3, p = cols; + long have_off = 0, want_off = 0; + + rewind(fpi); + + while ((c = getc(fpi)) != EOF) + { + if (c == '\r') /* Doze style input file? */ + continue; + + /* Allow multiple spaces. This doesn't work when there is normal text + * after the hex codes in the last line that looks like hex, thus only + * use it for PostScript format. */ + if (hextype == HEX_POSTSCRIPT && (c == ' ' || c == '\n' || c == '\t')) + continue; + + n3 = n2; + n2 = n1; + + n1 = parse_hex_digit(c); + if (n1 == -1 && ign_garb) + continue; + + ign_garb = 0; + + if (!hextype && (p >= cols)) + { + if (n1 < 0) + { + p = 0; + continue; + } + want_off = (want_off << 4) | n1; + continue; + } + + if (base_off + want_off != have_off) + { + if (fflush(fpo) != 0) + perror_exit(3); +#ifdef TRY_SEEK + if (fseek(fpo, base_off + want_off - have_off, SEEK_CUR) >= 0) + have_off = base_off + want_off; +#endif + if (base_off + want_off < have_off) + error_exit(5, "Sorry, cannot seek backwards."); + for (; have_off < base_off + want_off; have_off++) + putc_or_die(0, fpo); + } + + if (n2 >= 0 && n1 >= 0) + { + putc_or_die((n2 << 4) | n1, fpo); + have_off++; + want_off++; + n1 = -1; + if (!hextype && (++p >= cols)) + /* skip the rest of the line as garbage */ + c = skip_to_eol(fpi, c); + } + else if (n1 < 0 && n2 < 0 && n3 < 0) + /* already stumbled into garbage, skip line, wait and see */ + c = skip_to_eol(fpi, c); + + if (c == '\n') + { + if (!hextype) + want_off = 0; + p = cols; + ign_garb = 1; + } + } + if (fflush(fpo) != 0) + perror_exit(3); +#ifdef TRY_SEEK + fseek(fpo, 0L, SEEK_END); +#endif + fclose_or_die(fpi, fpo); + return 0; +} + +/* + * Print line l. If nz is false, xxdline regards the line a line of + * zeroes. If there are three or more consecutive lines of zeroes, + * they are replaced by a single '*' character. + * + * If the output ends with more than two lines of zeroes, you + * should call xxdline again with l being the last line and nz + * negative. This ensures that the last line is shown even when + * it is all zeroes. + * + * If nz is always positive, lines are never suppressed. + */ + static void +xxdline(FILE *fp, char *l, int nz) +{ + static char z[LLEN+1]; + static int zero_seen = 0; + + if (!nz && zero_seen == 1) + strcpy(z, l); + + if (nz || !zero_seen++) + { + if (nz) + { + if (nz < 0) + zero_seen--; + if (zero_seen == 2) + fputs_or_die(z, fp); + if (zero_seen > 2) + fputs_or_die("*\n", fp); + } + if (nz >= 0 || zero_seen > 0) + fputs_or_die(l, fp); + if (nz) + zero_seen = 0; + } +} + +/* This is an EBCDIC to ASCII conversion table */ +/* from a proposed BTL standard April 16, 1979 */ +static unsigned char etoa64[] = +{ + 0040,0240,0241,0242,0243,0244,0245,0246, + 0247,0250,0325,0056,0074,0050,0053,0174, + 0046,0251,0252,0253,0254,0255,0256,0257, + 0260,0261,0041,0044,0052,0051,0073,0176, + 0055,0057,0262,0263,0264,0265,0266,0267, + 0270,0271,0313,0054,0045,0137,0076,0077, + 0272,0273,0274,0275,0276,0277,0300,0301, + 0302,0140,0072,0043,0100,0047,0075,0042, + 0303,0141,0142,0143,0144,0145,0146,0147, + 0150,0151,0304,0305,0306,0307,0310,0311, + 0312,0152,0153,0154,0155,0156,0157,0160, + 0161,0162,0136,0314,0315,0316,0317,0320, + 0321,0345,0163,0164,0165,0166,0167,0170, + 0171,0172,0322,0323,0324,0133,0326,0327, + 0330,0331,0332,0333,0334,0335,0336,0337, + 0340,0341,0342,0343,0344,0135,0346,0347, + 0173,0101,0102,0103,0104,0105,0106,0107, + 0110,0111,0350,0351,0352,0353,0354,0355, + 0175,0112,0113,0114,0115,0116,0117,0120, + 0121,0122,0356,0357,0360,0361,0362,0363, + 0134,0237,0123,0124,0125,0126,0127,0130, + 0131,0132,0364,0365,0366,0367,0370,0371, + 0060,0061,0062,0063,0064,0065,0066,0067, + 0070,0071,0372,0373,0374,0375,0376,0377 +}; + + int +main(int argc, char *argv[]) +{ + FILE *fp, *fpo; + int c, e, p = 0, relseek = 1, negseek = 0, revert = 0; + int cols = 0, colsgiven = 0, nonzero = 0, autoskip = 0, hextype = HEX_NORMAL; + int capitalize = 0, decimal_offset = 0; + int ebcdic = 0; + int octspergrp = -1; /* number of octets grouped in output */ + int grplen; /* total chars per octet group */ + long length = -1, n = 0, seekoff = 0; + unsigned long displayoff = 0; + static char l[LLEN+1]; /* static because it may be too big for stack */ + char *pp; + char *varname = NULL; + int addrlen = 9; + +#ifdef AMIGA + /* This program doesn't work when started from the Workbench */ + if (argc == 0) + exit(1); +#endif + + pname = argv[0]; + for (pp = pname; *pp; ) + if (*pp++ == PATH_SEP) + pname = pp; +#ifdef FILE_SEP + for (pp = pname; *pp; pp++) + if (*pp == FILE_SEP) + { + *pp = '\0'; + break; + } +#endif + + while (argc >= 2) + { + pp = argv[1] + (!STRNCMP(argv[1], "--", 2) && argv[1][2]); + if (!STRNCMP(pp, "-a", 2)) autoskip = 1 - autoskip; + else if (!STRNCMP(pp, "-b", 2)) hextype = HEX_BITS; + else if (!STRNCMP(pp, "-e", 2)) hextype = HEX_LITTLEENDIAN; + else if (!STRNCMP(pp, "-u", 2)) hexx = hexxa + 16; + else if (!STRNCMP(pp, "-p", 2)) hextype = HEX_POSTSCRIPT; + else if (!STRNCMP(pp, "-i", 2)) hextype = HEX_CINCLUDE; + else if (!STRNCMP(pp, "-C", 2)) capitalize = 1; + else if (!STRNCMP(pp, "-d", 2)) decimal_offset = 1; + else if (!STRNCMP(pp, "-r", 2)) revert++; + else if (!STRNCMP(pp, "-E", 2)) ebcdic++; + else if (!STRNCMP(pp, "-v", 2)) + { + fprintf(stderr, "%s%s\n", version, osver); + exit(0); + } + else if (!STRNCMP(pp, "-c", 2)) + { + if (pp[2] && !STRNCMP("apitalize", pp + 2, 9)) + capitalize = 1; + else if (pp[2] && STRNCMP("ols", pp + 2, 3)) + { + colsgiven = 1; + cols = (int)strtol(pp + 2, NULL, 0); + } + else + { + if (!argv[2]) + exit_with_usage(); + colsgiven = 1; + cols = (int)strtol(argv[2], NULL, 0); + argv++; + argc--; + } + } + else if (!STRNCMP(pp, "-g", 2)) + { + if (pp[2] && STRNCMP("roup", pp + 2, 4)) + octspergrp = (int)strtol(pp + 2, NULL, 0); + else + { + if (!argv[2]) + exit_with_usage(); + octspergrp = (int)strtol(argv[2], NULL, 0); + argv++; + argc--; + } + } + else if (!STRNCMP(pp, "-o", 2)) + { + int reloffset = 0; + int negoffset = 0; + if (pp[2] && STRNCMP("ffset", pp + 2, 5)) + displayoff = strtoul(pp + 2, NULL, 0); + else + { + if (!argv[2]) + exit_with_usage(); + + if (argv[2][0] == '+') + reloffset++; + if (argv[2][reloffset] == '-') + negoffset++; + + if (negoffset) + displayoff = ULONG_MAX - strtoul(argv[2] + reloffset+negoffset, NULL, 0) + 1; + else + displayoff = strtoul(argv[2] + reloffset+negoffset, NULL, 0); + + argv++; + argc--; + } + } + else if (!STRNCMP(pp, "-s", 2)) + { + relseek = 0; + negseek = 0; + if (pp[2] && STRNCMP("kip", pp+2, 3) && STRNCMP("eek", pp+2, 3)) + { +#ifdef TRY_SEEK + if (pp[2] == '+') + relseek++; + if (pp[2+relseek] == '-') + negseek++; +#endif + seekoff = strtol(pp + 2+relseek+negseek, (char **)NULL, 0); + } + else + { + if (!argv[2]) + exit_with_usage(); +#ifdef TRY_SEEK + if (argv[2][0] == '+') + relseek++; + if (argv[2][relseek] == '-') + negseek++; +#endif + seekoff = strtol(argv[2] + relseek+negseek, (char **)NULL, 0); + argv++; + argc--; + } + } + else if (!STRNCMP(pp, "-l", 2)) + { + if (pp[2] && STRNCMP("en", pp + 2, 2)) + length = strtol(pp + 2, (char **)NULL, 0); + else + { + if (!argv[2]) + exit_with_usage(); + length = strtol(argv[2], (char **)NULL, 0); + argv++; + argc--; + } + } + else if (!STRNCMP(pp, "-n", 2)) + { + if (pp[2] && STRNCMP("ame", pp + 2, 3)) + varname = pp + 2; + else + { + if (!argv[2]) + exit_with_usage(); + varname = argv[2]; + argv++; + argc--; + } + } + else if (!strcmp(pp, "--")) /* end of options */ + { + argv++; + argc--; + break; + } + else if (pp[0] == '-' && pp[1]) /* unknown option */ + exit_with_usage(); + else + break; /* not an option */ + + argv++; /* advance to next argument */ + argc--; + } + + if (!colsgiven || (!cols && hextype != HEX_POSTSCRIPT)) + switch (hextype) + { + case HEX_POSTSCRIPT: cols = 30; break; + case HEX_CINCLUDE: cols = 12; break; + case HEX_BITS: cols = 6; break; + case HEX_NORMAL: + case HEX_LITTLEENDIAN: + default: cols = 16; break; + } + + if (octspergrp < 0) + switch (hextype) + { + case HEX_BITS: octspergrp = 1; break; + case HEX_NORMAL: octspergrp = 2; break; + case HEX_LITTLEENDIAN: octspergrp = 4; break; + case HEX_POSTSCRIPT: + case HEX_CINCLUDE: + default: octspergrp = 0; break; + } + + if ((hextype == HEX_POSTSCRIPT && cols < 0) || + (hextype != HEX_POSTSCRIPT && cols < 1) || + ((hextype == HEX_NORMAL || hextype == HEX_BITS || hextype == HEX_LITTLEENDIAN) + && (cols > COLS))) + { + fprintf(stderr, "%s: invalid number of columns (max. %d).\n", pname, COLS); + exit(1); + } + + if (octspergrp < 1 || octspergrp > cols) + octspergrp = cols; + else if (hextype == HEX_LITTLEENDIAN && (octspergrp & (octspergrp-1))) + error_exit(1, "number of octets per group must be a power of 2 with -e."); + + if (argc > 3) + exit_with_usage(); + + if (argc == 1 || (argv[1][0] == '-' && !argv[1][1])) + BIN_ASSIGN(fp = stdin, !revert); + else + { + if ((fp = fopen(argv[1], BIN_READ(!revert))) == NULL) + { + fprintf(stderr,"%s: ", pname); + perror(argv[1]); + return 2; + } + } + + if (argc < 3 || (argv[2][0] == '-' && !argv[2][1])) + BIN_ASSIGN(fpo = stdout, revert); + else + { + int fd; + int mode = revert ? O_WRONLY : (O_TRUNC|O_WRONLY); + + if (((fd = OPEN(argv[2], mode | BIN_CREAT(revert), 0666)) < 0) || + (fpo = fdopen(fd, BIN_WRITE(revert))) == NULL) + { + fprintf(stderr, "%s: ", pname); + perror(argv[2]); + return 3; + } + rewind(fpo); + } + + if (revert) + { + if (hextype && (hextype != HEX_POSTSCRIPT)) + error_exit(-1, "Sorry, cannot revert this type of hexdump"); + return huntype(fp, fpo, cols, hextype, + negseek ? -seekoff : seekoff); + } + + if (seekoff || negseek || !relseek) + { +#ifdef TRY_SEEK + if (relseek) + e = fseek(fp, negseek ? -seekoff : seekoff, SEEK_CUR); + else + e = fseek(fp, negseek ? -seekoff : seekoff, + negseek ? SEEK_END : SEEK_SET); + if (e < 0 && negseek) + error_exit(4, "Sorry, cannot seek."); + if (e >= 0) + seekoff = ftell(fp); + else +#endif + { + long s = seekoff; + + while (s--) + if (getc_or_die(fp) == EOF) + { + error_exit(4, "Sorry, cannot seek."); + } + } + } + + if (hextype == HEX_CINCLUDE) + { + /* A user-set variable name overrides fp == stdin */ + if (varname == NULL && fp != stdin) + varname = argv[1]; + + if (varname != NULL) + { + FPRINTF_OR_DIE((fpo, "unsigned char %s", isdigit((int)varname[0]) ? "__" : "")); + for (e = 0; (c = varname[e]) != 0; e++) + putc_or_die(isalnum(c) ? CONDITIONAL_CAPITALIZE(c) : '_', fpo); + fputs_or_die("[] = {\n", fpo); + } + + p = 0; + while ((length < 0 || p < length) && (c = getc_or_die(fp)) != EOF) + { + FPRINTF_OR_DIE((fpo, (hexx == hexxa) ? "%s0x%02x" : "%s0X%02X", + (p % cols) ? ", " : (!p ? " " : ",\n "), c)); + p++; + } + + if (p) + fputs_or_die("\n", fpo); + + if (varname != NULL) + { + fputs_or_die("};\n", fpo); + FPRINTF_OR_DIE((fpo, "unsigned int %s", isdigit((int)varname[0]) ? "__" : "")); + for (e = 0; (c = varname[e]) != 0; e++) + putc_or_die(isalnum(c) ? CONDITIONAL_CAPITALIZE(c) : '_', fpo); + FPRINTF_OR_DIE((fpo, "_%s = %d;\n", capitalize ? "LEN" : "len", p)); + } + + fclose_or_die(fp, fpo); + return 0; + } + + if (hextype == HEX_POSTSCRIPT) + { + p = cols; + while ((length < 0 || n < length) && (e = getc_or_die(fp)) != EOF) + { + putc_or_die(hexx[(e >> 4) & 0xf], fpo); + putc_or_die(hexx[e & 0xf], fpo); + n++; + if (cols > 0 && !--p) + { + putc_or_die('\n', fpo); + p = cols; + } + } + if (cols == 0 || p < cols) + putc_or_die('\n', fpo); + fclose_or_die(fp, fpo); + return 0; + } + + /* hextype: HEX_NORMAL or HEX_BITS or HEX_LITTLEENDIAN */ + + if (hextype != HEX_BITS) + grplen = octspergrp + octspergrp + 1; /* chars per octet group */ + else /* hextype == HEX_BITS */ + grplen = 8 * octspergrp + 1; + + while ((length < 0 || n < length) && (e = getc_or_die(fp)) != EOF) + { + int x; + + if (p == 0) + { + addrlen = sprintf(l, decimal_offset ? "%08ld:" : "%08lx:", + ((unsigned long)(n + seekoff + displayoff))); + for (c = addrlen; c < LLEN; l[c++] = ' ') + ; + } + x = hextype == HEX_LITTLEENDIAN ? p ^ (octspergrp-1) : p; + c = addrlen + 1 + (grplen * x) / octspergrp; + if (hextype == HEX_NORMAL || hextype == HEX_LITTLEENDIAN) + { + l[c] = hexx[(e >> 4) & 0xf]; + l[++c] = hexx[e & 0xf]; + } + else /* hextype == HEX_BITS */ + { + int i; + for (i = 7; i >= 0; i--) + l[c++] = (e & (1 << i)) ? '1' : '0'; + } + if (e) + nonzero++; + if (ebcdic) + e = (e < 64) ? '.' : etoa64[e-64]; + /* When changing this update definition of LLEN above. */ + if (hextype == HEX_LITTLEENDIAN) + /* last group will be fully used, round up */ + c = grplen * ((cols + octspergrp - 1) / octspergrp); + else + c = (grplen * cols - 1) / octspergrp; + c += addrlen + 3 + p; + l[c++] = +#ifdef __MVS__ + (e >= 64) +#else + (e > 31 && e < 127) +#endif + ? e : '.'; + n++; + if (++p == cols) + { + l[c] = '\n'; + l[++c] = '\0'; + xxdline(fpo, l, autoskip ? nonzero : 1); + nonzero = 0; + p = 0; + } + } + if (p) + { + l[c] = '\n'; + l[++c] = '\0'; + xxdline(fpo, l, 1); + } + else if (autoskip) + xxdline(fpo, l, -1); /* last chance to flush out suppressed lines */ + + fclose_or_die(fp, fpo); + return 0; +} + +/* vi:set ts=8 sw=4 sts=2 cino+={2 cino+=n-2 : */ diff --git a/src/univalue/CMakeLists.txt b/src/univalue/CMakeLists.txt new file mode 100644 index 0000000000..c16a48a7ea --- /dev/null +++ b/src/univalue/CMakeLists.txt @@ -0,0 +1,8 @@ +add_library(univalue STATIC + lib/univalue.cpp + lib/univalue_get.cpp + lib/univalue_read.cpp + lib/univalue_write.cpp +) + +target_include_directories(univalue PUBLIC "${CMAKE_CURRENT_SOURCE_DIR}/include") diff --git a/src/util.h b/src/util.h index c21fc77d29..43480785ee 100644 --- a/src/util.h +++ b/src/util.h @@ -208,19 +208,6 @@ inline std::string leftTrim(std::string src, char chr) return src; } -inline int64_t GetPerformanceCounter() -{ - int64_t nCounter = 0; -#ifdef WIN32 - QueryPerformanceCounter((LARGE_INTEGER*)&nCounter); -#else - timeval t; - gettimeofday(&t, nullptr); - nCounter = (int64_t) t.tv_sec * 1000000 + t.tv_usec; -#endif - return nCounter; -} - /** Median filter over a stream of values. * Returns the median of the last N numbers */ diff --git a/src/validation.cpp b/src/validation.cpp index 775dc60181..ee78d93bf5 100644 --- a/src/validation.cpp +++ b/src/validation.cpp @@ -1752,7 +1752,10 @@ bool AddToBlockIndex(CBlock& block, unsigned int nFile, unsigned int nBlockPos, } bool CheckBlock(const CBlock& block, int height1, bool fCheckPOW, bool fCheckMerkleRoot, bool fCheckSig, bool fLoadingIndex) + EXCLUSIVE_LOCKS_REQUIRED(cs_main) { + AssertLockHeld(cs_main); + // Allow the genesis block to pass. if(block.hashPrevBlock.IsNull() && block.GetHash(true) == (fTestNet ? hashGenesisBlockTestNet : hashGenesisBlock)) diff --git a/src/wallet/wallet.cpp b/src/wallet/wallet.cpp index c12db1915e..df01975f35 100644 --- a/src/wallet/wallet.cpp +++ b/src/wallet/wallet.cpp @@ -322,12 +322,12 @@ bool CWallet::EncryptWallet(const SecureString& strWalletPassphrase) CKeyingMaterial vMasterKey; vMasterKey.resize(WALLET_CRYPTO_KEY_SIZE); - GetStrongRandBytes(vMasterKey.data(), WALLET_CRYPTO_KEY_SIZE); + GetStrongRandBytes(vMasterKey); CMasterKey kMasterKey(nDerivationMethodIndex); kMasterKey.vchSalt.resize(WALLET_CRYPTO_SALT_SIZE); - GetStrongRandBytes(kMasterKey.vchSalt.data(), WALLET_CRYPTO_SALT_SIZE); + GetStrongRandBytes(kMasterKey.vchSalt); CCrypter crypter; int64_t nStartTime = GetTimeMillis(); @@ -2178,7 +2178,7 @@ bool CWallet::CreateTransaction(const vector >& vecSend, } // Insert change output at random position in the transaction: - vector::iterator position = wtxNew.vout.begin()+GetRandInt(wtxNew.vout.size()); + vector::iterator position = wtxNew.vout.begin() + GetRand(wtxNew.vout.size()); wtxNew.vout.insert(position, CTxOut(nChange, scriptChange)); } else diff --git a/test/lint/lint-format-strings.sh b/test/lint/lint-format-strings.sh index 88919940e0..12035971d0 100755 --- a/test/lint/lint-format-strings.sh +++ b/test/lint/lint-format-strings.sh @@ -35,7 +35,7 @@ if ! python3 -m doctest test/lint/lint-format-strings.py; then fi for S in "${FUNCTION_NAMES_AND_NUMBER_OF_LEADING_ARGUMENTS[@]}"; do IFS="," read -r FUNCTION_NAME SKIP_ARGUMENTS <<< "${S}" - for MATCHING_FILE in $(git grep --full-name -l "${FUNCTION_NAME}" -- "*.c" "*.cpp" "*.h" | sort | grep -vE "^src/(leveldb|secp256k1|tinyformat|univalue|bdb53|test/fuzz/strprintf.cpp)"); do + for MATCHING_FILE in $(git grep --full-name -l "${FUNCTION_NAME}" -- "*.c" "*.cpp" "*.h" | sort | grep -vE "^src/(leveldb|secp256k1|tinyformat|univalue|bdb53|test/fuzz/strprintf.cpp|test/xxd/xxd.c)"); do MATCHING_FILES+=("${MATCHING_FILE}") done if ! test/lint/lint-format-strings.py --skip-arguments "${SKIP_ARGUMENTS}" "${FUNCTION_NAME}" "${MATCHING_FILES[@]}"; then diff --git a/test/lint/lint-spelling.ignore-words.txt b/test/lint/lint-spelling.ignore-words.txt index efdbcc0d46..9cbee6e709 100644 --- a/test/lint/lint-spelling.ignore-words.txt +++ b/test/lint/lint-spelling.ignore-words.txt @@ -1,3 +1,4 @@ +MSDOS hights mor mut