mirror of
git://git.yoctoproject.org/linux-yocto.git
synced 2025-12-18 08:36:21 +01:00
Apparently as of version 2.42, glibc headers define AT_RENAME_NOREPLACE
and some of the other flags for renameat2() and friends in <stdio.h>.
Which would all be fine, except for inexplicable reasons glibc decided
to define them _differently_ from the kernel definitions, which then
makes some of our sample code that includes both kernel headers and user
space headers unhappy, because the compiler will (correctly) complain
about redefining things.
Now, mixing kernel headers and user space headers is always a somewhat
iffy proposition due to namespacing issues, but it's kind of inevitable
in our sample and selftest code. And this is just glibc being stupid.
Those defines come from the kernel, glibc is exposing the kernel
interfaces, and glibc shouldn't make up some random new expressions for
these values.
It's not like glibc headers changed the actual result values, but they
arbitrarily just decided to use a different expression to describe those
values. The kernel just does
#define AT_RENAME_NOREPLACE 0x0001
while glibc does
# define RENAME_NOREPLACE (1 << 0)
# define AT_RENAME_NOREPLACE RENAME_NOREPLACE
instead. Same value in the end, but very different macro definition.
For absolutely no reason.
This has since been fixed in the glibc development tree, so eventually
we'll end up with the canonical expressions and no clashes. But in the
meantime the broken headers are in the glibc-2.42 release and have made
it out into distributions.
Do a minimal work-around to make the samples build cleanly by just
undefining the affected macros in between the user space header include
and the kernel header includes.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
272 lines
6.6 KiB
C
272 lines
6.6 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/* Test the statx() system call.
|
|
*
|
|
* Note that the output of this program is intended to look like the output of
|
|
* /bin/stat where possible.
|
|
*
|
|
* Copyright (C) 2015 Red Hat, Inc. All Rights Reserved.
|
|
* Written by David Howells (dhowells@redhat.com)
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#define _ATFILE_SOURCE
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <ctype.h>
|
|
#include <errno.h>
|
|
#include <time.h>
|
|
#include <sys/syscall.h>
|
|
#include <sys/types.h>
|
|
|
|
// Work around glibc header silliness
|
|
#undef AT_RENAME_NOREPLACE
|
|
#undef AT_RENAME_EXCHANGE
|
|
#undef AT_RENAME_WHITEOUT
|
|
|
|
#include <linux/stat.h>
|
|
#include <linux/fcntl.h>
|
|
#define statx foo
|
|
#define statx_timestamp foo_timestamp
|
|
struct statx;
|
|
struct statx_timestamp;
|
|
#include <sys/stat.h>
|
|
#undef statx
|
|
#undef statx_timestamp
|
|
|
|
#define AT_STATX_SYNC_TYPE 0x6000
|
|
#define AT_STATX_SYNC_AS_STAT 0x0000
|
|
#define AT_STATX_FORCE_SYNC 0x2000
|
|
#define AT_STATX_DONT_SYNC 0x4000
|
|
|
|
#ifndef __NR_statx
|
|
#define __NR_statx -1
|
|
#endif
|
|
|
|
static __attribute__((unused))
|
|
ssize_t statx(int dfd, const char *filename, unsigned flags,
|
|
unsigned int mask, struct statx *buffer)
|
|
{
|
|
return syscall(__NR_statx, dfd, filename, flags, mask, buffer);
|
|
}
|
|
|
|
static void print_time(const char *field, struct statx_timestamp *ts)
|
|
{
|
|
struct tm tm;
|
|
time_t tim;
|
|
char buffer[100];
|
|
int len;
|
|
|
|
tim = ts->tv_sec;
|
|
if (!localtime_r(&tim, &tm)) {
|
|
perror("localtime_r");
|
|
exit(1);
|
|
}
|
|
len = strftime(buffer, 100, "%F %T", &tm);
|
|
if (len == 0) {
|
|
perror("strftime");
|
|
exit(1);
|
|
}
|
|
printf("%s", field);
|
|
fwrite(buffer, 1, len, stdout);
|
|
printf(".%09u", ts->tv_nsec);
|
|
len = strftime(buffer, 100, "%z", &tm);
|
|
if (len == 0) {
|
|
perror("strftime2");
|
|
exit(1);
|
|
}
|
|
fwrite(buffer, 1, len, stdout);
|
|
printf("\n");
|
|
}
|
|
|
|
static void dump_statx(struct statx *stx)
|
|
{
|
|
char buffer[256], ft = '?';
|
|
|
|
printf("results=%x\n", stx->stx_mask);
|
|
|
|
printf(" ");
|
|
if (stx->stx_mask & STATX_SIZE)
|
|
printf(" Size: %-15llu", (unsigned long long)stx->stx_size);
|
|
if (stx->stx_mask & STATX_BLOCKS)
|
|
printf(" Blocks: %-10llu", (unsigned long long)stx->stx_blocks);
|
|
printf(" IO Block: %-6llu", (unsigned long long)stx->stx_blksize);
|
|
if (stx->stx_mask & STATX_TYPE) {
|
|
switch (stx->stx_mode & S_IFMT) {
|
|
case S_IFIFO: printf(" FIFO\n"); ft = 'p'; break;
|
|
case S_IFCHR: printf(" character special file\n"); ft = 'c'; break;
|
|
case S_IFDIR: printf(" directory\n"); ft = 'd'; break;
|
|
case S_IFBLK: printf(" block special file\n"); ft = 'b'; break;
|
|
case S_IFREG: printf(" regular file\n"); ft = '-'; break;
|
|
case S_IFLNK: printf(" symbolic link\n"); ft = 'l'; break;
|
|
case S_IFSOCK: printf(" socket\n"); ft = 's'; break;
|
|
default:
|
|
printf(" unknown type (%o)\n", stx->stx_mode & S_IFMT);
|
|
break;
|
|
}
|
|
} else {
|
|
printf(" no type\n");
|
|
}
|
|
|
|
sprintf(buffer, "%02x:%02x", stx->stx_dev_major, stx->stx_dev_minor);
|
|
printf("Device: %-15s", buffer);
|
|
if (stx->stx_mask & STATX_INO)
|
|
printf(" Inode: %-11llu", (unsigned long long) stx->stx_ino);
|
|
if (stx->stx_mask & STATX_NLINK)
|
|
printf(" Links: %-5u", stx->stx_nlink);
|
|
if (stx->stx_mask & STATX_TYPE) {
|
|
switch (stx->stx_mode & S_IFMT) {
|
|
case S_IFBLK:
|
|
case S_IFCHR:
|
|
printf(" Device type: %u,%u",
|
|
stx->stx_rdev_major, stx->stx_rdev_minor);
|
|
break;
|
|
}
|
|
}
|
|
printf("\n");
|
|
|
|
if (stx->stx_mask & STATX_MODE)
|
|
printf("Access: (%04o/%c%c%c%c%c%c%c%c%c%c) ",
|
|
stx->stx_mode & 07777,
|
|
ft,
|
|
stx->stx_mode & S_IRUSR ? 'r' : '-',
|
|
stx->stx_mode & S_IWUSR ? 'w' : '-',
|
|
stx->stx_mode & S_IXUSR ? 'x' : '-',
|
|
stx->stx_mode & S_IRGRP ? 'r' : '-',
|
|
stx->stx_mode & S_IWGRP ? 'w' : '-',
|
|
stx->stx_mode & S_IXGRP ? 'x' : '-',
|
|
stx->stx_mode & S_IROTH ? 'r' : '-',
|
|
stx->stx_mode & S_IWOTH ? 'w' : '-',
|
|
stx->stx_mode & S_IXOTH ? 'x' : '-');
|
|
if (stx->stx_mask & STATX_UID)
|
|
printf("Uid: %5d ", stx->stx_uid);
|
|
if (stx->stx_mask & STATX_GID)
|
|
printf("Gid: %5d\n", stx->stx_gid);
|
|
|
|
if (stx->stx_mask & STATX_ATIME)
|
|
print_time("Access: ", &stx->stx_atime);
|
|
if (stx->stx_mask & STATX_MTIME)
|
|
print_time("Modify: ", &stx->stx_mtime);
|
|
if (stx->stx_mask & STATX_CTIME)
|
|
print_time("Change: ", &stx->stx_ctime);
|
|
if (stx->stx_mask & STATX_BTIME)
|
|
print_time(" Birth: ", &stx->stx_btime);
|
|
|
|
if (stx->stx_attributes_mask) {
|
|
unsigned char bits, mbits;
|
|
int loop, byte;
|
|
|
|
static char attr_representation[64 + 1] =
|
|
/* STATX_ATTR_ flags: */
|
|
"????????" /* 63-56 */
|
|
"????????" /* 55-48 */
|
|
"????????" /* 47-40 */
|
|
"????????" /* 39-32 */
|
|
"????????" /* 31-24 0x00000000-ff000000 */
|
|
"????????" /* 23-16 0x00000000-00ff0000 */
|
|
"???me???" /* 15- 8 0x00000000-0000ff00 */
|
|
"?dai?c??" /* 7- 0 0x00000000-000000ff */
|
|
;
|
|
|
|
printf("Attributes: %016llx (",
|
|
(unsigned long long)stx->stx_attributes);
|
|
for (byte = 64 - 8; byte >= 0; byte -= 8) {
|
|
bits = stx->stx_attributes >> byte;
|
|
mbits = stx->stx_attributes_mask >> byte;
|
|
for (loop = 7; loop >= 0; loop--) {
|
|
int bit = byte + loop;
|
|
|
|
if (!(mbits & 0x80))
|
|
putchar('.'); /* Not supported */
|
|
else if (bits & 0x80)
|
|
putchar(attr_representation[63 - bit]);
|
|
else
|
|
putchar('-'); /* Not set */
|
|
bits <<= 1;
|
|
mbits <<= 1;
|
|
}
|
|
if (byte)
|
|
putchar(' ');
|
|
}
|
|
printf(")\n");
|
|
}
|
|
}
|
|
|
|
static void dump_hex(unsigned long long *data, int from, int to)
|
|
{
|
|
unsigned offset, print_offset = 1, col = 0;
|
|
|
|
from /= 8;
|
|
to = (to + 7) / 8;
|
|
|
|
for (offset = from; offset < to; offset++) {
|
|
if (print_offset) {
|
|
printf("%04x: ", offset * 8);
|
|
print_offset = 0;
|
|
}
|
|
printf("%016llx", data[offset]);
|
|
col++;
|
|
if ((col & 3) == 0) {
|
|
printf("\n");
|
|
print_offset = 1;
|
|
} else {
|
|
printf(" ");
|
|
}
|
|
}
|
|
|
|
if (!print_offset)
|
|
printf("\n");
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
struct statx stx;
|
|
int ret, raw = 0, atflag = AT_SYMLINK_NOFOLLOW;
|
|
|
|
unsigned int mask = STATX_BASIC_STATS | STATX_BTIME;
|
|
|
|
for (argv++; *argv; argv++) {
|
|
if (strcmp(*argv, "-F") == 0) {
|
|
atflag &= ~AT_STATX_SYNC_TYPE;
|
|
atflag |= AT_STATX_FORCE_SYNC;
|
|
continue;
|
|
}
|
|
if (strcmp(*argv, "-D") == 0) {
|
|
atflag &= ~AT_STATX_SYNC_TYPE;
|
|
atflag |= AT_STATX_DONT_SYNC;
|
|
continue;
|
|
}
|
|
if (strcmp(*argv, "-L") == 0) {
|
|
atflag &= ~AT_SYMLINK_NOFOLLOW;
|
|
continue;
|
|
}
|
|
if (strcmp(*argv, "-O") == 0) {
|
|
mask &= ~STATX_BASIC_STATS;
|
|
continue;
|
|
}
|
|
if (strcmp(*argv, "-A") == 0) {
|
|
atflag |= AT_NO_AUTOMOUNT;
|
|
continue;
|
|
}
|
|
if (strcmp(*argv, "-R") == 0) {
|
|
raw = 1;
|
|
continue;
|
|
}
|
|
|
|
memset(&stx, 0xbf, sizeof(stx));
|
|
ret = statx(AT_FDCWD, *argv, atflag, mask, &stx);
|
|
printf("statx(%s) = %d\n", *argv, ret);
|
|
if (ret < 0) {
|
|
perror(*argv);
|
|
exit(1);
|
|
}
|
|
|
|
if (raw)
|
|
dump_hex((unsigned long long *)&stx, 0, sizeof(stx));
|
|
|
|
dump_statx(&stx);
|
|
}
|
|
return 0;
|
|
}
|