test_io.cpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. #include <AK/Assertions.h>
  2. #include <AK/Types.h>
  3. #include <fcntl.h>
  4. #include <stdio.h>
  5. #include <sys/mman.h>
  6. #include <unistd.h>
  7. #define EXPECT_ERROR_2(err, syscall, arg1, arg2) \
  8. do { \
  9. rc = syscall(arg1, arg2); \
  10. if (rc >= 0 || errno != err) { \
  11. fprintf(stderr, __FILE__ ":%d: Expected " #err ": " #syscall "(%p, %p), got rc=%d, errno=%d\n", __LINE__, (const void*)(arg1), (const void*)arg2, rc, errno); \
  12. } \
  13. } while (0)
  14. #define EXPECT_ERROR_3(err, syscall, arg1, arg2, arg3) \
  15. do { \
  16. rc = syscall(arg1, arg2, arg3); \
  17. if (rc >= 0 || errno != err) { \
  18. fprintf(stderr, __FILE__ ":%d: Expected " #err ": " #syscall "(%p, %p, %p), got rc=%d, errno=%d\n", __LINE__, (const void*)(arg1), (const void*)(arg2), (const void*)(arg3), rc, errno); \
  19. } \
  20. } while (0)
  21. void test_read_from_directory()
  22. {
  23. char buffer[BUFSIZ];
  24. int fd = open("/", O_DIRECTORY | O_RDONLY);
  25. ASSERT(fd >= 0);
  26. int rc;
  27. EXPECT_ERROR_3(EISDIR, read, fd, buffer, sizeof(buffer));
  28. rc = close(fd);
  29. ASSERT(rc == 0);
  30. }
  31. void test_write_to_directory()
  32. {
  33. char str[] = "oh frick";
  34. int fd = open("/", O_DIRECTORY | O_RDONLY);
  35. if (fd < 0)
  36. perror("open");
  37. ASSERT(fd >= 0);
  38. int rc;
  39. EXPECT_ERROR_3(EBADF, write, fd, str, sizeof(str));
  40. rc = close(fd);
  41. ASSERT(rc == 0);
  42. }
  43. void test_read_from_writeonly()
  44. {
  45. char buffer[BUFSIZ];
  46. int fd = open("/tmp/xxxx123", O_CREAT | O_WRONLY);
  47. ASSERT(fd >= 0);
  48. int rc;
  49. EXPECT_ERROR_3(EBADF, read, fd, buffer, sizeof(buffer));
  50. rc = close(fd);
  51. ASSERT(rc == 0);
  52. }
  53. void test_write_to_readonly()
  54. {
  55. char str[] = "hello";
  56. int fd = open("/tmp/abcd123", O_CREAT | O_RDONLY);
  57. ASSERT(fd >= 0);
  58. int rc;
  59. EXPECT_ERROR_3(EBADF, write, fd, str, sizeof(str));
  60. rc = close(fd);
  61. ASSERT(rc == 0);
  62. }
  63. void test_read_past_eof()
  64. {
  65. char buffer[BUFSIZ];
  66. int fd = open("/home/anon/myfile.txt", O_RDONLY);
  67. if (fd < 0)
  68. perror("open");
  69. ASSERT(fd >= 0);
  70. int rc;
  71. rc = lseek(fd, 9999, SEEK_SET);
  72. if (rc < 0)
  73. perror("lseek");
  74. rc = read(fd, buffer, sizeof(buffer));
  75. if (rc < 0)
  76. perror("read");
  77. if (rc > 0)
  78. fprintf(stderr, "read %d bytes past EOF\n", rc);
  79. rc = close(fd);
  80. ASSERT(rc == 0);
  81. }
  82. void test_ftruncate_readonly()
  83. {
  84. int fd = open("/tmp/trunctest", O_RDONLY | O_CREAT, 0666);
  85. ASSERT(fd >= 0);
  86. int rc;
  87. EXPECT_ERROR_2(EBADF, ftruncate, fd, 0);
  88. close(fd);
  89. }
  90. void test_ftruncate_negative()
  91. {
  92. int fd = open("/tmp/trunctest", O_RDWR | O_CREAT, 0666);
  93. ASSERT(fd >= 0);
  94. int rc;
  95. EXPECT_ERROR_2(EINVAL, ftruncate, fd, -1);
  96. close(fd);
  97. }
  98. void test_mmap_directory()
  99. {
  100. int fd = open("/tmp", O_RDONLY | O_DIRECTORY);
  101. ASSERT(fd >= 0);
  102. auto* ptr = mmap(nullptr, 4096, PROT_READ, MAP_FILE | MAP_SHARED, fd, 0);
  103. if (ptr != MAP_FAILED) {
  104. fprintf(stderr, "Boo! mmap() of a directory succeeded!\n");
  105. return;
  106. }
  107. if (errno != ENODEV) {
  108. fprintf(stderr, "Boo! mmap() of a directory gave errno=%d instead of ENODEV!\n", errno);
  109. return;
  110. }
  111. close(fd);
  112. }
  113. void test_tmpfs_read_past_end()
  114. {
  115. int fd = open("/tmp/x", O_RDWR | O_CREAT | O_TRUNC, 0600);
  116. ASSERT(fd >= 0);
  117. int rc = ftruncate(fd, 1);
  118. ASSERT(rc == 0);
  119. rc = lseek(fd, 4096, SEEK_SET);
  120. ASSERT(rc == 4096);
  121. char buffer[16];
  122. int nread = read(fd, buffer, sizeof(buffer));
  123. if (nread != 0) {
  124. fprintf(stderr, "Expected 0-length read past end of file in /tmp\n");
  125. }
  126. close(fd);
  127. }
  128. void test_procfs_read_past_end()
  129. {
  130. int fd = open("/proc/uptime", O_RDONLY);
  131. ASSERT(fd >= 0);
  132. int rc = lseek(fd, 4096, SEEK_SET);
  133. ASSERT(rc == 4096);
  134. char buffer[16];
  135. int nread = read(fd, buffer, sizeof(buffer));
  136. if (nread != 0) {
  137. fprintf(stderr, "Expected 0-length read past end of file in /proc\n");
  138. }
  139. close(fd);
  140. }
  141. int main(int, char**)
  142. {
  143. int rc;
  144. EXPECT_ERROR_2(ENOTDIR, open, "/dev/zero", (O_DIRECTORY | O_RDONLY));
  145. EXPECT_ERROR_2(EINVAL, open, "/dev/zero", (O_DIRECTORY | O_CREAT | O_RDWR));
  146. EXPECT_ERROR_2(EEXIST, open, "/dev/zero", (O_CREAT | O_EXCL | O_RDWR));
  147. EXPECT_ERROR_2(EINVAL, open, "/tmp/abcdef", (O_DIRECTORY | O_CREAT | O_RDWR));
  148. EXPECT_ERROR_2(EACCES, open, "/proc/all", (O_RDWR));
  149. EXPECT_ERROR_2(ENOENT, open, "/boof/baaf/nonexistent", (O_CREAT | O_RDWR));
  150. EXPECT_ERROR_2(EISDIR, open, "/tmp", (O_DIRECTORY | O_RDWR));
  151. test_read_from_directory();
  152. test_write_to_directory();
  153. test_read_from_writeonly();
  154. test_write_to_readonly();
  155. test_read_past_eof();
  156. test_ftruncate_readonly();
  157. test_ftruncate_negative();
  158. test_mmap_directory();
  159. test_tmpfs_read_past_end();
  160. test_procfs_read_past_end();
  161. return 0;
  162. }