Skip to content

Commit 02512bc

Browse files
gbottesiGuillaume BOTTESIsean-parent
authored
No longer a header-only library. Added support for src files to make it simpler to integrate with dynamic libraries with hidden symbols.
Added .cpp files to hold thread pool and pre-exit support code to avoid duplicates when built as/in a DLL. Declared pre_exit and at_pre_exit as extern "C" to make them simpler to export from a DLL in an export list. Adding an issue to make it simpler to export the default executor as well. --------- Co-authored-by: Guillaume BOTTESI <[email protected]> Co-authored-by: Sean Parent <[email protected]>
1 parent a97f0a3 commit 02512bc

15 files changed

+212
-120
lines changed

CMakeLists.txt

+4-19
Original file line numberDiff line numberDiff line change
@@ -69,13 +69,7 @@ if((STLAB_MAIN_EXECUTOR STREQUAL "qt6") AND NOT Qt6Core_FOUND)
6969
message(SEND_ERROR "STLAB_MAIN_EXECUTOR is set to \"qt6\", but a Qt6 installation was not found.")
7070
endif()
7171

72-
#
73-
# stlab has no compiled components. As such, we declare it as an `INTERFACE`
74-
# library, which denotes a collection of target propeties to be applied
75-
# transitively to linking targets. In our case, this ammounts to an include
76-
# directory, compile flags, linking flags, and links to system libraries.
77-
#
78-
add_library(stlab INTERFACE)
72+
add_library(stlab)
7973
add_library(stlab::stlab ALIAS stlab)
8074

8175
#
@@ -84,8 +78,8 @@ add_library(stlab::stlab ALIAS stlab)
8478
# on how the configuration under which this library is being consumed.
8579
#
8680
target_include_directories(stlab INTERFACE
87-
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}>/include
88-
$<BUILD_INTERFACE:${PROJECT_BINARY_DIR}>/include
81+
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/include>
82+
$<BUILD_INTERFACE:${PROJECT_BINARY_DIR}/include>
8983
$<INSTALL_INTERFACE:include>)
9084

9185
#
@@ -98,6 +92,7 @@ target_include_directories(stlab INTERFACE
9892
target_compile_definitions(stlab INTERFACE $<$<CXX_COMPILER_ID:MSVC>:NOMINMAX>)
9993

10094
add_subdirectory(include/stlab)
95+
add_subdirectory(src)
10196

10297
if(NOT STLAB_THREAD_SYSTEM STREQUAL "none")
10398
target_link_libraries(stlab INTERFACE Threads::Threads)
@@ -130,16 +125,6 @@ write_basic_package_version_file(
130125
VERSION ${stlab_VERSION}
131126
COMPATIBILITY SameMajorVersion)
132127

133-
#
134-
# As a header-only library, there are no target components to be installed
135-
# directly (the PUBLIC_HEADER property is not white listed for INTERFACE
136-
# targets for some reason).
137-
#
138-
# However, it is worthwhile export our target description in order to later
139-
# generate a CMake configuration file for consumption by CMake's `find_package`
140-
# intrinsic
141-
#
142-
143128
install(
144129
TARGETS stlab
145130
EXPORT stlabTargets

CMakePresets.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -144,7 +144,8 @@
144144
"cacheVariables": {
145145
"CMAKE_CXX_STANDARD": "17",
146146
"CMAKE_BUILD_TYPE": "DEBUG",
147-
"CMAKE_CXX_INCLUDE_WHAT_YOU_USE": "include-what-you-use;-Xiwyu;--mapping_file=/usr/local/Cellar/include-what-you-use/0.23/libexec/share/include-what-you-use/boost-all.imp"
147+
"CMAKE_CXX_INCLUDE_WHAT_YOU_USE":
148+
"include-what-you-use;-Xiwyu;--mapping_file=/usr/local/Cellar/include-what-you-use/0.23/libexec/share/include-what-you-use/boost-all.imp"
148149
}
149150
}
150151
],
@@ -170,7 +171,7 @@
170171
"description": "",
171172
"displayName": "",
172173
"configurePreset": "ninja-cpp20-debug-thread-undefined",
173-
"output":{
174+
"output": {
174175
"verbosity": "verbose"
175176
}
176177
}

docs/_data/contributors.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,7 @@
4141
"type": "User",
4242
"user_view_type": "public",
4343
"site_admin": false,
44-
"contributions": 167
44+
"contributions": 170
4545
},
4646
{
4747
"login": "fosterbrereton",

docs/include/stlab/pre_exit.hpp/f_at_pre_exit.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ hyde:
88
- function
99
inline:
1010
brief:
11-
- Register a pre-exit handler. The pre-exit-handler may not throw. With C++17 or later it is required to be `noexcept`.
11+
- Register a pre-exit handler. The `pre-exit-handler` may not throw. With C++17 or later it is required to be `noexcept`.
1212
defined_in_file: stlab/pre_exit.hpp
1313
overloads:
1414
void at_pre_exit(stlab::pre_exit_handler):
@@ -19,10 +19,10 @@ hyde:
1919
description: Register the `pre_exit_handler` to be called on [`pre_exit()`](./f_pre_exit.html). The functions are called in the reverse order that they are registered.
2020
inline:
2121
description:
22-
- Register a pre-exit handler. The pre-exit-handler may not throw. With C++17 or later it is required to be `noexcept`.
22+
- Register a pre-exit handler. The `pre-exit-handler` may not throw. With C++17 or later it is required to be `noexcept`.
2323
return: __OPTIONAL__
2424
signature_with_names: void at_pre_exit(stlab::pre_exit_handler f)
2525
namespace:
2626
- stlab
27-
- v1
27+
- v2
2828
---

docs/include/stlab/pre_exit.hpp/f_pre_exit.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,5 +20,5 @@ hyde:
2020
signature_with_names: void pre_exit()
2121
namespace:
2222
- stlab
23-
- v1
23+
- v2
2424
---
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
---
2+
layout: function
3+
title: stlab_at_pre_exit
4+
hyde:
5+
owner: __MISSING__
6+
brief: __INLINED__
7+
tags:
8+
- function
9+
inline:
10+
brief:
11+
- An `extern "C"` vector for `at_pre-exit()` to make it simpler to export the function from a shared library.
12+
defined_in_file: stlab/pre_exit.hpp
13+
overloads:
14+
void stlab_at_pre_exit(stlab::pre_exit_handler):
15+
arguments:
16+
- description: __OPTIONAL__
17+
name: f
18+
type: stlab::pre_exit_handler
19+
description: __INLINED__
20+
inline:
21+
description:
22+
- An `extern "C"` vector for `at_pre-exit()` to make it simpler to export the function from a shared library.
23+
return: __OPTIONAL__
24+
signature_with_names: void stlab_at_pre_exit(stlab::pre_exit_handler f)
25+
namespace:
26+
- stlab
27+
- v2
28+
---
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
---
2+
layout: function
3+
title: stlab_pre_exit
4+
hyde:
5+
owner: __MISSING__
6+
brief: __INLINED__
7+
tags:
8+
- function
9+
inline:
10+
brief:
11+
- An `extern "C"` vector for `pre-exit()` to make it simpler to export the function from a shared library.
12+
defined_in_file: stlab/pre_exit.hpp
13+
overloads:
14+
void stlab_pre_exit():
15+
description: __INLINED__
16+
inline:
17+
description:
18+
- An `extern "C"` vector for `pre-exit()` to make it simpler to export the function from a shared library.
19+
return: __OPTIONAL__
20+
signature_with_names: void stlab_pre_exit()
21+
namespace:
22+
- stlab
23+
- v2
24+
---

docs/tools/docker-tools/README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ This should leave you at a bash prompt that looks like this:
7575
app@fc7590a63ba3:~$
7676
```
7777

78-
The hex number is the docker image container ID and may be different. Going foreward I refer to this as the _docker_ prompt to distinguish it from the _local_ prompt.
78+
The hex number is the docker image container ID and may be different. Going foreword I refer to this as the _docker_ prompt to distinguish it from the _local_ prompt.
7979

8080
```
8181
cd /mnt/host

include/stlab/CMakeLists.txt

+2-2
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ stlab_generate_config_file()
4444
target_sources( stlab INTERFACE
4545
FILE_SET stlab
4646
TYPE HEADERS
47-
BASE_DIRS ${PROJECT_BINARY_DIR}
47+
BASE_DIRS ${PROJECT_BINARY_DIR}/include
4848
FILES
49-
${PROJECT_BINARY_DIR}/stlab/config.hpp
49+
${PROJECT_BINARY_DIR}/include/stlab/config.hpp
5050
)

include/stlab/concurrency/default_executor.hpp

+7-28
Original file line numberDiff line numberDiff line change
@@ -11,25 +11,22 @@
1111

1212
#include <stlab/config.hpp>
1313

14-
#include <stlab/pre_exit.hpp>
15-
16-
#include <stlab/concurrency/set_current_thread_name.hpp>
17-
#include <stlab/concurrency/task.hpp>
18-
1914
#include <cassert>
20-
#include <chrono>
2115
#include <cstdint>
22-
#include <functional>
23-
#include <memory>
2416
#include <type_traits>
2517
#include <utility>
2618

2719
#if STLAB_TASK_SYSTEM(LIBDISPATCH)
2820
#include <dispatch/dispatch.h>
2921
#elif STLAB_TASK_SYSTEM(WINDOWS)
22+
#include <stlab/pre_exit.hpp>
23+
3024
#include <Windows.h>
3125
#include <memory>
3226
#elif STLAB_TASK_SYSTEM(PORTABLE)
27+
#include <stlab/concurrency/set_current_thread_name.hpp>
28+
#include <stlab/concurrency/task.hpp>
29+
3330
#include <algorithm>
3431
#include <atomic>
3532
#include <climits>
@@ -79,17 +76,7 @@ struct group_t {
7976
}
8077
};
8178

82-
inline auto group() -> group_t& {
83-
// Use an immediately executed lambda to atomically register pre-exit handler
84-
// and create the dispatch group.
85-
static group_t g{[] {
86-
at_pre_exit([]() noexcept { // <br>
87-
dispatch_group_wait(g._group, DISPATCH_TIME_FOREVER);
88-
});
89-
return group_t{};
90-
}()};
91-
return g;
92-
}
79+
group_t& group();
9380

9481
template <executor_priority P = executor_priority::medium>
9582
struct executor_type {
@@ -452,15 +439,7 @@ class priority_task_system {
452439
/// Returns an instance of the task system singleton. An immediately executed lambda is used
453440
/// to register the the task system for tear down pre-exit in a thread safe manner.
454441

455-
inline priority_task_system& pts() {
456-
// Uses the `nullptr` constructor with an immediate executed lambda to register the task
457-
// system in a thread safe manner.
458-
static priority_task_system only_task_system{[] {
459-
at_pre_exit([]() noexcept { only_task_system.join(); });
460-
return nullptr;
461-
}()};
462-
return only_task_system;
463-
}
442+
priority_task_system& pts();
464443

465444
#endif
466445

include/stlab/pre_exit.hpp

+11-60
Original file line numberDiff line numberDiff line change
@@ -10,16 +10,10 @@
1010

1111
/**************************************************************************************************/
1212

13-
#include <cassert>
14-
#include <mutex>
15-
#include <vector>
16-
17-
/**************************************************************************************************/
18-
1913
// The namespace for pre_exit cannot be changed without an ABI break. If making an ABI breaking
2014
// change in this file it needs to be done in a way supporting this version as well.
2115
namespace stlab {
22-
inline namespace v1 {
16+
inline namespace v2 {
2317

2418
/**************************************************************************************************/
2519

@@ -30,68 +24,25 @@ using pre_exit_handler = void (*)() noexcept;
3024
using pre_exit_handler = void (*)();
3125
#endif
3226

33-
namespace detail {
34-
35-
/// Thread-safe stack of pre-exit handlers. The stack closes after the last element is popped.
36-
struct _pre_exit_stack_t {
37-
using lock_t = std::unique_lock<std::mutex>;
38-
39-
std::mutex _mutex;
40-
std::vector<pre_exit_handler> _stack;
41-
bool _closed{false};
42-
43-
/// Push an exit handler. Precondition that stack is not closed.
44-
void push(pre_exit_handler f) {
45-
lock_t lock{_mutex};
46-
assert(
47-
!_closed &&
48-
"WARNING: Adding pre-exit handler with `at_pre_exit()` after `pre_exit()` completed.");
49-
_stack.push_back(f);
50-
}
51-
52-
/// Pop one exit handler, returns `nullptr` and closes stack if empty.
53-
auto pop() -> pre_exit_handler {
54-
lock_t lock{_mutex};
55-
if (_stack.empty()) {
56-
assert(!_closed && "WARNING `pre_exit()` invoked more than once.");
57-
_closed = true;
58-
return nullptr;
59-
}
60-
auto result = _stack.back();
61-
_stack.pop_back();
62-
return result;
63-
}
64-
65-
~_pre_exit_stack_t() {
66-
assert(_closed && "WARNING: `pre_exit()` not called before program exit.");
67-
}
68-
};
69-
70-
/// Returns a reference to the pre-exit stack singleton.
71-
inline auto _pre_exit_stack() -> auto& {
72-
static _pre_exit_stack_t _q;
73-
return _q;
74-
}
75-
76-
} // namespace detail
27+
/// An `extern "C"` vector for `pre-exit()` to make it simpler to
28+
/// export the function from a shared library.
29+
extern "C" void stlab_pre_exit();
30+
/// An `extern "C"` vector for `at_pre-exit()` to make it simpler to
31+
/// export the function from a shared library.
32+
extern "C" void stlab_at_pre_exit(pre_exit_handler f);
7733

7834
/// Invoke all registered pre-exit handlers in the reverse order they are registered. It is safe
7935
/// to register additional handlers during this operation. Must be invoked exactly once prior to
8036
/// program exit.
81-
inline void pre_exit() {
82-
auto& _s = detail::_pre_exit_stack();
83-
while (auto f = _s.pop()) {
84-
f();
85-
};
86-
}
37+
inline void pre_exit() { stlab_pre_exit(); }
8738

88-
/// Register a pre-exit handler. The pre-exit-handler may not throw. With C++17 or later it
39+
/// Register a pre-exit handler. The `pre-exit-handler` may not throw. With C++17 or later it
8940
/// is required to be `noexcept`.
90-
inline void at_pre_exit(pre_exit_handler f) { detail::_pre_exit_stack().push(f); }
41+
inline void at_pre_exit(pre_exit_handler f) { stlab_at_pre_exit(f); }
9142

9243
/**************************************************************************************************/
9344

94-
} // namespace v1
45+
} // namespace v2
9546
} // namespace stlab
9647

9748
/**************************************************************************************************/

include/stlab/utility.hpp

+5-3
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,10 @@
1414
#include <stlab/config.hpp>
1515

1616
#include <array>
17-
#include <type_traits>
17+
#include <cstddef> // for size_t
18+
#include <initializer_list> // for initializer_list
19+
#include <type_traits> // for remove_reference, is_same, is_const
20+
#include <utility> // for index_sequence, forward
1821

1922
/**************************************************************************************************/
2023

@@ -76,8 +79,7 @@ template <class Seq, template <std::size_t> class F, std::size_t Index, std::siz
7679
struct index_sequence_transform;
7780

7881
template <class Seq,
79-
template <std::size_t>
80-
class F,
82+
template <std::size_t> class F,
8183
std::size_t Index = 0,
8284
std::size_t Count = Seq::size()>
8385
using index_sequence_transform_t = typename index_sequence_transform<Seq, F, Index, Count>::type;

src/CMakeLists.txt

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
target_sources(stlab PRIVATE pre_exit.cpp concurrency/default_executor.cpp)
2+
target_include_directories(stlab PRIVATE ${CMAKE_SOURCE_DIR}/include ${CMAKE_BINARY_DIR}/include)
3+
target_compile_definitions(stlab PRIVATE $<$<CXX_COMPILER_ID:MSVC>:NOMINMAX>)
4+
5+
if(BUILD_SHARED_LIBS)
6+
set_target_properties(stlab PROPERTIES WINDOWS_EXPORT_ALL_SYMBOLS ON)
7+
endif()

0 commit comments

Comments
 (0)