compute/test/test_event.cpp
Lakshay Garg a6879d2c2c add future::then() method
resolves #94
2018-03-04 16:56:03 +05:30

144 lines
4.2 KiB
C++

//---------------------------------------------------------------------------//
// Copyright (c) 2013-2014 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// Distributed under the Boost Software License, Version 1.0
// See accompanying file LICENSE_1_0.txt or copy at
// http://www.boost.org/LICENSE_1_0.txt
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#define BOOST_TEST_MODULE TestEvent
#include <boost/test/unit_test.hpp>
#include <vector>
#ifdef BOOST_COMPUTE_USE_CPP11
#include <mutex>
#include <future>
#endif // BOOST_COMPUTE_USE_CPP11
#include <boost/compute/async/future.hpp>
#include <boost/compute/event.hpp>
#include "context_setup.hpp"
BOOST_AUTO_TEST_CASE(null_event)
{
boost::compute::event null;
BOOST_CHECK(null.get() == cl_event());
}
#if defined(BOOST_COMPUTE_CL_VERSION_1_1) && defined(BOOST_COMPUTE_USE_CPP11)
std::mutex callback_mutex;
std::condition_variable callback_condition_variable;
static bool callback_invoked = false;
static void BOOST_COMPUTE_CL_CALLBACK
callback(cl_event event, cl_int status, void *user_data)
{
std::lock_guard<std::mutex> lock(callback_mutex);
callback_invoked = true;
callback_condition_variable.notify_one();
}
BOOST_AUTO_TEST_CASE(event_callback)
{
REQUIRES_OPENCL_VERSION(1,2);
// ensure callback has not yet been executed
BOOST_CHECK_EQUAL(callback_invoked, false);
// enqueue marker and set callback to be invoked
boost::compute::event marker = queue.enqueue_marker();
marker.set_callback(callback);
marker.wait();
// wait up to one second for the callback to be executed
std::unique_lock<std::mutex> lock(callback_mutex);
callback_condition_variable.wait_for(
lock, std::chrono::seconds(1), [&](){ return callback_invoked; }
);
// ensure callback has been executed
BOOST_CHECK_EQUAL(callback_invoked, true);
}
BOOST_AUTO_TEST_CASE(lambda_callback)
{
REQUIRES_OPENCL_VERSION(1,2);
bool lambda_invoked = false;
boost::compute::event marker = queue.enqueue_marker();
marker.set_callback([&](){
std::lock_guard<std::mutex> lock(callback_mutex);
lambda_invoked = true;
callback_condition_variable.notify_one();
});
marker.wait();
// wait up to one second for the callback to be executed
std::unique_lock<std::mutex> lock(callback_mutex);
callback_condition_variable.wait_for(
lock, std::chrono::seconds(1), [&](){ return lambda_invoked; }
);
BOOST_CHECK_EQUAL(lambda_invoked, true);
}
BOOST_AUTO_TEST_CASE(future_then_callback)
{
REQUIRES_OPENCL_VERSION(1,2);
bool callback_invoked = false;
boost::compute::future<void> future(queue.enqueue_marker());
future.then([&](){
std::lock_guard<std::mutex> lock(callback_mutex);
callback_invoked = true;
callback_condition_variable.notify_one();
});
future.wait();
// wait up to one second for the callback to be executed
std::unique_lock<std::mutex> lock(callback_mutex);
callback_condition_variable.wait_for(
lock, std::chrono::seconds(1), [&](){ return callback_invoked; }
);
BOOST_CHECK_EQUAL(callback_invoked, true);
}
void BOOST_COMPUTE_CL_CALLBACK
event_promise_fulfiller_callback(cl_event event, cl_int status, void *user_data)
{
auto *promise = static_cast<std::promise<void> *>(user_data);
promise->set_value();
delete promise;
}
BOOST_AUTO_TEST_CASE(event_to_std_future)
{
REQUIRES_OPENCL_VERSION(1,2);
// enqueue an asynchronous copy to the device
std::vector<float> vector(1000, 3.14f);
boost::compute::buffer buffer(context, 1000 * sizeof(float));
auto event = queue.enqueue_write_buffer_async(
buffer, 0, 1000 * sizeof(float), vector.data()
);
// create a promise and future to be set by the callback
auto *promise = new std::promise<void>;
std::future<void> future = promise->get_future();
event.set_callback(event_promise_fulfiller_callback, CL_COMPLETE, promise);
// ensure commands are submitted to the device before waiting
queue.flush();
// wait for future to become ready
future.wait();
}
#endif // BOOST_COMPUTE_CL_VERSION_1_1
BOOST_AUTO_TEST_SUITE_END()