1bb611c8fSApple OSS Distributions //
2bb611c8fSApple OSS Distributions // Tests for
3bb611c8fSApple OSS Distributions //  T& operator[](std::ptrdiff_t n) const;
4bb611c8fSApple OSS Distributions //
5bb611c8fSApple OSS Distributions 
6bb611c8fSApple OSS Distributions #include <libkern/c++/bounded_ptr.h>
7bb611c8fSApple OSS Distributions #include <array>
8bb611c8fSApple OSS Distributions #include <cstddef>
9bb611c8fSApple OSS Distributions #include <darwintest.h>
10bb611c8fSApple OSS Distributions #include <darwintest_utils.h>
11bb611c8fSApple OSS Distributions #include "test_utils.h"
12bb611c8fSApple OSS Distributions 
13bb611c8fSApple OSS Distributions #define _assert(...) T_ASSERT_TRUE((__VA_ARGS__), # __VA_ARGS__)
14bb611c8fSApple OSS Distributions 
15bb611c8fSApple OSS Distributions struct T {
16bb611c8fSApple OSS Distributions 	int i;
17bb611c8fSApple OSS Distributions 	friend constexpr bool
operator ==(T const & a,T const & b)18bb611c8fSApple OSS Distributions 	operator==(T const& a, T const& b)
19bb611c8fSApple OSS Distributions 	{
20bb611c8fSApple OSS Distributions 		return a.i == b.i;
21bb611c8fSApple OSS Distributions 	}
22bb611c8fSApple OSS Distributions };
23bb611c8fSApple OSS Distributions 
24bb611c8fSApple OSS Distributions namespace {
25bb611c8fSApple OSS Distributions struct tracking_policy {
26bb611c8fSApple OSS Distributions 	static bool did_trap;
27bb611c8fSApple OSS Distributions 	static void
trap__anond5fd282a0111::tracking_policy28bb611c8fSApple OSS Distributions 	trap(char const*)
29bb611c8fSApple OSS Distributions 	{
30bb611c8fSApple OSS Distributions 		did_trap = true;
31bb611c8fSApple OSS Distributions 	}
32bb611c8fSApple OSS Distributions };
33bb611c8fSApple OSS Distributions bool tracking_policy::did_trap = false;
34bb611c8fSApple OSS Distributions }
35bb611c8fSApple OSS Distributions 
36bb611c8fSApple OSS Distributions template <typename T, typename QualT>
37bb611c8fSApple OSS Distributions static void
tests()38bb611c8fSApple OSS Distributions tests()
39bb611c8fSApple OSS Distributions {
40bb611c8fSApple OSS Distributions 	std::array<T, 5> array = {T{0}, T{1}, T{2}, T{3}, T{4}};
41bb611c8fSApple OSS Distributions 
42bb611c8fSApple OSS Distributions 	{
43bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
44bb611c8fSApple OSS Distributions 		//   ^                                                ^
45bb611c8fSApple OSS Distributions 		//   |                                                |
46bb611c8fSApple OSS Distributions 		// begin, ptr                                        end
47bb611c8fSApple OSS Distributions 		test_bounded_ptr<QualT> ptr(array.begin() + 0, array.begin(), array.end());
48bb611c8fSApple OSS Distributions 		QualT& ref0 = ptr[0];
49bb611c8fSApple OSS Distributions 		_assert(&ref0 == &array[0]);
50bb611c8fSApple OSS Distributions 
51bb611c8fSApple OSS Distributions 		QualT& ref1 = ptr[1];
52bb611c8fSApple OSS Distributions 		_assert(&ref1 == &array[1]);
53bb611c8fSApple OSS Distributions 
54bb611c8fSApple OSS Distributions 		QualT& ref2 = ptr[2];
55bb611c8fSApple OSS Distributions 		_assert(&ref2 == &array[2]);
56bb611c8fSApple OSS Distributions 
57bb611c8fSApple OSS Distributions 		QualT& ref3 = ptr[3];
58bb611c8fSApple OSS Distributions 		_assert(&ref3 == &array[3]);
59bb611c8fSApple OSS Distributions 
60bb611c8fSApple OSS Distributions 		QualT& ref4 = ptr[4];
61bb611c8fSApple OSS Distributions 		_assert(&ref4 == &array[4]);
62bb611c8fSApple OSS Distributions 	}
63bb611c8fSApple OSS Distributions 	{
64bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
65bb611c8fSApple OSS Distributions 		//   ^        ^                                       ^
66bb611c8fSApple OSS Distributions 		//   |        |                                       |
67bb611c8fSApple OSS Distributions 		// begin     ptr                                     end
68bb611c8fSApple OSS Distributions 		test_bounded_ptr<QualT> ptr(array.begin() + 1, array.begin(), array.end());
69bb611c8fSApple OSS Distributions 		QualT& ref0 = ptr[-1];
70bb611c8fSApple OSS Distributions 		_assert(&ref0 == &array[0]);
71bb611c8fSApple OSS Distributions 
72bb611c8fSApple OSS Distributions 		QualT& ref1 = ptr[0];
73bb611c8fSApple OSS Distributions 		_assert(&ref1 == &array[1]);
74bb611c8fSApple OSS Distributions 
75bb611c8fSApple OSS Distributions 		QualT& ref2 = ptr[1];
76bb611c8fSApple OSS Distributions 		_assert(&ref2 == &array[2]);
77bb611c8fSApple OSS Distributions 
78bb611c8fSApple OSS Distributions 		QualT& ref3 = ptr[2];
79bb611c8fSApple OSS Distributions 		_assert(&ref3 == &array[3]);
80bb611c8fSApple OSS Distributions 
81bb611c8fSApple OSS Distributions 		QualT& ref4 = ptr[3];
82bb611c8fSApple OSS Distributions 		_assert(&ref4 == &array[4]);
83bb611c8fSApple OSS Distributions 	}
84bb611c8fSApple OSS Distributions 	{
85bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
86bb611c8fSApple OSS Distributions 		//   ^                 ^                              ^
87bb611c8fSApple OSS Distributions 		//   |                 |                              |
88bb611c8fSApple OSS Distributions 		// begin              ptr                            end
89bb611c8fSApple OSS Distributions 		test_bounded_ptr<QualT> ptr(array.begin() + 2, array.begin(), array.end());
90bb611c8fSApple OSS Distributions 		QualT& ref0 = ptr[-2];
91bb611c8fSApple OSS Distributions 		_assert(&ref0 == &array[0]);
92bb611c8fSApple OSS Distributions 
93bb611c8fSApple OSS Distributions 		QualT& ref1 = ptr[-1];
94bb611c8fSApple OSS Distributions 		_assert(&ref1 == &array[1]);
95bb611c8fSApple OSS Distributions 
96bb611c8fSApple OSS Distributions 		QualT& ref2 = ptr[0];
97bb611c8fSApple OSS Distributions 		_assert(&ref2 == &array[2]);
98bb611c8fSApple OSS Distributions 
99bb611c8fSApple OSS Distributions 		QualT& ref3 = ptr[1];
100bb611c8fSApple OSS Distributions 		_assert(&ref3 == &array[3]);
101bb611c8fSApple OSS Distributions 
102bb611c8fSApple OSS Distributions 		QualT& ref4 = ptr[2];
103bb611c8fSApple OSS Distributions 		_assert(&ref4 == &array[4]);
104bb611c8fSApple OSS Distributions 	}
105bb611c8fSApple OSS Distributions 	{
106bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
107bb611c8fSApple OSS Distributions 		//   ^                                   ^            ^
108bb611c8fSApple OSS Distributions 		//   |                                   |            |
109bb611c8fSApple OSS Distributions 		// begin                                ptr          end
110bb611c8fSApple OSS Distributions 		test_bounded_ptr<QualT> ptr(array.begin() + 4, array.begin(), array.end());
111bb611c8fSApple OSS Distributions 		QualT& ref0 = ptr[-4];
112bb611c8fSApple OSS Distributions 		_assert(&ref0 == &array[0]);
113bb611c8fSApple OSS Distributions 
114bb611c8fSApple OSS Distributions 		QualT& ref1 = ptr[-3];
115bb611c8fSApple OSS Distributions 		_assert(&ref1 == &array[1]);
116bb611c8fSApple OSS Distributions 
117bb611c8fSApple OSS Distributions 		QualT& ref2 = ptr[-2];
118bb611c8fSApple OSS Distributions 		_assert(&ref2 == &array[2]);
119bb611c8fSApple OSS Distributions 
120bb611c8fSApple OSS Distributions 		QualT& ref3 = ptr[-1];
121bb611c8fSApple OSS Distributions 		_assert(&ref3 == &array[3]);
122bb611c8fSApple OSS Distributions 
123bb611c8fSApple OSS Distributions 		QualT& ref4 = ptr[0];
124bb611c8fSApple OSS Distributions 		_assert(&ref4 == &array[4]);
125bb611c8fSApple OSS Distributions 	}
126bb611c8fSApple OSS Distributions 	{
127bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
128bb611c8fSApple OSS Distributions 		//   ^                                                ^
129bb611c8fSApple OSS Distributions 		//   |                                                |
130bb611c8fSApple OSS Distributions 		// begin                                           end,ptr
131bb611c8fSApple OSS Distributions 		test_bounded_ptr<QualT> ptr(array.end(), array.begin(), array.end());
132bb611c8fSApple OSS Distributions 		QualT& ref0 = ptr[-5];
133bb611c8fSApple OSS Distributions 		_assert(&ref0 == &array[0]);
134bb611c8fSApple OSS Distributions 
135bb611c8fSApple OSS Distributions 		QualT& ref1 = ptr[-4];
136bb611c8fSApple OSS Distributions 		_assert(&ref1 == &array[1]);
137bb611c8fSApple OSS Distributions 
138bb611c8fSApple OSS Distributions 		QualT& ref2 = ptr[-3];
139bb611c8fSApple OSS Distributions 		_assert(&ref2 == &array[2]);
140bb611c8fSApple OSS Distributions 
141bb611c8fSApple OSS Distributions 		QualT& ref3 = ptr[-2];
142bb611c8fSApple OSS Distributions 		_assert(&ref3 == &array[3]);
143bb611c8fSApple OSS Distributions 
144bb611c8fSApple OSS Distributions 		QualT& ref4 = ptr[-1];
145bb611c8fSApple OSS Distributions 		_assert(&ref4 == &array[4]);
146bb611c8fSApple OSS Distributions 	}
147bb611c8fSApple OSS Distributions 
148bb611c8fSApple OSS Distributions 	// Make sure we trap when we subscript a pointer at an out-of-bounds offset
149bb611c8fSApple OSS Distributions 	{
150bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
151bb611c8fSApple OSS Distributions 		//   ^                          ^        ^
152bb611c8fSApple OSS Distributions 		//   |                          |        |
153bb611c8fSApple OSS Distributions 		// begin                       end      ptr
154bb611c8fSApple OSS Distributions 		libkern::bounded_ptr<QualT, tracking_policy> ptr(array.end() - 1, array.begin(), array.end() - 2);
155bb611c8fSApple OSS Distributions 
156bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
157bb611c8fSApple OSS Distributions 		(void)ptr[-4];
158bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
159bb611c8fSApple OSS Distributions 
160bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
161bb611c8fSApple OSS Distributions 		(void)ptr[-3];
162bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
163bb611c8fSApple OSS Distributions 
164bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
165bb611c8fSApple OSS Distributions 		(void)ptr[-2];
166bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
167bb611c8fSApple OSS Distributions 
168bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
169bb611c8fSApple OSS Distributions 		(void)ptr[-1]; // trap
170bb611c8fSApple OSS Distributions 		_assert(tracking_policy::did_trap);
171bb611c8fSApple OSS Distributions 
172bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
173bb611c8fSApple OSS Distributions 		(void)ptr[0]; // trap
174bb611c8fSApple OSS Distributions 		_assert(tracking_policy::did_trap);
175bb611c8fSApple OSS Distributions 	}
176bb611c8fSApple OSS Distributions 	{
177bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
178bb611c8fSApple OSS Distributions 		//   ^        ^                                        ^
179bb611c8fSApple OSS Distributions 		//   |        |                                        |
180bb611c8fSApple OSS Distributions 		// begin     ptr                                      end
181bb611c8fSApple OSS Distributions 		libkern::bounded_ptr<QualT, tracking_policy> ptr(array.begin() + 1, array.begin(), array.end());
182bb611c8fSApple OSS Distributions 
183bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
184bb611c8fSApple OSS Distributions 		(void)ptr[-1];
185bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
186bb611c8fSApple OSS Distributions 
187bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
188bb611c8fSApple OSS Distributions 		(void)ptr[0];
189bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
190bb611c8fSApple OSS Distributions 
191bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
192bb611c8fSApple OSS Distributions 		(void)ptr[1];
193bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
194bb611c8fSApple OSS Distributions 
195bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
196bb611c8fSApple OSS Distributions 		(void)ptr[2];
197bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
198bb611c8fSApple OSS Distributions 
199bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
200bb611c8fSApple OSS Distributions 		(void)ptr[3];
201bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
202bb611c8fSApple OSS Distributions 
203bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
204bb611c8fSApple OSS Distributions 		(void)ptr[4]; // trap
205bb611c8fSApple OSS Distributions 		_assert(tracking_policy::did_trap);
206bb611c8fSApple OSS Distributions 	}
207bb611c8fSApple OSS Distributions 	{
208bb611c8fSApple OSS Distributions 		// T{0}     T{1}     T{2}     T{3}     T{4}     <one-past-last>
209bb611c8fSApple OSS Distributions 		//   ^        ^                          ^
210bb611c8fSApple OSS Distributions 		//   |        |                          |
211bb611c8fSApple OSS Distributions 		//  ptr     begin                       end
212bb611c8fSApple OSS Distributions 		libkern::bounded_ptr<QualT, tracking_policy> ptr(array.begin(), array.begin() + 1, array.end() - 1);
213bb611c8fSApple OSS Distributions 
214bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
215bb611c8fSApple OSS Distributions 		(void)ptr[0]; // trap
216bb611c8fSApple OSS Distributions 		_assert(tracking_policy::did_trap);
217bb611c8fSApple OSS Distributions 
218bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
219bb611c8fSApple OSS Distributions 		(void)ptr[1];
220bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
221bb611c8fSApple OSS Distributions 
222bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
223bb611c8fSApple OSS Distributions 		(void)ptr[2];
224bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
225bb611c8fSApple OSS Distributions 
226bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
227bb611c8fSApple OSS Distributions 		(void)ptr[3];
228bb611c8fSApple OSS Distributions 		_assert(!tracking_policy::did_trap);
229bb611c8fSApple OSS Distributions 
230bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
231bb611c8fSApple OSS Distributions 		(void)ptr[4]; // trap
232bb611c8fSApple OSS Distributions 		_assert(tracking_policy::did_trap);
233bb611c8fSApple OSS Distributions 
234bb611c8fSApple OSS Distributions 		tracking_policy::did_trap = false;
235bb611c8fSApple OSS Distributions 		(void)ptr[5]; // trap
236bb611c8fSApple OSS Distributions 		_assert(tracking_policy::did_trap);
237bb611c8fSApple OSS Distributions 	}
238bb611c8fSApple OSS Distributions }
239bb611c8fSApple OSS Distributions 
240*8d741a5dSApple OSS Distributions T_DECL(subscript, "bounded_ptr.subscript", T_META_TAG_VM_PREFERRED) {
241bb611c8fSApple OSS Distributions 	tests<T, T>();
242bb611c8fSApple OSS Distributions 	tests<T, T const>();
243bb611c8fSApple OSS Distributions 	tests<T, T volatile>();
244bb611c8fSApple OSS Distributions 	tests<T, T const volatile>();
245bb611c8fSApple OSS Distributions 
246bb611c8fSApple OSS Distributions 	// Make sure that we don't hard-error in the definition of operator[]
247bb611c8fSApple OSS Distributions 	// when instantiating a `bounded_ptr<cv-void>`
248bb611c8fSApple OSS Distributions 	test_bounded_ptr<void> p1;
249bb611c8fSApple OSS Distributions 	test_bounded_ptr<void const> p2;
250bb611c8fSApple OSS Distributions 	test_bounded_ptr<void volatile> p3;
251bb611c8fSApple OSS Distributions 	test_bounded_ptr<void const volatile> p4;
252bb611c8fSApple OSS Distributions }
253