2023-04-06 21:54:49 +00:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0 */
|
2022-07-19 23:44:35 +02:00
|
|
|
|
2022-07-19 23:44:44 +02:00
|
|
|
#define _GNU_SOURCE
|
2023-07-08 02:26:24 +08:00
|
|
|
#define _LARGEFILE64_SOURCE
|
2022-07-19 23:44:44 +02:00
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
/* libc-specific include files
|
2022-07-19 23:44:44 +02:00
|
|
|
* The program may be built in 3 ways:
|
2022-07-19 23:44:35 +02:00
|
|
|
* $(CC) -nostdlib -include /path/to/nolibc.h => NOLIBC already defined
|
2022-07-19 23:44:44 +02:00
|
|
|
* $(CC) -nostdlib -I/path/to/nolibc/sysroot => _NOLIBC_* guards are present
|
|
|
|
* $(CC) with default libc => NOLIBC* never defined
|
2022-07-19 23:44:35 +02:00
|
|
|
*/
|
|
|
|
#ifndef NOLIBC
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2022-07-19 23:44:44 +02:00
|
|
|
#ifndef _NOLIBC_STDIO_H
|
|
|
|
/* standard libcs need more includes */
|
|
|
|
#include <sys/io.h>
|
|
|
|
#include <sys/ioctl.h>
|
2023-04-02 20:48:06 +02:00
|
|
|
#include <sys/mman.h>
|
2022-07-19 23:44:44 +02:00
|
|
|
#include <sys/mount.h>
|
2023-05-26 09:51:19 +02:00
|
|
|
#include <sys/prctl.h>
|
2022-07-19 23:44:44 +02:00
|
|
|
#include <sys/reboot.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/syscall.h>
|
|
|
|
#include <sys/sysmacros.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <poll.h>
|
|
|
|
#include <sched.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <stdarg.h>
|
2023-04-02 10:13:57 +02:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
2022-07-19 23:44:44 +02:00
|
|
|
#include <unistd.h>
|
selftests/nolibc: not include limits.h for nolibc
When compile nolibc-test.c with 2.31 glibc, we got such error:
In file included from /usr/riscv64-linux-gnu/include/sys/cdefs.h:452,
from /usr/riscv64-linux-gnu/include/features.h:461,
from /usr/riscv64-linux-gnu/include/bits/libc-header-start.h:33,
from /usr/riscv64-linux-gnu/include/limits.h:26,
from /usr/lib/gcc-cross/riscv64-linux-gnu/9/include/limits.h:194,
from /usr/lib/gcc-cross/riscv64-linux-gnu/9/include/syslimits.h:7,
from /usr/lib/gcc-cross/riscv64-linux-gnu/9/include/limits.h:34,
from /labs/linux-lab/src/linux-stable/tools/testing/selftests/nolibc/nolibc-test.c:6:
/usr/riscv64-linux-gnu/include/bits/wordsize.h:28:3: error: #error "rv32i-based targets are not supported"
28 | # error "rv32i-based targets are not supported"
Glibc (>= 2.33) commit 5b6113d62efa ("RISC-V: Support the 32-bit ABI
implementation") fixed up above error.
As suggested by Thomas, defining INT_MIN/INT_MAX for nolibc can remove
the including of limits.h, and therefore no above error. of course, the
other libcs still require limits.h, move it to the right place.
The LONG_MIN/LONG_MAX are also defined too.
Suggested-by: Thomas Weißschuh <linux@weissschuh.net>
Link: https://lore.kernel.org/linux-riscv/09d60dc2-e298-4c22-8e2f-8375861bd9be@t-8ch.de/
Signed-off-by: Zhangjin Wu <falcon@tinylab.org>
Reviewed-by: Thomas Weißschuh <linux@weissschuh.net>
Signed-off-by: Willy Tarreau <w@1wt.eu>
Signed-off-by: Paul E. McKenney <paulmck@kernel.org>
2023-06-03 16:06:57 +08:00
|
|
|
#include <limits.h>
|
2022-07-19 23:44:44 +02:00
|
|
|
#endif
|
2022-07-19 23:44:35 +02:00
|
|
|
#endif
|
|
|
|
|
2023-07-08 02:27:40 +08:00
|
|
|
/* for the type of int_fast16_t and int_fast32_t, musl differs from glibc and nolibc */
|
|
|
|
#define SINT_MAX_OF_TYPE(type) (((type)1 << (sizeof(type) * 8 - 2)) - (type)1 + ((type)1 << (sizeof(type) * 8 - 2)))
|
|
|
|
#define SINT_MIN_OF_TYPE(type) (-SINT_MAX_OF_TYPE(type) - 1)
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
/* will be used by nolibc by getenv() */
|
|
|
|
char **environ;
|
|
|
|
|
2023-07-07 23:01:09 +08:00
|
|
|
/* will be used by some test cases as readable file, please don't write it */
|
|
|
|
static const char *argv0;
|
|
|
|
|
2022-07-19 23:44:36 +02:00
|
|
|
/* definition of a series of tests */
|
|
|
|
struct test {
|
2023-04-06 21:54:49 +00:00
|
|
|
const char *name; /* test name */
|
|
|
|
int (*func)(int min, int max); /* handler */
|
2022-07-19 23:44:36 +02:00
|
|
|
};
|
|
|
|
|
2022-07-19 23:44:44 +02:00
|
|
|
#ifndef _NOLIBC_STDLIB_H
|
|
|
|
char *itoa(int i)
|
|
|
|
{
|
|
|
|
static char buf[12];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = snprintf(buf, sizeof(buf), "%d", i);
|
|
|
|
return (ret >= 0 && ret < sizeof(buf)) ? buf : "#err";
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
#define CASE_ERR(err) \
|
|
|
|
case err: return #err
|
|
|
|
|
|
|
|
/* returns the error name (e.g. "ENOENT") for common errors, "SUCCESS" for 0,
|
|
|
|
* or the decimal value for less common ones.
|
|
|
|
*/
|
|
|
|
const char *errorname(int err)
|
|
|
|
{
|
|
|
|
switch (err) {
|
|
|
|
case 0: return "SUCCESS";
|
|
|
|
CASE_ERR(EPERM);
|
|
|
|
CASE_ERR(ENOENT);
|
|
|
|
CASE_ERR(ESRCH);
|
|
|
|
CASE_ERR(EINTR);
|
|
|
|
CASE_ERR(EIO);
|
|
|
|
CASE_ERR(ENXIO);
|
|
|
|
CASE_ERR(E2BIG);
|
|
|
|
CASE_ERR(ENOEXEC);
|
|
|
|
CASE_ERR(EBADF);
|
|
|
|
CASE_ERR(ECHILD);
|
|
|
|
CASE_ERR(EAGAIN);
|
|
|
|
CASE_ERR(ENOMEM);
|
|
|
|
CASE_ERR(EACCES);
|
|
|
|
CASE_ERR(EFAULT);
|
|
|
|
CASE_ERR(ENOTBLK);
|
|
|
|
CASE_ERR(EBUSY);
|
|
|
|
CASE_ERR(EEXIST);
|
|
|
|
CASE_ERR(EXDEV);
|
|
|
|
CASE_ERR(ENODEV);
|
|
|
|
CASE_ERR(ENOTDIR);
|
|
|
|
CASE_ERR(EISDIR);
|
|
|
|
CASE_ERR(EINVAL);
|
|
|
|
CASE_ERR(ENFILE);
|
|
|
|
CASE_ERR(EMFILE);
|
|
|
|
CASE_ERR(ENOTTY);
|
|
|
|
CASE_ERR(ETXTBSY);
|
|
|
|
CASE_ERR(EFBIG);
|
|
|
|
CASE_ERR(ENOSPC);
|
|
|
|
CASE_ERR(ESPIPE);
|
|
|
|
CASE_ERR(EROFS);
|
|
|
|
CASE_ERR(EMLINK);
|
|
|
|
CASE_ERR(EPIPE);
|
|
|
|
CASE_ERR(EDOM);
|
|
|
|
CASE_ERR(ERANGE);
|
|
|
|
CASE_ERR(ENOSYS);
|
2023-05-25 01:46:54 +08:00
|
|
|
CASE_ERR(EOVERFLOW);
|
2022-07-19 23:44:35 +02:00
|
|
|
default:
|
|
|
|
return itoa(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-17 18:01:31 +02:00
|
|
|
static void putcharn(char c, size_t n)
|
|
|
|
{
|
|
|
|
char buf[64];
|
|
|
|
|
|
|
|
memset(buf, c, n);
|
|
|
|
buf[n] = '\0';
|
|
|
|
fputs(buf, stdout);
|
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
static int pad_spc(int llen, int cnt, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
int ret;
|
|
|
|
|
2023-04-17 18:01:31 +02:00
|
|
|
putcharn(' ', cnt - llen);
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
va_start(args, fmt);
|
|
|
|
ret = vfprintf(stdout, fmt, args);
|
|
|
|
va_end(args);
|
2023-04-17 18:01:31 +02:00
|
|
|
return ret < 0 ? ret : ret + cnt - llen;
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* The tests below are intended to be used by the macroes, which evaluate
|
|
|
|
* expression <expr>, print the status to stdout, and update the "ret"
|
|
|
|
* variable to count failures. The functions themselves return the number
|
|
|
|
* of failures, thus either 0 or 1.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define EXPECT_ZR(cond, expr) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_zr(expr, llen); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_zr(int expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = !(expr == 0);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_NZ(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_nz(expr, llen; } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_nz(int expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = !(expr != 0);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_EQ(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_eq(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
2023-03-04 15:26:05 +01:00
|
|
|
static int expect_eq(uint64_t expr, int llen, uint64_t val)
|
2022-07-19 23:44:35 +02:00
|
|
|
{
|
|
|
|
int ret = !(expr == val);
|
|
|
|
|
2023-06-03 16:05:30 +08:00
|
|
|
llen += printf(" = %lld ", (long long)expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_NE(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ne(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_ne(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = !(expr != val);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_GE(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ge(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_ge(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = !(expr >= val);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_GT(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_gt(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_gt(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = !(expr > val);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_LE(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_le(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_le(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = !(expr <= val);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_LT(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_lt(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_lt(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = !(expr < val);
|
|
|
|
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_SYSZR(cond, expr) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_syszr(expr, llen); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_syszr(int expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (expr) {
|
|
|
|
ret = 1;
|
|
|
|
llen += printf(" = %d %s ", expr, errorname(errno));
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_SYSEQ(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_syseq(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_syseq(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (expr != val) {
|
|
|
|
ret = 1;
|
|
|
|
llen += printf(" = %d %s ", expr, errorname(errno));
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_SYSNE(cond, expr, val) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_sysne(expr, llen, val); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_sysne(int expr, int llen, int val)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (expr == val) {
|
|
|
|
ret = 1;
|
|
|
|
llen += printf(" = %d %s ", expr, errorname(errno));
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
|
|
|
llen += printf(" = %d ", expr);
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-06-03 16:13:40 +08:00
|
|
|
#define EXPECT_SYSER2(cond, expr, expret, experr1, experr2) \
|
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_syserr2(expr, expret, experr1, experr2, llen); } while (0)
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
#define EXPECT_SYSER(cond, expr, expret, experr) \
|
2023-06-03 16:13:40 +08:00
|
|
|
EXPECT_SYSER2(cond, expr, expret, experr, 0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
2023-06-03 16:13:40 +08:00
|
|
|
static int expect_syserr2(int expr, int expret, int experr1, int experr2, int llen)
|
2022-07-19 23:44:35 +02:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
int _errno = errno;
|
|
|
|
|
|
|
|
llen += printf(" = %d %s ", expr, errorname(_errno));
|
2023-06-03 16:13:40 +08:00
|
|
|
if (expr != expret || (_errno != experr1 && _errno != experr2)) {
|
2022-07-19 23:44:35 +02:00
|
|
|
ret = 1;
|
2023-06-03 16:13:40 +08:00
|
|
|
if (experr2 == 0)
|
|
|
|
llen += printf(" != (%d %s) ", expret, errorname(experr1));
|
|
|
|
else
|
|
|
|
llen += printf(" != (%d %s %s) ", expret, errorname(experr1), errorname(experr2));
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_PTRZR(cond, expr) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ptrzr(expr, llen); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_ptrzr(const void *expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%p> ", expr);
|
|
|
|
if (expr) {
|
|
|
|
ret = 1;
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_PTRNZ(cond, expr) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ptrnz(expr, llen); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_ptrnz(const void *expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%p> ", expr);
|
|
|
|
if (!expr) {
|
|
|
|
ret = 1;
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-07-07 23:03:26 +08:00
|
|
|
#define EXPECT_PTREQ(cond, expr, cmp) \
|
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ptreq(expr, llen, cmp); } while (0)
|
|
|
|
|
|
|
|
static int expect_ptreq(const void *expr, int llen, const void *cmp)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%p> ", expr);
|
|
|
|
if (expr != cmp) {
|
|
|
|
ret = 1;
|
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
} else {
|
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define EXPECT_PTRNE(cond, expr, cmp) \
|
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ptrne(expr, llen, cmp); } while (0)
|
|
|
|
|
|
|
|
static int expect_ptrne(const void *expr, int llen, const void *cmp)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%p> ", expr);
|
|
|
|
if (expr == cmp) {
|
|
|
|
ret = 1;
|
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
} else {
|
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define EXPECT_PTRER2(cond, expr, expret, experr1, experr2) \
|
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_ptrerr2(expr, expret, experr1, experr2, llen); } while (0)
|
|
|
|
|
|
|
|
#define EXPECT_PTRER(cond, expr, expret, experr) \
|
|
|
|
EXPECT_PTRER2(cond, expr, expret, experr, 0)
|
|
|
|
|
|
|
|
static int expect_ptrerr2(const void *expr, const void *expret, int experr1, int experr2, int llen)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
int _errno = errno;
|
|
|
|
|
|
|
|
llen += printf(" = <%p> %s ", expr, errorname(_errno));
|
|
|
|
if (expr != expret || (_errno != experr1 && _errno != experr2)) {
|
|
|
|
ret = 1;
|
|
|
|
if (experr2 == 0)
|
|
|
|
llen += printf(" != (<%p> %s) ", expret, errorname(experr1));
|
|
|
|
else
|
|
|
|
llen += printf(" != (<%p> %s %s) ", expret, errorname(experr1), errorname(experr2));
|
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
} else {
|
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
#define EXPECT_STRZR(cond, expr) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_strzr(expr, llen); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_strzr(const char *expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%s> ", expr);
|
|
|
|
if (expr) {
|
|
|
|
ret = 1;
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_STRNZ(cond, expr) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_strnz(expr, llen); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_strnz(const char *expr, int llen)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%s> ", expr);
|
|
|
|
if (!expr) {
|
|
|
|
ret = 1;
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_STREQ(cond, expr, cmp) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_streq(expr, llen, cmp); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_streq(const char *expr, int llen, const char *cmp)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%s> ", expr);
|
|
|
|
if (strcmp(expr, cmp) != 0) {
|
|
|
|
ret = 1;
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPECT_STRNE(cond, expr, cmp) \
|
2023-03-04 15:26:05 +01:00
|
|
|
do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_strne(expr, llen, cmp); } while (0)
|
2022-07-19 23:44:35 +02:00
|
|
|
|
|
|
|
static int expect_strne(const char *expr, int llen, const char *cmp)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
llen += printf(" = <%s> ", expr);
|
|
|
|
if (strcmp(expr, cmp) == 0) {
|
|
|
|
ret = 1;
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, "[FAIL]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
} else {
|
2023-03-04 15:26:05 +01:00
|
|
|
llen += pad_spc(llen, 64, " [OK]\n");
|
2022-07-19 23:44:35 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:36 +02:00
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
/* declare tests based on line numbers. There must be exactly one test per line. */
|
|
|
|
#define CASE_TEST(name) \
|
|
|
|
case __LINE__: llen += printf("%d %s", test, #name);
|
|
|
|
|
|
|
|
|
2022-07-19 23:44:37 +02:00
|
|
|
/* used by some syscall tests below */
|
|
|
|
int test_getdents64(const char *dir)
|
|
|
|
{
|
|
|
|
char buffer[4096];
|
|
|
|
int fd, ret;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
ret = fd = open(dir, O_RDONLY | O_DIRECTORY, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = getdents64(fd, (void *)buffer, sizeof(buffer));
|
|
|
|
err = errno;
|
|
|
|
close(fd);
|
|
|
|
|
|
|
|
errno = err;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-01-10 08:24:34 +01:00
|
|
|
static int test_getpagesize(void)
|
|
|
|
{
|
|
|
|
long x = getpagesize();
|
|
|
|
int c;
|
|
|
|
|
|
|
|
if (x < 0)
|
|
|
|
return x;
|
|
|
|
|
|
|
|
#if defined(__x86_64__) || defined(__i386__) || defined(__i486__) || defined(__i586__) || defined(__i686__)
|
|
|
|
/*
|
|
|
|
* x86 family is always 4K page.
|
|
|
|
*/
|
|
|
|
c = (x == 4096);
|
|
|
|
#elif defined(__aarch64__)
|
|
|
|
/*
|
|
|
|
* Linux aarch64 supports three values of page size: 4K, 16K, and 64K
|
|
|
|
* which are selected at kernel compilation time.
|
|
|
|
*/
|
|
|
|
c = (x == 4096 || x == (16 * 1024) || x == (64 * 1024));
|
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* Assuming other architectures must have at least 4K page.
|
|
|
|
*/
|
|
|
|
c = (x >= 4096);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return !c;
|
|
|
|
}
|
|
|
|
|
2023-04-15 23:28:48 +02:00
|
|
|
static int test_fork(void)
|
|
|
|
{
|
|
|
|
int status;
|
selftests/nolibc: test_fork: fix up duplicated print
running nolibc-test with glibc on x86_64 got such print issue:
29 execve_root = -1 EACCES [OK]
30 fork30 fork = 0 [OK]
31 getdents64_root = 712 [OK]
The fork test case has three printf calls:
(1) llen += printf("%d %s", test, #name);
(2) llen += printf(" = %d %s ", expr, errorname(errno));
(3) llen += pad_spc(llen, 64, "[FAIL]\n"); --> vfprintf()
In the following scene, the above issue happens:
(a) The parent calls (1)
(b) The parent calls fork()
(c) The child runs and shares the print buffer of (1)
(d) The child exits, flushs the print buffer and closes its own stdout/stderr
* "30 fork" is printed at the first time.
(e) The parent calls (2) and (3), with "\n" in (3), it flushs the whole buffer
* "30 fork = 0 ..." is printed
Therefore, there are two "30 fork" in the stdout.
Between (a) and (b), if flush the stdout (and the sterr), the child in
stage (c) will not be able to 'see' the print buffer.
Signed-off-by: Zhangjin Wu <falcon@tinylab.org>
Reviewed-by: Thomas Weißschuh <linux@weissschuh.net>
Signed-off-by: Willy Tarreau <w@1wt.eu>
Signed-off-by: Paul E. McKenney <paulmck@kernel.org>
2023-05-30 14:03:33 +08:00
|
|
|
pid_t pid;
|
|
|
|
|
|
|
|
/* flush the printf buffer to avoid child flush it */
|
|
|
|
fflush(stdout);
|
|
|
|
fflush(stderr);
|
|
|
|
|
|
|
|
pid = fork();
|
2023-04-15 23:28:48 +02:00
|
|
|
|
|
|
|
switch (pid) {
|
|
|
|
case -1:
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
exit(123);
|
|
|
|
|
|
|
|
default:
|
|
|
|
pid = waitpid(pid, &status, 0);
|
|
|
|
|
|
|
|
return pid == -1 || !WIFEXITED(status) || WEXITSTATUS(status) != 123;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-27 14:56:42 +02:00
|
|
|
static int test_stat_timestamps(void)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
|
|
|
|
if (sizeof(st.st_atim.tv_sec) != sizeof(st.st_atime))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (stat("/proc/self/", &st))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (st.st_atim.tv_sec != st.st_atime || st.st_atim.tv_nsec > 1000000000)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (st.st_mtim.tv_sec != st.st_mtime || st.st_mtim.tv_nsec > 1000000000)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (st.st_ctim.tv_sec != st.st_ctime || st.st_ctim.tv_nsec > 1000000000)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-07-07 23:08:20 +08:00
|
|
|
int test_mmap_munmap(void)
|
|
|
|
{
|
|
|
|
int ret, fd, i;
|
|
|
|
void *mem;
|
|
|
|
size_t page_size, file_size, length;
|
|
|
|
off_t offset, pa_offset;
|
|
|
|
struct stat stat_buf;
|
|
|
|
const char * const files[] = {
|
|
|
|
"/dev/zero",
|
|
|
|
"/proc/1/exe", "/proc/self/exe",
|
|
|
|
argv0,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
page_size = getpagesize();
|
|
|
|
if (page_size < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* find a right file to mmap, existed and accessible */
|
|
|
|
for (i = 0; files[i] != NULL; i++) {
|
|
|
|
ret = fd = open(files[i], O_RDONLY);
|
|
|
|
if (ret == -1)
|
|
|
|
continue;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (ret == -1)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = stat(files[i], &stat_buf);
|
|
|
|
if (ret == -1)
|
|
|
|
goto end;
|
|
|
|
|
|
|
|
/* file size of the special /dev/zero is 0, let's assign one manually */
|
|
|
|
if (i == 0)
|
|
|
|
file_size = 3*page_size;
|
|
|
|
else
|
|
|
|
file_size = stat_buf.st_size;
|
|
|
|
|
|
|
|
offset = file_size - 1;
|
|
|
|
if (offset < 0)
|
|
|
|
offset = 0;
|
|
|
|
length = file_size - offset;
|
|
|
|
pa_offset = offset & ~(page_size - 1);
|
|
|
|
|
|
|
|
mem = mmap(NULL, length + offset - pa_offset, PROT_READ, MAP_SHARED, fd, pa_offset);
|
|
|
|
if (mem == MAP_FAILED) {
|
|
|
|
ret = -1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = munmap(mem, length + offset - pa_offset);
|
|
|
|
|
|
|
|
end:
|
|
|
|
close(fd);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-19 23:44:37 +02:00
|
|
|
/* Run syscall tests between IDs <min> and <max>.
|
|
|
|
* Return 0 on success, non-zero on failure.
|
|
|
|
*/
|
|
|
|
int run_syscall(int min, int max)
|
|
|
|
{
|
2023-06-03 16:16:07 +08:00
|
|
|
struct timeval tv;
|
|
|
|
struct timezone tz;
|
2022-07-19 23:44:37 +02:00
|
|
|
struct stat stat_buf;
|
2023-03-04 15:28:41 +01:00
|
|
|
int euid0;
|
2022-07-19 23:44:43 +02:00
|
|
|
int proc;
|
2022-07-19 23:44:37 +02:00
|
|
|
int test;
|
|
|
|
int tmp;
|
|
|
|
int ret = 0;
|
|
|
|
void *p1, *p2;
|
2023-07-08 02:25:11 +08:00
|
|
|
int has_gettid = 1;
|
2022-07-19 23:44:37 +02:00
|
|
|
|
2022-07-19 23:44:43 +02:00
|
|
|
/* <proc> indicates whether or not /proc is mounted */
|
|
|
|
proc = stat("/proc", &stat_buf) == 0;
|
|
|
|
|
2023-03-04 15:28:41 +01:00
|
|
|
/* this will be used to skip certain tests that can't be run unprivileged */
|
|
|
|
euid0 = geteuid() == 0;
|
|
|
|
|
2023-07-08 02:25:11 +08:00
|
|
|
/* from 2.30, glibc provides gettid() */
|
|
|
|
#if defined(__GLIBC_MINOR__) && defined(__GLIBC__)
|
|
|
|
has_gettid = __GLIBC__ > 2 || (__GLIBC__ == 2 && __GLIBC_MINOR__ >= 30);
|
|
|
|
#endif
|
|
|
|
|
2022-07-19 23:44:37 +02:00
|
|
|
for (test = min; test >= 0 && test <= max; test++) {
|
2023-04-06 21:54:49 +00:00
|
|
|
int llen = 0; /* line length */
|
2022-07-19 23:44:37 +02:00
|
|
|
|
|
|
|
/* avoid leaving empty lines below, this will insert holes into
|
|
|
|
* test numbers.
|
|
|
|
*/
|
|
|
|
switch (test + __LINE__ + 1) {
|
|
|
|
CASE_TEST(getpid); EXPECT_SYSNE(1, getpid(), -1); break;
|
|
|
|
CASE_TEST(getppid); EXPECT_SYSNE(1, getppid(), -1); break;
|
2023-07-08 02:25:11 +08:00
|
|
|
CASE_TEST(gettid); EXPECT_SYSNE(has_gettid, gettid(), -1); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(getpgid_self); EXPECT_SYSNE(1, getpgid(0), -1); break;
|
|
|
|
CASE_TEST(getpgid_bad); EXPECT_SYSER(1, getpgid(-1), -1, ESRCH); break;
|
|
|
|
CASE_TEST(kill_0); EXPECT_SYSZR(1, kill(getpid(), 0)); break;
|
|
|
|
CASE_TEST(kill_CONT); EXPECT_SYSZR(1, kill(getpid(), 0)); break;
|
|
|
|
CASE_TEST(kill_BADPID); EXPECT_SYSER(1, kill(INT_MAX, 0), -1, ESRCH); break;
|
2023-07-07 23:04:38 +08:00
|
|
|
CASE_TEST(sbrk_0); EXPECT_PTRNE(1, sbrk(0), (void *)-1); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(sbrk); if ((p1 = p2 = sbrk(4096)) != (void *)-1) p2 = sbrk(-4096); EXPECT_SYSZR(1, (p2 == (void *)-1) || p2 == p1); break;
|
|
|
|
CASE_TEST(brk); EXPECT_SYSZR(1, brk(sbrk(0))); break;
|
|
|
|
CASE_TEST(chdir_root); EXPECT_SYSZR(1, chdir("/")); break;
|
|
|
|
CASE_TEST(chdir_dot); EXPECT_SYSZR(1, chdir(".")); break;
|
|
|
|
CASE_TEST(chdir_blah); EXPECT_SYSER(1, chdir("/blah"), -1, ENOENT); break;
|
2022-07-19 23:44:43 +02:00
|
|
|
CASE_TEST(chmod_net); EXPECT_SYSZR(proc, chmod("/proc/self/net", 0555)); break;
|
|
|
|
CASE_TEST(chmod_self); EXPECT_SYSER(proc, chmod("/proc/self", 0555), -1, EPERM); break;
|
|
|
|
CASE_TEST(chown_self); EXPECT_SYSER(proc, chown("/proc/self", 0, 0), -1, EPERM); break;
|
2023-03-04 15:28:41 +01:00
|
|
|
CASE_TEST(chroot_root); EXPECT_SYSZR(euid0, chroot("/")); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(chroot_blah); EXPECT_SYSER(1, chroot("/proc/self/blah"), -1, ENOENT); break;
|
2022-07-19 23:44:43 +02:00
|
|
|
CASE_TEST(chroot_exe); EXPECT_SYSER(proc, chroot("/proc/self/exe"), -1, ENOTDIR); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(close_m1); EXPECT_SYSER(1, close(-1), -1, EBADF); break;
|
|
|
|
CASE_TEST(close_dup); EXPECT_SYSZR(1, close(dup(0))); break;
|
|
|
|
CASE_TEST(dup_0); tmp = dup(0); EXPECT_SYSNE(1, tmp, -1); close(tmp); break;
|
|
|
|
CASE_TEST(dup_m1); tmp = dup(-1); EXPECT_SYSER(1, tmp, -1, EBADF); if (tmp != -1) close(tmp); break;
|
|
|
|
CASE_TEST(dup2_0); tmp = dup2(0, 100); EXPECT_SYSNE(1, tmp, -1); close(tmp); break;
|
|
|
|
CASE_TEST(dup2_m1); tmp = dup2(-1, 100); EXPECT_SYSER(1, tmp, -1, EBADF); if (tmp != -1) close(tmp); break;
|
|
|
|
CASE_TEST(dup3_0); tmp = dup3(0, 100, 0); EXPECT_SYSNE(1, tmp, -1); close(tmp); break;
|
|
|
|
CASE_TEST(dup3_m1); tmp = dup3(-1, 100, 0); EXPECT_SYSER(1, tmp, -1, EBADF); if (tmp != -1) close(tmp); break;
|
|
|
|
CASE_TEST(execve_root); EXPECT_SYSER(1, execve("/", (char*[]){ [0] = "/", [1] = NULL }, NULL), -1, EACCES); break;
|
2023-04-15 23:28:48 +02:00
|
|
|
CASE_TEST(fork); EXPECT_SYSZR(1, test_fork()); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(getdents64_root); EXPECT_SYSNE(1, test_getdents64("/"), -1); break;
|
|
|
|
CASE_TEST(getdents64_null); EXPECT_SYSER(1, test_getdents64("/dev/null"), -1, ENOTDIR); break;
|
2023-06-03 16:16:07 +08:00
|
|
|
CASE_TEST(gettimeofday_tv); EXPECT_SYSZR(1, gettimeofday(&tv, NULL)); break;
|
|
|
|
CASE_TEST(gettimeofday_tv_tz);EXPECT_SYSZR(1, gettimeofday(&tv, &tz)); break;
|
2023-01-10 08:24:34 +01:00
|
|
|
CASE_TEST(getpagesize); EXPECT_SYSZR(1, test_getpagesize()); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(ioctl_tiocinq); EXPECT_SYSZR(1, ioctl(0, TIOCINQ, &tmp)); break;
|
|
|
|
CASE_TEST(ioctl_tiocinq); EXPECT_SYSZR(1, ioctl(0, TIOCINQ, &tmp)); break;
|
|
|
|
CASE_TEST(link_root1); EXPECT_SYSER(1, link("/", "/"), -1, EEXIST); break;
|
|
|
|
CASE_TEST(link_blah); EXPECT_SYSER(1, link("/proc/self/blah", "/blah"), -1, ENOENT); break;
|
2023-03-04 15:28:41 +01:00
|
|
|
CASE_TEST(link_dir); EXPECT_SYSER(euid0, link("/", "/blah"), -1, EPERM); break;
|
2022-07-19 23:44:43 +02:00
|
|
|
CASE_TEST(link_cross); EXPECT_SYSER(proc, link("/proc/self/net", "/blah"), -1, EXDEV); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(lseek_m1); EXPECT_SYSER(1, lseek(-1, 0, SEEK_SET), -1, EBADF); break;
|
|
|
|
CASE_TEST(lseek_0); EXPECT_SYSER(1, lseek(0, 0, SEEK_SET), -1, ESPIPE); break;
|
|
|
|
CASE_TEST(mkdir_root); EXPECT_SYSER(1, mkdir("/", 0755), -1, EEXIST); break;
|
2023-07-07 23:05:49 +08:00
|
|
|
CASE_TEST(mmap_bad); EXPECT_PTRER(1, mmap(NULL, 0, PROT_READ, MAP_PRIVATE, 0, 0), MAP_FAILED, EINVAL); break;
|
2023-07-07 23:07:13 +08:00
|
|
|
CASE_TEST(munmap_bad); EXPECT_SYSER(1, munmap((void *)1, 0), -1, EINVAL); break;
|
2023-07-07 23:08:20 +08:00
|
|
|
CASE_TEST(mmap_munmap_good); EXPECT_SYSZR(1, test_mmap_munmap()); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(open_tty); EXPECT_SYSNE(1, tmp = open("/dev/null", 0), -1); if (tmp != -1) close(tmp); break;
|
|
|
|
CASE_TEST(open_blah); EXPECT_SYSER(1, tmp = open("/proc/self/blah", 0), -1, ENOENT); if (tmp != -1) close(tmp); break;
|
|
|
|
CASE_TEST(poll_null); EXPECT_SYSZR(1, poll(NULL, 0, 0)); break;
|
|
|
|
CASE_TEST(poll_stdout); EXPECT_SYSNE(1, ({ struct pollfd fds = { 1, POLLOUT, 0}; poll(&fds, 1, 0); }), -1); break;
|
|
|
|
CASE_TEST(poll_fault); EXPECT_SYSER(1, poll((void *)1, 1, 0), -1, EFAULT); break;
|
2023-05-26 09:51:19 +02:00
|
|
|
CASE_TEST(prctl); EXPECT_SYSER(1, prctl(PR_SET_NAME, (unsigned long)NULL, 0, 0, 0), -1, EFAULT); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(read_badf); EXPECT_SYSER(1, read(-1, &tmp, 1), -1, EBADF); break;
|
|
|
|
CASE_TEST(sched_yield); EXPECT_SYSZR(1, sched_yield()); break;
|
|
|
|
CASE_TEST(select_null); EXPECT_SYSZR(1, ({ struct timeval tv = { 0 }; select(0, NULL, NULL, NULL, &tv); })); break;
|
|
|
|
CASE_TEST(select_stdout); EXPECT_SYSNE(1, ({ fd_set fds; FD_ZERO(&fds); FD_SET(1, &fds); select(2, NULL, &fds, NULL, NULL); }), -1); break;
|
|
|
|
CASE_TEST(select_fault); EXPECT_SYSER(1, select(1, (void *)1, NULL, NULL, 0), -1, EFAULT); break;
|
|
|
|
CASE_TEST(stat_blah); EXPECT_SYSER(1, stat("/proc/self/blah", &stat_buf), -1, ENOENT); break;
|
selftests/nolibc: stat_fault: silence NULL argument warning with glibc
Use another invalid address (void *)1 instead of NULL to silence this
compile warning with glibc:
$ make libc-test
CC libc-test
nolibc-test.c: In function ‘run_syscall’:
nolibc-test.c:622:49: warning: null argument where non-null required (argument 1) [-Wnonnull]
622 | CASE_TEST(stat_fault); EXPECT_SYSER(1, stat(NULL, &stat_buf), -1, EFAULT); break;
| ^~~~
nolibc-test.c:304:79: note: in definition of macro ‘EXPECT_SYSER2’
304 | do { if (!cond) pad_spc(llen, 64, "[SKIPPED]\n"); else ret += expect_syserr2(expr, expret, experr1, experr2, llen); } while (0)
| ^~~~
nolibc-test.c:622:33: note: in expansion of macro ‘EXPECT_SYSER’
622 | CASE_TEST(stat_fault); EXPECT_SYSER(1, stat(NULL, &stat_buf), -1, EFAULT); break;
Reviewed-by: Thomas Weißschuh <linux@weissschuh.net>
Signed-off-by: Zhangjin Wu <falcon@tinylab.org>
Signed-off-by: Willy Tarreau <w@1wt.eu>
2023-07-08 02:23:45 +08:00
|
|
|
CASE_TEST(stat_fault); EXPECT_SYSER(1, stat((void *)1, &stat_buf), -1, EFAULT); break;
|
2023-05-27 14:56:42 +02:00
|
|
|
CASE_TEST(stat_timestamps); EXPECT_SYSZR(1, test_stat_timestamps()); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
CASE_TEST(symlink_root); EXPECT_SYSER(1, symlink("/", "/"), -1, EEXIST); break;
|
|
|
|
CASE_TEST(unlink_root); EXPECT_SYSER(1, unlink("/"), -1, EISDIR); break;
|
|
|
|
CASE_TEST(unlink_blah); EXPECT_SYSER(1, unlink("/proc/self/blah"), -1, ENOENT); break;
|
|
|
|
CASE_TEST(wait_child); EXPECT_SYSER(1, wait(&tmp), -1, ECHILD); break;
|
|
|
|
CASE_TEST(waitpid_min); EXPECT_SYSER(1, waitpid(INT_MIN, &tmp, WNOHANG), -1, ESRCH); break;
|
|
|
|
CASE_TEST(waitpid_child); EXPECT_SYSER(1, waitpid(getpid(), &tmp, WNOHANG), -1, ECHILD); break;
|
|
|
|
CASE_TEST(write_badf); EXPECT_SYSER(1, write(-1, &tmp, 1), -1, EBADF); break;
|
|
|
|
CASE_TEST(write_zero); EXPECT_SYSZR(1, write(1, &tmp, 0)); break;
|
2023-05-20 09:58:57 +02:00
|
|
|
CASE_TEST(syscall_noargs); EXPECT_SYSEQ(1, syscall(__NR_getpid), getpid()); break;
|
2023-06-03 16:00:46 +08:00
|
|
|
CASE_TEST(syscall_args); EXPECT_SYSER(1, syscall(__NR_statx, 0, NULL, 0, 0, NULL), -1, EFAULT); break;
|
2022-07-19 23:44:37 +02:00
|
|
|
case __LINE__:
|
|
|
|
return ret; /* must be last */
|
|
|
|
/* note: do not set any defaults so as to permit holes above */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:38 +02:00
|
|
|
int run_stdlib(int min, int max)
|
|
|
|
{
|
|
|
|
int test;
|
|
|
|
int tmp;
|
|
|
|
int ret = 0;
|
|
|
|
void *p1, *p2;
|
|
|
|
|
|
|
|
for (test = min; test >= 0 && test <= max; test++) {
|
2023-04-06 21:54:49 +00:00
|
|
|
int llen = 0; /* line length */
|
2022-07-19 23:44:38 +02:00
|
|
|
|
|
|
|
/* avoid leaving empty lines below, this will insert holes into
|
|
|
|
* test numbers.
|
|
|
|
*/
|
|
|
|
switch (test + __LINE__ + 1) {
|
|
|
|
CASE_TEST(getenv_TERM); EXPECT_STRNZ(1, getenv("TERM")); break;
|
|
|
|
CASE_TEST(getenv_blah); EXPECT_STRZR(1, getenv("blah")); break;
|
|
|
|
CASE_TEST(setcmp_blah_blah); EXPECT_EQ(1, strcmp("blah", "blah"), 0); break;
|
|
|
|
CASE_TEST(setcmp_blah_blah2); EXPECT_NE(1, strcmp("blah", "blah2"), 0); break;
|
|
|
|
CASE_TEST(setncmp_blah_blah); EXPECT_EQ(1, strncmp("blah", "blah", 10), 0); break;
|
|
|
|
CASE_TEST(setncmp_blah_blah4); EXPECT_EQ(1, strncmp("blah", "blah4", 4), 0); break;
|
|
|
|
CASE_TEST(setncmp_blah_blah5); EXPECT_NE(1, strncmp("blah", "blah5", 5), 0); break;
|
|
|
|
CASE_TEST(setncmp_blah_blah6); EXPECT_NE(1, strncmp("blah", "blah6", 6), 0); break;
|
|
|
|
CASE_TEST(strchr_foobar_o); EXPECT_STREQ(1, strchr("foobar", 'o'), "oobar"); break;
|
|
|
|
CASE_TEST(strchr_foobar_z); EXPECT_STRZR(1, strchr("foobar", 'z')); break;
|
|
|
|
CASE_TEST(strrchr_foobar_o); EXPECT_STREQ(1, strrchr("foobar", 'o'), "obar"); break;
|
|
|
|
CASE_TEST(strrchr_foobar_z); EXPECT_STRZR(1, strrchr("foobar", 'z')); break;
|
2022-10-21 08:03:40 +02:00
|
|
|
CASE_TEST(memcmp_20_20); EXPECT_EQ(1, memcmp("aaa\x20", "aaa\x20", 4), 0); break;
|
|
|
|
CASE_TEST(memcmp_20_60); EXPECT_LT(1, memcmp("aaa\x20", "aaa\x60", 4), 0); break;
|
|
|
|
CASE_TEST(memcmp_60_20); EXPECT_GT(1, memcmp("aaa\x60", "aaa\x20", 4), 0); break;
|
|
|
|
CASE_TEST(memcmp_20_e0); EXPECT_LT(1, memcmp("aaa\x20", "aaa\xe0", 4), 0); break;
|
|
|
|
CASE_TEST(memcmp_e0_20); EXPECT_GT(1, memcmp("aaa\xe0", "aaa\x20", 4), 0); break;
|
|
|
|
CASE_TEST(memcmp_80_e0); EXPECT_LT(1, memcmp("aaa\x80", "aaa\xe0", 4), 0); break;
|
|
|
|
CASE_TEST(memcmp_e0_80); EXPECT_GT(1, memcmp("aaa\xe0", "aaa\x80", 4), 0); break;
|
2023-03-04 15:26:06 +01:00
|
|
|
CASE_TEST(limit_int8_max); EXPECT_EQ(1, INT8_MAX, (int8_t) 0x7f); break;
|
|
|
|
CASE_TEST(limit_int8_min); EXPECT_EQ(1, INT8_MIN, (int8_t) 0x80); break;
|
|
|
|
CASE_TEST(limit_uint8_max); EXPECT_EQ(1, UINT8_MAX, (uint8_t) 0xff); break;
|
|
|
|
CASE_TEST(limit_int16_max); EXPECT_EQ(1, INT16_MAX, (int16_t) 0x7fff); break;
|
|
|
|
CASE_TEST(limit_int16_min); EXPECT_EQ(1, INT16_MIN, (int16_t) 0x8000); break;
|
|
|
|
CASE_TEST(limit_uint16_max); EXPECT_EQ(1, UINT16_MAX, (uint16_t) 0xffff); break;
|
|
|
|
CASE_TEST(limit_int32_max); EXPECT_EQ(1, INT32_MAX, (int32_t) 0x7fffffff); break;
|
|
|
|
CASE_TEST(limit_int32_min); EXPECT_EQ(1, INT32_MIN, (int32_t) 0x80000000); break;
|
|
|
|
CASE_TEST(limit_uint32_max); EXPECT_EQ(1, UINT32_MAX, (uint32_t) 0xffffffff); break;
|
|
|
|
CASE_TEST(limit_int64_max); EXPECT_EQ(1, INT64_MAX, (int64_t) 0x7fffffffffffffff); break;
|
|
|
|
CASE_TEST(limit_int64_min); EXPECT_EQ(1, INT64_MIN, (int64_t) 0x8000000000000000); break;
|
|
|
|
CASE_TEST(limit_uint64_max); EXPECT_EQ(1, UINT64_MAX, (uint64_t) 0xffffffffffffffff); break;
|
|
|
|
CASE_TEST(limit_int_least8_max); EXPECT_EQ(1, INT_LEAST8_MAX, (int_least8_t) 0x7f); break;
|
|
|
|
CASE_TEST(limit_int_least8_min); EXPECT_EQ(1, INT_LEAST8_MIN, (int_least8_t) 0x80); break;
|
|
|
|
CASE_TEST(limit_uint_least8_max); EXPECT_EQ(1, UINT_LEAST8_MAX, (uint_least8_t) 0xff); break;
|
|
|
|
CASE_TEST(limit_int_least16_max); EXPECT_EQ(1, INT_LEAST16_MAX, (int_least16_t) 0x7fff); break;
|
|
|
|
CASE_TEST(limit_int_least16_min); EXPECT_EQ(1, INT_LEAST16_MIN, (int_least16_t) 0x8000); break;
|
|
|
|
CASE_TEST(limit_uint_least16_max); EXPECT_EQ(1, UINT_LEAST16_MAX, (uint_least16_t) 0xffff); break;
|
|
|
|
CASE_TEST(limit_int_least32_max); EXPECT_EQ(1, INT_LEAST32_MAX, (int_least32_t) 0x7fffffff); break;
|
|
|
|
CASE_TEST(limit_int_least32_min); EXPECT_EQ(1, INT_LEAST32_MIN, (int_least32_t) 0x80000000); break;
|
|
|
|
CASE_TEST(limit_uint_least32_max); EXPECT_EQ(1, UINT_LEAST32_MAX, (uint_least32_t) 0xffffffffU); break;
|
|
|
|
CASE_TEST(limit_int_least64_min); EXPECT_EQ(1, INT_LEAST64_MIN, (int_least64_t) 0x8000000000000000LL); break;
|
|
|
|
CASE_TEST(limit_int_least64_max); EXPECT_EQ(1, INT_LEAST64_MAX, (int_least64_t) 0x7fffffffffffffffLL); break;
|
|
|
|
CASE_TEST(limit_uint_least64_max); EXPECT_EQ(1, UINT_LEAST64_MAX, (uint_least64_t) 0xffffffffffffffffULL); break;
|
|
|
|
CASE_TEST(limit_int_fast8_max); EXPECT_EQ(1, INT_FAST8_MAX, (int_fast8_t) 0x7f); break;
|
|
|
|
CASE_TEST(limit_int_fast8_min); EXPECT_EQ(1, INT_FAST8_MIN, (int_fast8_t) 0x80); break;
|
|
|
|
CASE_TEST(limit_uint_fast8_max); EXPECT_EQ(1, UINT_FAST8_MAX, (uint_fast8_t) 0xff); break;
|
2023-07-08 02:27:40 +08:00
|
|
|
CASE_TEST(limit_int_fast16_min); EXPECT_EQ(1, INT_FAST16_MIN, (int_fast16_t) SINT_MIN_OF_TYPE(int_fast16_t)); break;
|
|
|
|
CASE_TEST(limit_int_fast16_max); EXPECT_EQ(1, INT_FAST16_MAX, (int_fast16_t) SINT_MAX_OF_TYPE(int_fast16_t)); break;
|
2023-03-04 15:26:06 +01:00
|
|
|
CASE_TEST(limit_uint_fast16_max); EXPECT_EQ(1, UINT_FAST16_MAX, (uint_fast16_t) UINTPTR_MAX); break;
|
2023-07-08 02:27:40 +08:00
|
|
|
CASE_TEST(limit_int_fast32_min); EXPECT_EQ(1, INT_FAST32_MIN, (int_fast32_t) SINT_MIN_OF_TYPE(int_fast32_t)); break;
|
|
|
|
CASE_TEST(limit_int_fast32_max); EXPECT_EQ(1, INT_FAST32_MAX, (int_fast32_t) SINT_MAX_OF_TYPE(int_fast32_t)); break;
|
2023-03-04 15:26:06 +01:00
|
|
|
CASE_TEST(limit_uint_fast32_max); EXPECT_EQ(1, UINT_FAST32_MAX, (uint_fast32_t) UINTPTR_MAX); break;
|
2023-05-30 11:18:00 +02:00
|
|
|
CASE_TEST(limit_int_fast64_min); EXPECT_EQ(1, INT_FAST64_MIN, (int_fast64_t) INT64_MIN); break;
|
|
|
|
CASE_TEST(limit_int_fast64_max); EXPECT_EQ(1, INT_FAST64_MAX, (int_fast64_t) INT64_MAX); break;
|
|
|
|
CASE_TEST(limit_uint_fast64_max); EXPECT_EQ(1, UINT_FAST64_MAX, (uint_fast64_t) UINT64_MAX); break;
|
2023-03-04 15:26:06 +01:00
|
|
|
#if __SIZEOF_LONG__ == 8
|
|
|
|
CASE_TEST(limit_intptr_min); EXPECT_EQ(1, INTPTR_MIN, (intptr_t) 0x8000000000000000LL); break;
|
|
|
|
CASE_TEST(limit_intptr_max); EXPECT_EQ(1, INTPTR_MAX, (intptr_t) 0x7fffffffffffffffLL); break;
|
|
|
|
CASE_TEST(limit_uintptr_max); EXPECT_EQ(1, UINTPTR_MAX, (uintptr_t) 0xffffffffffffffffULL); break;
|
|
|
|
CASE_TEST(limit_ptrdiff_min); EXPECT_EQ(1, PTRDIFF_MIN, (ptrdiff_t) 0x8000000000000000LL); break;
|
|
|
|
CASE_TEST(limit_ptrdiff_max); EXPECT_EQ(1, PTRDIFF_MAX, (ptrdiff_t) 0x7fffffffffffffffLL); break;
|
|
|
|
CASE_TEST(limit_size_max); EXPECT_EQ(1, SIZE_MAX, (size_t) 0xffffffffffffffffULL); break;
|
|
|
|
#elif __SIZEOF_LONG__ == 4
|
|
|
|
CASE_TEST(limit_intptr_min); EXPECT_EQ(1, INTPTR_MIN, (intptr_t) 0x80000000); break;
|
|
|
|
CASE_TEST(limit_intptr_max); EXPECT_EQ(1, INTPTR_MAX, (intptr_t) 0x7fffffff); break;
|
|
|
|
CASE_TEST(limit_uintptr_max); EXPECT_EQ(1, UINTPTR_MAX, (uintptr_t) 0xffffffffU); break;
|
|
|
|
CASE_TEST(limit_ptrdiff_min); EXPECT_EQ(1, PTRDIFF_MIN, (ptrdiff_t) 0x80000000); break;
|
|
|
|
CASE_TEST(limit_ptrdiff_max); EXPECT_EQ(1, PTRDIFF_MAX, (ptrdiff_t) 0x7fffffff); break;
|
|
|
|
CASE_TEST(limit_size_max); EXPECT_EQ(1, SIZE_MAX, (size_t) 0xffffffffU); break;
|
|
|
|
#else
|
|
|
|
# warning "__SIZEOF_LONG__ is undefined"
|
|
|
|
#endif /* __SIZEOF_LONG__ */
|
2022-07-19 23:44:38 +02:00
|
|
|
case __LINE__:
|
|
|
|
return ret; /* must be last */
|
|
|
|
/* note: do not set any defaults so as to permit holes above */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-04-02 20:48:06 +02:00
|
|
|
#define EXPECT_VFPRINTF(c, expected, fmt, ...) \
|
|
|
|
ret += expect_vfprintf(llen, c, expected, fmt, ##__VA_ARGS__)
|
|
|
|
|
|
|
|
static int expect_vfprintf(int llen, size_t c, const char *expected, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
int ret, fd, w, r;
|
|
|
|
char buf[100];
|
|
|
|
FILE *memfile;
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
fd = memfd_create("vfprintf", 0);
|
|
|
|
if (fd == -1) {
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memfile = fdopen(fd, "w+");
|
|
|
|
if (!memfile) {
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
va_start(args, fmt);
|
|
|
|
w = vfprintf(memfile, fmt, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
if (w != c) {
|
|
|
|
llen += printf(" written(%d) != %d", w, (int) c);
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fflush(memfile);
|
|
|
|
lseek(fd, 0, SEEK_SET);
|
|
|
|
|
|
|
|
r = read(fd, buf, sizeof(buf) - 1);
|
|
|
|
buf[r] = '\0';
|
|
|
|
|
|
|
|
fclose(memfile);
|
|
|
|
|
|
|
|
if (r != w) {
|
|
|
|
llen += printf(" written(%d) != read(%d)", w, r);
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
llen += printf(" \"%s\" = \"%s\"", expected, buf);
|
|
|
|
ret = strncmp(expected, buf, c);
|
|
|
|
|
|
|
|
pad_spc(llen, 64, ret ? "[FAIL]\n" : " [OK]\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int run_vfprintf(int min, int max)
|
|
|
|
{
|
|
|
|
int test;
|
|
|
|
int tmp;
|
|
|
|
int ret = 0;
|
|
|
|
void *p1, *p2;
|
|
|
|
|
|
|
|
for (test = min; test >= 0 && test <= max; test++) {
|
2023-04-06 21:54:49 +00:00
|
|
|
int llen = 0; /* line length */
|
2023-04-02 20:48:06 +02:00
|
|
|
|
|
|
|
/* avoid leaving empty lines below, this will insert holes into
|
|
|
|
* test numbers.
|
|
|
|
*/
|
|
|
|
switch (test + __LINE__ + 1) {
|
|
|
|
CASE_TEST(empty); EXPECT_VFPRINTF(0, "", ""); break;
|
|
|
|
CASE_TEST(simple); EXPECT_VFPRINTF(3, "foo", "foo"); break;
|
|
|
|
CASE_TEST(string); EXPECT_VFPRINTF(3, "foo", "%s", "foo"); break;
|
|
|
|
CASE_TEST(number); EXPECT_VFPRINTF(4, "1234", "%d", 1234); break;
|
|
|
|
CASE_TEST(negnumber); EXPECT_VFPRINTF(5, "-1234", "%d", -1234); break;
|
|
|
|
CASE_TEST(unsigned); EXPECT_VFPRINTF(5, "12345", "%u", 12345); break;
|
|
|
|
CASE_TEST(char); EXPECT_VFPRINTF(1, "c", "%c", 'c'); break;
|
|
|
|
CASE_TEST(hex); EXPECT_VFPRINTF(1, "f", "%x", 0xf); break;
|
|
|
|
CASE_TEST(pointer); EXPECT_VFPRINTF(3, "0x1", "%p", (void *) 0x1); break;
|
|
|
|
case __LINE__:
|
|
|
|
return ret; /* must be last */
|
|
|
|
/* note: do not set any defaults so as to permit holes above */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-03-25 16:45:14 +01:00
|
|
|
static int smash_stack(void)
|
|
|
|
{
|
|
|
|
char buf[100];
|
2023-03-28 18:18:45 +02:00
|
|
|
volatile char *ptr = buf;
|
2023-04-02 10:13:56 +02:00
|
|
|
size_t i;
|
2023-03-25 16:45:14 +01:00
|
|
|
|
2023-04-02 10:13:56 +02:00
|
|
|
for (i = 0; i < 200; i++)
|
2023-03-28 18:18:45 +02:00
|
|
|
ptr[i] = 'P';
|
2023-03-25 16:45:14 +01:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int run_protection(int min, int max)
|
|
|
|
{
|
|
|
|
pid_t pid;
|
|
|
|
int llen = 0, status;
|
|
|
|
|
|
|
|
llen += printf("0 -fstackprotector ");
|
|
|
|
|
2023-05-21 11:36:34 +02:00
|
|
|
#if !defined(_NOLIBC_STACKPROTECTOR)
|
2023-03-25 16:45:14 +01:00
|
|
|
llen += printf("not supported");
|
|
|
|
pad_spc(llen, 64, "[SKIPPED]\n");
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
|
2023-05-21 11:36:34 +02:00
|
|
|
#if defined(_NOLIBC_STACKPROTECTOR)
|
2023-05-21 11:36:32 +02:00
|
|
|
if (!__stack_chk_guard) {
|
|
|
|
llen += printf("__stack_chk_guard not initialized");
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-03-25 16:45:14 +01:00
|
|
|
pid = -1;
|
|
|
|
pid = fork();
|
|
|
|
|
|
|
|
switch (pid) {
|
|
|
|
case -1:
|
|
|
|
llen += printf("fork()");
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
close(STDOUT_FILENO);
|
|
|
|
close(STDERR_FILENO);
|
|
|
|
|
2023-05-26 09:51:20 +02:00
|
|
|
prctl(PR_SET_DUMPABLE, 0, 0, 0, 0);
|
2023-03-25 16:45:14 +01:00
|
|
|
smash_stack();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
default:
|
|
|
|
pid = waitpid(pid, &status, 0);
|
|
|
|
|
|
|
|
if (pid == -1 || !WIFSIGNALED(status) || WTERMSIG(status) != SIGABRT) {
|
|
|
|
llen += printf("waitpid()");
|
|
|
|
pad_spc(llen, 64, "[FAIL]\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
pad_spc(llen, 64, " [OK]\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:42 +02:00
|
|
|
/* prepare what needs to be prepared for pid 1 (stdio, /dev, /proc, etc) */
|
|
|
|
int prepare(void)
|
|
|
|
{
|
|
|
|
struct stat stat_buf;
|
|
|
|
|
|
|
|
/* It's possible that /dev doesn't even exist or was not mounted, so
|
|
|
|
* we'll try to create it, mount it, or create minimal entries into it.
|
|
|
|
* We want at least /dev/null and /dev/console.
|
|
|
|
*/
|
|
|
|
if (stat("/dev/.", &stat_buf) == 0 || mkdir("/dev", 0755) == 0) {
|
|
|
|
if (stat("/dev/console", &stat_buf) != 0 ||
|
2023-07-07 23:02:16 +08:00
|
|
|
stat("/dev/null", &stat_buf) != 0 ||
|
|
|
|
stat("/dev/zero", &stat_buf) != 0) {
|
2022-07-19 23:44:42 +02:00
|
|
|
/* try devtmpfs first, otherwise fall back to manual creation */
|
|
|
|
if (mount("/dev", "/dev", "devtmpfs", 0, 0) != 0) {
|
|
|
|
mknod("/dev/console", 0600 | S_IFCHR, makedev(5, 1));
|
|
|
|
mknod("/dev/null", 0666 | S_IFCHR, makedev(1, 3));
|
2023-07-07 23:02:16 +08:00
|
|
|
mknod("/dev/zero", 0666 | S_IFCHR, makedev(1, 5));
|
2022-07-19 23:44:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If no /dev/console was found before calling init, stdio is closed so
|
|
|
|
* we need to reopen it from /dev/console. If it failed above, it will
|
|
|
|
* still fail here and we cannot emit a message anyway.
|
|
|
|
*/
|
|
|
|
if (close(dup(1)) == -1) {
|
|
|
|
int fd = open("/dev/console", O_RDWR);
|
|
|
|
|
|
|
|
if (fd >= 0) {
|
|
|
|
if (fd != 0)
|
|
|
|
dup2(fd, 0);
|
|
|
|
if (fd != 1)
|
|
|
|
dup2(fd, 1);
|
|
|
|
if (fd != 2)
|
|
|
|
dup2(fd, 2);
|
|
|
|
if (fd > 2)
|
|
|
|
close(fd);
|
|
|
|
puts("\nSuccessfully reopened /dev/console.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* try to mount /proc if not mounted. Silently fail otherwise */
|
|
|
|
if (stat("/proc/.", &stat_buf) == 0 || mkdir("/proc", 0755) == 0) {
|
|
|
|
if (stat("/proc/self", &stat_buf) != 0)
|
|
|
|
mount("/proc", "/proc", "proc", 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2022-07-19 23:44:37 +02:00
|
|
|
|
2022-07-19 23:44:36 +02:00
|
|
|
/* This is the definition of known test names, with their functions */
|
2023-03-25 16:45:11 +01:00
|
|
|
static const struct test test_names[] = {
|
2022-07-19 23:44:36 +02:00
|
|
|
/* add new tests here */
|
2023-03-25 16:45:14 +01:00
|
|
|
{ .name = "syscall", .func = run_syscall },
|
|
|
|
{ .name = "stdlib", .func = run_stdlib },
|
2023-04-02 20:48:06 +02:00
|
|
|
{ .name = "vfprintf", .func = run_vfprintf },
|
2023-03-25 16:45:14 +01:00
|
|
|
{ .name = "protection", .func = run_protection },
|
2022-07-19 23:44:36 +02:00
|
|
|
{ 0 }
|
|
|
|
};
|
|
|
|
|
2023-07-08 02:32:05 +08:00
|
|
|
int is_setting_valid(char *test)
|
|
|
|
{
|
|
|
|
int idx, len, test_len, valid = 0;
|
|
|
|
char delimiter;
|
|
|
|
|
|
|
|
if (!test)
|
|
|
|
return valid;
|
|
|
|
|
|
|
|
test_len = strlen(test);
|
|
|
|
|
|
|
|
for (idx = 0; test_names[idx].name; idx++) {
|
|
|
|
len = strlen(test_names[idx].name);
|
|
|
|
if (test_len < len)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (strncmp(test, test_names[idx].name, len) != 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
delimiter = test[len];
|
|
|
|
if (delimiter != ':' && delimiter != ',' && delimiter != '\0')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
valid = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
int main(int argc, char **argv, char **envp)
|
|
|
|
{
|
|
|
|
int min = 0;
|
2023-06-03 16:08:12 +08:00
|
|
|
int max = INT_MAX;
|
2022-07-19 23:44:35 +02:00
|
|
|
int ret = 0;
|
2022-07-19 23:44:36 +02:00
|
|
|
int err;
|
|
|
|
int idx;
|
|
|
|
char *test;
|
2022-07-19 23:44:35 +02:00
|
|
|
|
2023-07-07 23:01:09 +08:00
|
|
|
argv0 = argv[0];
|
2022-07-19 23:44:35 +02:00
|
|
|
environ = envp;
|
|
|
|
|
2022-07-19 23:44:42 +02:00
|
|
|
/* when called as init, it's possible that no console was opened, for
|
|
|
|
* example if no /dev file system was provided. We'll check that fd#1
|
|
|
|
* was opened, and if not we'll attempt to create and open /dev/console
|
|
|
|
* and /dev/null that we'll use for later tests.
|
|
|
|
*/
|
|
|
|
if (getpid() == 1)
|
|
|
|
prepare();
|
|
|
|
|
2022-07-19 23:44:36 +02:00
|
|
|
/* the definition of a series of tests comes from either argv[1] or the
|
|
|
|
* "NOLIBC_TEST" environment variable. It's made of a comma-delimited
|
|
|
|
* series of test names and optional ranges:
|
|
|
|
* syscall:5-15[:.*],stdlib:8-10
|
|
|
|
*/
|
|
|
|
test = argv[1];
|
2023-07-08 02:32:05 +08:00
|
|
|
if (!is_setting_valid(test))
|
2022-07-19 23:44:36 +02:00
|
|
|
test = getenv("NOLIBC_TEST");
|
|
|
|
|
2023-07-08 02:32:05 +08:00
|
|
|
if (is_setting_valid(test)) {
|
2022-07-19 23:44:36 +02:00
|
|
|
char *comma, *colon, *dash, *value;
|
|
|
|
|
|
|
|
do {
|
|
|
|
comma = strchr(test, ',');
|
|
|
|
if (comma)
|
|
|
|
*(comma++) = '\0';
|
|
|
|
|
|
|
|
colon = strchr(test, ':');
|
|
|
|
if (colon)
|
|
|
|
*(colon++) = '\0';
|
|
|
|
|
|
|
|
for (idx = 0; test_names[idx].name; idx++) {
|
|
|
|
if (strcmp(test, test_names[idx].name) == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (test_names[idx].name) {
|
|
|
|
/* The test was named, it will be called at least
|
|
|
|
* once. We may have an optional range at <colon>
|
|
|
|
* here, which defaults to the full range.
|
|
|
|
*/
|
|
|
|
do {
|
2023-06-03 16:08:12 +08:00
|
|
|
min = 0; max = INT_MAX;
|
2022-07-19 23:44:36 +02:00
|
|
|
value = colon;
|
|
|
|
if (value && *value) {
|
|
|
|
colon = strchr(value, ':');
|
|
|
|
if (colon)
|
|
|
|
*(colon++) = '\0';
|
|
|
|
|
|
|
|
dash = strchr(value, '-');
|
|
|
|
if (dash)
|
|
|
|
*(dash++) = '\0';
|
|
|
|
|
|
|
|
/* support :val: :min-max: :min-: :-max: */
|
|
|
|
if (*value)
|
|
|
|
min = atoi(value);
|
|
|
|
if (!dash)
|
|
|
|
max = min;
|
|
|
|
else if (*dash)
|
|
|
|
max = atoi(dash);
|
|
|
|
|
|
|
|
value = colon;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* now's time to call the test */
|
|
|
|
printf("Running test '%s'\n", test_names[idx].name);
|
|
|
|
err = test_names[idx].func(min, max);
|
|
|
|
ret += err;
|
|
|
|
printf("Errors during this test: %d\n\n", err);
|
|
|
|
} while (colon && *colon);
|
|
|
|
} else
|
|
|
|
printf("Ignoring unknown test name '%s'\n", test);
|
|
|
|
|
|
|
|
test = comma;
|
|
|
|
} while (test && *test);
|
|
|
|
} else {
|
|
|
|
/* no test mentioned, run everything */
|
|
|
|
for (idx = 0; test_names[idx].name; idx++) {
|
|
|
|
printf("Running test '%s'\n", test_names[idx].name);
|
|
|
|
err = test_names[idx].func(min, max);
|
|
|
|
ret += err;
|
|
|
|
printf("Errors during this test: %d\n\n", err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
printf("Total number of errors: %d\n", ret);
|
2022-07-19 23:44:40 +02:00
|
|
|
|
|
|
|
if (getpid() == 1) {
|
|
|
|
/* we're running as init, there's no other process on the
|
|
|
|
* system, thus likely started from a VM for a quick check.
|
|
|
|
* Exiting will provoke a kernel panic that may be reported
|
|
|
|
* as an error by Qemu or the hypervisor, while stopping
|
|
|
|
* cleanly will often be reported as a success. This allows
|
|
|
|
* to use the output of this program for bisecting kernels.
|
|
|
|
*/
|
|
|
|
printf("Leaving init with final status: %d\n", !!ret);
|
|
|
|
if (ret == 0)
|
2023-07-08 02:30:58 +08:00
|
|
|
reboot(RB_POWER_OFF);
|
2022-07-19 23:44:41 +02:00
|
|
|
#if defined(__x86_64__)
|
|
|
|
/* QEMU started with "-device isa-debug-exit -no-reboot" will
|
|
|
|
* exit with status code 2N+1 when N is written to 0x501. We
|
|
|
|
* hard-code the syscall here as it's arch-dependent.
|
|
|
|
*/
|
2023-07-03 10:11:08 +02:00
|
|
|
else if (syscall(__NR_ioperm, 0x501, 1, 1) == 0)
|
2023-04-06 21:54:47 +00:00
|
|
|
__asm__ volatile ("outb %%al, %%dx" :: "d"(0x501), "a"(0));
|
2022-07-19 23:44:41 +02:00
|
|
|
/* if it does nothing, fall back to the regular panic */
|
|
|
|
#endif
|
2022-07-19 23:44:40 +02:00
|
|
|
}
|
|
|
|
|
2022-07-19 23:44:35 +02:00
|
|
|
printf("Exiting with status %d\n", !!ret);
|
|
|
|
return !!ret;
|
|
|
|
}
|