小组成员:姚凯文(kevinyao0901),姜嘉琪
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

453 lines
16 KiB

leveldb: Fix PosixWritableFile::Sync() on Apple systems. Apple doesn't follow POSIX specifications for fsync(). Instead, fsync() guarantees to flush the buffer cache to the device, which means the data will survive kernel panics, but may not survive power outages. Applications that need stronger guarantees (like databases) need to use fcntl(F_FULLFSYNC). This CL switches PosixWritableFile::Sync() to get the stronger guarantees on Apple systems. The improved implementation follows the same principles as SQLite [1] and node.js [2]. Research for the fcntl() to fsync() fallback strategy: Apple's released source code at https://opensource.apple.com/ shows at least three different error codes being returned when a filesystem does not support F_FULLFSYNC. fcntl() is implemented in xnu-4903.221.2 in bsd/kern/kern_descrip.c, where it delegates to fcntl_nocancel(). The documentation for fcntl_nocancel() mentions error codes for some operations, but does not include F_FULLFSYNC. The F_FULLSYNC branch in fcntl_nocancel() calls VNOP_IOCTL(_, F_FULLSYNC, NULL, 0, _), whose return value sets the error code. VNOP_IOCTL() is implemented in bsd/vfs/kpi_vfs.c and calls the ioctl function in the vnode's operation vector. The per-filesystem function names follow the pattern _vnop_ioctl() for all the instances in opensource code: {hfs,msdosfs,nfs,ntfs,smbfs,webdav,zfs}_vnop_ioctl(). hfs-407.30.1, msdosfs-229.200.3, and nfs in xnu-4903.221.2 handle F_FULLFSYNC. ntfs-94.200.1 and smb-759.40.1 do not handle F_FULLFSYNC, and the default branch returns ENOSUP. webdav-380.200.1 also does not handle F_FULLFSYNC, but the default branch returns EINVAL. zfs-59 also does not handle F_FULLSYNC, and its default branch returns ENOTTY. From a different angle, Apple's ntfs-94.200.1 includes utility code that uses fcntl(F_FULLFSYNC) and falls back to fsync() just like we do, supporting the hypothesis that there is no good way to detect lack of F_FULLFSYNC support. Also, Apple's fcntl() man page [3] does not mention a way to detect lack of F_FULLFSYNC support. [1] https://www.sqlite.org/src/doc/trunk/src/os_unix.c [2] https://github.com/libuv/libuv/blob/master/src/unix/fs.c [3] https://developer.apple.com/library/archive/documentatiVon/System/Conceptual/ManPages_iPhoneOS/man2/fcntl.2.html Tested: https://travis-ci.org/pwnall/leveldb/builds/477318498 TAP global presubmit ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=228593729
6 years ago
Fix snapshot compaction bug Closes google/leveldb#320 During compaction it was possible that records from a block b1=(l1,u1) would be pushed down from level i to level i+1. If there is a block b2=(l2,u2) at level i with k1 = user_key(u1) = user_key(l2) then a subsequent search for k1 will yield the record l2 which has a smaller sequence number than u1 because the sort order for records sorts increasing by user key but decreaing by sequence number. This change add a call to a new function AddBoundaryInputs to SetupOtherInputs. AddBoundaryInputs searches for a block b2 matching the criteria above and adds it to the set of files to be compacted. Whenever AddBoundaryInputs is called it is important that the compaction fileset in level i+1 (known as c->inputs_[1] in the code) be recomputed. Each call to AddBoundaryInputs is followed by a call to GetOverlappingInputs. SetupOtherInputs is called on both manual and automated compaction passes. It is called for both level zero and for levels greater than 0. The original change posted in https://github.com/google/leveldb/pull/339 has been modified to also include changed made by Chris Mumford<cmumford@google.com> in https://github.com/cmumford/leveldb/commit/4b72cb14f8da2aab12451c24b8e205aff686e9dc 1. Releasing snapshots during test cleanup to avoid memory leak warnings. 2. Refactored test to use testutil.h to be in line with other issue tests and to create the test database in the correct temporary location. 3. Added copyright banner. Otherwise, just minor formatting and limiting character width to 80 characters. Additionally the change was rebased on top of current master and changes previously made to the Makefile were ported to the CMakeLists.txt. Testing Done: A test program (issue320_test) was constructed that performs mutations while snapshots are active. issue320_test fails without this bug fix after 64k writes. It passes with this bug fix. It was run with 200M writes and passed. Unit tests were written for the new function that was added to the code. Make test was run and seen to pass. Signed-off-by: Richard Cole <richcole@amazon.com>
9 years ago
Add move constructor to Status. This will result in smaller code generation when Status instances are passed around. Benchmarks don't indicate a significant change either way. CPU: 48 * Intel(R) Xeon(R) CPU E5-2690 v3 @ 2.60GHz CPUCache: 30720 KB Keys: 16 bytes each Values: 100 bytes each (50 bytes after compression) Entries: 1000000 RawSize: 110.6 MB (estimated) FileSize: 62.9 MB (estimated) Baseline: fillseq : 3.589 micros/op; 30.8 MB/s fillsync : 4165.299 micros/op; 0.0 MB/s (1000 ops) fillrandom : 5.864 micros/op; 18.9 MB/s overwrite : 7.830 micros/op; 14.1 MB/s readrandom : 5.534 micros/op; (1000000 of 1000000 found) readrandom : 4.292 micros/op; (1000000 of 1000000 found) readseq : 0.312 micros/op; 354.1 MB/s readreverse : 0.501 micros/op; 220.8 MB/s compact : 886211.000 micros/op; readrandom : 3.518 micros/op; (1000000 of 1000000 found) readseq : 0.251 micros/op; 441.2 MB/s readreverse : 0.456 micros/op; 242.4 MB/s fill100K : 1329.723 micros/op; 71.7 MB/s (1000 ops) crc32c : 1.976 micros/op; 1976.7 MB/s (4K per op) snappycomp : 4.705 micros/op; 830.2 MB/s (output: 55.1%) snappyuncomp : 0.958 micros/op; 4079.1 MB/s acquireload : 0.727 micros/op; (each op is 1000 loads) New: fillseq : 3.129 micros/op; 35.4 MB/s fillsync : 2748.099 micros/op; 0.0 MB/s (1000 ops) fillrandom : 5.394 micros/op; 20.5 MB/s overwrite : 7.253 micros/op; 15.3 MB/s readrandom : 5.655 micros/op; (1000000 of 1000000 found) readrandom : 4.425 micros/op; (1000000 of 1000000 found) readseq : 0.298 micros/op; 371.3 MB/s readreverse : 0.508 micros/op; 217.9 MB/s compact : 885842.000 micros/op; readrandom : 3.545 micros/op; (1000000 of 1000000 found) readseq : 0.252 micros/op; 438.2 MB/s readreverse : 0.425 micros/op; 260.2 MB/s fill100K : 1418.347 micros/op; 67.2 MB/s (1000 ops) crc32c : 1.987 micros/op; 1966.0 MB/s (4K per op) snappycomp : 4.767 micros/op; 819.4 MB/s (output: 55.1%) snappyuncomp : 0.916 micros/op; 4264.9 MB/s acquireload : 0.665 micros/op; (each op is 1000 loads) ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=194002392
6 years ago
  1. # Copyright 2017 The LevelDB Authors. All rights reserved.
  2. # Use of this source code is governed by a BSD-style license that can be
  3. # found in the LICENSE file. See the AUTHORS file for names of contributors.
  4. cmake_minimum_required(VERSION 3.9)
  5. # Keep the version below in sync with the one in db.h
  6. project(leveldb VERSION 1.22.0 LANGUAGES C CXX)
  7. # This project can use C11, but will gracefully decay down to C89.
  8. set(CMAKE_C_STANDARD 11)
  9. set(CMAKE_C_STANDARD_REQUIRED OFF)
  10. set(CMAKE_C_EXTENSIONS OFF)
  11. # This project requires C++11.
  12. set(CMAKE_CXX_STANDARD 11)
  13. set(CMAKE_CXX_STANDARD_REQUIRED ON)
  14. set(CMAKE_CXX_EXTENSIONS OFF)
  15. if (WIN32)
  16. set(LEVELDB_PLATFORM_NAME LEVELDB_PLATFORM_WINDOWS)
  17. # TODO(cmumford): Make UNICODE configurable for Windows.
  18. add_definitions(-D_UNICODE -DUNICODE)
  19. else (WIN32)
  20. set(LEVELDB_PLATFORM_NAME LEVELDB_PLATFORM_POSIX)
  21. endif (WIN32)
  22. option(LEVELDB_BUILD_TESTS "Build LevelDB's unit tests" ON)
  23. option(LEVELDB_BUILD_BENCHMARKS "Build LevelDB's benchmarks" ON)
  24. option(LEVELDB_INSTALL "Install LevelDB's header and library" ON)
  25. include(TestBigEndian)
  26. test_big_endian(LEVELDB_IS_BIG_ENDIAN)
  27. include(CheckIncludeFile)
  28. check_include_file("unistd.h" HAVE_UNISTD_H)
  29. include(CheckLibraryExists)
  30. check_library_exists(crc32c crc32c_value "" HAVE_CRC32C)
  31. check_library_exists(snappy snappy_compress "" HAVE_SNAPPY)
  32. check_library_exists(tcmalloc malloc "" HAVE_TCMALLOC)
  33. include(CheckCXXSymbolExists)
  34. # Using check_cxx_symbol_exists() instead of check_c_symbol_exists() because
  35. # we're including the header from C++, and feature detection should use the same
  36. # compiler language that the project will use later. Principles aside, some
  37. # versions of do not expose fdatasync() in <unistd.h> in standard C mode
  38. # (-std=c11), but do expose the function in standard C++ mode (-std=c++11).
  39. check_cxx_symbol_exists(fdatasync "unistd.h" HAVE_FDATASYNC)
  40. check_cxx_symbol_exists(F_FULLFSYNC "fcntl.h" HAVE_FULLFSYNC)
  41. include(CheckCXXSourceCompiles)
  42. set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wstrict-prototypes")
  43. # Test whether -Wthread-safety is available. See
  44. # https://clang.llvm.org/docs/ThreadSafetyAnalysis.html
  45. # -Werror is necessary because unknown attributes only generate warnings.
  46. set(OLD_CMAKE_REQUIRED_FLAGS ${CMAKE_REQUIRED_FLAGS})
  47. list(APPEND CMAKE_REQUIRED_FLAGS -Werror -Wthread-safety)
  48. check_cxx_source_compiles("
  49. struct __attribute__((lockable)) Lock {
  50. void Acquire() __attribute__((exclusive_lock_function()));
  51. void Release() __attribute__((unlock_function()));
  52. };
  53. struct ThreadSafeType {
  54. Lock lock_;
  55. int data_ __attribute__((guarded_by(lock_)));
  56. };
  57. int main() { return 0; }
  58. " HAVE_CLANG_THREAD_SAFETY)
  59. set(CMAKE_REQUIRED_FLAGS ${OLD_CMAKE_REQUIRED_FLAGS})
  60. # Test whether C++17 __has_include is available.
  61. check_cxx_source_compiles("
  62. #if defined(__has_include) && __has_include(<string>)
  63. #include <string>
  64. #endif
  65. int main() { std::string str; return 0; }
  66. " HAVE_CXX17_HAS_INCLUDE)
  67. set(LEVELDB_PUBLIC_INCLUDE_DIR "include/leveldb")
  68. set(LEVELDB_PORT_CONFIG_DIR "include/port")
  69. configure_file(
  70. "${PROJECT_SOURCE_DIR}/port/port_config.h.in"
  71. "${PROJECT_BINARY_DIR}/${LEVELDB_PORT_CONFIG_DIR}/port_config.h"
  72. )
  73. include_directories(
  74. "${PROJECT_BINARY_DIR}/include"
  75. "${PROJECT_SOURCE_DIR}"
  76. )
  77. if(BUILD_SHARED_LIBS)
  78. # Only export LEVELDB_EXPORT symbols from the shared library.
  79. add_compile_options(-fvisibility=hidden)
  80. endif(BUILD_SHARED_LIBS)
  81. add_library(leveldb "")
  82. target_sources(leveldb
  83. PRIVATE
  84. "${PROJECT_BINARY_DIR}/${LEVELDB_PORT_CONFIG_DIR}/port_config.h"
  85. "${PROJECT_SOURCE_DIR}/db/builder.cc"
  86. "${PROJECT_SOURCE_DIR}/db/builder.h"
  87. "${PROJECT_SOURCE_DIR}/db/c.cc"
  88. "${PROJECT_SOURCE_DIR}/db/db_impl.cc"
  89. "${PROJECT_SOURCE_DIR}/db/db_impl.h"
  90. "${PROJECT_SOURCE_DIR}/db/db_iter.cc"
  91. "${PROJECT_SOURCE_DIR}/db/db_iter.h"
  92. "${PROJECT_SOURCE_DIR}/db/dbformat.cc"
  93. "${PROJECT_SOURCE_DIR}/db/dbformat.h"
  94. "${PROJECT_SOURCE_DIR}/db/dumpfile.cc"
  95. "${PROJECT_SOURCE_DIR}/db/filename.cc"
  96. "${PROJECT_SOURCE_DIR}/db/filename.h"
  97. "${PROJECT_SOURCE_DIR}/db/log_format.h"
  98. "${PROJECT_SOURCE_DIR}/db/log_reader.cc"
  99. "${PROJECT_SOURCE_DIR}/db/log_reader.h"
  100. "${PROJECT_SOURCE_DIR}/db/log_writer.cc"
  101. "${PROJECT_SOURCE_DIR}/db/log_writer.h"
  102. "${PROJECT_SOURCE_DIR}/db/memtable.cc"
  103. "${PROJECT_SOURCE_DIR}/db/memtable.h"
  104. "${PROJECT_SOURCE_DIR}/db/repair.cc"
  105. "${PROJECT_SOURCE_DIR}/db/skiplist.h"
  106. "${PROJECT_SOURCE_DIR}/db/snapshot.h"
  107. "${PROJECT_SOURCE_DIR}/db/table_cache.cc"
  108. "${PROJECT_SOURCE_DIR}/db/table_cache.h"
  109. "${PROJECT_SOURCE_DIR}/db/version_edit.cc"
  110. "${PROJECT_SOURCE_DIR}/db/version_edit.h"
  111. "${PROJECT_SOURCE_DIR}/db/version_set.cc"
  112. "${PROJECT_SOURCE_DIR}/db/version_set.h"
  113. "${PROJECT_SOURCE_DIR}/db/write_batch_internal.h"
  114. "${PROJECT_SOURCE_DIR}/db/write_batch.cc"
  115. "${PROJECT_SOURCE_DIR}/port/port_stdcxx.h"
  116. "${PROJECT_SOURCE_DIR}/port/port.h"
  117. "${PROJECT_SOURCE_DIR}/port/thread_annotations.h"
  118. "${PROJECT_SOURCE_DIR}/table/block_builder.cc"
  119. "${PROJECT_SOURCE_DIR}/table/block_builder.h"
  120. "${PROJECT_SOURCE_DIR}/table/block.cc"
  121. "${PROJECT_SOURCE_DIR}/table/block.h"
  122. "${PROJECT_SOURCE_DIR}/table/filter_block.cc"
  123. "${PROJECT_SOURCE_DIR}/table/filter_block.h"
  124. "${PROJECT_SOURCE_DIR}/table/format.cc"
  125. "${PROJECT_SOURCE_DIR}/table/format.h"
  126. "${PROJECT_SOURCE_DIR}/table/iterator_wrapper.h"
  127. "${PROJECT_SOURCE_DIR}/table/iterator.cc"
  128. "${PROJECT_SOURCE_DIR}/table/merger.cc"
  129. "${PROJECT_SOURCE_DIR}/table/merger.h"
  130. "${PROJECT_SOURCE_DIR}/table/table_builder.cc"
  131. "${PROJECT_SOURCE_DIR}/table/table.cc"
  132. "${PROJECT_SOURCE_DIR}/table/two_level_iterator.cc"
  133. "${PROJECT_SOURCE_DIR}/table/two_level_iterator.h"
  134. "${PROJECT_SOURCE_DIR}/util/arena.cc"
  135. "${PROJECT_SOURCE_DIR}/util/arena.h"
  136. "${PROJECT_SOURCE_DIR}/util/bloom.cc"
  137. "${PROJECT_SOURCE_DIR}/util/cache.cc"
  138. "${PROJECT_SOURCE_DIR}/util/coding.cc"
  139. "${PROJECT_SOURCE_DIR}/util/coding.h"
  140. "${PROJECT_SOURCE_DIR}/util/comparator.cc"
  141. "${PROJECT_SOURCE_DIR}/util/crc32c.cc"
  142. "${PROJECT_SOURCE_DIR}/util/crc32c.h"
  143. "${PROJECT_SOURCE_DIR}/util/env.cc"
  144. "${PROJECT_SOURCE_DIR}/util/filter_policy.cc"
  145. "${PROJECT_SOURCE_DIR}/util/hash.cc"
  146. "${PROJECT_SOURCE_DIR}/util/hash.h"
  147. "${PROJECT_SOURCE_DIR}/util/logging.cc"
  148. "${PROJECT_SOURCE_DIR}/util/logging.h"
  149. "${PROJECT_SOURCE_DIR}/util/mutexlock.h"
  150. "${PROJECT_SOURCE_DIR}/util/no_destructor.h"
  151. "${PROJECT_SOURCE_DIR}/util/options.cc"
  152. "${PROJECT_SOURCE_DIR}/util/random.h"
  153. "${PROJECT_SOURCE_DIR}/util/status.cc"
  154. # Only CMake 3.3+ supports PUBLIC sources in targets exported by "install".
  155. $<$<VERSION_GREATER:CMAKE_VERSION,3.2>:PUBLIC>
  156. "${LEVELDB_PUBLIC_INCLUDE_DIR}/c.h"
  157. "${LEVELDB_PUBLIC_INCLUDE_DIR}/cache.h"
  158. "${LEVELDB_PUBLIC_INCLUDE_DIR}/comparator.h"
  159. "${LEVELDB_PUBLIC_INCLUDE_DIR}/db.h"
  160. "${LEVELDB_PUBLIC_INCLUDE_DIR}/dumpfile.h"
  161. "${LEVELDB_PUBLIC_INCLUDE_DIR}/env.h"
  162. "${LEVELDB_PUBLIC_INCLUDE_DIR}/export.h"
  163. "${LEVELDB_PUBLIC_INCLUDE_DIR}/filter_policy.h"
  164. "${LEVELDB_PUBLIC_INCLUDE_DIR}/iterator.h"
  165. "${LEVELDB_PUBLIC_INCLUDE_DIR}/options.h"
  166. "${LEVELDB_PUBLIC_INCLUDE_DIR}/slice.h"
  167. "${LEVELDB_PUBLIC_INCLUDE_DIR}/status.h"
  168. "${LEVELDB_PUBLIC_INCLUDE_DIR}/table_builder.h"
  169. "${LEVELDB_PUBLIC_INCLUDE_DIR}/table.h"
  170. "${LEVELDB_PUBLIC_INCLUDE_DIR}/write_batch.h"
  171. )
  172. if (WIN32)
  173. target_sources(leveldb
  174. PRIVATE
  175. "${PROJECT_SOURCE_DIR}/util/env_windows.cc"
  176. "${PROJECT_SOURCE_DIR}/util/windows_logger.h"
  177. )
  178. else (WIN32)
  179. target_sources(leveldb
  180. PRIVATE
  181. "${PROJECT_SOURCE_DIR}/util/env_posix.cc"
  182. "${PROJECT_SOURCE_DIR}/util/posix_logger.h"
  183. )
  184. endif (WIN32)
  185. # MemEnv is not part of the interface and could be pulled to a separate library.
  186. target_sources(leveldb
  187. PRIVATE
  188. "${PROJECT_SOURCE_DIR}/helpers/memenv/memenv.cc"
  189. "${PROJECT_SOURCE_DIR}/helpers/memenv/memenv.h"
  190. )
  191. target_include_directories(leveldb
  192. PUBLIC
  193. $<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/include>
  194. $<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}>
  195. )
  196. set_target_properties(leveldb
  197. PROPERTIES VERSION ${PROJECT_VERSION} SOVERSION ${PROJECT_VERSION_MAJOR})
  198. target_compile_definitions(leveldb
  199. PRIVATE
  200. # Used by include/export.h when building shared libraries.
  201. LEVELDB_COMPILE_LIBRARY
  202. # Used by port/port.h.
  203. ${LEVELDB_PLATFORM_NAME}=1
  204. )
  205. if (NOT HAVE_CXX17_HAS_INCLUDE)
  206. target_compile_definitions(leveldb
  207. PRIVATE
  208. LEVELDB_HAS_PORT_CONFIG_H=1
  209. )
  210. endif(NOT HAVE_CXX17_HAS_INCLUDE)
  211. if(BUILD_SHARED_LIBS)
  212. target_compile_definitions(leveldb
  213. PUBLIC
  214. # Used by include/export.h.
  215. LEVELDB_SHARED_LIBRARY
  216. )
  217. endif(BUILD_SHARED_LIBS)
  218. if(HAVE_CLANG_THREAD_SAFETY)
  219. target_compile_options(leveldb
  220. PUBLIC
  221. -Werror -Wthread-safety)
  222. endif(HAVE_CLANG_THREAD_SAFETY)
  223. if(HAVE_CRC32C)
  224. target_link_libraries(leveldb crc32c)
  225. endif(HAVE_CRC32C)
  226. if(HAVE_SNAPPY)
  227. target_link_libraries(leveldb snappy)
  228. endif(HAVE_SNAPPY)
  229. if(HAVE_TCMALLOC)
  230. target_link_libraries(leveldb tcmalloc)
  231. endif(HAVE_TCMALLOC)
  232. # Needed by port_stdcxx.h
  233. find_package(Threads REQUIRED)
  234. target_link_libraries(leveldb Threads::Threads)
  235. add_executable(leveldbutil
  236. "${PROJECT_SOURCE_DIR}/db/leveldbutil.cc"
  237. )
  238. target_link_libraries(leveldbutil leveldb)
  239. if(LEVELDB_BUILD_TESTS)
  240. enable_testing()
  241. function(leveldb_test test_file)
  242. get_filename_component(test_target_name "${test_file}" NAME_WE)
  243. add_executable("${test_target_name}" "")
  244. target_sources("${test_target_name}"
  245. PRIVATE
  246. "${PROJECT_BINARY_DIR}/${LEVELDB_PORT_CONFIG_DIR}/port_config.h"
  247. "${PROJECT_SOURCE_DIR}/util/testharness.cc"
  248. "${PROJECT_SOURCE_DIR}/util/testharness.h"
  249. "${PROJECT_SOURCE_DIR}/util/testutil.cc"
  250. "${PROJECT_SOURCE_DIR}/util/testutil.h"
  251. "${test_file}"
  252. )
  253. target_link_libraries("${test_target_name}" leveldb)
  254. target_compile_definitions("${test_target_name}"
  255. PRIVATE
  256. ${LEVELDB_PLATFORM_NAME}=1
  257. )
  258. if (NOT HAVE_CXX17_HAS_INCLUDE)
  259. target_compile_definitions("${test_target_name}"
  260. PRIVATE
  261. LEVELDB_HAS_PORT_CONFIG_H=1
  262. )
  263. endif(NOT HAVE_CXX17_HAS_INCLUDE)
  264. add_test(NAME "${test_target_name}" COMMAND "${test_target_name}")
  265. endfunction(leveldb_test)
  266. leveldb_test("${PROJECT_SOURCE_DIR}/db/c_test.c")
  267. leveldb_test("${PROJECT_SOURCE_DIR}/db/fault_injection_test.cc")
  268. leveldb_test("${PROJECT_SOURCE_DIR}/issues/issue178_test.cc")
  269. leveldb_test("${PROJECT_SOURCE_DIR}/issues/issue200_test.cc")
  270. leveldb_test("${PROJECT_SOURCE_DIR}/issues/issue320_test.cc")
  271. leveldb_test("${PROJECT_SOURCE_DIR}/util/env_test.cc")
  272. leveldb_test("${PROJECT_SOURCE_DIR}/util/status_test.cc")
  273. leveldb_test("${PROJECT_SOURCE_DIR}/util/no_destructor_test.cc")
  274. if(NOT BUILD_SHARED_LIBS)
  275. leveldb_test("${PROJECT_SOURCE_DIR}/db/autocompact_test.cc")
  276. leveldb_test("${PROJECT_SOURCE_DIR}/db/corruption_test.cc")
  277. leveldb_test("${PROJECT_SOURCE_DIR}/db/db_test.cc")
  278. leveldb_test("${PROJECT_SOURCE_DIR}/db/dbformat_test.cc")
  279. leveldb_test("${PROJECT_SOURCE_DIR}/db/filename_test.cc")
  280. leveldb_test("${PROJECT_SOURCE_DIR}/db/log_test.cc")
  281. leveldb_test("${PROJECT_SOURCE_DIR}/db/recovery_test.cc")
  282. leveldb_test("${PROJECT_SOURCE_DIR}/db/skiplist_test.cc")
  283. leveldb_test("${PROJECT_SOURCE_DIR}/db/version_edit_test.cc")
  284. leveldb_test("${PROJECT_SOURCE_DIR}/db/version_set_test.cc")
  285. leveldb_test("${PROJECT_SOURCE_DIR}/db/write_batch_test.cc")
  286. leveldb_test("${PROJECT_SOURCE_DIR}/helpers/memenv/memenv_test.cc")
  287. leveldb_test("${PROJECT_SOURCE_DIR}/table/filter_block_test.cc")
  288. leveldb_test("${PROJECT_SOURCE_DIR}/table/table_test.cc")
  289. leveldb_test("${PROJECT_SOURCE_DIR}/util/arena_test.cc")
  290. leveldb_test("${PROJECT_SOURCE_DIR}/util/bloom_test.cc")
  291. leveldb_test("${PROJECT_SOURCE_DIR}/util/cache_test.cc")
  292. leveldb_test("${PROJECT_SOURCE_DIR}/util/coding_test.cc")
  293. leveldb_test("${PROJECT_SOURCE_DIR}/util/crc32c_test.cc")
  294. leveldb_test("${PROJECT_SOURCE_DIR}/util/hash_test.cc")
  295. leveldb_test("${PROJECT_SOURCE_DIR}/util/logging_test.cc")
  296. # TODO(costan): This test also uses
  297. # "${PROJECT_SOURCE_DIR}/util/env_{posix|windows}_test_helper.h"
  298. if (WIN32)
  299. leveldb_test("${PROJECT_SOURCE_DIR}/util/env_windows_test.cc")
  300. else (WIN32)
  301. leveldb_test("${PROJECT_SOURCE_DIR}/util/env_posix_test.cc")
  302. endif (WIN32)
  303. endif(NOT BUILD_SHARED_LIBS)
  304. endif(LEVELDB_BUILD_TESTS)
  305. if(LEVELDB_BUILD_BENCHMARKS)
  306. function(leveldb_benchmark bench_file)
  307. get_filename_component(bench_target_name "${bench_file}" NAME_WE)
  308. add_executable("${bench_target_name}" "")
  309. target_sources("${bench_target_name}"
  310. PRIVATE
  311. "${PROJECT_BINARY_DIR}/${LEVELDB_PORT_CONFIG_DIR}/port_config.h"
  312. "${PROJECT_SOURCE_DIR}/util/histogram.cc"
  313. "${PROJECT_SOURCE_DIR}/util/histogram.h"
  314. "${PROJECT_SOURCE_DIR}/util/testharness.cc"
  315. "${PROJECT_SOURCE_DIR}/util/testharness.h"
  316. "${PROJECT_SOURCE_DIR}/util/testutil.cc"
  317. "${PROJECT_SOURCE_DIR}/util/testutil.h"
  318. "${bench_file}"
  319. )
  320. target_link_libraries("${bench_target_name}" leveldb)
  321. target_compile_definitions("${bench_target_name}"
  322. PRIVATE
  323. ${LEVELDB_PLATFORM_NAME}=1
  324. )
  325. if (NOT HAVE_CXX17_HAS_INCLUDE)
  326. target_compile_definitions("${bench_target_name}"
  327. PRIVATE
  328. LEVELDB_HAS_PORT_CONFIG_H=1
  329. )
  330. endif(NOT HAVE_CXX17_HAS_INCLUDE)
  331. endfunction(leveldb_benchmark)
  332. if(NOT BUILD_SHARED_LIBS)
  333. leveldb_benchmark("${PROJECT_SOURCE_DIR}/benchmarks/db_bench.cc")
  334. endif(NOT BUILD_SHARED_LIBS)
  335. check_library_exists(sqlite3 sqlite3_open "" HAVE_SQLITE3)
  336. if(HAVE_SQLITE3)
  337. leveldb_benchmark("${PROJECT_SOURCE_DIR}/benchmarks/db_bench_sqlite3.cc")
  338. target_link_libraries(db_bench_sqlite3 sqlite3)
  339. endif(HAVE_SQLITE3)
  340. # check_library_exists is insufficient here because the library names have
  341. # different manglings when compiled with clang or gcc, at least when installed
  342. # with Homebrew on Mac.
  343. set(OLD_CMAKE_REQURED_LIBRARIES ${CMAKE_REQUIRED_LIBRARIES})
  344. list(APPEND CMAKE_REQUIRED_LIBRARIES kyotocabinet)
  345. check_cxx_source_compiles("
  346. #include <kcpolydb.h>
  347. int main() {
  348. kyotocabinet::TreeDB* db = new kyotocabinet::TreeDB();
  349. delete db;
  350. return 0;
  351. }
  352. " HAVE_KYOTOCABINET)
  353. set(CMAKE_REQUIRED_LIBRARIES ${OLD_CMAKE_REQURED_LIBRARIES})
  354. if(HAVE_KYOTOCABINET)
  355. leveldb_benchmark("${PROJECT_SOURCE_DIR}/benchmarks/db_bench_tree_db.cc")
  356. target_link_libraries(db_bench_tree_db kyotocabinet)
  357. endif(HAVE_KYOTOCABINET)
  358. endif(LEVELDB_BUILD_BENCHMARKS)
  359. if(LEVELDB_INSTALL)
  360. include(GNUInstallDirs)
  361. install(TARGETS leveldb
  362. EXPORT leveldbTargets
  363. RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
  364. LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
  365. ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
  366. )
  367. install(
  368. FILES
  369. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/c.h"
  370. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/cache.h"
  371. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/comparator.h"
  372. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/db.h"
  373. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/dumpfile.h"
  374. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/env.h"
  375. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/export.h"
  376. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/filter_policy.h"
  377. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/iterator.h"
  378. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/options.h"
  379. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/slice.h"
  380. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/status.h"
  381. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/table_builder.h"
  382. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/table.h"
  383. "${PROJECT_SOURCE_DIR}/${LEVELDB_PUBLIC_INCLUDE_DIR}/write_batch.h"
  384. DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/leveldb
  385. )
  386. include(CMakePackageConfigHelpers)
  387. write_basic_package_version_file(
  388. "${PROJECT_BINARY_DIR}/leveldbConfigVersion.cmake"
  389. COMPATIBILITY SameMajorVersion
  390. )
  391. install(
  392. EXPORT leveldbTargets
  393. NAMESPACE leveldb::
  394. DESTINATION "${CMAKE_INSTALL_LIBDIR}/cmake/leveldb"
  395. )
  396. install(
  397. FILES
  398. "${PROJECT_SOURCE_DIR}/cmake/leveldbConfig.cmake"
  399. "${PROJECT_BINARY_DIR}/leveldbConfigVersion.cmake"
  400. DESTINATION "${CMAKE_INSTALL_LIBDIR}/cmake/leveldb"
  401. )
  402. endif(LEVELDB_INSTALL)