Skip to content

Commit 8bec449

Browse files
Merge pull request #362 from stlab/develop
1.6.0 release
2 parents c9d5de0 + 314193c commit 8bec449

31 files changed

+649
-438
lines changed

.appveyor.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -54,4 +54,5 @@ build_script:
5454
- cd build
5555
- cmake -G "%CMAKE_TOOLSET%" -D BOOST_ROOT="%BOOST_ROOT%" -D Boost_USE_STATIC_LIBS=ON ..
5656
- cmake --build . --config Release
57-
- ctest -C Release
57+
- set TESTS_ARGUMENTS=--log_level=message
58+
- ctest -C Release --verbose --no-compress-output

.travis/build.sh

+2-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,8 @@ if [ $? -ne 0 ]; then exit 1; fi
4242

4343
if $coverage; then lcov -c -i -b .. -d . -o Coverage.baseline; fi
4444

45-
ctest --output-on-failure -j$NPROC
45+
export TESTS_ARGUMENTS=--log_level=message
46+
ctest -C Release -j$NPROC --verbose --no-compress-output
4647
if [ $? -ne 0 ]; then exit 1; fi
4748

4849
if $coverage; then

CHANGES.md

+5
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,8 @@
1+
## v1.6.0 - 2021 - February 9
2+
- Backport the library to support C++14 and later
3+
- Better auto-configuration - no need for compiler flags.
4+
- Supported for threaded wasm using the portable tasking system (auto-configured).
5+
16
## v1.5.6 - 2021 - February 1
27
- Fixed issues
38
= [#352]:(https://github.com/stlab/libraries/issues/352) Non portable 'warning' preprocessor command

CMakeLists.txt

+11-4
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ endif()
1616

1717
project( stlab VERSION 1.5.6 LANGUAGES CXX )
1818

19-
set(CMAKE_CXX_STANDARD 17)
19+
# Overriden from Conan?
20+
# set(CMAKE_CXX_STANDARD 17)
2021
set(CMAKE_CXX_STANDARD_REQUIRED ON)
2122

2223
include( CTest )
@@ -40,7 +41,7 @@ option( stlab.boost_variant "Prefer Boost::variant to std::variant" OFF )
4041
option( stlab.boost_optional "Prefer Boost::optional to std::optional" OFF )
4142
option( stlab.coroutines "Leverage the coroutine TS in stlab" OFF )
4243

43-
set(stlab.task_system "portable" CACHE STRING "Select the task system (portable|libdispatch|emscripten|pnacl|windows).")
44+
set(stlab.task_system "header" CACHE STRING "Select the task system (header|portable|libdispatch|emscripten|pnacl|windows).")
4445

4546
#
4647
# On apple we have to force the usage of boost.variant, because Apple's
@@ -67,7 +68,7 @@ add_library( stlab::stlab ALIAS stlab )
6768
# features ensures that the corresponding C++ standard flag is populated in
6869
# targets linking to stlab.
6970
#
70-
target_compile_features( stlab INTERFACE cxx_std_17 )
71+
# target_compile_features( stlab INTERFACE cxx_std_14)
7172

7273
#
7374
# The include directory for stlab can be expected to vary between build
@@ -106,6 +107,9 @@ if( EXISTS ${CMAKE_BINARY_DIR}/conanbuildinfo.cmake)
106107
# conanfile.txt
107108
#
108109
conan_basic_setup( TARGETS KEEP_RPATHS )
110+
111+
# REVISIT (sparent) : This should not be necessary but appears to be
112+
set(CMAKE_CXX_STANDARD ${CONAN_SETTINGS_COMPILER_CPPSTD})
109113
endif()
110114

111115
if ( stlab.testing OR stlab.boost_variant OR stlab.boost_optional )
@@ -168,7 +172,10 @@ if (NOT APPLE AND (${stlab.task_system} STREQUAL "libdispatch"))
168172
target_link_libraries(${CMAKE_PROJECT_NAME} INTERFACE libdispatch::libdispatch)
169173
endif()
170174

171-
if (${stlab.task_system} STREQUAL "portable")
175+
if (${stlab.task_system} STREQUAL "header")
176+
# Task System is selected in concurrency/config_task_system.hpp
177+
# Nothing to define.
178+
elseif (${stlab.task_system} STREQUAL "portable")
172179
target_compile_definitions( stlab INTERFACE -DSTLAB_FORCE_TASK_SYSTEM_PORTABLE )
173180
elseif (${stlab.task_system} STREQUAL "libdispatch")
174181
target_compile_definitions( stlab INTERFACE -DSTLAB_FORCE_TASK_SYSTEM_LIBDISPATCH )

build.py

+6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,12 @@
99
filtered_builds = []
1010
for settings, options, env_vars, build_requires, reference in builder.items:
1111

12+
#TODO(fernando): auto and portable Task Systems are disabled until we fix the deadlock
13+
# using portable in Windows.
14+
# opts_auto = copy.deepcopy(options)
15+
# opts_auto["stlab:task_system"] = "auto"
16+
# filtered_builds.append([settings, opts_auto, env_vars, build_requires, reference])
17+
1218
if settings["compiler"] == "clang":
1319
opts_libdispatch = copy.deepcopy(options)
1420
opts_libdispatch["stlab:task_system"] = "libdispatch"

cmake/RunTests.cmake

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
if(NOT DEFINED ENV{TESTS_ARGUMENTS})
2+
set(ENV{TESTS_ARGUMENTS} "")
3+
endif()
4+
execute_process(COMMAND ${TEST_EXECUTABLE} $ENV{TESTS_ARGUMENTS} RESULT_VARIABLE result)
5+
if(NOT "${result}" STREQUAL "0")
6+
message(FATAL_ERROR "Test failed with return value '${result}'")
7+
endif()

conanfile.py

+7-5
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ class StlabLibrariesConan(ConanFile):
2828
"boost_optional": [True, False],
2929
"boost_variant": [True, False],
3030
"coroutines": [True, False],
31-
"task_system": ["portable", "libdispatch", "emscripten", "pnacl", "windows", "auto"],
31+
"task_system": ["header", "portable", "libdispatch", "emscripten", "pnacl", "windows", "auto"],
3232
}
3333

3434
default_options = {
@@ -37,7 +37,7 @@ class StlabLibrariesConan(ConanFile):
3737
"boost_optional": False,
3838
"boost_variant": False,
3939
"coroutines": False,
40-
"task_system": "auto",
40+
"task_system": "header",
4141
}
4242

4343
def _log(self, str):
@@ -109,7 +109,8 @@ def _configure_boost(self):
109109
def _fix_boost_components(self):
110110
if self.settings.os != "Macos": return
111111
if self.settings.compiler != "apple-clang": return
112-
if float(str(self.settings.compiler.version)) >= 12: return
112+
113+
if (float(str(self.settings.compiler.version)) >= 12) and not ((self.settings.compiler.cppstd == "14") or (self.settings.compiler.cppstd == "gnu14")): return
113114

114115
#
115116
# On Apple we have to force the usage of boost.variant, because Apple's implementation of C++17 is not complete.
@@ -171,6 +172,8 @@ def _configure_task_system(self):
171172
def configure(self):
172173
ConanFile.configure(self)
173174

175+
tools.check_min_cppstd(self, "14")
176+
174177
self._fix_boost_components()
175178

176179
if self._use_boost():
@@ -195,11 +198,10 @@ def build(self):
195198
cmake.configure()
196199
cmake.build()
197200
cmake.test(output_on_failure=True)
198-
201+
199202
def package(self):
200203
self.copy("*.hpp")
201204

202205
def imports(self):
203206
self.copy("*.dll", "./bin", "bin")
204207
self.copy("*.dylib", "./bin", "lib")
205-

setup_xcode.sh setup_xcode_cpp14.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
mkdir -p build
44
pushd build
55

6-
conan install .. --build=missing -s build_type=Debug
6+
conan install .. --build=missing -s build_type=Debug -o testing=True -s compiler.cppstd=14
77

88
cmake -GXcode -D CMAKE_BUILD_TYPE=debug -D stlab_testing=ON ..
99

setup_xcode_cpp17.sh

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
#!/bin/bash
2+
3+
mkdir -p build
4+
pushd build
5+
6+
conan install .. --build=missing -s build_type=Debug -o testing=True -s compiler.cppstd=17
7+
8+
cmake -GXcode -D CMAKE_BUILD_TYPE=debug -D stlab_testing=ON ..
9+
10+
popd

stlab/concurrency/channel.hpp

+115-21
Original file line numberDiff line numberDiff line change
@@ -229,15 +229,21 @@ struct invoke_variant_dispatcher {
229229
template <>
230230
struct invoke_variant_dispatcher<1> {
231231
template <typename F, typename T, typename Arg>
232-
static auto invoke_(F&& f, T& t) {
233-
if constexpr (std::is_same<Arg, void>::value) {
234-
return;
235-
} else if constexpr (std::is_same<Arg, detail::avoid_>::value) {
236-
return std::forward<F>(f)();
237-
} else {
238-
return std::forward<F>(f)(std::move(stlab::get<Arg>(std::get<0>(t))));
239-
}
232+
static auto invoke_(F&&, T&) -> std::enable_if_t<std::is_same<Arg, void>::value, void> {
233+
return;
240234
}
235+
template <typename F, typename T, typename Arg>
236+
static auto invoke_(F&& f, T&) -> std::enable_if_t<std::is_same<Arg, detail::avoid_>::value,
237+
decltype(std::forward<F>(f)())> {
238+
return std::forward<F>(f)();
239+
}
240+
template <typename F, typename T, typename Arg>
241+
static auto invoke_(F&& f, T& t) -> std::enable_if_t<
242+
!(std::is_same<Arg, detail::avoid_>::value || std::is_same<Arg, detail::avoid_>::value),
243+
decltype(std::forward<F>(f)(std::move(stlab::get<Arg>(std::get<0>(t)))))> {
244+
return std::forward<F>(f)(std::move(stlab::get<Arg>(std::get<0>(t))));
245+
}
246+
241247
template <typename F, typename T, typename... Args>
242248
static auto invoke(F&& f, T& t) {
243249
return invoke_<F, T, first_t<Args...>>(std::forward<F>(f), t);
@@ -699,7 +705,7 @@ struct downstream<
699705
_data = std::forward<F>(f);
700706
}
701707

702-
void clear() { _data = nullopt; }
708+
void clear() { _data = stlab::nullopt; }
703709

704710
std::size_t size() const { return 1; }
705711

@@ -828,7 +834,7 @@ struct shared_process
828834
if (do_final) {
829835
std::unique_lock<std::mutex> lock(_downstream_mutex);
830836
_downstream.clear(); // This will propagate the close to anything downstream
831-
_process = nullopt;
837+
_process = stlab::nullopt;
832838
}
833839
}
834840

@@ -910,8 +916,14 @@ struct shared_process
910916
return bool(message);
911917
}
912918

919+
/*
920+
REVISIT (sparent) : Next two cases are nearly identical, complicated by the need to
921+
remove constexpr if to support C++14.
922+
*/
923+
913924
template <typename U>
914-
auto step() -> std::enable_if_t<has_process_yield_v<unwrap_reference_t<U>>> {
925+
auto step() -> std::enable_if_t<has_process_yield_v<unwrap_reference_t<U>> &&
926+
!has_process_await_v<unwrap_reference_t<T>, Args...>> {
915927
// in case that the timeout function is just been executed then we have to re-schedule
916928
// the current run
917929
lock_t lock(_timeout_function_control, std::try_to_lock);
@@ -926,12 +938,90 @@ struct shared_process
926938
is done on yield()
927939
*/
928940
try {
929-
if constexpr (has_process_await_v<unwrap_reference_t<T>, Args...>) {
930-
while (get_process_state(_process).first == process_state::await) {
931-
if (!dequeue()) break;
932-
}
941+
if (get_process_state(_process).first == process_state::await) return;
942+
943+
// Workaround until we can use structured bindings
944+
auto tmp = get_process_state(_process);
945+
const auto& state = tmp.first;
946+
const auto& duration = tmp.second;
947+
948+
/*
949+
Once we hit yield, go ahead and call it. If the yield is delayed then schedule it.
950+
This process will be considered running until it executes.
951+
*/
952+
if (state == process_state::yield) {
953+
if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) <=
954+
std::chrono::nanoseconds::min())
955+
broadcast(unwrap(*_process).yield());
956+
else
957+
execute_at(duration,
958+
_executor)([_weak_this = make_weak_ptr(this->shared_from_this())] {
959+
auto _this = _weak_this.lock();
960+
if (!_this) return;
961+
_this->try_broadcast();
962+
});
963+
}
964+
965+
/*
966+
We are in an await state and the queue is empty.
967+
968+
If we await forever then task_done() leaving us in an await state.
969+
else if we await with an expired timeout then go ahead and yield now.
970+
else schedule a timeout when we will yield if not canceled by intervening await.
971+
*/
972+
else if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) ==
973+
std::chrono::nanoseconds::max()) {
974+
task_done();
975+
} else if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) <=
976+
std::chrono::nanoseconds::min()) {
977+
broadcast(unwrap(*_process).yield());
933978
} else {
934-
if (get_process_state(_process).first == process_state::await) return;
979+
/* Schedule a timeout. */
980+
_timeout_function_active = true;
981+
execute_at(duration,
982+
_executor)([_weak_this = make_weak_ptr(this->shared_from_this())] {
983+
auto _this = _weak_this.lock();
984+
// It may be that the complete channel is gone in the meanwhile
985+
if (!_this) return;
986+
987+
// try_lock can fail spuriously
988+
while (true) {
989+
lock_t lock(_this->_timeout_function_control, std::try_to_lock);
990+
if (!lock) continue;
991+
992+
// we were cancelled
993+
if (get_process_state(_this->_process).first != process_state::yield) {
994+
_this->try_broadcast();
995+
_this->_timeout_function_active = false;
996+
}
997+
return;
998+
}
999+
});
1000+
}
1001+
} catch (...) { // this catches exceptions during _process.await() and _process.yield()
1002+
broadcast(std::move(std::current_exception()));
1003+
}
1004+
}
1005+
1006+
template <typename U>
1007+
auto step() -> std::enable_if_t<has_process_yield_v<unwrap_reference_t<U>> &&
1008+
has_process_await_v<unwrap_reference_t<T>, Args...>> {
1009+
// in case that the timeout function is just been executed then we have to re-schedule
1010+
// the current run
1011+
lock_t lock(_timeout_function_control, std::try_to_lock);
1012+
if (!lock) {
1013+
run();
1014+
return;
1015+
}
1016+
_timeout_function_active = false;
1017+
1018+
/*
1019+
While we are waiting we will flush the queue. The assumption here is that work
1020+
is done on yield()
1021+
*/
1022+
try {
1023+
while (get_process_state(_process).first == process_state::await) {
1024+
if (!dequeue()) break;
9351025
}
9361026

9371027
// Workaround until we can use structured bindings
@@ -944,7 +1034,8 @@ struct shared_process
9441034
This process will be considered running until it executes.
9451035
*/
9461036
if (state == process_state::yield) {
947-
if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) <= std::chrono::nanoseconds::min())
1037+
if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) <=
1038+
std::chrono::nanoseconds::min())
9481039
broadcast(unwrap(*_process).yield());
9491040
else
9501041
execute_at(duration,
@@ -962,14 +1053,17 @@ struct shared_process
9621053
else if we await with an expired timeout then go ahead and yield now.
9631054
else schedule a timeout when we will yield if not canceled by intervening await.
9641055
*/
965-
else if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) == std::chrono::nanoseconds::max()) {
1056+
else if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) ==
1057+
std::chrono::nanoseconds::max()) {
9661058
task_done();
967-
} else if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) <= std::chrono::nanoseconds::min()) {
1059+
} else if (std::chrono::duration_cast<std::chrono::nanoseconds>(duration) <=
1060+
std::chrono::nanoseconds::min()) {
9681061
broadcast(unwrap(*_process).yield());
9691062
} else {
9701063
/* Schedule a timeout. */
9711064
_timeout_function_active = true;
972-
execute_at(duration, _executor)([_weak_this = make_weak_ptr(this->shared_from_this())] {
1065+
execute_at(duration,
1066+
_executor)([_weak_this = make_weak_ptr(this->shared_from_this())] {
9731067
auto _this = _weak_this.lock();
9741068
// It may be that the complete channel is gone in the meanwhile
9751069
if (!_this) return;
@@ -1367,7 +1461,7 @@ detail::annotated_process<F> operator&(detail::annotated_process<F>&& a, buffer_
13671461
/**************************************************************************************************/
13681462

13691463
template <typename T>
1370-
class [[nodiscard]] receiver {
1464+
class STLAB_NODISCARD() receiver {
13711465
using ptr_t = std::shared_ptr<detail::shared_process_receiver<T>>;
13721466

13731467
ptr_t _p;

stlab/concurrency/config.hpp

+9
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,15 @@
6161
#endif
6262
#endif
6363

64+
/**************************************************************************************************/
65+
66+
#if __cplusplus < 201703L
67+
#define STLAB_NODISCARD()
68+
#else
69+
#define STLAB_NODISCARD() [[nodiscard]]
70+
#endif
71+
72+
6473
/**************************************************************************************************/
6574

6675
#endif // STLAB_CONCURRENCY_CONFIG_HPP

0 commit comments

Comments
 (0)