2018-05-04 07:28:18 +08:00
|
|
|
set(FDB_C_SRCS
|
|
|
|
fdb_c.cpp
|
|
|
|
foundationdb/fdb_c.h
|
|
|
|
ThreadCleanup.cpp)
|
|
|
|
|
|
|
|
file(MAKE_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/foundationdb)
|
|
|
|
|
2019-02-06 11:29:01 +08:00
|
|
|
set(asm_file ${CMAKE_CURRENT_BINARY_DIR}/fdb_c.g.S)
|
|
|
|
|
2021-11-11 11:05:38 +08:00
|
|
|
set(os "linux")
|
|
|
|
set(cpu "intel")
|
2018-05-04 07:28:18 +08:00
|
|
|
if(APPLE)
|
2021-11-11 11:05:38 +08:00
|
|
|
set(os "osx")
|
2019-02-06 11:29:01 +08:00
|
|
|
elseif(WIN32)
|
2021-11-11 11:05:38 +08:00
|
|
|
set(os "windows")
|
2019-02-06 11:29:01 +08:00
|
|
|
set(asm_file ${CMAKE_CURRENT_BINARY_DIR}/fdb_c.g.asm)
|
2021-11-11 11:05:38 +08:00
|
|
|
endif()
|
|
|
|
|
|
|
|
if(CMAKE_SYSTEM_PROCESSOR MATCHES "aarch64" OR CMAKE_SYSTEM_PROCESSOR MATCHES "arm64")
|
|
|
|
set(cpu "aarch64")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
set(IS_ARM_MAC NO)
|
|
|
|
if(APPLE AND CMAKE_SYSTEM_PROCESSOR STREQUAL "arm64")
|
|
|
|
set(IS_ARM_MAC YES)
|
2018-05-04 07:28:18 +08:00
|
|
|
endif()
|
|
|
|
|
2019-02-06 11:29:01 +08:00
|
|
|
add_custom_command(OUTPUT ${asm_file} ${CMAKE_CURRENT_BINARY_DIR}/fdb_c_function_pointers.g.h
|
2021-11-11 11:05:38 +08:00
|
|
|
COMMAND $<TARGET_FILE:Python::Interpreter> ${CMAKE_CURRENT_SOURCE_DIR}/generate_asm.py ${os} ${cpu}
|
2018-05-04 07:28:18 +08:00
|
|
|
${CMAKE_CURRENT_SOURCE_DIR}/fdb_c.cpp
|
2019-02-06 11:29:01 +08:00
|
|
|
${asm_file}
|
2018-05-04 07:28:18 +08:00
|
|
|
${CMAKE_CURRENT_BINARY_DIR}/fdb_c_function_pointers.g.h
|
|
|
|
DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/generate_asm.py ${CMAKE_CURRENT_SOURCE_DIR}/fdb_c.cpp
|
|
|
|
COMMENT "Generate C bindings")
|
2019-02-06 11:29:01 +08:00
|
|
|
add_custom_target(fdb_c_generated DEPENDS ${asm_file}
|
2018-05-04 07:28:18 +08:00
|
|
|
${CMAKE_CURRENT_BINARY_DIR}/fdb_c_function_pointers.g.h)
|
|
|
|
|
2019-02-05 13:25:10 +08:00
|
|
|
vexillographer_compile(TARGET fdb_c_options LANG c OUT ${CMAKE_CURRENT_BINARY_DIR}/foundationdb/fdb_c_options.g.h
|
|
|
|
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/foundationdb/fdb_c_options.g.h)
|
2018-05-04 07:28:18 +08:00
|
|
|
|
2018-05-05 06:35:27 +08:00
|
|
|
include(GenerateExportHeader)
|
|
|
|
|
2019-02-18 09:54:54 +08:00
|
|
|
if(OPEN_FOR_IDE)
|
|
|
|
add_library(fdb_c OBJECT ${FDB_C_SRCS} ${asm_file})
|
|
|
|
else()
|
|
|
|
add_library(fdb_c SHARED ${FDB_C_SRCS} ${asm_file})
|
2019-03-21 11:27:10 +08:00
|
|
|
strip_debug_symbols(fdb_c)
|
2019-02-18 09:54:54 +08:00
|
|
|
endif()
|
2018-05-04 07:28:18 +08:00
|
|
|
add_dependencies(fdb_c fdb_c_generated fdb_c_options)
|
2021-09-22 02:35:55 +08:00
|
|
|
add_dependencies(fdbclient fdb_c_options)
|
|
|
|
add_dependencies(fdbclient_sampling fdb_c_options)
|
2019-05-24 02:46:31 +08:00
|
|
|
target_link_libraries(fdb_c PUBLIC $<BUILD_INTERFACE:fdbclient>)
|
2020-02-13 11:35:12 +08:00
|
|
|
if(APPLE)
|
|
|
|
set(symbols ${CMAKE_CURRENT_BINARY_DIR}/fdb_c.symbols)
|
|
|
|
add_custom_command(OUTPUT ${symbols}
|
|
|
|
COMMAND $<TARGET_FILE:Python::Interpreter> ${CMAKE_CURRENT_SOURCE_DIR}/symbolify.py
|
|
|
|
${CMAKE_CURRENT_SOURCE_DIR}/foundationdb/fdb_c.h
|
|
|
|
${symbols}
|
|
|
|
DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/symbolify.py ${CMAKE_CURRENT_SOURCE_DIR}/foundationdb/fdb_c.h
|
|
|
|
COMMENT "Generate exported_symbols_list")
|
|
|
|
add_custom_target(exported_symbols_list DEPENDS ${symbols})
|
|
|
|
add_dependencies(fdb_c exported_symbols_list)
|
2020-02-19 02:49:43 +08:00
|
|
|
target_link_options(fdb_c PRIVATE "LINKER:-no_weak_exports,-exported_symbols_list,${symbols}")
|
2020-02-13 11:35:12 +08:00
|
|
|
elseif(WIN32)
|
|
|
|
else()
|
2021-12-08 02:51:10 +08:00
|
|
|
target_link_options(fdb_c PRIVATE "LINKER:--version-script=${CMAKE_CURRENT_SOURCE_DIR}/fdb_c.map,-z,nodelete,-z,noexecstack")
|
2020-02-13 11:35:12 +08:00
|
|
|
endif()
|
2018-05-05 06:35:27 +08:00
|
|
|
target_include_directories(fdb_c PUBLIC
|
2019-05-24 02:15:39 +08:00
|
|
|
$<BUILD_INTERFACE:${CMAKE_CURRENT_BINARY_DIR}>
|
|
|
|
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}>
|
|
|
|
$<BUILD_INTERFACE:${CMAKE_CURRENT_BINARY_DIR}/foundationdb>)
|
2019-02-06 11:29:01 +08:00
|
|
|
if(WIN32)
|
|
|
|
enable_language(ASM_MASM)
|
|
|
|
set_property(SOURCE ${asm_file} PROPERTY LANGUAGE ASM_MASM)
|
|
|
|
endif()
|
2019-02-10 07:13:25 +08:00
|
|
|
|
2021-11-11 11:05:38 +08:00
|
|
|
# The tests don't build on windows and ARM macs
|
|
|
|
# doctest doesn't seem to compile on ARM macs, we should
|
|
|
|
# check later whether this works
|
|
|
|
if(NOT WIN32 AND NOT IS_ARM_MAC)
|
2019-06-07 07:58:11 +08:00
|
|
|
set(MAKO_SRCS
|
|
|
|
test/mako/mako.c
|
|
|
|
test/mako/mako.h
|
|
|
|
test/mako/utils.c
|
2019-09-18 04:04:54 +08:00
|
|
|
test/mako/utils.h)
|
2020-10-01 08:38:58 +08:00
|
|
|
add_subdirectory(test/unit/third_party)
|
2020-10-03 13:01:57 +08:00
|
|
|
find_package(Threads REQUIRED)
|
2020-10-01 08:38:58 +08:00
|
|
|
set(UNIT_TEST_SRCS
|
|
|
|
test/unit/unit_tests.cpp
|
|
|
|
test/unit/fdb_api.cpp
|
|
|
|
test/unit/fdb_api.hpp)
|
2019-06-07 07:58:11 +08:00
|
|
|
|
2021-06-05 11:19:19 +08:00
|
|
|
set(UNIT_TEST_VERSION_510_SRCS test/unit/unit_tests_version_510.cpp)
|
2021-08-03 10:29:56 +08:00
|
|
|
set(TRACE_PARTIAL_FILE_SUFFIX_TEST_SRCS test/unit/trace_partial_file_suffix_test.cpp)
|
2021-09-29 12:59:43 +08:00
|
|
|
set(DISCONNECTED_TIMEOUT_UNIT_TEST_SRCS
|
|
|
|
test/unit/disconnected_timeout_tests.cpp
|
|
|
|
test/unit/fdb_api.cpp
|
|
|
|
test/unit/fdb_api.hpp)
|
2021-06-05 11:19:19 +08:00
|
|
|
|
2019-02-18 11:29:33 +08:00
|
|
|
if(OPEN_FOR_IDE)
|
|
|
|
add_library(fdb_c_performance_test OBJECT test/performance_test.c test/test.h)
|
|
|
|
add_library(fdb_c_ryw_benchmark OBJECT test/ryw_benchmark.c test/test.h)
|
2019-06-29 01:15:37 +08:00
|
|
|
add_library(fdb_c_txn_size_test OBJECT test/txn_size_test.c test/test.h)
|
2019-06-07 07:58:11 +08:00
|
|
|
add_library(mako OBJECT ${MAKO_SRCS})
|
2020-10-01 08:38:58 +08:00
|
|
|
add_library(fdb_c_setup_tests OBJECT test/unit/setup_tests.cpp)
|
|
|
|
add_library(fdb_c_unit_tests OBJECT ${UNIT_TEST_SRCS})
|
2021-06-05 11:19:19 +08:00
|
|
|
add_library(fdb_c_unit_tests_version_510 OBJECT ${UNIT_TEST_VERSION_510_SRCS})
|
2021-08-03 10:29:56 +08:00
|
|
|
add_library(trace_partial_file_suffix_test OBJECT ${TRACE_PARTIAL_FILE_SUFFIX_TEST_SRCS})
|
2021-09-29 12:59:43 +08:00
|
|
|
add_library(disconnected_timeout_unit_tests OBJECT ${DISCONNECTED_TIMEOUT_UNIT_TEST_SRCS})
|
2019-02-18 11:29:33 +08:00
|
|
|
else()
|
|
|
|
add_executable(fdb_c_performance_test test/performance_test.c test/test.h)
|
|
|
|
add_executable(fdb_c_ryw_benchmark test/ryw_benchmark.c test/test.h)
|
2019-06-29 01:15:37 +08:00
|
|
|
add_executable(fdb_c_txn_size_test test/txn_size_test.c test/test.h)
|
2019-06-07 07:58:11 +08:00
|
|
|
add_executable(mako ${MAKO_SRCS})
|
2020-10-01 08:38:58 +08:00
|
|
|
add_executable(fdb_c_setup_tests test/unit/setup_tests.cpp)
|
|
|
|
add_executable(fdb_c_unit_tests ${UNIT_TEST_SRCS})
|
2021-06-05 11:19:19 +08:00
|
|
|
add_executable(fdb_c_unit_tests_version_510 ${UNIT_TEST_VERSION_510_SRCS})
|
2021-08-03 10:29:56 +08:00
|
|
|
add_executable(trace_partial_file_suffix_test ${TRACE_PARTIAL_FILE_SUFFIX_TEST_SRCS})
|
2021-09-29 12:59:43 +08:00
|
|
|
add_executable(disconnected_timeout_unit_tests ${DISCONNECTED_TIMEOUT_UNIT_TEST_SRCS})
|
2019-03-21 11:27:10 +08:00
|
|
|
strip_debug_symbols(fdb_c_performance_test)
|
|
|
|
strip_debug_symbols(fdb_c_ryw_benchmark)
|
2019-06-29 01:15:37 +08:00
|
|
|
strip_debug_symbols(fdb_c_txn_size_test)
|
2019-02-18 11:29:33 +08:00
|
|
|
endif()
|
2019-02-11 13:29:03 +08:00
|
|
|
target_link_libraries(fdb_c_performance_test PRIVATE fdb_c)
|
|
|
|
target_link_libraries(fdb_c_ryw_benchmark PRIVATE fdb_c)
|
2019-06-29 01:15:37 +08:00
|
|
|
target_link_libraries(fdb_c_txn_size_test PRIVATE fdb_c)
|
2020-10-01 08:38:58 +08:00
|
|
|
|
2020-10-03 09:57:07 +08:00
|
|
|
add_dependencies(fdb_c_setup_tests doctest)
|
|
|
|
add_dependencies(fdb_c_unit_tests doctest)
|
2021-08-03 10:29:56 +08:00
|
|
|
add_dependencies(fdb_c_unit_tests_version_510 doctest)
|
2021-09-29 12:59:43 +08:00
|
|
|
add_dependencies(disconnected_timeout_unit_tests doctest)
|
2020-10-01 08:38:58 +08:00
|
|
|
target_include_directories(fdb_c_setup_tests PUBLIC ${DOCTEST_INCLUDE_DIR})
|
|
|
|
target_include_directories(fdb_c_unit_tests PUBLIC ${DOCTEST_INCLUDE_DIR})
|
2021-06-05 11:19:19 +08:00
|
|
|
target_include_directories(fdb_c_unit_tests_version_510 PUBLIC ${DOCTEST_INCLUDE_DIR})
|
2021-09-29 12:59:43 +08:00
|
|
|
target_include_directories(disconnected_timeout_unit_tests PUBLIC ${DOCTEST_INCLUDE_DIR})
|
2020-10-01 08:38:58 +08:00
|
|
|
target_link_libraries(fdb_c_setup_tests PRIVATE fdb_c Threads::Threads)
|
|
|
|
target_link_libraries(fdb_c_unit_tests PRIVATE fdb_c Threads::Threads)
|
2021-06-05 11:19:19 +08:00
|
|
|
target_link_libraries(fdb_c_unit_tests_version_510 PRIVATE fdb_c Threads::Threads)
|
2021-08-03 10:29:56 +08:00
|
|
|
target_link_libraries(trace_partial_file_suffix_test PRIVATE fdb_c Threads::Threads)
|
2021-09-29 12:59:43 +08:00
|
|
|
target_link_libraries(disconnected_timeout_unit_tests PRIVATE fdb_c Threads::Threads)
|
2020-10-01 08:38:58 +08:00
|
|
|
|
2019-06-07 05:53:26 +08:00
|
|
|
# do not set RPATH for mako
|
|
|
|
set_property(TARGET mako PROPERTY SKIP_BUILD_RPATH TRUE)
|
2019-06-05 08:35:47 +08:00
|
|
|
target_link_libraries(mako PRIVATE fdb_c)
|
2020-10-01 08:38:58 +08:00
|
|
|
|
2021-01-26 08:09:32 +08:00
|
|
|
if(NOT OPEN_FOR_IDE)
|
|
|
|
# Make sure that fdb_c.h is compatible with c90
|
|
|
|
add_executable(fdb_c90_test test/fdb_c90_test.c)
|
|
|
|
set_property(TARGET fdb_c90_test PROPERTY C_STANDARD 90)
|
|
|
|
target_compile_options(fdb_c90_test PRIVATE -Wall -Wextra -Wpedantic -Werror)
|
|
|
|
target_link_libraries(fdb_c90_test PRIVATE fdb_c)
|
|
|
|
endif()
|
2020-12-23 02:30:07 +08:00
|
|
|
|
2021-08-24 07:25:19 +08:00
|
|
|
if(OPEN_FOR_IDE)
|
|
|
|
set(FDB_C_TARGET $<TARGET_OBJECTS:fdb_c>)
|
|
|
|
else()
|
|
|
|
set(FDB_C_TARGET $<TARGET_FILE:fdb_c>)
|
|
|
|
endif()
|
2022-01-08 09:04:16 +08:00
|
|
|
add_custom_command(
|
|
|
|
TARGET fdb_c POST_BUILD
|
|
|
|
COMMAND ${CMAKE_COMMAND} -E copy ${FDB_C_TARGET} ${CMAKE_BINARY_DIR}/lib/libfdb_c_external.so)
|
2021-02-05 11:37:52 +08:00
|
|
|
add_custom_command(
|
|
|
|
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/libfdb_c.so
|
2021-08-24 07:25:19 +08:00
|
|
|
COMMAND ${CMAKE_COMMAND} -E copy ${FDB_C_TARGET} ${CMAKE_CURRENT_BINARY_DIR}/libfdb_c.so
|
2021-02-05 11:37:52 +08:00
|
|
|
DEPENDS fdb_c
|
2021-02-05 09:24:45 +08:00
|
|
|
COMMENT "Copy libfdb_c to use as external client for test")
|
|
|
|
add_custom_target(external_client DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/libfdb_c.so)
|
|
|
|
add_dependencies(fdb_c_unit_tests external_client)
|
2021-09-30 04:30:48 +08:00
|
|
|
add_dependencies(disconnected_timeout_unit_tests external_client)
|
2021-02-05 09:24:45 +08:00
|
|
|
|
2020-10-01 08:38:58 +08:00
|
|
|
add_fdbclient_test(
|
|
|
|
NAME fdb_c_setup_tests
|
|
|
|
COMMAND $<TARGET_FILE:fdb_c_setup_tests>)
|
|
|
|
add_fdbclient_test(
|
|
|
|
NAME fdb_c_unit_tests
|
|
|
|
COMMAND $<TARGET_FILE:fdb_c_unit_tests>
|
|
|
|
@CLUSTER_FILE@
|
|
|
|
fdb)
|
2021-06-05 11:19:19 +08:00
|
|
|
add_fdbclient_test(
|
|
|
|
NAME fdb_c_unit_tests_version_510
|
|
|
|
COMMAND $<TARGET_FILE:fdb_c_unit_tests_version_510>
|
|
|
|
@CLUSTER_FILE@
|
|
|
|
fdb)
|
2021-08-03 10:29:56 +08:00
|
|
|
add_fdbclient_test(
|
|
|
|
NAME trace_partial_file_suffix_test
|
|
|
|
COMMAND $<TARGET_FILE:trace_partial_file_suffix_test>
|
|
|
|
@CLUSTER_FILE@
|
|
|
|
fdb)
|
2021-02-05 09:24:45 +08:00
|
|
|
add_fdbclient_test(
|
|
|
|
NAME fdb_c_external_client_unit_tests
|
|
|
|
COMMAND $<TARGET_FILE:fdb_c_unit_tests>
|
|
|
|
@CLUSTER_FILE@
|
|
|
|
fdb
|
|
|
|
${CMAKE_CURRENT_BINARY_DIR}/libfdb_c.so
|
|
|
|
)
|
2021-09-29 12:59:43 +08:00
|
|
|
add_unavailable_fdbclient_test(
|
|
|
|
NAME disconnected_timeout_unit_tests
|
|
|
|
COMMAND $<TARGET_FILE:disconnected_timeout_unit_tests>
|
|
|
|
@CLUSTER_FILE@
|
2021-09-29 23:13:34 +08:00
|
|
|
)
|
|
|
|
add_unavailable_fdbclient_test(
|
|
|
|
NAME disconnected_timeout_external_client_unit_tests
|
|
|
|
COMMAND $<TARGET_FILE:disconnected_timeout_unit_tests>
|
|
|
|
@CLUSTER_FILE@
|
2021-09-29 12:59:43 +08:00
|
|
|
${CMAKE_CURRENT_BINARY_DIR}/libfdb_c.so
|
|
|
|
)
|
2019-02-11 13:29:03 +08:00
|
|
|
endif()
|
2019-02-10 07:13:25 +08:00
|
|
|
|
2019-06-06 04:24:06 +08:00
|
|
|
set(c_workloads_srcs
|
|
|
|
test/workloads/workloads.cpp
|
2019-06-08 05:57:56 +08:00
|
|
|
test/workloads/workloads.h
|
|
|
|
test/workloads/SimpleWorkload.cpp)
|
2019-06-06 04:24:06 +08:00
|
|
|
|
|
|
|
if(OPEN_FOR_IDE)
|
|
|
|
add_library(c_workloads OBJECT ${c_workloads_srcs})
|
|
|
|
else()
|
|
|
|
add_library(c_workloads SHARED ${c_workloads_srcs})
|
|
|
|
endif()
|
|
|
|
set_target_properties(c_workloads PROPERTIES
|
|
|
|
LIBRARY_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/share/foundationdb")
|
2019-06-08 05:57:56 +08:00
|
|
|
target_link_libraries(c_workloads PUBLIC fdb_c)
|
2019-06-06 04:24:06 +08:00
|
|
|
|
2021-06-06 05:05:29 +08:00
|
|
|
if (NOT WIN32 AND NOT APPLE AND NOT OPEN_FOR_IDE)
|
|
|
|
target_link_options(c_workloads PRIVATE "LINKER:--version-script=${CMAKE_CURRENT_SOURCE_DIR}/external_workload.map,-z,nodelete")
|
|
|
|
endif()
|
|
|
|
|
2019-02-28 12:17:11 +08:00
|
|
|
# TODO: re-enable once the old vcxproj-based build system is removed.
|
|
|
|
#generate_export_header(fdb_c EXPORT_MACRO_NAME "DLLEXPORT"
|
|
|
|
# EXPORT_FILE_NAME ${CMAKE_CURRENT_BINARY_DIR}/foundationdb/fdb_c_export.h)
|
2019-05-24 03:39:27 +08:00
|
|
|
|
|
|
|
set(targets_export_name "FoundationDB-Client")
|
|
|
|
set(generated_dir "${CMAKE_CURRENT_BINARY_DIR}/generated")
|
|
|
|
set(version_config "${generated_dir}/${targets_export_name}ConfigVersion.cmake")
|
|
|
|
set(project_config "${generated_dir}/${targets_export_name}Config.cmake")
|
|
|
|
include(CMakePackageConfigHelpers)
|
|
|
|
write_basic_package_version_file(
|
|
|
|
"${version_config}" VERSION ${GENERIC_LIB_VERSION} COMPATIBILITY AnyNewerVersion
|
|
|
|
)
|
|
|
|
configure_file("${PROJECT_SOURCE_DIR}/cmake/Config.cmake.in" "${project_config}" @ONLY)
|
|
|
|
|
2019-05-24 02:15:39 +08:00
|
|
|
fdb_install(
|
|
|
|
TARGETS fdb_c
|
2019-05-24 03:39:27 +08:00
|
|
|
EXPORT ${targets_export_name}
|
2019-02-28 12:17:11 +08:00
|
|
|
DESTINATION lib
|
|
|
|
COMPONENT clients)
|
|
|
|
fdb_install(
|
|
|
|
FILES foundationdb/fdb_c.h
|
|
|
|
${CMAKE_CURRENT_BINARY_DIR}/foundationdb/fdb_c_options.g.h
|
|
|
|
${CMAKE_SOURCE_DIR}/fdbclient/vexillographer/fdb.options
|
2019-05-24 01:43:15 +08:00
|
|
|
DESTINATION include
|
|
|
|
DESTINATION_SUFFIX /foundationdb
|
|
|
|
COMPONENT clients)
|
2019-05-24 02:46:31 +08:00
|
|
|
fdb_install(
|
2019-05-24 03:39:27 +08:00
|
|
|
FILES "${project_config}" "${version_config}"
|
|
|
|
DESTINATION lib
|
|
|
|
DESTINATION_SUFFIX "/cmake/${targets_export_name}"
|
|
|
|
COMPONENT clients)
|
2019-05-24 05:13:07 +08:00
|
|
|
fdb_configure_and_install(
|
|
|
|
FILE "${PROJECT_SOURCE_DIR}/cmake/foundationdb-client.pc.in"
|
|
|
|
DESTINATION lib
|
|
|
|
DESTINATION_SUFFIX "/pkgconfig"
|
|
|
|
COMPONENT clients)
|
2019-05-24 03:39:27 +08:00
|
|
|
fdb_install(
|
|
|
|
EXPORT ${targets_export_name}
|
2019-05-24 02:46:31 +08:00
|
|
|
DESTINATION lib
|
2019-05-24 03:39:27 +08:00
|
|
|
DESTINATION_SUFFIX "/cmake/${targets_export_name}"
|
2019-05-24 02:46:31 +08:00
|
|
|
COMPONENT clients)
|