On 6/23/26 18:13, Nhat Pham wrote:
> On Tue, Jun 23, 2026 at 4:15 AM Pavel Tikhomirov
> <[email protected]> wrote:
>>
>> Mount an overlayfs, create and write a file in the merged directory, and
>> run cachestat() on it, reusing the existing test_cachestat() helper.
>>
>> Also bump NR_TESTS to the actual number of tests run: it was 9 while
>> ten tests were already executed, and this adds an eleventh.
>>
>> Signed-off-by: Pavel Tikhomirov <[email protected]>
>> ---
>>  .../selftests/cachestat/test_cachestat.c      | 75 ++++++++++++++++++-
> 
> Hmm this looks strange...?

That's just how git-format-patch formats it for long path and big
change ("++...++") it hides part of the path.

> 
>>  1 file changed, 74 insertions(+), 1 deletion(-)
>>
>> diff --git a/tools/testing/selftests/cachestat/test_cachestat.c 
>> b/tools/testing/selftests/cachestat/test_cachestat.c
>> index 542cd09cb4434..1662d9817c50b 100644
>> --- a/tools/testing/selftests/cachestat/test_cachestat.c
>> +++ b/tools/testing/selftests/cachestat/test_cachestat.c
>> @@ -4,21 +4,25 @@
>>
>>  #include <stdio.h>
>>  #include <stdbool.h>
>> +#include <stdlib.h>
>>  #include <linux/kernel.h>
>>  #include <linux/magic.h>
>>  #include <linux/mman.h>
>>  #include <sys/mman.h>
>> +#include <sys/mount.h>
>>  #include <sys/shm.h>
>> +#include <sys/stat.h>
>>  #include <sys/syscall.h>
>>  #include <sys/vfs.h>
>>  #include <unistd.h>
>>  #include <string.h>
>>  #include <fcntl.h>
>>  #include <errno.h>
>> +#include <limits.h>
>>
>>  #include "kselftest.h"
>>
>> -#define NR_TESTS       9
>> +#define NR_TESTS       11
>>
>>  static const char * const dev_files[] = {
>>         "/dev/zero", "/dev/null", "/dev/urandom",
>> @@ -294,6 +298,62 @@ bool run_cachestat_test(enum file_type type)
>>         return ret;
>>  }
>>
>> +/*
>> + * Set up an overlayfs mount and run cachestat on a freshly created file in 
>> the
>> + * merged directory. Overlayfs forwards data I/O to the underlying (upper)
>> + * inode, so the page cache lives there and not in the overlay inode's 
>> mapping.
>> + * This is a regression test for cachestat returning all zeroes on 
>> overlayfs.
>> + */
>> +static int run_cachestat_overlayfs_test(void)
>> +{
>> +       char tmpl[] = "/tmp/cachestat_ovl.XXXXXX";
>> +       char lower[PATH_MAX], upper[PATH_MAX], work[PATH_MAX];
>> +       char merged[PATH_MAX], opts[4 * PATH_MAX], file[PATH_MAX];
>> +       char *base;
>> +       int ret;
>> +
>> +       base = mkdtemp(tmpl);
>> +       if (!base) {
>> +               ksft_print_msg("Unable to create overlayfs base dir: %s\n",
>> +                              strerror(errno));
>> +               return KSFT_FAIL;
>> +       }
>> +
>> +       snprintf(lower, sizeof(lower), "%s/lower", base);
>> +       snprintf(upper, sizeof(upper), "%s/upper", base);
>> +       snprintf(work, sizeof(work), "%s/work", base);
>> +       snprintf(merged, sizeof(merged), "%s/merged", base);
>> +
>> +       if (mkdir(lower, 0755) || mkdir(upper, 0755) ||
>> +           mkdir(work, 0755) || mkdir(merged, 0755)) {
>> +               ksft_print_msg("Unable to create overlayfs dirs: %s\n",
>> +                              strerror(errno));
>> +               ret = KSFT_FAIL;
>> +               goto cleanup;
>> +       }
>> +
>> +       snprintf(opts, sizeof(opts), "lowerdir=%s,upperdir=%s,workdir=%s",
>> +                lower, upper, work);
>> +
>> +       if (mount("overlay", merged, "overlay", 0, opts)) {
>> +               ksft_print_msg("Unable to mount overlayfs (need root?): 
>> %s\n",
>> +                              strerror(errno));
>> +               ret = KSFT_SKIP;
>> +               goto cleanup;
>> +       }
>> +
>> +       snprintf(file, sizeof(file), "%s/merged/cachestat", base);
>> +       ret = test_cachestat(file, true, true, false, 4, O_CREAT | O_RDWR, 
>> 0600);
>> +
>> +       umount(merged);
>> +cleanup:
>> +       /* Best-effort recursive cleanup of the temporary tree. */
>> +       snprintf(opts, sizeof(opts), "rm -rf %s", base);
>> +       if (system(opts))
>> +               ksft_print_msg("Unable to clean up %s\n", base);
> 
> nit: no helper for these? :)

This is similar to e.g. 
./tools/testing/selftests/bpf/prog_tests/test_local_storage.c's
cleanup, I didn't find a common selftest helper for temporary directory removal.

> 
>> +       return ret;
>> +}
>> +
>>  int main(void)
>>  {
>>         int ret;
>> @@ -361,5 +421,18 @@ int main(void)
>>                 ksft_test_result_fail("cachestat fails with a mmap file\n");
>>                 ret = 1;
>>         }
>> +
>> +       switch (run_cachestat_overlayfs_test()) {
>> +       case KSFT_FAIL:
>> +               ksft_test_result_fail("cachestat fails with an overlayfs 
>> file\n");
>> +               ret = 1;
>> +               break;
>> +       case KSFT_PASS:
>> +               ksft_test_result_pass("cachestat works with an overlayfs 
>> file\n");
>> +               break;
>> +       case KSFT_SKIP:
>> +               ksft_test_result_skip("overlayfs not available\n");
>> +               break;
>> +       }
>>         return ret;
>>  }
>> --
>> 2.54.0
>>

-- 
Best regards, Pavel Tikhomirov
Senior Software Developer, Virtuozzo.


Reply via email to