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 "ppapi/tests/test_file_mapping.h"
6
7 #include <string.h>
8
9 #include <limits>
10 #include <string>
11
12 #include "ppapi/c/pp_errors.h"
13 #include "ppapi/c/ppb_file_io.h"
14 #include "ppapi/c/ppb_file_mapping.h"
15 #include "ppapi/cpp/file_io.h"
16 #include "ppapi/cpp/file_ref.h"
17 #include "ppapi/cpp/file_system.h"
18 #include "ppapi/cpp/instance.h"
19 #include "ppapi/cpp/module.h"
20 #include "ppapi/tests/test_utils.h"
21
22 REGISTER_TEST_CASE(FileMapping);
23
24 namespace {
25
26 // TODO(dmichael): Move these to test_utils so we can share them?
ReadEntireFile(PP_Instance instance,pp::FileIO * file_io,int32_t offset,std::string * data,CallbackType callback_type)27 int32_t ReadEntireFile(PP_Instance instance,
28 pp::FileIO* file_io,
29 int32_t offset,
30 std::string* data,
31 CallbackType callback_type) {
32 TestCompletionCallback callback(instance, callback_type);
33 char buf[256];
34 int32_t read_offset = offset;
35
36 for (;;) {
37 callback.WaitForResult(
38 file_io->Read(read_offset, buf, sizeof(buf), callback.GetCallback()));
39 if (callback.result() < 0)
40 return callback.result();
41 if (callback.result() == 0)
42 break;
43 read_offset += callback.result();
44 data->append(buf, callback.result());
45 }
46
47 return PP_OK;
48 }
49
WriteEntireBuffer(PP_Instance instance,pp::FileIO * file_io,int32_t offset,const std::string & data,CallbackType callback_type)50 int32_t WriteEntireBuffer(PP_Instance instance,
51 pp::FileIO* file_io,
52 int32_t offset,
53 const std::string& data,
54 CallbackType callback_type) {
55 TestCompletionCallback callback(instance, callback_type);
56 int32_t write_offset = offset;
57 const char* buf = data.c_str();
58 int32_t size = data.size();
59
60 while (write_offset < offset + size) {
61 callback.WaitForResult(file_io->Write(write_offset,
62 &buf[write_offset - offset],
63 size - write_offset + offset,
64 callback.GetCallback()));
65 if (callback.result() < 0)
66 return callback.result();
67 if (callback.result() == 0)
68 return PP_ERROR_FAILED;
69 write_offset += callback.result();
70 }
71 callback.WaitForResult(file_io->Flush(callback.GetCallback()));
72 return callback.result();
73 }
74
75 } // namespace
76
MapAndCheckResults(uint32_t prot,uint32_t flags)77 std::string TestFileMapping::MapAndCheckResults(uint32_t prot,
78 uint32_t flags) {
79 TestCompletionCallback callback(instance_->pp_instance(), callback_type());
80
81 pp::FileSystem file_system(instance_, PP_FILESYSTEMTYPE_LOCALTEMPORARY);
82 pp::FileRef file_ref(file_system, "/mapped_file");
83
84 callback.WaitForResult(file_system.Open(1024, callback.GetCallback()));
85 ASSERT_EQ(PP_OK, callback.result());
86
87 const int64_t page_size =
88 file_mapping_if_->GetMapPageSize(instance_->pp_instance());
89 const int64_t kNumPages = 4;
90 // Make a string that's big enough that it spans all of the first |n-1| pages,
91 // plus a little bit of the |nth| page.
92 std::string file_contents((page_size * (kNumPages - 1)) + 128, 'a');
93
94 pp::FileIO file_io(instance_);
95 callback.WaitForResult(file_io.Open(file_ref,
96 PP_FILEOPENFLAG_CREATE |
97 PP_FILEOPENFLAG_TRUNCATE |
98 PP_FILEOPENFLAG_READ |
99 PP_FILEOPENFLAG_WRITE,
100 callback.GetCallback()));
101 ASSERT_EQ(PP_OK, callback.result());
102 ASSERT_EQ(PP_OK, WriteEntireBuffer(instance_->pp_instance(),
103 &file_io,
104 0,
105 file_contents,
106 callback_type()));
107
108 // TODO(dmichael): Use C++ interface.
109 void* address = NULL;
110 callback.WaitForResult(
111 file_mapping_if_->Map(
112 instance_->pp_instance(),
113 file_io.pp_resource(),
114 kNumPages * page_size,
115 prot,
116 flags,
117 0,
118 &address,
119 callback.GetCallback().pp_completion_callback()));
120 CHECK_CALLBACK_BEHAVIOR(callback);
121 ASSERT_EQ(PP_OK, callback.result());
122 ASSERT_NE(NULL, address);
123
124 if (prot & PP_FILEMAPPROTECTION_READ) {
125 // Make sure we can read.
126 std::string mapped_data(static_cast<char*>(address), file_contents.size());
127 // The initial data should match.
128 ASSERT_EQ(file_contents, mapped_data);
129
130 // Now write some data and flush it.
131 const std::string file_contents2(file_contents.size(), 'x');
132 ASSERT_EQ(PP_OK, WriteEntireBuffer(instance_->pp_instance(),
133 &file_io,
134 0,
135 file_contents2,
136 callback_type()));
137 // If the region was mapped SHARED, it should get updated.
138 std::string mapped_data2(static_cast<char*>(address), file_contents.size());
139 if (flags & PP_FILEMAPFLAG_SHARED)
140 ASSERT_EQ(file_contents2, mapped_data2);
141 // In POSIX, it is unspecified in the PRIVATE case whether changes to the
142 // file are visible to the mapped region. So we can't really test anything
143 // here in that case.
144 // TODO(dmichael): Make sure our Pepper documentation reflects this.
145 }
146 if (prot & PP_FILEMAPPROTECTION_WRITE) {
147 std::string old_file_contents;
148 ASSERT_EQ(PP_OK, ReadEntireFile(instance_->pp_instance(),
149 &file_io,
150 0,
151 &old_file_contents,
152 callback_type()));
153 // Write something else to the mapped region, then unmap, and see if it
154 // gets written to the file. (Note we have to Unmap to make sure that the
155 // write is committed).
156 memset(address, 'y', file_contents.size());
157 // Note, we might not have read access to the mapped region here, so we
158 // make a string with the same contents without actually reading.
159 std::string mapped_data3(file_contents.size(), 'y');
160 callback.WaitForResult(
161 file_mapping_if_->Unmap(
162 instance_->pp_instance(), address, file_contents.size(),
163 callback.GetCallback().pp_completion_callback()));
164 CHECK_CALLBACK_BEHAVIOR(callback);
165 ASSERT_EQ(PP_OK, callback.result());
166 std::string new_file_contents;
167 ASSERT_EQ(PP_OK, ReadEntireFile(instance_->pp_instance(),
168 &file_io,
169 0,
170 &new_file_contents,
171 callback_type()));
172
173 // Sanity-check that the data we wrote isn't the same as what was already
174 // there, otherwise our test is invalid.
175 ASSERT_NE(mapped_data3, old_file_contents);
176 // If it's SHARED, the file should match what we wrote to the mapped region.
177 // Otherwise, it should not have changed.
178 if (flags & PP_FILEMAPFLAG_SHARED)
179 ASSERT_EQ(mapped_data3, new_file_contents);
180 else
181 ASSERT_EQ(old_file_contents, new_file_contents);
182 } else {
183 // We didn't do the "WRITE" test, but we still want to Unmap.
184 callback.WaitForResult(
185 file_mapping_if_->Unmap(
186 instance_->pp_instance(), address, file_contents.size(),
187 callback.GetCallback().pp_completion_callback()));
188 CHECK_CALLBACK_BEHAVIOR(callback);
189 ASSERT_EQ(PP_OK, callback.result());
190 }
191 PASS();
192 }
193
Init()194 bool TestFileMapping::Init() {
195 // TODO(dmichael): Use unversioned string when this goes to stable?
196 file_mapping_if_ = static_cast<const PPB_FileMapping_0_1*>(
197 pp::Module::Get()->GetBrowserInterface(PPB_FILEMAPPING_INTERFACE_0_1));
198 return !!file_mapping_if_ && CheckTestingInterface() &&
199 EnsureRunningOverHTTP();
200 }
201
RunTests(const std::string & filter)202 void TestFileMapping::RunTests(const std::string& filter) {
203 RUN_CALLBACK_TEST(TestFileMapping, BadParameters, filter);
204 RUN_CALLBACK_TEST(TestFileMapping, Map, filter);
205 RUN_CALLBACK_TEST(TestFileMapping, PartialRegions, filter);
206 }
207
TestBadParameters()208 std::string TestFileMapping::TestBadParameters() {
209 TestCompletionCallback callback(instance_->pp_instance(), callback_type());
210
211 pp::FileSystem file_system(instance_, PP_FILESYSTEMTYPE_LOCALTEMPORARY);
212 pp::FileRef file_ref(file_system, "/mapped_file");
213
214 callback.WaitForResult(file_system.Open(1024, callback.GetCallback()));
215 ASSERT_EQ(PP_OK, callback.result());
216
217 const int64_t page_size =
218 file_mapping_if_->GetMapPageSize(instance_->pp_instance());
219 // const int64_t kNumPages = 4;
220 // Make a string that's big enough that it spans 3 pages, plus a little extra.
221 //std::string file_contents((page_size * (kNumPages - 1)) + 128, 'a');
222 std::string file_contents(page_size, 'a');
223
224 pp::FileIO file_io(instance_);
225 callback.WaitForResult(file_io.Open(file_ref,
226 PP_FILEOPENFLAG_CREATE |
227 PP_FILEOPENFLAG_TRUNCATE |
228 PP_FILEOPENFLAG_READ |
229 PP_FILEOPENFLAG_WRITE,
230 callback.GetCallback()));
231 ASSERT_EQ(PP_OK, callback.result());
232 ASSERT_EQ(PP_OK, WriteEntireBuffer(instance_->pp_instance(),
233 &file_io,
234 0,
235 file_contents,
236 callback_type()));
237
238 // Bad instance.
239 void* address = NULL;
240 callback.WaitForResult(
241 file_mapping_if_->Map(
242 PP_Instance(0xbadbad),
243 file_io.pp_resource(),
244 page_size,
245 PP_FILEMAPPROTECTION_READ,
246 PP_FILEMAPFLAG_PRIVATE,
247 0,
248 &address,
249 callback.GetCallback().pp_completion_callback()));
250 CHECK_CALLBACK_BEHAVIOR(callback);
251 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
252 ASSERT_EQ(NULL, address);
253
254 // Bad resource.
255 callback.WaitForResult(
256 file_mapping_if_->Map(
257 instance_->pp_instance(),
258 PP_Resource(0xbadbad),
259 page_size,
260 PP_FILEMAPPROTECTION_READ,
261 PP_FILEMAPFLAG_PRIVATE,
262 0,
263 &address,
264 callback.GetCallback().pp_completion_callback()));
265 CHECK_CALLBACK_BEHAVIOR(callback);
266 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
267 ASSERT_EQ(NULL, address);
268
269 // Length too big.
270 callback.WaitForResult(
271 file_mapping_if_->Map(
272 instance_->pp_instance(),
273 file_io.pp_resource(),
274 std::numeric_limits<int64_t>::max(),
275 PP_FILEMAPPROTECTION_READ,
276 PP_FILEMAPFLAG_PRIVATE,
277 0,
278 &address,
279 callback.GetCallback().pp_completion_callback()));
280 CHECK_CALLBACK_BEHAVIOR(callback);
281 ASSERT_EQ(PP_ERROR_NOMEMORY, callback.result());
282 ASSERT_EQ(NULL, address);
283
284 // Length too small.
285 callback.WaitForResult(
286 file_mapping_if_->Map(
287 instance_->pp_instance(),
288 file_io.pp_resource(),
289 -1,
290 PP_FILEMAPPROTECTION_READ,
291 PP_FILEMAPFLAG_PRIVATE,
292 0,
293 &address,
294 callback.GetCallback().pp_completion_callback()));
295 CHECK_CALLBACK_BEHAVIOR(callback);
296 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
297 ASSERT_EQ(NULL, address);
298 // TODO(dmichael): Check & test length that is not a multiple of page size???
299
300 // Bad protection.
301 callback.WaitForResult(
302 file_mapping_if_->Map(
303 instance_->pp_instance(),
304 file_io.pp_resource(),
305 page_size,
306 ~static_cast<uint32_t>(PP_FILEMAPPROTECTION_READ),
307 PP_FILEMAPFLAG_PRIVATE,
308 0,
309 &address,
310 callback.GetCallback().pp_completion_callback()));
311 CHECK_CALLBACK_BEHAVIOR(callback);
312 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
313 ASSERT_EQ(NULL, address);
314
315 // No flags.
316 callback.WaitForResult(
317 file_mapping_if_->Map(
318 instance_->pp_instance(),
319 file_io.pp_resource(),
320 page_size,
321 PP_FILEMAPPROTECTION_READ,
322 0,
323 0,
324 &address,
325 callback.GetCallback().pp_completion_callback()));
326 CHECK_CALLBACK_BEHAVIOR(callback);
327 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
328 ASSERT_EQ(NULL, address);
329
330 // Both flags.
331 callback.WaitForResult(
332 file_mapping_if_->Map(
333 instance_->pp_instance(),
334 file_io.pp_resource(),
335 page_size,
336 PP_FILEMAPPROTECTION_READ,
337 PP_FILEMAPFLAG_SHARED | PP_FILEMAPFLAG_PRIVATE,
338 0,
339 &address,
340 callback.GetCallback().pp_completion_callback()));
341 CHECK_CALLBACK_BEHAVIOR(callback);
342 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
343 ASSERT_EQ(NULL, address);
344
345 // Bad flags.
346 callback.WaitForResult(
347 file_mapping_if_->Map(
348 instance_->pp_instance(),
349 file_io.pp_resource(),
350 page_size,
351 PP_FILEMAPPROTECTION_READ,
352 ~static_cast<uint32_t>(PP_FILEMAPFLAG_SHARED),
353 0,
354 &address,
355 callback.GetCallback().pp_completion_callback()));
356 CHECK_CALLBACK_BEHAVIOR(callback);
357 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
358 ASSERT_EQ(NULL, address);
359
360 // Bad offset; not a multiple of page size.
361 callback.WaitForResult(
362 file_mapping_if_->Map(
363 instance_->pp_instance(),
364 file_io.pp_resource(),
365 page_size,
366 PP_FILEMAPPROTECTION_READ,
367 ~static_cast<uint32_t>(PP_FILEMAPFLAG_SHARED),
368 1,
369 &address,
370 callback.GetCallback().pp_completion_callback()));
371 CHECK_CALLBACK_BEHAVIOR(callback);
372 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
373 ASSERT_EQ(NULL, address);
374
375 // Unmap NULL.
376 callback.WaitForResult(
377 file_mapping_if_->Unmap(
378 instance_->pp_instance(),
379 NULL,
380 page_size,
381 callback.GetCallback().pp_completion_callback()));
382 CHECK_CALLBACK_BEHAVIOR(callback);
383 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
384
385 // Unmap bad address.
386 callback.WaitForResult(
387 file_mapping_if_->Unmap(
388 instance_->pp_instance(),
389 reinterpret_cast<const void*>(0xdeadbeef),
390 page_size,
391 callback.GetCallback().pp_completion_callback()));
392 CHECK_CALLBACK_BEHAVIOR(callback);
393 ASSERT_EQ(PP_ERROR_BADARGUMENT, callback.result());
394
395 PASS();
396 }
397
TestMap()398 std::string TestFileMapping::TestMap() {
399 ASSERT_SUBTEST_SUCCESS(MapAndCheckResults(PP_FILEMAPPROTECTION_READ,
400 PP_FILEMAPFLAG_SHARED));
401 ASSERT_SUBTEST_SUCCESS(MapAndCheckResults(PP_FILEMAPPROTECTION_WRITE,
402 PP_FILEMAPFLAG_SHARED));
403 ASSERT_SUBTEST_SUCCESS(
404 MapAndCheckResults(PP_FILEMAPPROTECTION_WRITE | PP_FILEMAPPROTECTION_READ,
405 PP_FILEMAPFLAG_SHARED));
406 ASSERT_SUBTEST_SUCCESS(MapAndCheckResults(PP_FILEMAPPROTECTION_READ,
407 PP_FILEMAPFLAG_PRIVATE));
408 ASSERT_SUBTEST_SUCCESS(MapAndCheckResults(PP_FILEMAPPROTECTION_WRITE,
409 PP_FILEMAPFLAG_PRIVATE));
410 ASSERT_SUBTEST_SUCCESS(
411 MapAndCheckResults(PP_FILEMAPPROTECTION_WRITE | PP_FILEMAPPROTECTION_READ,
412 PP_FILEMAPFLAG_PRIVATE));
413 PASS();
414 }
415
TestPartialRegions()416 std::string TestFileMapping::TestPartialRegions() {
417 TestCompletionCallback callback(instance_->pp_instance(), callback_type());
418
419 pp::FileSystem file_system(instance_, PP_FILESYSTEMTYPE_LOCALTEMPORARY);
420 pp::FileRef file_ref1(file_system, "/mapped_file1");
421 pp::FileRef file_ref2(file_system, "/mapped_file2");
422
423 callback.WaitForResult(file_system.Open(1024, callback.GetCallback()));
424 ASSERT_EQ(PP_OK, callback.result());
425
426 const int64_t page_size =
427 file_mapping_if_->GetMapPageSize(instance_->pp_instance());
428 const int64_t kNumPages = 3;
429 std::string file_contents1(kNumPages * page_size, 'a');
430
431 pp::FileIO file_io1(instance_);
432 callback.WaitForResult(file_io1.Open(file_ref1,
433 PP_FILEOPENFLAG_CREATE |
434 PP_FILEOPENFLAG_TRUNCATE |
435 PP_FILEOPENFLAG_READ |
436 PP_FILEOPENFLAG_WRITE,
437 callback.GetCallback()));
438 ASSERT_EQ(PP_OK, callback.result());
439 ASSERT_EQ(PP_OK, WriteEntireBuffer(instance_->pp_instance(),
440 &file_io1,
441 0,
442 file_contents1,
443 callback_type()));
444
445 // TODO(dmichael): Use C++ interface.
446 void* address = NULL;
447 callback.WaitForResult(
448 file_mapping_if_->Map(
449 instance_->pp_instance(),
450 file_io1.pp_resource(),
451 kNumPages * page_size,
452 PP_FILEMAPPROTECTION_WRITE | PP_FILEMAPPROTECTION_READ,
453 PP_FILEMAPFLAG_SHARED,
454 0,
455 &address,
456 callback.GetCallback().pp_completion_callback()));
457 CHECK_CALLBACK_BEHAVIOR(callback);
458 ASSERT_EQ(PP_OK, callback.result());
459 ASSERT_NE(NULL, address);
460
461 // Unmap only the middle page.
462 void* address_of_middle_page =
463 reinterpret_cast<void*>(reinterpret_cast<uintptr_t>(address) + page_size);
464 callback.WaitForResult(
465 file_mapping_if_->Unmap(
466 instance_->pp_instance(),
467 address_of_middle_page,
468 page_size,
469 callback.GetCallback().pp_completion_callback()));
470 CHECK_CALLBACK_BEHAVIOR(callback);
471 ASSERT_EQ(PP_OK, callback.result());
472
473 // Write another file, map it in to the middle hole that was left above.
474 pp::FileIO file_io2(instance_);
475 callback.WaitForResult(file_io2.Open(file_ref2,
476 PP_FILEOPENFLAG_CREATE |
477 PP_FILEOPENFLAG_TRUNCATE |
478 PP_FILEOPENFLAG_READ |
479 PP_FILEOPENFLAG_WRITE,
480 callback.GetCallback()));
481 ASSERT_EQ(PP_OK, callback.result());
482 // This second file will have 1 page worth of data.
483 std::string file_contents2(page_size, 'b');
484 ASSERT_EQ(PP_OK, WriteEntireBuffer(instance_->pp_instance(),
485 &file_io2,
486 0,
487 file_contents2,
488 callback_type()));
489 callback.WaitForResult(
490 file_mapping_if_->Map(
491 instance_->pp_instance(),
492 file_io2.pp_resource(),
493 page_size,
494 PP_FILEMAPPROTECTION_WRITE | PP_FILEMAPPROTECTION_READ,
495 PP_FILEMAPFLAG_SHARED | PP_FILEMAPFLAG_FIXED,
496 0,
497 &address_of_middle_page,
498 callback.GetCallback().pp_completion_callback()));
499 CHECK_CALLBACK_BEHAVIOR(callback);
500 ASSERT_EQ(PP_OK, callback.result());
501
502 // Write something else to the mapped region, then unmap, and see if it
503 // gets written to both files. (Note we have to Unmap to make sure that the
504 // write is committed).
505 memset(address, 'c', kNumPages * page_size);
506 callback.WaitForResult(
507 file_mapping_if_->Unmap(
508 instance_->pp_instance(), address, kNumPages * page_size,
509 callback.GetCallback().pp_completion_callback()));
510 CHECK_CALLBACK_BEHAVIOR(callback);
511 ASSERT_EQ(PP_OK, callback.result());
512 // The first and third page should have been written with 'c', but the
513 // second page should be untouched.
514 std::string expected_file_contents1 = std::string(page_size, 'c') +
515 std::string(page_size, 'a') +
516 std::string(page_size, 'c');
517 std::string new_file_contents1;
518 ASSERT_EQ(PP_OK, ReadEntireFile(instance_->pp_instance(),
519 &file_io1,
520 0,
521 &new_file_contents1,
522 callback_type()));
523 ASSERT_EQ(expected_file_contents1, new_file_contents1);
524
525 // The second file should have been entirely over-written.
526 std::string expected_file_contents2 = std::string(page_size, 'c');
527 std::string new_file_contents2;
528 ASSERT_EQ(PP_OK, ReadEntireFile(instance_->pp_instance(),
529 &file_io2,
530 0,
531 &new_file_contents2,
532 callback_type()));
533 ASSERT_EQ(expected_file_contents2, new_file_contents2);
534
535 // TODO(dmichael): Test non-zero offset
536
537 PASS();
538 }
539
540