1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
/*
* Иногда требуется создать умные указатели (shared_ptr) на константные объекты.
* Если при этом используется enable_shared_from_this, то первый shared_ptr не
* инициализирует weak_ptr лежищий внутри enable_shared_from_this.
* Это приводит к тому, что метод shared_from_this() кидает исключение.
* Это происходит из-за того, что шаблонная функция ipcdetails::sp_enable_shared_from_this,
* вызываемая в конструкторе shared_ptr не может сматчить входящие аргументы.
* Данная ошибка исправляется путем довления константности к типу входящего артумента 'pe'.
*/
#include <library/cpp/testing/unittest/registar.h>
#include <boost/interprocess/smart_ptr/shared_ptr.hpp>
#include <boost/interprocess/smart_ptr/enable_shared_from_this.hpp>
#include <type_traits>
#include <stdlib.h>
using Allocator = std::allocator<void>;
template <bool Const>
struct TestTypes {
class TestClass;
class Deleter {
public:
using const_pointer = TestClass const*;
using pointer = const_pointer;
Deleter& operator ()(pointer p) {
delete p;
return *this;
}
};
class TestClass:
public boost::interprocess::enable_shared_from_this<typename std::conditional<Const, const TestClass, TestClass>::type, Allocator, Deleter>
{
};
using shared_ptr = boost::interprocess::shared_ptr<typename std::conditional<Const, const TestClass, TestClass>::type, Allocator, Deleter>;
};
template <bool ConstPtr, bool ConstSharedPtr>
void test() {
using T = typename TestTypes<ConstSharedPtr>::TestClass;
using shared_ptr = typename TestTypes<ConstSharedPtr>::shared_ptr;
T* p = new T;
typename std::conditional<ConstPtr, T* const, T*>::type ptr = p;
shared_ptr sptr1(ptr);
UNIT_ASSERT_VALUES_EQUAL(sptr1.use_count(), 1);
{
shared_ptr sptr2 = p->shared_from_this();
UNIT_ASSERT_VALUES_EQUAL(sptr2.use_count(), 2);
}
UNIT_ASSERT_VALUES_EQUAL(sptr1.use_count(), 1);
}
Y_UNIT_TEST_SUITE(TestSharedPtr) {
Y_UNIT_TEST(NonConst_NonConst) {
test<false, false>();
}
Y_UNIT_TEST(NonConst_Const) {
test<false, true>();
}
Y_UNIT_TEST(Const_Const) {
test<true, true>();
}
};
|