Blob Blame History Raw
/*
 * Copyright 2016-2019, Intel Corporation
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 *     * Redistributions of source code must retain the above copyright
 *       notice, this list of conditions and the following disclaimer.
 *
 *     * Redistributions in binary form must reproduce the above copyright
 *       notice, this list of conditions and the following disclaimer in
 *       the documentation and/or other materials provided with the
 *       distribution.
 *
 *     * Neither the name of the copyright holder nor the names of its
 *       contributors may be used to endorse or promote products derived
 *       from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

/*
 * obj_cpp_make_persistent_atomic.cpp -- cpp make_persistent_atomic test for
 * objects
 */

#include "unittest.hpp"

#include <libpmemobj++/make_persistent_atomic.hpp>
#include <libpmemobj++/p.hpp>
#include <libpmemobj++/persistent_ptr.hpp>
#include <libpmemobj++/pool.hpp>
#include <libpmemobj/ctl.h>

#define LAYOUT "cpp"

namespace nvobj = pmem::obj;

namespace
{

const int TEST_ARR_SIZE = 10;

struct force_throw {
};

class foo {
public:
	foo() : bar(1)
	{
		for (int i = 0; i < TEST_ARR_SIZE; ++i)
			this->arr[i] = 1;
	}

	foo(const int &val) : bar(val)
	{
		for (int i = 0; i < TEST_ARR_SIZE; ++i)
			this->arr[i] = static_cast<char>(val);
	}

	foo(const int &val, char arr_val) : bar(val)
	{
		for (int i = 0; i < TEST_ARR_SIZE; ++i)
			this->arr[i] = arr_val;
	}

	explicit foo(struct force_throw &val)
	{
		throw std::bad_alloc();
	}

	/*
	 * Assert values of foo.
	 */
	void
	check_foo(int val, char arr_val)
	{
		UT_ASSERTeq(val, this->bar);
		for (int i = 0; i < TEST_ARR_SIZE; ++i)
			UT_ASSERTeq(arr_val, this->arr[i]);
	}

	nvobj::p<int> bar;
	nvobj::p<char> arr[TEST_ARR_SIZE];
};

static int var_bar_copy_ctors_called = 0;
static int var_bar_move_ctors_called = 0;

struct var_bar {

	/* Expects 'a' to be rvalue ref and 'b' and 'c' to be lvalue ref. */
	template <typename A, typename B, typename C>
	var_bar(A &&a, B &&b, C &&c)
	{
		static_assert(std::is_rvalue_reference<decltype(a)>::value, "");
		static_assert(std::is_lvalue_reference<decltype(b)>::value, "");
		static_assert(std::is_lvalue_reference<decltype(c)>::value, "");
	}

	var_bar(const var_bar &a)
	{
		var_bar_copy_ctors_called++;
	}

	var_bar(var_bar &&a)
	{
		var_bar_move_ctors_called++;
	}
};

struct root {
	nvobj::persistent_ptr<foo> pfoo;
	nvobj::persistent_ptr<var_bar> pvbar;
};

/*
 * test_make_no_args -- (internal) test make_persistent without arguments
 */
void
test_make_no_args(nvobj::pool<struct root> &pop)
{
	nvobj::persistent_ptr<root> r = pop.root();

	UT_ASSERT(r->pfoo == nullptr);

	nvobj::make_persistent_atomic<foo>(pop, r->pfoo);
	r->pfoo->check_foo(1, 1);

	nvobj::delete_persistent_atomic<foo>(r->pfoo);
}

/*
 * test_make_args -- (internal) test make_persistent with arguments
 */
void
test_make_args(nvobj::pool<struct root> &pop)
{
	nvobj::persistent_ptr<root> r = pop.root();
	UT_ASSERT(r->pfoo == nullptr);

	nvobj::make_persistent_atomic<foo>(pop, r->pfoo, 2);
	r->pfoo->check_foo(2, 2);

	nvobj::delete_persistent_atomic<foo>(r->pfoo);

	nvobj::make_persistent_atomic<foo>(pop, r->pfoo, 3, 4);
	r->pfoo->check_foo(3, 4);

	nvobj::delete_persistent_atomic<foo>(r->pfoo);
}

/*
 * test_throw -- (internal) test if make_persistent_atomic rethrows constructor
 * exception
 */
void
test_throw(nvobj::pool<struct root> &pop)
{
	nvobj::persistent_ptr<root> r = pop.root();
	UT_ASSERT(r->pfoo == nullptr);

	bool exception_thrown = false;
	force_throw val;
	try {
		nvobj::make_persistent_atomic<foo>(pop, r->pfoo, val);
	} catch (std::bad_alloc &) {
		exception_thrown = true;
	} catch (std::exception &e) {
		UT_FATALexc(e);
	}

	UT_ASSERT(exception_thrown);
	UT_ASSERT(r->pfoo == nullptr);
}

/*
 * test_delete_null -- (internal) test atomic delete nullptr
 */
void
test_delete_null(nvobj::pool<struct root> &pop)
{
	nvobj::persistent_ptr<foo> pfoo;

	UT_ASSERT(pfoo == nullptr);

	try {
		nvobj::delete_persistent_atomic<foo>(pfoo);
	} catch (...) {
		UT_ASSERT(0);
	}
}

/*
 * test_flags -- (internal) test proper handling of flags
 */
void
test_flags(nvobj::pool<struct root> &pop)
{
	nvobj::persistent_ptr<root> r = pop.root();

	auto alloc_class = pop.ctl_set<struct pobj_alloc_class_desc>(
		"heap.alloc_class.new.desc",
		{sizeof(foo) + 16, 0, 200, POBJ_HEADER_COMPACT, 0});

	try {
		nvobj::make_persistent_atomic<foo>(
			pop, r->pfoo,
			nvobj::allocation_flag_atomic::class_id(
				alloc_class.class_id));
	} catch (std::exception &e) {
		UT_FATALexc(e);
	}

	UT_ASSERTeq(pmemobj_alloc_usable_size(r->pfoo.raw()), sizeof(foo));

	try {
		nvobj::delete_persistent_atomic<foo>(r->pfoo);
	} catch (std::exception &e) {
		UT_FATALexc(e);
	}

	try {
		nvobj::make_persistent_atomic<foo>(
			pop, r->pfoo,
			nvobj::allocation_flag_atomic::class_id(
				alloc_class.class_id),
			1, 2);
	} catch (std::exception &e) {
		UT_FATALexc(e);
	}

	r->pfoo->check_foo(1, 2);

	UT_ASSERTeq(pmemobj_alloc_usable_size(r->pfoo.raw()), sizeof(foo));

	try {
		nvobj::delete_persistent_atomic<foo>(r->pfoo);
	} catch (std::exception &e) {
		UT_FATALexc(e);
	}
}

/*
 * test_rlvalue_parameters -- (internal) test proper forwarding of arguments
 * to the constructor (maintaining rvalue and lvalue reference types)
 */
void
test_rlvalue_parameters(nvobj::pool<struct root> &pop)
{
	auto r = pop.root();

	int a = 1, b = 2, c = 3;
	nvobj::make_persistent_atomic<var_bar>(pop, r->pvbar, std::move(a), b,
					       c);

	nvobj::persistent_ptr<var_bar> vbar1;
	nvobj::persistent_ptr<var_bar> vbar2;
	nvobj::persistent_ptr<var_bar> vbar3;

	nvobj::make_persistent_atomic<var_bar>(pop, vbar1, *r->pvbar);
	UT_ASSERT(var_bar_copy_ctors_called == 1);
	UT_ASSERT(var_bar_move_ctors_called == 0);

	nvobj::make_persistent_atomic<var_bar>(pop, vbar2, *r->pvbar);
	UT_ASSERT(var_bar_copy_ctors_called == 2);
	UT_ASSERT(var_bar_move_ctors_called == 0);

	nvobj::make_persistent_atomic<var_bar>(pop, vbar3,
					       std::move(*r->pvbar));

	UT_ASSERT(var_bar_copy_ctors_called == 2);
	UT_ASSERT(var_bar_move_ctors_called == 1);
}

/*
 * test_make_invalid -- (internal) test failure of atomic make_persistent
 */
void
test_make_invalid(nvobj::pool<struct root> &pop)
{
	nvobj::persistent_ptr<foo> pfoo;

	bool thrown = false;
	try {
		nvobj::make_persistent_atomic<foo>(
			pop, pfoo,
			nvobj::allocation_flag_atomic::class_id(254));
	} catch (std::bad_alloc &e) {
		thrown = true;
	} catch (std::exception &e) {
		UT_FATALexc(e);
	}

	UT_ASSERT(thrown);
}
}

int
main(int argc, char *argv[])
{
	START();

	if (argc != 2)
		UT_FATAL("usage: %s file-name", argv[0]);

	const char *path = argv[1];

	nvobj::pool<struct root> pop;

	try {
		pop = nvobj::pool<struct root>::create(
			path, LAYOUT, PMEMOBJ_MIN_POOL, S_IWUSR | S_IRUSR);
	} catch (pmem::pool_error &pe) {
		UT_FATAL("!pool::create: %s %s", pe.what(), path);
	}

	test_make_no_args(pop);
	test_make_args(pop);
	test_throw(pop);
	test_delete_null(pop);
	test_flags(pop);
	test_rlvalue_parameters(pop);
	test_make_invalid(pop);

	pop.close();

	return 0;
}