First
This commit is contained in:
@@ -0,0 +1,290 @@
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// Copyright (c) Lewis Baker
|
||||
// Licenced under MIT license. See LICENSE.txt for details.
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#include <cppcoro/schedule_on.hpp>
|
||||
#include <cppcoro/resume_on.hpp>
|
||||
#include <cppcoro/io_service.hpp>
|
||||
#include <cppcoro/sync_wait.hpp>
|
||||
#include <cppcoro/when_all_ready.hpp>
|
||||
#include <cppcoro/on_scope_exit.hpp>
|
||||
#include <cppcoro/fmap.hpp>
|
||||
|
||||
#include "io_service_fixture.hpp"
|
||||
|
||||
#include <ostream>
|
||||
#include "doctest/cppcoro_doctest.h"
|
||||
|
||||
TEST_SUITE_BEGIN("schedule/resume_on");
|
||||
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "schedule_on task<> function")
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
std::thread::id ioThreadId;
|
||||
|
||||
auto start = [&]() -> cppcoro::task<>
|
||||
{
|
||||
ioThreadId = std::this_thread::get_id();
|
||||
CHECK(ioThreadId != mainThreadId);
|
||||
co_return;
|
||||
};
|
||||
|
||||
cppcoro::sync_wait([&]() -> cppcoro::task<>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
|
||||
co_await schedule_on(io_service(), start());
|
||||
|
||||
// TODO: Uncomment this check once the implementation of task<T>
|
||||
// guarantees that the continuation will resume on the same thread
|
||||
// that the task completed on. Currently it's possible to resume on
|
||||
// the thread that launched the task if it completes on another thread
|
||||
// before the current thread could attach the continuation after it
|
||||
// suspended. See cppcoro issue #79.
|
||||
//
|
||||
// The long-term solution here is to use the symmetric-transfer capability
|
||||
// to avoid the use of atomics and races, but we're still waiting for MSVC to
|
||||
// implement this (doesn't seem to be implemented as of VS 2017.8 Preview 5)
|
||||
//CHECK(std::this_thread::get_id() == ioThreadId);
|
||||
}());
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "schedule_on async_generator<> function")
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
std::thread::id ioThreadId;
|
||||
|
||||
auto makeSequence = [&]() -> cppcoro::async_generator<int>
|
||||
{
|
||||
ioThreadId = std::this_thread::get_id();
|
||||
CHECK(ioThreadId != mainThreadId);
|
||||
|
||||
co_yield 1;
|
||||
|
||||
CHECK(std::this_thread::get_id() == ioThreadId);
|
||||
|
||||
co_yield 2;
|
||||
|
||||
CHECK(std::this_thread::get_id() == ioThreadId);
|
||||
|
||||
co_yield 3;
|
||||
|
||||
CHECK(std::this_thread::get_id() == ioThreadId);
|
||||
|
||||
co_return;
|
||||
};
|
||||
|
||||
cppcoro::io_service otherIoService;
|
||||
|
||||
cppcoro::sync_wait(cppcoro::when_all_ready(
|
||||
[&]() -> cppcoro::task<>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
|
||||
auto seq = schedule_on(io_service(), makeSequence());
|
||||
|
||||
int expected = 1;
|
||||
for (auto iter = co_await seq.begin(); iter != seq.end(); co_await ++iter)
|
||||
{
|
||||
int value = *iter;
|
||||
CHECK(value == expected++);
|
||||
|
||||
// Transfer exection back to main thread before
|
||||
// awaiting next item in the loop to chck that
|
||||
// the generator is resumed on io_service() thread.
|
||||
co_await otherIoService.schedule();
|
||||
}
|
||||
|
||||
otherIoService.stop();
|
||||
}(),
|
||||
[&]() -> cppcoro::task<>
|
||||
{
|
||||
otherIoService.process_events();
|
||||
co_return;
|
||||
}()));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "resume_on task<> function")
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
auto start = [&]() -> cppcoro::task<>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
co_return;
|
||||
};
|
||||
|
||||
cppcoro::sync_wait([&]() -> cppcoro::task<>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
|
||||
co_await resume_on(io_service(), start());
|
||||
|
||||
// NOTE: This check could potentially spuriously fail with the current
|
||||
// implementation of task<T>. See cppcoro issue #79.
|
||||
CHECK(std::this_thread::get_id() != mainThreadId);
|
||||
}());
|
||||
}
|
||||
|
||||
constexpr bool isMsvc15_4X86Optimised =
|
||||
#if defined(_MSC_VER) && _MSC_VER == 1911 && defined(_M_IX86) && !defined(_DEBUG)
|
||||
true;
|
||||
#else
|
||||
false;
|
||||
#endif
|
||||
|
||||
// Disable under MSVC 15.4 X86 Optimised due to presumed compiler bug that causes
|
||||
// an access violation. Seems to be fixed under MSVC 15.5.
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "resume_on async_generator<> function"
|
||||
* doctest::skip{ isMsvc15_4X86Optimised })
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
std::thread::id ioThreadId;
|
||||
|
||||
auto makeSequence = [&]() -> cppcoro::async_generator<int>
|
||||
{
|
||||
co_await io_service().schedule();
|
||||
|
||||
ioThreadId = std::this_thread::get_id();
|
||||
|
||||
CHECK(ioThreadId != mainThreadId);
|
||||
|
||||
co_yield 1;
|
||||
|
||||
co_yield 2;
|
||||
|
||||
co_await io_service().schedule();
|
||||
|
||||
co_yield 3;
|
||||
|
||||
co_await io_service().schedule();
|
||||
|
||||
co_return;
|
||||
};
|
||||
|
||||
cppcoro::io_service otherIoService;
|
||||
|
||||
cppcoro::sync_wait(cppcoro::when_all_ready(
|
||||
[&]() -> cppcoro::task<>
|
||||
{
|
||||
auto stopOnExit = cppcoro::on_scope_exit([&] { otherIoService.stop(); });
|
||||
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
|
||||
auto seq = resume_on(otherIoService, makeSequence());
|
||||
|
||||
int expected = 1;
|
||||
for (auto iter = co_await seq.begin(); iter != seq.end(); co_await ++iter)
|
||||
{
|
||||
int value = *iter;
|
||||
// Every time we receive a value it should be on our requested
|
||||
// scheduler (ie. main thread)
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
CHECK(value == expected++);
|
||||
|
||||
// Occasionally transfer execution to a different thread before
|
||||
// awaiting next element.
|
||||
if (value == 2)
|
||||
{
|
||||
co_await io_service().schedule();
|
||||
}
|
||||
}
|
||||
|
||||
otherIoService.stop();
|
||||
}(),
|
||||
[&]() -> cppcoro::task<>
|
||||
{
|
||||
otherIoService.process_events();
|
||||
co_return;
|
||||
}()));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "schedule_on task<> pipe syntax")
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
auto makeTask = [&]() -> cppcoro::task<int>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() != mainThreadId);
|
||||
co_return 123;
|
||||
};
|
||||
|
||||
auto triple = [&](int x)
|
||||
{
|
||||
CHECK(std::this_thread::get_id() != mainThreadId);
|
||||
return x * 3;
|
||||
};
|
||||
|
||||
CHECK(cppcoro::sync_wait(makeTask() | schedule_on(io_service())) == 123);
|
||||
|
||||
// Shouldn't matter where in sequence schedule_on() appears since it applies
|
||||
// at the start of the pipeline (ie. before first task starts).
|
||||
CHECK(cppcoro::sync_wait(makeTask() | schedule_on(io_service()) | cppcoro::fmap(triple)) == 369);
|
||||
CHECK(cppcoro::sync_wait(makeTask() | cppcoro::fmap(triple) | schedule_on(io_service())) == 369);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "resume_on task<> pipe syntax")
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
auto makeTask = [&]() -> cppcoro::task<int>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
co_return 123;
|
||||
};
|
||||
|
||||
cppcoro::sync_wait([&]() -> cppcoro::task<>
|
||||
{
|
||||
cppcoro::task<int> t = makeTask() | cppcoro::resume_on(io_service());
|
||||
CHECK(co_await t == 123);
|
||||
CHECK(std::this_thread::get_id() != mainThreadId);
|
||||
}());
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(io_service_fixture, "resume_on task<> pipe syntax multiple uses")
|
||||
{
|
||||
auto mainThreadId = std::this_thread::get_id();
|
||||
|
||||
auto makeTask = [&]() -> cppcoro::task<int>
|
||||
{
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
co_return 123;
|
||||
};
|
||||
|
||||
auto triple = [&](int x)
|
||||
{
|
||||
CHECK(std::this_thread::get_id() != mainThreadId);
|
||||
return x * 3;
|
||||
};
|
||||
|
||||
cppcoro::io_service otherIoService;
|
||||
|
||||
cppcoro::sync_wait(cppcoro::when_all_ready(
|
||||
[&]() -> cppcoro::task<>
|
||||
{
|
||||
auto stopOnExit = cppcoro::on_scope_exit([&] { otherIoService.stop(); });
|
||||
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
|
||||
cppcoro::task<int> t =
|
||||
makeTask()
|
||||
| cppcoro::resume_on(io_service())
|
||||
| cppcoro::fmap(triple)
|
||||
| cppcoro::resume_on(otherIoService);
|
||||
|
||||
CHECK(co_await t == 369);
|
||||
|
||||
CHECK(std::this_thread::get_id() == mainThreadId);
|
||||
}(),
|
||||
[&]() -> cppcoro::task<>
|
||||
{
|
||||
otherIoService.process_events();
|
||||
co_return;
|
||||
}()));
|
||||
}
|
||||
|
||||
TEST_SUITE_END();
|
||||
Reference in New Issue
Block a user