Message ID | 20221027055428.406908-1-alistair.francis@opensource.wdc.com |
---|---|
State | New |
Headers | show |
Series | [v2] Tests: select_poll_epoll: Add support for _time64 | expand |
On Thu, Oct 27, 2022 at 3:54 PM Alistair Francis <alistair.francis at opensource.wdc.com> wrote: > > From: Alistair Francis <alistair.francis at wdc.com> > > Add support for the 64-bit time_t syscalls SYS_ppoll_time64 > and SYS_pselect6_time64. > > These are the syscalls that exist 32-bit platforms since the 5.1 kernel. > 32-bit platforms with a 64-bit time_t only have these and don't have the > original syscalls (such as 32-bit RISC-V). > > Fixes: https://github.com/lttng/lttng-tools/pull/162 > Signed-off-by: Alistair Francis <alistair.francis at wdc.com> Ping! Alistair > --- > To keep the test_cases[] array clean I have implemented the functions > for all builds, but the functions are a no-op if the syscall is missing. > > v2: > - Split out a seperate _time64 test > > tests/regression/kernel/select_poll_epoll.cpp | 184 ++++++++++++++++++ > 1 file changed, 184 insertions(+) > > diff --git a/tests/regression/kernel/select_poll_epoll.cpp b/tests/regression/kernel/select_poll_epoll.cpp > index c0b688217..dfaab52c8 100644 > --- a/tests/regression/kernel/select_poll_epoll.cpp > +++ b/tests/regression/kernel/select_poll_epoll.cpp > @@ -5,6 +5,7 @@ > * > */ > > +#include <errno.h> > #include <fcntl.h> > #include <limits.h> > #include <poll.h> > @@ -48,10 +49,14 @@ int lttng_opt_quiet, lttng_opt_verbose, lttng_opt_mi; > > static void run_working_cases(FILE *validation_output_file); > static void pselect_invalid_fd(FILE *validation_output_file); > +static void pselect_time64_invalid_fd(FILE *validation_output_file); > static void test_ppoll_big(FILE *validation_output_file); > static void ppoll_fds_buffer_overflow(FILE *validation_output_file); > +static void ppoll_time64_fds_buffer_overflow(FILE *validation_output_file); > static void pselect_invalid_pointer(FILE *validation_output_file); > +static void pselect_time64_invalid_pointer(FILE *validation_output_file); > static void ppoll_fds_ulong_max(FILE *validation_output_file); > +static void ppoll_time64_fds_ulong_max(FILE *validation_output_file); > static void epoll_pwait_invalid_pointer(FILE *validation_output_file); > static void epoll_pwait_int_max(FILE *validation_output_file); > static void ppoll_concurrent_write(FILE *validation_output_file); > @@ -69,10 +74,14 @@ const struct test_case { > { .run = run_working_cases, .produces_validation_info = true, .timeout = -1 }, > { .run = run_working_cases, .produces_validation_info = true, .timeout = 1 }, > { .run = pselect_invalid_fd, .produces_validation_info = false, .timeout = 0 }, > + { .run = pselect_time64_invalid_fd, .produces_validation_info = false, .timeout = 0 }, > { .run = test_ppoll_big, .produces_validation_info = false, .timeout = 0 }, > { .run = ppoll_fds_buffer_overflow, .produces_validation_info = false, .timeout = 0 }, > + { .run = ppoll_time64_fds_buffer_overflow, .produces_validation_info = false, .timeout = 0 }, > { .run = pselect_invalid_pointer, .produces_validation_info = false, .timeout = 0 }, > + { .run = pselect_time64_invalid_pointer, .produces_validation_info = false, .timeout = 0 }, > { .run = ppoll_fds_ulong_max, .produces_validation_info = false, .timeout = 0 }, > + { .run = ppoll_time64_fds_ulong_max, .produces_validation_info = false, .timeout = 0 }, > { .run = epoll_pwait_invalid_pointer, .produces_validation_info = true, .timeout = 0 }, > { .run = epoll_pwait_int_max, .produces_validation_info = true, .timeout = 0 }, > { .run = ppoll_concurrent_write, .produces_validation_info = false, .timeout = 0 }, > @@ -440,6 +449,44 @@ end: > return; > } > > +/* > + * Ask for 100 FDs in a buffer for allocated for only 1 FD, should > + * segfault (eventually with a "*** stack smashing detected ***" message). > + * The event should contain an array of 100 FDs filled with garbage. > + */ > +static > +void ppoll_time64_fds_buffer_overflow( > + FILE *validation_output_file __attribute__((unused))) > +{ > +#ifdef SYS_ppoll_time64 > + struct pollfd ufds[NB_FD]; > + char buf[BUF_SIZE]; > + int ret; > + > + ufds[0].fd = wait_fd; > + ufds[0].events = POLLIN|POLLPRI; > + > + /* > + * As there is no timeout value, we don't convert to/from > + * 64/32-bit time_t. > + */ > + ret = syscall(SYS_ppoll_time64, ufds, 100, NULL, NULL); > + /* > + * There is no fallback to SYS_ppoll, we expect SYS_ppoll_time64 > + * to work and if it doesn't we fail. > + */ > + > + if (ret < 0) { > + PERROR("ppoll_time64"); > + } else if (ret > 0) { > + ret = read(wait_fd, buf, BUF_SIZE); > + if (ret < 0) { > + PERROR("[ppoll_time64] read"); > + } > + } > +#endif > +} > + > /* > * Ask for 100 FDs in a buffer for allocated for only 1 FD, should > * segfault (eventually with a "*** stack smashing detected ***" message). > @@ -449,6 +496,7 @@ static > void ppoll_fds_buffer_overflow( > FILE *validation_output_file __attribute__((unused))) > { > +#ifdef SYS_ppoll > struct pollfd ufds[NB_FD]; > char buf[BUF_SIZE]; > int ret; > @@ -466,6 +514,46 @@ void ppoll_fds_buffer_overflow( > PERROR("[ppoll] read"); > } > } > +#endif > +} > + > +/* > + * Ask for ULONG_MAX FDs in a buffer for allocated for only 1 FD, should > + * cleanly fail with a "Invalid argument". > + * The event should contain an empty array of FDs and overflow = 1. > + */ > +static > +void ppoll_time64_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) > +{ > +#ifdef SYS_ppoll_time64 > + struct pollfd ufds[NB_FD]; > + char buf[BUF_SIZE]; > + int ret; > + > + ufds[0].fd = wait_fd; > + ufds[0].events = POLLIN|POLLPRI; > + > + /* > + * As there is no timeout value, we don't convert to/from > + * 64/32-bit time_t. > + */ > + ret = syscall(SYS_ppoll_time64, ufds, ULONG_MAX, NULL, NULL); > + /* > + * There is no fallback to SYS_ppoll, we expect SYS_ppoll_time64 > + * to work and if it doesn't we fail. > + */ > + > + if (ret < 0 && errno != ENOSYS) { > + /* Expected error. */ > + } else if (errno == ENOSYS) { > + PERROR("[ppoll_time64] missing syscall"); > + } else if (ret > 0) { > + ret = read(wait_fd, buf, BUF_SIZE); > + if (ret < 0) { > + PERROR("[ppoll_time64] read"); > + } > + } > +#endif > } > > /* > @@ -476,6 +564,7 @@ void ppoll_fds_buffer_overflow( > static > void ppoll_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) > { > +#ifdef SYS_ppoll > struct pollfd ufds[NB_FD]; > char buf[BUF_SIZE]; > int ret; > @@ -492,6 +581,59 @@ void ppoll_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) > PERROR("[ppoll] read"); > } > } > +#endif > +} > + > +/* > + * Pass an invalid file descriptor to pselect6(). The syscall should return > + * -EBADF. The recorded event should contain a "ret = -EBADF (-9)". > + */ > +static > +void pselect_time64_invalid_fd(FILE *validation_output_file __attribute__((unused))) > +{ > +#ifdef SYS_pselect6_time64 > + fd_set rfds; > + int ret; > + int fd; > + char buf[BUF_SIZE]; > + > + /* > + * Open a file, close it and use the closed FD in the pselect6 call. > + */ > + fd = open("/dev/null", O_RDONLY); > + if (fd == -1) { > + PERROR("open"); > + goto error; > + } > + > + ret = close(fd); > + if (ret == -1) { > + PERROR("close"); > + goto error; > + } > + > + FD_ZERO(&rfds); > + FD_SET(fd, &rfds); > + > + ret = syscall(SYS_pselect6_time64, fd + 1, &rfds, NULL, NULL, NULL, NULL); > + /* > + * There is no fallback to SYS_pselect6, we expect SYS_pselect6_time64 > + * to work and if it doesn't we fail. > + */ > + > + if (ret == -1 && errno != ENOSYS) { > + /* Expected error. */ > + } else if (errno == ENOSYS) { > + PERROR("[pselect_time64] missing syscall"); > + } else if (ret) { > + ret = read(wait_fd, buf, BUF_SIZE); > + if (ret < 0) { > + PERROR("[pselect_time64] read"); > + } > + } > +error: > + return; > +#endif > } > > /* > @@ -501,6 +643,7 @@ void ppoll_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) > static > void pselect_invalid_fd(FILE *validation_output_file __attribute__((unused))) > { > +#ifdef SYS_pselect6 > fd_set rfds; > int ret; > int fd; > @@ -525,6 +668,7 @@ void pselect_invalid_fd(FILE *validation_output_file __attribute__((unused))) > FD_SET(fd, &rfds); > > ret = syscall(SYS_pselect6, fd + 1, &rfds, NULL, NULL, NULL, NULL); > + > if (ret == -1) { > /* Expected error. */ > } else if (ret) { > @@ -535,6 +679,44 @@ void pselect_invalid_fd(FILE *validation_output_file __attribute__((unused))) > } > error: > return; > +#endif > +} > + > +/* > + * Invalid pointer as writefds, should output a ppoll event > + * with 0 FDs. > + */ > +static > +void pselect_time64_invalid_pointer( > + FILE *validation_output_file __attribute__((unused))) > +{ > +#ifdef SYS_pselect6_time64 > + fd_set rfds; > + int ret; > + char buf[BUF_SIZE]; > + void *invalid = (void *) 0x42; > + > + FD_ZERO(&rfds); > + FD_SET(wait_fd, &rfds); > + > + ret = syscall(SYS_pselect6_time64, 1, &rfds, (fd_set *) invalid, NULL, NULL, > + NULL); > + /* > + * There is no fallback to SYS_pselect6, we expect SYS_pselect6_time64 > + * to work and if it doesn't we fail. > + */ > + > + if (ret == -1 && errno != ENOSYS) { > + /* Expected error. */ > + } else if (errno == ENOSYS) { > + PERROR("[pselect_time64] missing syscall"); > + } else if (ret) { > + ret = read(wait_fd, buf, BUF_SIZE); > + if (ret < 0) { > + PERROR("[pselect_time64] read"); > + } > + } > +#endif > } > > /* > @@ -545,6 +727,7 @@ static > void pselect_invalid_pointer( > FILE *validation_output_file __attribute__((unused))) > { > +#ifdef SYS_pselect6 > fd_set rfds; > int ret; > char buf[BUF_SIZE]; > @@ -563,6 +746,7 @@ void pselect_invalid_pointer( > PERROR("[pselect] read"); > } > } > +#endif > } > > /* > -- > 2.37.3 >
Hi Alistair, The patch you submitted doesn't pass on x86 and x86-64. I have written an alternative patch that works on the 32/64 variants of ARM and x86. I could only verify that it builds on RISC-V 64. Are you able to compile-test it on RISC-V 32? https://review.lttng.org/c/lttng-tools/+/8907 Thanks, J?r?mie -- J?r?mie Galarneau EfficiOS Inc. https://www.efficios.com
On Thu, Dec 15, 2022 at 6:20 AM J?r?mie Galarneau <jgalar at efficios.com> wrote: > > Hi Alistair, > > The patch you submitted doesn't pass on x86 and x86-64. Are you able to provide the failures? It should just be a simple fix > > I have written an alternative patch that works on the 32/64 variants of ARM and x86. I could only verify that it builds on RISC-V 64. > > Are you able to compile-test it on RISC-V 32? > > https://review.lttng.org/c/lttng-tools/+/8907 Thanks! I am currently having some trouble building it. The requirement on liburcu >= 0.14 is proving difficult to meet and the patch conflicts with earlier versions of lttng. I had a look at the patch though. It seems like you still call SYS_ppoll, which won't work on 64-bit time_t 32-bit systems. Changes like this: + #ifdef sys_pselect6_time64 + test_pselect_time64(); + #else test_pselect(); + #endif /* sys_pselect6_time64 */ will mean that test_pselect() isn't called on 32-bit platforms with a 5.4+ kernel. Which I thought is what you wanted to avoid. Alistair
diff --git a/tests/regression/kernel/select_poll_epoll.cpp b/tests/regression/kernel/select_poll_epoll.cpp index c0b688217..dfaab52c8 100644 --- a/tests/regression/kernel/select_poll_epoll.cpp +++ b/tests/regression/kernel/select_poll_epoll.cpp @@ -5,6 +5,7 @@ * */ +#include <errno.h> #include <fcntl.h> #include <limits.h> #include <poll.h> @@ -48,10 +49,14 @@ int lttng_opt_quiet, lttng_opt_verbose, lttng_opt_mi; static void run_working_cases(FILE *validation_output_file); static void pselect_invalid_fd(FILE *validation_output_file); +static void pselect_time64_invalid_fd(FILE *validation_output_file); static void test_ppoll_big(FILE *validation_output_file); static void ppoll_fds_buffer_overflow(FILE *validation_output_file); +static void ppoll_time64_fds_buffer_overflow(FILE *validation_output_file); static void pselect_invalid_pointer(FILE *validation_output_file); +static void pselect_time64_invalid_pointer(FILE *validation_output_file); static void ppoll_fds_ulong_max(FILE *validation_output_file); +static void ppoll_time64_fds_ulong_max(FILE *validation_output_file); static void epoll_pwait_invalid_pointer(FILE *validation_output_file); static void epoll_pwait_int_max(FILE *validation_output_file); static void ppoll_concurrent_write(FILE *validation_output_file); @@ -69,10 +74,14 @@ const struct test_case { { .run = run_working_cases, .produces_validation_info = true, .timeout = -1 }, { .run = run_working_cases, .produces_validation_info = true, .timeout = 1 }, { .run = pselect_invalid_fd, .produces_validation_info = false, .timeout = 0 }, + { .run = pselect_time64_invalid_fd, .produces_validation_info = false, .timeout = 0 }, { .run = test_ppoll_big, .produces_validation_info = false, .timeout = 0 }, { .run = ppoll_fds_buffer_overflow, .produces_validation_info = false, .timeout = 0 }, + { .run = ppoll_time64_fds_buffer_overflow, .produces_validation_info = false, .timeout = 0 }, { .run = pselect_invalid_pointer, .produces_validation_info = false, .timeout = 0 }, + { .run = pselect_time64_invalid_pointer, .produces_validation_info = false, .timeout = 0 }, { .run = ppoll_fds_ulong_max, .produces_validation_info = false, .timeout = 0 }, + { .run = ppoll_time64_fds_ulong_max, .produces_validation_info = false, .timeout = 0 }, { .run = epoll_pwait_invalid_pointer, .produces_validation_info = true, .timeout = 0 }, { .run = epoll_pwait_int_max, .produces_validation_info = true, .timeout = 0 }, { .run = ppoll_concurrent_write, .produces_validation_info = false, .timeout = 0 }, @@ -440,6 +449,44 @@ end: return; } +/* + * Ask for 100 FDs in a buffer for allocated for only 1 FD, should + * segfault (eventually with a "*** stack smashing detected ***" message). + * The event should contain an array of 100 FDs filled with garbage. + */ +static +void ppoll_time64_fds_buffer_overflow( + FILE *validation_output_file __attribute__((unused))) +{ +#ifdef SYS_ppoll_time64 + struct pollfd ufds[NB_FD]; + char buf[BUF_SIZE]; + int ret; + + ufds[0].fd = wait_fd; + ufds[0].events = POLLIN|POLLPRI; + + /* + * As there is no timeout value, we don't convert to/from + * 64/32-bit time_t. + */ + ret = syscall(SYS_ppoll_time64, ufds, 100, NULL, NULL); + /* + * There is no fallback to SYS_ppoll, we expect SYS_ppoll_time64 + * to work and if it doesn't we fail. + */ + + if (ret < 0) { + PERROR("ppoll_time64"); + } else if (ret > 0) { + ret = read(wait_fd, buf, BUF_SIZE); + if (ret < 0) { + PERROR("[ppoll_time64] read"); + } + } +#endif +} + /* * Ask for 100 FDs in a buffer for allocated for only 1 FD, should * segfault (eventually with a "*** stack smashing detected ***" message). @@ -449,6 +496,7 @@ static void ppoll_fds_buffer_overflow( FILE *validation_output_file __attribute__((unused))) { +#ifdef SYS_ppoll struct pollfd ufds[NB_FD]; char buf[BUF_SIZE]; int ret; @@ -466,6 +514,46 @@ void ppoll_fds_buffer_overflow( PERROR("[ppoll] read"); } } +#endif +} + +/* + * Ask for ULONG_MAX FDs in a buffer for allocated for only 1 FD, should + * cleanly fail with a "Invalid argument". + * The event should contain an empty array of FDs and overflow = 1. + */ +static +void ppoll_time64_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) +{ +#ifdef SYS_ppoll_time64 + struct pollfd ufds[NB_FD]; + char buf[BUF_SIZE]; + int ret; + + ufds[0].fd = wait_fd; + ufds[0].events = POLLIN|POLLPRI; + + /* + * As there is no timeout value, we don't convert to/from + * 64/32-bit time_t. + */ + ret = syscall(SYS_ppoll_time64, ufds, ULONG_MAX, NULL, NULL); + /* + * There is no fallback to SYS_ppoll, we expect SYS_ppoll_time64 + * to work and if it doesn't we fail. + */ + + if (ret < 0 && errno != ENOSYS) { + /* Expected error. */ + } else if (errno == ENOSYS) { + PERROR("[ppoll_time64] missing syscall"); + } else if (ret > 0) { + ret = read(wait_fd, buf, BUF_SIZE); + if (ret < 0) { + PERROR("[ppoll_time64] read"); + } + } +#endif } /* @@ -476,6 +564,7 @@ void ppoll_fds_buffer_overflow( static void ppoll_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) { +#ifdef SYS_ppoll struct pollfd ufds[NB_FD]; char buf[BUF_SIZE]; int ret; @@ -492,6 +581,59 @@ void ppoll_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) PERROR("[ppoll] read"); } } +#endif +} + +/* + * Pass an invalid file descriptor to pselect6(). The syscall should return + * -EBADF. The recorded event should contain a "ret = -EBADF (-9)". + */ +static +void pselect_time64_invalid_fd(FILE *validation_output_file __attribute__((unused))) +{ +#ifdef SYS_pselect6_time64 + fd_set rfds; + int ret; + int fd; + char buf[BUF_SIZE]; + + /* + * Open a file, close it and use the closed FD in the pselect6 call. + */ + fd = open("/dev/null", O_RDONLY); + if (fd == -1) { + PERROR("open"); + goto error; + } + + ret = close(fd); + if (ret == -1) { + PERROR("close"); + goto error; + } + + FD_ZERO(&rfds); + FD_SET(fd, &rfds); + + ret = syscall(SYS_pselect6_time64, fd + 1, &rfds, NULL, NULL, NULL, NULL); + /* + * There is no fallback to SYS_pselect6, we expect SYS_pselect6_time64 + * to work and if it doesn't we fail. + */ + + if (ret == -1 && errno != ENOSYS) { + /* Expected error. */ + } else if (errno == ENOSYS) { + PERROR("[pselect_time64] missing syscall"); + } else if (ret) { + ret = read(wait_fd, buf, BUF_SIZE); + if (ret < 0) { + PERROR("[pselect_time64] read"); + } + } +error: + return; +#endif } /* @@ -501,6 +643,7 @@ void ppoll_fds_ulong_max(FILE *validation_output_file __attribute__((unused))) static void pselect_invalid_fd(FILE *validation_output_file __attribute__((unused))) { +#ifdef SYS_pselect6 fd_set rfds; int ret; int fd; @@ -525,6 +668,7 @@ void pselect_invalid_fd(FILE *validation_output_file __attribute__((unused))) FD_SET(fd, &rfds); ret = syscall(SYS_pselect6, fd + 1, &rfds, NULL, NULL, NULL, NULL); + if (ret == -1) { /* Expected error. */ } else if (ret) { @@ -535,6 +679,44 @@ void pselect_invalid_fd(FILE *validation_output_file __attribute__((unused))) } error: return; +#endif +} + +/* + * Invalid pointer as writefds, should output a ppoll event + * with 0 FDs. + */ +static +void pselect_time64_invalid_pointer( + FILE *validation_output_file __attribute__((unused))) +{ +#ifdef SYS_pselect6_time64 + fd_set rfds; + int ret; + char buf[BUF_SIZE]; + void *invalid = (void *) 0x42; + + FD_ZERO(&rfds); + FD_SET(wait_fd, &rfds); + + ret = syscall(SYS_pselect6_time64, 1, &rfds, (fd_set *) invalid, NULL, NULL, + NULL); + /* + * There is no fallback to SYS_pselect6, we expect SYS_pselect6_time64 + * to work and if it doesn't we fail. + */ + + if (ret == -1 && errno != ENOSYS) { + /* Expected error. */ + } else if (errno == ENOSYS) { + PERROR("[pselect_time64] missing syscall"); + } else if (ret) { + ret = read(wait_fd, buf, BUF_SIZE); + if (ret < 0) { + PERROR("[pselect_time64] read"); + } + } +#endif } /* @@ -545,6 +727,7 @@ static void pselect_invalid_pointer( FILE *validation_output_file __attribute__((unused))) { +#ifdef SYS_pselect6 fd_set rfds; int ret; char buf[BUF_SIZE]; @@ -563,6 +746,7 @@ void pselect_invalid_pointer( PERROR("[pselect] read"); } } +#endif } /*