The following patch seems to make the test pass, but unfortunately by skipping the stat_filename test:
diff --git a/src/lib/sandbox/sandbox.c b/src/lib/sandbox/sandbox.cindex aed7e3706f..4a980284c5 100644--- a/src/lib/sandbox/sandbox.c+++ b/src/lib/sandbox/sandbox.c@@ -257,7 +257,6 @@ static int filter_nopar_gen[] = { SCMP_SYS(madvise), #ifdef __NR_stat64- // getaddrinfo uses this.. SCMP_SYS(stat64), #endif@@ -624,6 +623,33 @@ sb_chown(scmp_filter_ctx ctx, sandbox_cfg_t *filter) return 0; }+#ifdef __NR_chown32+static int+sb_chown32(scmp_filter_ctx ctx, sandbox_cfg_t *filter)+{+ int rc;+ sandbox_cfg_t *elem = NULL;++ // for each dynamic parameter filters+ for (elem = filter; elem != NULL; elem = elem->next) {+ smp_param_t *param = elem->param;++ if (param != NULL && param->prot == 1 && param->syscall+ == SCMP_SYS(chown32)) {+ rc = seccomp_rule_add_1(ctx, SCMP_ACT_ALLOW, SCMP_SYS(chown32),+ SCMP_CMP_STR(0, SCMP_CMP_EQ, param->value));+ if (rc != 0) {+ log_err(LD_BUG,"(Sandbox) failed to add chown32 syscall, received "+ "libseccomp error %d", rc);+ return rc;+ }+ }+ }++ return 0;+}+#endif // __NR_chown32+ /** * Function responsible for setting up the rename syscall for * the seccomp filter sandbox.@@ -1271,6 +1297,9 @@ static sandbox_filter_func_t filter_func[] = { sb_mmap2, #endif sb_chown,+#ifdef __NR_chown32+ sb_chown32,+#endif sb_chmod, sb_open, sb_openat,@@ -1597,7 +1626,14 @@ sandbox_cfg_allow_chown_filename(sandbox_cfg_t **cfg, char *file) elem = new_element(SCMP_SYS(chown), file); elem->next = *cfg;++#if __NR_chown32+ elem = new_element(SCMP_SYS(chown32), file);++ elem->next = *cfg;+ *cfg = elem; *cfg = elem;+#endif return 0; }diff --git a/src/test/test_sandbox.c b/src/test/test_sandbox.cindex e5064c58ec..c8366924e7 100644--- a/src/test/test_sandbox.c+++ b/src/test/test_sandbox.c@@ -337,7 +337,7 @@ struct testcase_t sandbox_tests[] = { * * Skip testing sandbox_cfg_allow_stat_filename() if it seems the likely the * function will have no effect and the test will therefore not succeed. */-#if !defined(__NR_newfstatat) && (!defined(__NR_stat) || defined(__NR_stat64))+#if !defined(__NR_newfstatat) && (!defined(__NR_stat) || !defined(__NR_stat64)) SANDBOX_TEST_IN_SANDBOX(stat_filename), #else SANDBOX_TEST_SKIPPED(stat_filename),
The change in src/test/test_sandbox.c is not correct as it is right now.
I suspect this might be related to #40382 (closed) too. The i386 debian stable I tested with was using glibc 2.31 and seems to use stat64() to implement stat() with.
I wonder if @nickm or @ssouth have opinions about the x86-32 situatoin here with our Sandbox tests and stat(). I think the chown32() code above is fine, but I am of course open to suggestions :-)
Some extra debug info. This is a Debian i386 running Debian 11.1 (bullseye).
The glibc version:
GNU C Library (Debian GLIBC 2.31-13+deb11u2) stable release version 2.31.Copyright (C) 2020 Free Software Foundation, Inc.This is free software; see the source for copying conditions.There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR APARTICULAR PURPOSE.Compiled by GNU CC version 10.2.1 20210110.libc ABIs: UNIQUE IFUNC ABSOLUTEFor bug reporting instructions, please see:<http://www.debian.org/Bugs/>.
A small test C program to see what stat() turns into system-call wise:
I've created !480 (merged) with the changes I'd like to propose to resolve this.
Alexander (@ahf), this borrows from the code you posted but changes it a bit to fit in better with the existing sandbox code and to work reliably on x86-64 as well. There's also an additional change, to allow the clock_gettime64 system call, required to get the entire test suite passing on x86.
These changes work for me on a virtualized i686 with glibc 2.31, and I'll re-test with glibc 2.33 once I have my test system updated.