/* * Copyright (c) 2015 Dmitry V. Levin <ldv@altlinux.org> * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #if defined TEST_SYSCALL_NAME \ && defined HAVE_FTRUNCATE && defined HAVE_FUTIMENS # ifndef TEST_SYSCALL_INVOKE # error TEST_SYSCALL_INVOKE must be defined # endif # ifndef PRINT_SYSCALL_HEADER # error PRINT_SYSCALL_HEADER must be defined # endif # ifndef PRINT_SYSCALL_FOOTER # error PRINT_SYSCALL_FOOTER must be defined # endif # include <errno.h> # include <stdio.h> # include <stddef.h> # include <time.h> # include <unistd.h> # if defined MAJOR_IN_SYSMACROS # include <sys/sysmacros.h> # elif defined MAJOR_IN_MKDEV # include <sys/mkdev.h> # else # include <sys/types.h> # endif static void print_time(const time_t t) { if (!t) { printf("0"); return; } struct tm *p = localtime(&t); if (p) printf("%02d/%02d/%02d-%02d:%02d:%02d", p->tm_year + 1900, p->tm_mon + 1, p->tm_mday, p->tm_hour, p->tm_min, p->tm_sec); else printf("%llu", (unsigned long long) t); } typedef off_t libc_off_t; # ifdef USE_ASM_STAT # define stat libc_stat # define stat64 libc_stat64 # endif # include <fcntl.h> # include <sys/stat.h> # ifdef USE_ASM_STAT # undef stat # undef stat64 # endif # ifdef USE_ASM_STAT # undef st_atime # undef st_mtime # undef st_ctime # undef dev_t # undef gid_t # undef ino_t # undef loff_t # undef mode_t # undef nlink_t # undef off64_t # undef off_t # undef time_t # undef uid_t # define dev_t __kernel_dev_t # define gid_t __kernel_gid_t # define ino_t __kernel_ino_t # define loff_t __kernel_loff_t # define mode_t __kernel_mode_t # define nlink_t __kernel_nlink_t # define off64_t __kernel_off64_t # define off_t __kernel_off_t # define time_t __kernel_time_t # define uid_t __kernel_uid_t # include "asm_stat.h" # else # undef HAVE_STRUCT_STAT_ST_ATIME_NSEC # ifdef HAVE_STRUCT_STAT_ST_ATIM_TV_NSEC # define HAVE_STRUCT_STAT_ST_ATIME_NSEC 1 # undef st_atime_nsec # define st_atime_nsec st_atim.tv_nsec # endif # undef HAVE_STRUCT_STAT_ST_MTIME_NSEC # ifdef HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC # define HAVE_STRUCT_STAT_ST_MTIME_NSEC 1 # undef st_mtime_nsec # define st_mtime_nsec st_mtim.tv_nsec # endif # undef HAVE_STRUCT_STAT_ST_CTIME_NSEC # ifdef HAVE_STRUCT_STAT_ST_CTIM_TV_NSEC # define HAVE_STRUCT_STAT_ST_CTIME_NSEC 1 # undef st_ctime_nsec # define st_ctime_nsec st_ctim.tv_nsec # endif # endif # ifndef STRUCT_STAT # define STRUCT_STAT struct stat # endif # ifndef SAMPLE_SIZE # define SAMPLE_SIZE 43147718418 # endif static void print_ftype(const unsigned int mode) { if (S_ISREG(mode)) printf("S_IFREG"); else if (S_ISDIR(mode)) printf("S_IFDIR"); else if (S_ISCHR(mode)) printf("S_IFCHR"); else if (S_ISBLK(mode)) printf("S_IFBLK"); else printf("%#o", mode & S_IFMT); } static void print_perms(const unsigned int mode) { printf("%#o", mode & ~S_IFMT); } static void print_stat(const STRUCT_STAT *st) { printf("{st_dev=makedev(%u, %u)", (unsigned int) major(st->st_dev), (unsigned int) minor(st->st_dev)); printf(", st_ino=%llu", (unsigned long long) st->st_ino); printf(", st_mode="); print_ftype(st->st_mode); printf("|"); print_perms(st->st_mode); printf(", st_nlink=%u", (unsigned int) st->st_nlink); printf(", st_uid=%u", (unsigned int) st->st_uid); printf(", st_gid=%u", (unsigned int) st->st_gid); printf(", st_blksize=%u", (unsigned int) st->st_blksize); printf(", st_blocks=%u", (unsigned int) st->st_blocks); switch (st->st_mode & S_IFMT) { case S_IFCHR: case S_IFBLK: printf(", st_rdev=makedev(%u, %u)", (unsigned int) major(st->st_rdev), (unsigned int) minor(st->st_rdev)); break; default: printf(", st_size=%llu", (unsigned long long) st->st_size); } printf(", st_atime="); print_time(st->st_atime); # ifdef HAVE_STRUCT_STAT_ST_ATIME_NSEC if (st->st_atime_nsec) printf(".%09lu", (unsigned long) st->st_atime_nsec); # endif printf(", st_mtime="); print_time(st->st_mtime); # ifdef HAVE_STRUCT_STAT_ST_MTIME_NSEC if (st->st_mtime_nsec) printf(".%09lu", (unsigned long) st->st_mtime_nsec); # endif printf(", st_ctime="); print_time(st->st_ctime); # ifdef HAVE_STRUCT_STAT_ST_CTIME_NSEC if (st->st_ctime_nsec) printf(".%09lu", (unsigned long) st->st_ctime_nsec); # endif printf("}"); } static int create_sample(const char *fname, const libc_off_t size) { static const struct timespec ts[] = { {-10843, 135}, {-10841, 246} }; (void) close(0); if (open(fname, O_RDWR | O_CREAT | O_TRUNC, 0640)) { perror(fname); return 77; } if (ftruncate(0, size)) { perror("ftruncate"); return 77; } if (futimens(0, ts)) { perror("futimens"); return 77; } return 0; } # define stringify_(arg) #arg # define stringify(arg) stringify_(arg) # define TEST_SYSCALL_STR stringify(TEST_SYSCALL_NAME) # define STRUCT_STAT_STR stringify(STRUCT_STAT) int main(void) { static const char sample[] = TEST_SYSCALL_STR ".sample"; STRUCT_STAT st[2]; int rc = create_sample(sample, SAMPLE_SIZE); if (rc) { (void) unlink(sample); return rc; } if (TEST_SYSCALL_INVOKE(sample, st)) { perror(TEST_SYSCALL_STR); (void) unlink(sample); return 77; } (void) unlink(sample); if ((unsigned long long) SAMPLE_SIZE != (unsigned long long) st[0].st_size) { fprintf(stderr, "Size mismatch: " "requested size(%llu) != st_size(%llu)\n", (unsigned long long) SAMPLE_SIZE, (unsigned long long) st[0].st_size); fprintf(stderr, "The most likely reason for this is incorrect" " definition of %s.\n" "Here is some diagnostics that might help:\n", STRUCT_STAT_STR); fprintf(stderr, "offsetof(%s, st_dev) = %zu" ", sizeof(st_dev) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_dev), sizeof(st[0].st_dev)); fprintf(stderr, "offsetof(%s, st_ino) = %zu" ", sizeof(st_ino) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_ino), sizeof(st[0].st_ino)); fprintf(stderr, "offsetof(%s, st_mode) = %zu" ", sizeof(st_mode) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_mode), sizeof(st[0].st_mode)); fprintf(stderr, "offsetof(%s, st_nlink) = %zu" ", sizeof(st_nlink) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_nlink), sizeof(st[0].st_nlink)); fprintf(stderr, "offsetof(%s, st_uid) = %zu" ", sizeof(st_uid) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_uid), sizeof(st[0].st_uid)); fprintf(stderr, "offsetof(%s, st_gid) = %zu" ", sizeof(st_gid) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_gid), sizeof(st[0].st_gid)); fprintf(stderr, "offsetof(%s, st_rdev) = %zu" ", sizeof(st_rdev) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_rdev), sizeof(st[0].st_rdev)); fprintf(stderr, "offsetof(%s, st_size) = %zu" ", sizeof(st_size) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_size), sizeof(st[0].st_size)); fprintf(stderr, "offsetof(%s, st_blksize) = %zu" ", sizeof(st_blksize) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_blksize), sizeof(st[0].st_blksize)); fprintf(stderr, "offsetof(%s, st_blocks) = %zu" ", sizeof(st_blocks) = %zu\n", STRUCT_STAT_STR, offsetof(STRUCT_STAT, st_blocks), sizeof(st[0].st_blocks)); return 77; } PRINT_SYSCALL_HEADER(sample); print_stat(st); PRINT_SYSCALL_FOOTER; puts("+++ exited with 0 +++"); return 0; } #else int main(void) { return 77; } #endif