ppoll.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. #include <errno.h>
  2. #define _GNU_SOURCE
  3. #include <fcntl.h>
  4. #include <poll.h>
  5. #include <signal.h>
  6. #include <stdio.h>
  7. #include <stdlib.h>
  8. #include <string.h>
  9. #include <sys/poll.h>
  10. #include <sys/signalfd.h>
  11. #include <sys/syscall.h>
  12. #include <sys/wait.h>
  13. #include <time.h>
  14. #include <unistd.h>
  15. #define RED "\x1B[31m"
  16. #define GREEN "\x1B[32m"
  17. #define RESET "\x1B[0m"
  18. // 测试用例1:基本功能测试(管道I/O)
  19. void test_basic_functionality() {
  20. int pipefd[2];
  21. struct pollfd fds[1];
  22. struct timespec timeout = {5, 0}; // 5秒超时
  23. printf("=== Test 1: Basic functionality test ===\n");
  24. // 创建管道
  25. if (pipe(pipefd) == -1) {
  26. perror("pipe creation failed");
  27. exit(EXIT_FAILURE);
  28. }
  29. // 设置监听读端管道
  30. fds[0].fd = pipefd[0];
  31. fds[0].events = POLLIN;
  32. printf("Test scenario 1: Wait with no data (should timeout)\n");
  33. int ret = ppoll(fds, 1, &timeout, NULL);
  34. if (ret == 0) {
  35. printf(GREEN "Test passed: Correct timeout\n" RESET);
  36. } else {
  37. printf(RED "Test failed: Return value %d\n" RESET, ret);
  38. }
  39. // 向管道写入数据
  40. const char *msg = "test data";
  41. write(pipefd[1], msg, strlen(msg));
  42. printf(
  43. "\nTest scenario 2: Should return immediately when data is available\n");
  44. timeout.tv_sec = 5;
  45. ret = ppoll(fds, 1, &timeout, NULL);
  46. if (ret > 0 && (fds[0].revents & POLLIN)) {
  47. printf(GREEN "Test passed: Data detected\n" RESET);
  48. } else {
  49. printf(RED "Test failed: Return value %d, revents %d\n" RESET, ret,
  50. fds[0].revents);
  51. }
  52. close(pipefd[0]);
  53. close(pipefd[1]);
  54. }
  55. // 测试用例2:信号屏蔽测试
  56. void test_signal_handling() {
  57. printf("\n=== Test 2: Signal handling test ===\n");
  58. sigset_t mask, orig_mask;
  59. struct timespec timeout = {5, 0};
  60. struct pollfd fds[1];
  61. fds[0].fd = -1;
  62. fds[0].events = 0;
  63. // 设置信号屏蔽
  64. sigemptyset(&mask);
  65. sigaddset(&mask, SIGUSR1);
  66. // 阻塞SIGUSR1,并保存原来的信号掩码
  67. if (sigprocmask(SIG_BLOCK, &mask, &orig_mask)) {
  68. perror("sigprocmask");
  69. exit(EXIT_FAILURE);
  70. }
  71. printf("Test scenario: Signal should not interrupt when masked\n");
  72. pid_t pid = fork();
  73. if (pid == 0) { // 子进程
  74. sleep(2); // 等待父进程进入ppoll
  75. kill(getppid(), SIGUSR1);
  76. exit(0);
  77. }
  78. int ret = ppoll(fds, 1, &timeout, &mask);
  79. if (ret == 0) {
  80. printf(GREEN "Test passed: Completed full 5 second wait\n" RESET);
  81. } else {
  82. printf(RED "Test failed: Premature return %d\n" RESET, errno);
  83. }
  84. waitpid(pid, NULL, 0);
  85. // 检查并消费挂起的SIGUSR1信号
  86. sigset_t pending;
  87. sigpending(&pending);
  88. if (sigismember(&pending, SIGUSR1)) {
  89. int sig;
  90. sigwait(&mask, &sig); // 主动消费信号
  91. printf("Consumed pending SIGUSR1 signal\n");
  92. }
  93. // 恢复原来的信号掩码
  94. sigprocmask(SIG_SETMASK, &orig_mask, NULL);
  95. }
  96. // 测试用例3:精确超时测试
  97. void test_timeout_accuracy() {
  98. printf("\n=== Test 3: Timeout accuracy test ===\n");
  99. struct timespec start, end, timeout = {0, 500000000};
  100. struct pollfd fds[1];
  101. fds[0].fd = -1;
  102. fds[0].events = 0;
  103. clock_gettime(CLOCK_MONOTONIC, &start);
  104. int ret = ppoll(fds, 1, &timeout, NULL);
  105. clock_gettime(CLOCK_MONOTONIC, &end);
  106. long elapsed = (end.tv_sec - start.tv_sec) * 1000000 +
  107. (end.tv_nsec - start.tv_nsec) / 1000;
  108. printf("Expected timeout: 500ms, Actual elapsed: %.3fms\n", elapsed / 1000.0);
  109. if (labs(elapsed - 500000) < 50000) { // 允许±50ms误差
  110. printf(GREEN "Test passed: Timeout within acceptable range\n" RESET);
  111. } else {
  112. printf(RED "Test failed: Timeout deviation too large\n" RESET);
  113. }
  114. }
  115. int main() {
  116. // 设置非阻塞标准输入
  117. fcntl(STDIN_FILENO, F_SETFL, O_NONBLOCK);
  118. test_basic_functionality();
  119. test_signal_handling();
  120. test_timeout_accuracy();
  121. return 0;
  122. }