Skip to content

Commit

Permalink
zdtm: Check fd from pidfd_getfd is C/Red correctly
Browse files Browse the repository at this point in the history
We get the read end of a pipe using `pidfd_getfd` and check if we can
read from it after C/R.

signed-off-by: Bhavik Sachdev <b.sachdev1904@gmail.com>
  • Loading branch information
bsach64 committed Aug 20, 2024
1 parent 7c42698 commit 1c6cc6b
Show file tree
Hide file tree
Showing 2 changed files with 106 additions and 0 deletions.
1 change: 1 addition & 0 deletions test/zdtm/static/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@ TST_NOFILE := \
pidfd_dead \
pidfd_child \
pidfd_kill \
fd_from_pidfd \
pipe00 \
pipe01 \
pipe02 \
Expand Down
105 changes: 105 additions & 0 deletions test/zdtm/static/fd_from_pidfd.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
#include <sys/syscall.h>
#include <signal.h>
#include <sys/wait.h>
#include <unistd.h>

#include "zdtmtst.h"

const char *test_doc = "Check if fd obtained from pidfd_get_fd is C/R correctly\n";
const char *test_author = "Bhavik Sachdev <b.sachdev1904@gmail.com>";

static int pidfd_open(pid_t pid, unsigned int flags)
{
return syscall(__NR_pidfd_open, pid, flags);
}

static int pidfd_getfd(int pidfd, int targetfd, unsigned int flags)
{
return syscall(__NR_pidfd_getfd, pidfd, targetfd, flags);
}

static int pidfd_send_signal(int pidfd, int sig, siginfo_t* info, unsigned int flags)

Check warning on line 21 in test/zdtm/static/fd_from_pidfd.c

View workflow job for this annotation

GitHub Actions / build

{
return syscall(__NR_pidfd_send_signal, pidfd, sig, info, flags);
}

int main(int argc, char* argv[])
{
#define READ 0
#define WRITE 1

int pidfd, child, p[2], child_read, read_data, status;
int data = 42;

test_init(argc, argv);

if (pipe(p)) {
pr_perror("pipe");
return 1;
}

child = fork();
if (child < 0) {
pr_perror("fork");
return 1;
}

if (child == 0) {
close(p[WRITE]);
test_waitsig();
return 0;
}

pidfd = pidfd_open(child, 0);
if (pidfd < 0) {
pr_perror("pidfd_open failed");
return 1;
}

close(p[READ]);
if (write(p[WRITE], &data, sizeof(data)) != sizeof(data)) {
pr_perror("write");
return 1;
}
close(p[WRITE]);

child_read = pidfd_getfd(pidfd, p[READ], 0);
if (child_read < 0) {
pr_perror("pidfd_getfd");
return 1;
}

test_daemon();
test_waitsig();

if (read(child_read, &read_data, sizeof(read_data)) != sizeof(read_data)) {
pr_perror("read");
return 1;
}

if (read_data != data) {
fail("data from fd obtained using pidfd_getfd incorrect");
goto out;
}

if (pidfd_send_signal(pidfd, SIGTERM, NULL, 0)) {
pr_perror("Could not send signal");
goto out;
}

if (waitpid(child, &status, 0) != child) {
pr_perror("waitpid()");
return 1;
}

if (status != 0) {
fail("%d:%d:%d:%d", WIFEXITED(status), WEXITSTATUS(status), WIFSIGNALED(status), WTERMSIG(status));
return 1;
}

pass();
out:
close(child_read);
close(pidfd);
return 0;
}

0 comments on commit 1c6cc6b

Please sign in to comment.