2
0
mirror of https://github.com/boostorg/fiber.git synced 2026-01-29 19:42:08 +00:00
Files
fiber/test/test_fss.cpp
Oliver Kowalke 2f19be6d67 use C++11
2014-12-27 19:07:42 +01:00

283 lines
6.0 KiB
C++

// Copyright (C) 2001-2003
// William E. Kempf
// Copyright (C) 2007 Anthony Williams
//
// 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)
#include <iostream>
#include <boost/test/unit_test.hpp>
#include <boost/fiber/all.hpp>
boost::fibers::mutex check_mutex;
boost::fibers::mutex fss_mutex;
int fss_instances = 0;
int fss_total = 0;
struct fss_value_t
{
fss_value_t()
{
std::unique_lock<boost::fibers::mutex> lock(fss_mutex);
++fss_instances;
++fss_total;
value = 0;
}
~fss_value_t()
{
std::unique_lock<boost::fibers::mutex> lock(fss_mutex);
--fss_instances;
}
int value;
};
boost::fibers::fiber_specific_ptr<fss_value_t> fss_value;
void fss_fiber()
{
fss_value.reset(new fss_value_t());
for (int i=0; i<1000; ++i)
{
int& n = fss_value->value;
if (n != i)
{
std::unique_lock<boost::fibers::mutex> lock(check_mutex);
BOOST_CHECK_EQUAL(n, i);
}
++n;
}
}
void fss()
{
fss_instances = 0;
fss_total = 0;
boost::fibers::fiber f1( fss_fiber);
boost::fibers::fiber f2( fss_fiber);
boost::fibers::fiber f3( fss_fiber);
boost::fibers::fiber f4( fss_fiber);
boost::fibers::fiber f5( fss_fiber);
f1.join();
f2.join();
f3.join();
f4.join();
f5.join();
std::cout
<< "fss_instances = " << fss_instances
<< "; fss_total = " << fss_total
<< "\n";
std::cout.flush();
BOOST_CHECK_EQUAL(fss_instances, 0);
BOOST_CHECK_EQUAL(fss_total, 5);
}
void test_fss()
{
boost::fibers::fiber( fss).join();
}
bool fss_cleanup_called=false;
struct Dummy
{};
void fss_custom_cleanup(Dummy* d)
{
delete d;
fss_cleanup_called=true;
}
boost::fibers::fiber_specific_ptr<Dummy> fss_with_cleanup(fss_custom_cleanup);
void fss_fiber_with_custom_cleanup()
{
fss_with_cleanup.reset(new Dummy);
}
void fss_with_custom_cleanup()
{
boost::fibers::fiber f(fss_fiber_with_custom_cleanup);
try
{
f.join();
}
catch(...)
{
f.interrupt();
f.join();
throw;
}
BOOST_CHECK(fss_cleanup_called);
}
void test_fss_with_custom_cleanup()
{
boost::fibers::fiber( fss_with_custom_cleanup).join();
}
Dummy* fss_object=new Dummy;
void fss_fiber_with_custom_cleanup_and_release()
{
fss_with_cleanup.reset(fss_object);
fss_with_cleanup.release();
}
void do_test_fss_does_no_cleanup_after_release()
{
fss_cleanup_called=false;
boost::fibers::fiber f(fss_fiber_with_custom_cleanup_and_release);
try
{
f.join();
}
catch(...)
{
f.interrupt();
f.join();
throw;
}
BOOST_CHECK(!fss_cleanup_called);
if(!fss_cleanup_called)
{
delete fss_object;
}
}
struct dummy_class_tracks_deletions
{
static unsigned deletions;
~dummy_class_tracks_deletions()
{
++deletions;
}
};
unsigned dummy_class_tracks_deletions::deletions=0;
boost::fibers::fiber_specific_ptr<dummy_class_tracks_deletions> fss_with_null_cleanup(NULL);
void fss_fiber_with_null_cleanup(dummy_class_tracks_deletions* delete_tracker)
{
fss_with_null_cleanup.reset(delete_tracker);
}
void do_test_fss_does_no_cleanup_with_null_cleanup_function()
{
dummy_class_tracks_deletions* delete_tracker=new dummy_class_tracks_deletions;
boost::fibers::fiber f(
std::bind( fss_fiber_with_null_cleanup,delete_tracker) );
try
{
f.join();
}
catch(...)
{
f.interrupt();
f.join();
throw;
}
BOOST_CHECK(!dummy_class_tracks_deletions::deletions);
if(!dummy_class_tracks_deletions::deletions)
{
delete delete_tracker;
}
}
void test_fss_does_no_cleanup_after_release()
{
boost::fibers::fiber( do_test_fss_does_no_cleanup_after_release).join();
}
void test_fss_does_no_cleanup_with_null_cleanup_function()
{
boost::fibers::fiber( do_test_fss_does_no_cleanup_with_null_cleanup_function).join();
}
void fiber_with_local_fss_ptr()
{
{
boost::fibers::fiber_specific_ptr<Dummy> local_fss(fss_custom_cleanup);
local_fss.reset(new Dummy);
}
BOOST_CHECK(fss_cleanup_called);
fss_cleanup_called=false;
}
void fss_does_not_call_cleanup_after_ptr_destroyed()
{
boost::fibers::fiber(fiber_with_local_fss_ptr).join();
BOOST_CHECK(!fss_cleanup_called);
}
void test_fss_does_not_call_cleanup_after_ptr_destroyed()
{
boost::fibers::fiber( fss_does_not_call_cleanup_after_ptr_destroyed).join();
}
void fss_cleanup_not_called_for_null_pointer()
{
boost::fibers::fiber_specific_ptr<Dummy> local_fss(fss_custom_cleanup);
local_fss.reset(new Dummy);
fss_cleanup_called=false;
local_fss.reset(0);
BOOST_CHECK(fss_cleanup_called);
fss_cleanup_called=false;
local_fss.reset(new Dummy);
BOOST_CHECK(!fss_cleanup_called);
}
void test_fss_cleanup_not_called_for_null_pointer()
{
boost::fibers::fiber( fss_cleanup_not_called_for_null_pointer).join();
}
void fss_at_the_same_adress()
{
for(int i=0; i<2; i++)
{
boost::fibers::fiber_specific_ptr<Dummy> local_fss(fss_custom_cleanup);
local_fss.reset(new Dummy);
fss_cleanup_called=false;
BOOST_CHECK(fss_cleanup_called);
fss_cleanup_called=false;
BOOST_CHECK(!fss_cleanup_called);
}
}
void test_fss_at_the_same_adress()
{
boost::fibers::fiber( fss_at_the_same_adress).join();
}
boost::unit_test::test_suite* init_unit_test_suite(int, char*[])
{
boost::unit_test::test_suite* test =
BOOST_TEST_SUITE("Boost.Fiber: fss test suite");
test->add(BOOST_TEST_CASE(test_fss));
test->add(BOOST_TEST_CASE(test_fss_with_custom_cleanup));
test->add(BOOST_TEST_CASE(test_fss_does_no_cleanup_after_release));
test->add(BOOST_TEST_CASE(test_fss_does_no_cleanup_with_null_cleanup_function));
test->add(BOOST_TEST_CASE(test_fss_does_not_call_cleanup_after_ptr_destroyed));
test->add(BOOST_TEST_CASE(test_fss_cleanup_not_called_for_null_pointer));
return test;
}