• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "mojo/system/memory.h"
6 
7 #include <stddef.h>
8 #include <stdint.h>
9 
10 #include <limits>
11 
12 #include "mojo/public/c/system/macros.h"
13 #include "testing/gtest/include/gtest/gtest.h"
14 
15 namespace mojo {
16 namespace system {
17 namespace {
18 
TEST(MemoryTest,Valid)19 TEST(MemoryTest, Valid) {
20   char my_char;
21   int32_t my_int32;
22   int64_t my_int64_array[5] = {};  // Zero initialize.
23 
24   UserPointer<char> my_char_ptr(&my_char);
25   UserPointer<int32_t> my_int32_ptr(&my_int32);
26   UserPointer<int64_t> my_int64_array_ptr(my_int64_array);
27 
28   // |UserPointer<>::IsNull()|:
29   EXPECT_FALSE(my_char_ptr.IsNull());
30   EXPECT_FALSE(my_int32_ptr.IsNull());
31   EXPECT_FALSE(my_int64_array_ptr.IsNull());
32 
33   // |UserPointer<>::Put()| and |UserPointer<>::Get()|:
34   my_char_ptr.Put('x');
35   EXPECT_EQ('x', my_char);
36   EXPECT_EQ('x', my_char_ptr.Get());
37   my_int32_ptr.Put(123);
38   EXPECT_EQ(123, my_int32);
39   EXPECT_EQ(123, my_int32_ptr.Get());
40   my_int64_array_ptr.Put(456);
41   EXPECT_EQ(456, my_int64_array[0]);
42   EXPECT_EQ(456, my_int64_array_ptr.Get());
43 
44   // |UserPointer<>::At()|, etc.:
45   my_int64_array_ptr.At(3).Put(789);
46   EXPECT_EQ(789, my_int64_array[3]);
47   {
48     // Copy construction:
49     UserPointer<int64_t> other(my_int64_array_ptr.At(3));
50     EXPECT_FALSE(other.IsNull());
51     EXPECT_EQ(789, other.Get());
52 
53     // Assignment:
54     other = my_int64_array_ptr;
55     EXPECT_FALSE(other.IsNull());
56     EXPECT_EQ(456, other.Get());
57 
58     // Assignment to |NullUserPointer()|:
59     other = NullUserPointer();
60     EXPECT_TRUE(other.IsNull());
61 
62     // |MakeUserPointer()|:
63     other = MakeUserPointer(&my_int64_array[1]);
64     other.Put(-123);
65     EXPECT_EQ(-123, my_int64_array_ptr.At(1).Get());
66   }
67 
68   // "const" |UserPointer<>|:
69   {
70     // Explicit constructor from |NullUserPointer()|:
71     UserPointer<const char> other((NullUserPointer()));
72     EXPECT_TRUE(other.IsNull());
73 
74     // Conversion to "const":
75     other = my_char_ptr;
76     EXPECT_EQ('x', other.Get());
77   }
78 
79   // Default constructor:
80   {
81     UserPointer<int32_t> other;
82     EXPECT_TRUE(other.IsNull());
83 
84     other = my_int32_ptr;
85     other.Put(-456);
86     EXPECT_EQ(-456, my_int32_ptr.Get());
87   }
88 
89   // |UserPointer<>::CheckArray()|:
90   my_int64_array_ptr.CheckArray(5);
91 
92   // |UserPointer<>::GetArray()|:
93   {
94     // From a "const" |UserPointer<>| (why not?):
95     UserPointer<const int64_t> other(my_int64_array_ptr);
96     int64_t array[3] = {1, 2, 3};
97     other.At(1).GetArray(array, 3);
98     EXPECT_EQ(-123, array[0]);
99     EXPECT_EQ(0, array[1]);
100     EXPECT_EQ(789, array[2]);
101   }
102 
103   // |UserPointer<>::PutArray()|:
104   {
105     const int64_t array[2] = {654, 321};
106     my_int64_array_ptr.At(3).PutArray(array, 2);
107     EXPECT_EQ(0, my_int64_array[2]);
108     EXPECT_EQ(654, my_int64_array[3]);
109     EXPECT_EQ(321, my_int64_array[4]);
110   }
111 
112   // |UserPointer<>::Reader|:
113   {
114     UserPointer<int64_t>::Reader reader(my_int64_array_ptr, 5);
115     EXPECT_EQ(456, reader.GetPointer()[0]);
116     EXPECT_EQ(321, reader.GetPointer()[4]);
117   }
118 
119   // Non-const to const:
120   {
121     UserPointer<const int64_t>::Reader reader(my_int64_array_ptr.At(3), 1);
122     const int64_t* ptr = reader.GetPointer();
123     EXPECT_EQ(654, *ptr);
124   }
125 
126   // |UserPointer<>::Writer|:
127   {
128     UserPointer<int64_t>::Writer writer(my_int64_array_ptr.At(2), 1);
129     int64_t* ptr = writer.GetPointer();
130     *ptr = 1234567890123LL;
131     writer.Commit();
132     EXPECT_EQ(1234567890123LL, my_int64_array[2]);
133   }
134 
135   // |UserPointer<>::ReaderWriter|:
136   {
137     UserPointer<int32_t>::ReaderWriter reader_writer(my_int32_ptr, 1);
138     int32_t* ptr = reader_writer.GetPointer();
139     EXPECT_EQ(-456, *ptr);
140     *ptr = 42;
141     reader_writer.Commit();
142     EXPECT_EQ(42, my_int32);
143   }
144 
145   // |UserPointer<>::ReinterpretCast<>|:
146   // (This assumes little-endian, etc.)
147   {
148     UserPointer<const char> other(my_int32_ptr.ReinterpretCast<char>());
149     EXPECT_EQ(42, other.Get());
150     EXPECT_EQ(0, other.At(1).Get());
151     EXPECT_EQ(0, other.At(2).Get());
152     EXPECT_EQ(0, other.At(3).Get());
153   }
154 
155   // |UserPointer<>::GetPointerValue()|:
156   {
157     UserPointer<int32_t> other;
158     EXPECT_EQ(0u, other.GetPointerValue());
159     other = my_int32_ptr;
160     EXPECT_EQ(reinterpret_cast<uintptr_t>(&my_int32), other.GetPointerValue());
161   }
162 }
163 
TEST(MemoryTest,InvalidDeath)164 TEST(MemoryTest, InvalidDeath) {
165   const char kMemoryCheckFailedRegex[] = "Check failed";
166 
167   // Note: |Check...()| are defined to be "best effort" checks (and may always
168   // return true). Thus these tests of invalid cases only reflect the current
169   // implementation.
170 
171   // These tests depend on |int32_t| and |int64_t| having nontrivial alignment.
172   static_assert(MOJO_ALIGNOF(int32_t) != 1,
173                 "int32_t does not require nontrivial alignment");
174   static_assert(MOJO_ALIGNOF(int64_t) != 1,
175                 "int64_t does not require nontrivial alignment");
176 
177   // Null:
178   {
179     UserPointer<char> ptr(nullptr);
180     char array[5] = {};
181     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
182     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
183     EXPECT_DEATH_IF_SUPPORTED(ptr.Put('x'), kMemoryCheckFailedRegex);
184     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
185     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
186     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(array, 5), kMemoryCheckFailedRegex);
187   }
188   {
189     UserPointer<int32_t> ptr(nullptr);
190     int32_t array[5] = {};
191     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
192     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
193     EXPECT_DEATH_IF_SUPPORTED(ptr.Put(123), kMemoryCheckFailedRegex);
194     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
195     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
196     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(array, 5), kMemoryCheckFailedRegex);
197   }
198   {
199     UserPointer<int64_t> ptr(nullptr);
200     int64_t array[5] = {};
201     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
202     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
203     EXPECT_DEATH_IF_SUPPORTED(ptr.Put(123), kMemoryCheckFailedRegex);
204     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
205     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
206     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(array, 5), kMemoryCheckFailedRegex);
207   }
208   // Also check a const pointer:
209   {
210     UserPointer<const int32_t> ptr(nullptr);
211     int32_t array[5] = {};
212     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
213     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
214     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
215     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
216   }
217 
218   // Unaligned:
219   {
220     int32_t x[10];
221     UserPointer<int32_t> ptr(
222         reinterpret_cast<int32_t*>(reinterpret_cast<uintptr_t>(x) + 1));
223     int32_t array[5] = {};
224     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
225     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
226     EXPECT_DEATH_IF_SUPPORTED(ptr.Put(123), kMemoryCheckFailedRegex);
227     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
228     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
229     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(array, 5), kMemoryCheckFailedRegex);
230   }
231   {
232     int64_t x[10];
233     UserPointer<int64_t> ptr(
234         reinterpret_cast<int64_t*>(reinterpret_cast<uintptr_t>(x) + 1));
235     int64_t array[5] = {};
236     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
237     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
238     EXPECT_DEATH_IF_SUPPORTED(ptr.Put(123), kMemoryCheckFailedRegex);
239     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
240     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
241     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(array, 5), kMemoryCheckFailedRegex);
242   }
243   // Also check a const pointer:
244   {
245     int32_t x[10];
246     UserPointer<const int32_t> ptr(
247         reinterpret_cast<const int32_t*>(reinterpret_cast<uintptr_t>(x) + 1));
248     int32_t array[5] = {};
249     EXPECT_DEATH_IF_SUPPORTED(ptr.Check(), kMemoryCheckFailedRegex);
250     EXPECT_DEATH_IF_SUPPORTED(ptr.Get(), kMemoryCheckFailedRegex);
251     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(5), kMemoryCheckFailedRegex);
252     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(array, 5), kMemoryCheckFailedRegex);
253   }
254 
255   // Count too big:
256   {
257     const size_t kTooBig =
258         std::numeric_limits<size_t>::max() / sizeof(int32_t) + 1;
259     int32_t x = 0;
260     UserPointer<int32_t> ptr(&x);
261     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(kTooBig), kMemoryCheckFailedRegex);
262     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(&x, kTooBig),
263                               kMemoryCheckFailedRegex);
264     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(&x, kTooBig),
265                               kMemoryCheckFailedRegex);
266   }
267   {
268     const size_t kTooBig =
269         std::numeric_limits<size_t>::max() / sizeof(int64_t) + 1;
270     int64_t x = 0;
271     UserPointer<int64_t> ptr(&x);
272     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(kTooBig), kMemoryCheckFailedRegex);
273     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(&x, kTooBig),
274                               kMemoryCheckFailedRegex);
275     EXPECT_DEATH_IF_SUPPORTED(ptr.PutArray(&x, kTooBig),
276                               kMemoryCheckFailedRegex);
277   }
278   // Also check a const pointer:
279   {
280     const size_t kTooBig =
281         std::numeric_limits<size_t>::max() / sizeof(int32_t) + 1;
282     int32_t x = 0;
283     UserPointer<const int32_t> ptr(&x);
284     EXPECT_DEATH_IF_SUPPORTED(ptr.CheckArray(kTooBig), kMemoryCheckFailedRegex);
285     EXPECT_DEATH_IF_SUPPORTED(ptr.GetArray(&x, kTooBig),
286                               kMemoryCheckFailedRegex);
287   }
288 
289   // TODO(vtl): Tests for |UserPointer{Reader,Writer,ReaderWriter}|.
290 }
291 
292 }  // namespace
293 }  // namespace system
294 }  // namespace mojo
295