310 lines
8.9 KiB
C
310 lines
8.9 KiB
C
#define _GNU_SOURCE
|
|
#include <sys/types.h>
|
|
#include <sys/vfs.h>
|
|
#include <fcntl.h>
|
|
#include <limits.h>
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#include "test_fs.h"
|
|
|
|
// ============================================================================
|
|
// Helper variable and function
|
|
// ============================================================================
|
|
|
|
// Contains the name that was used to invoke the calling program
|
|
extern char *program_invocation_short_name;
|
|
|
|
static int test_readlink_from_procfs(const char *proc_inode, char *buf, int buf_size,
|
|
const char *expected_target) {
|
|
int n = readlink(proc_inode, buf, buf_size);
|
|
if (n < 0) {
|
|
THROW_ERROR("failed to readlink from %s", proc_inode);
|
|
} else if (n != strlen(expected_target)) {
|
|
THROW_ERROR("readlink from %s length is wrong", proc_inode);
|
|
}
|
|
if (strncmp(buf, expected_target, n) != 0) {
|
|
THROW_ERROR("check the result from %s failed", proc_inode);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_procfs(const char *proc_inode) {
|
|
char buf[1024] = { 0 };
|
|
int len;
|
|
|
|
int fd = open(proc_inode, O_RDONLY);
|
|
if (fd < 0) {
|
|
THROW_ERROR("failed to open file: %s", proc_inode);
|
|
}
|
|
do {
|
|
len = read(fd, buf, sizeof(buf));
|
|
if (len < 0) {
|
|
THROW_ERROR("failed to read: %s", proc_inode);
|
|
}
|
|
} while (len == sizeof(buf));
|
|
close(fd);
|
|
printf("test read from %s:\n%s\n", proc_inode, buf);
|
|
return 0;
|
|
}
|
|
|
|
// ============================================================================
|
|
// Test cases for procfs
|
|
// ============================================================================
|
|
|
|
static int test_readlink_from_proc_self_exe() {
|
|
char exe_buf[PATH_MAX] = { 0 };
|
|
char absolute_path[PATH_MAX] = { 0 };
|
|
const char *proc_exe = "/proc/self/exe";
|
|
|
|
int n = snprintf(absolute_path, sizeof(absolute_path), "/bin/%s",
|
|
program_invocation_short_name);
|
|
if (n < 0) {
|
|
THROW_ERROR("failed to call snprintf");
|
|
}
|
|
if (test_readlink_from_procfs(proc_exe, exe_buf, PATH_MAX, absolute_path) < 0) {
|
|
THROW_ERROR("failed to call test_readlink_from_procfs");
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test_readlink_from_proc_self_cwd() {
|
|
char cwd_buf[PATH_MAX] = { 0 };
|
|
const char *proc_cwd = "/proc/self/cwd";
|
|
|
|
if (test_readlink_from_procfs(proc_cwd, cwd_buf, PATH_MAX, "/") < 0) {
|
|
THROW_ERROR("failed to call test_readlink_from_procfs");
|
|
}
|
|
if (chdir("/bin") < 0) {
|
|
THROW_ERROR("failed to chdir");
|
|
}
|
|
if (test_readlink_from_procfs(proc_cwd, cwd_buf, PATH_MAX, "/bin") < 0) {
|
|
THROW_ERROR("failed to call test_readlink_from_procfs after chdir");
|
|
}
|
|
if (chdir("/") < 0) {
|
|
THROW_ERROR("failed to chdir");
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_self_maps() {
|
|
const char *proc_maps = "/proc/self/maps";
|
|
|
|
if (test_read_from_procfs(proc_maps) < 0) {
|
|
THROW_ERROR("failed to read the cpuinfo");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int test_readlink_from_proc_self_root() {
|
|
char root_buf[PATH_MAX] = { 0 };
|
|
const char *proc_root = "/proc/self/root";
|
|
|
|
if (test_readlink_from_procfs(proc_root, root_buf, PATH_MAX, "/") < 0) {
|
|
THROW_ERROR("failed to call test_readlink_from_procfs");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int test_create_and_unlink_file_from_proc_self_root() {
|
|
const char *proc_root_file = "/proc/self/root/test_file";
|
|
int fd = open(proc_root_file, O_RDONLY | O_CREAT | O_TRUNC, 00666);
|
|
if (fd < 0) {
|
|
THROW_ERROR("failed to create a file");
|
|
}
|
|
close(fd);
|
|
if (unlink(proc_root_file) < 0) {
|
|
THROW_ERROR("failed to unlink the created file");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_self_cmdline() {
|
|
char absolute_path[PATH_MAX] = { 0 };
|
|
const char *proc_cmdline = "/proc/self/cmdline";
|
|
|
|
int n = snprintf(absolute_path, sizeof(absolute_path), "/bin/%s",
|
|
program_invocation_short_name);
|
|
if (n < 0) {
|
|
THROW_ERROR("failed to call snprintf");
|
|
}
|
|
char read_buf[PATH_MAX] = { 0 };
|
|
int fd = open(proc_cmdline, O_RDONLY);
|
|
size_t len = read(fd, read_buf, sizeof(read_buf));
|
|
if (len != strlen(absolute_path) + 1) {
|
|
THROW_ERROR("failed check the return value of reading from %s", proc_cmdline);
|
|
}
|
|
if (read_buf[strlen(absolute_path)] != '\0') {
|
|
THROW_ERROR("failed check the buffer of reading from %s", proc_cmdline);
|
|
}
|
|
if (strcmp(absolute_path, read_buf) != 0) {
|
|
THROW_ERROR("failed to check result in %s", proc_cmdline);
|
|
}
|
|
close(fd);
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_self_comm() {
|
|
// The name can be up to 16 bytes long, including the terminating null byte.
|
|
char comm_name[16] = { 0 };
|
|
const char *proc_comm = "/proc/self/comm";
|
|
|
|
if (snprintf(comm_name, sizeof(comm_name), "%s", program_invocation_short_name) < 0) {
|
|
THROW_ERROR("failed to call snprintf");
|
|
}
|
|
// The last byte shoud be '\n'
|
|
int end_idx = strlen(comm_name);
|
|
comm_name[end_idx] = '\n';
|
|
if (fs_check_file_content(proc_comm, comm_name) < 0) {
|
|
THROW_ERROR("failed to check result in %s", proc_comm);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_self_stat() {
|
|
const char *proc_self_stat = "/proc/self/stat";
|
|
FILE *fp = fopen(proc_self_stat, "r");
|
|
if (fp == NULL) {
|
|
THROW_ERROR("failed to fopen: %s", proc_self_stat);
|
|
}
|
|
|
|
int pid, ppid, pgrp;
|
|
char comm[32] = { 0 };
|
|
char state[32] = { 0 };
|
|
int ret = fscanf(fp, "%d %s %s %d %d", &pid, comm, state, &ppid, &pgrp);
|
|
if (ret != 5) {
|
|
THROW_ERROR("failed to parse the first 5 items");
|
|
}
|
|
if (pid != getpid()) {
|
|
THROW_ERROR("failed to check the result in %s", proc_self_stat);
|
|
}
|
|
printf("cat %s with the first 5 items:\n%d %s %s %d %d\n", proc_self_stat, pid, comm,
|
|
state, ppid, pgrp);
|
|
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_meminfo() {
|
|
const char *proc_meminfo = "/proc/meminfo";
|
|
|
|
if (test_read_from_procfs(proc_meminfo) < 0) {
|
|
THROW_ERROR("failed to read the meminfo");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_cpuinfo() {
|
|
const char *proc_cpuinfo = "/proc/cpuinfo";
|
|
|
|
if (test_read_from_procfs(proc_cpuinfo) < 0) {
|
|
THROW_ERROR("failed to read the cpuinfo");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int test_read_from_proc_stat() {
|
|
const char *proc_stat = "/proc/stat";
|
|
|
|
if (test_read_from_procfs(proc_stat) < 0) {
|
|
THROW_ERROR("failed to read the stat");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#define PROC_SUPER_MAGIC 0x9fa0
|
|
static int test_statfs() {
|
|
const char *file_path = "/proc/cpuinfo";
|
|
struct statfs statfs_buf;
|
|
int ret;
|
|
|
|
ret = statfs(file_path, &statfs_buf);
|
|
if (ret < 0) {
|
|
THROW_ERROR("failed to statfs the file");
|
|
}
|
|
if (statfs_buf.f_type != PROC_SUPER_MAGIC) {
|
|
THROW_ERROR("failed to check the f_type");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int test_readdir_root() {
|
|
const char *root = "/proc";
|
|
char pid[NAME_MAX] = { 0 };
|
|
snprintf(pid, sizeof(pid), "%d", getpid());
|
|
char expected_entries[4][NAME_MAX] = {
|
|
"self",
|
|
"meminfo",
|
|
"cpuinfo",
|
|
{ *pid },
|
|
};
|
|
|
|
if (check_readdir_with_expected_entries(root, expected_entries, 4) < 0) {
|
|
THROW_ERROR("failed to test readdir %s", root);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test_readdir_self() {
|
|
const char *self = "/proc/self";
|
|
char expected_entries[6][NAME_MAX] = {
|
|
"exe",
|
|
"cwd",
|
|
"root",
|
|
"fd",
|
|
"comm",
|
|
"cmdline",
|
|
};
|
|
|
|
if (check_readdir_with_expected_entries(self, expected_entries, 6) < 0) {
|
|
THROW_ERROR("failed to test readdir %s", self);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test_readdir_self_fd() {
|
|
const char *self_fd = "/proc/self/fd";
|
|
char expected_entries[3][NAME_MAX] = {
|
|
"0",
|
|
"1",
|
|
"2",
|
|
};
|
|
|
|
if (check_readdir_with_expected_entries(self_fd, expected_entries, 3) < 0) {
|
|
THROW_ERROR("failed to test readdir %s", self_fd);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// ============================================================================
|
|
// Test suite main
|
|
// ============================================================================
|
|
|
|
static test_case_t test_cases[] = {
|
|
TEST_CASE(test_readlink_from_proc_self_exe),
|
|
TEST_CASE(test_readlink_from_proc_self_cwd),
|
|
TEST_CASE(test_readlink_from_proc_self_root),
|
|
TEST_CASE(test_create_and_unlink_file_from_proc_self_root),
|
|
TEST_CASE(test_read_from_proc_self_cmdline),
|
|
TEST_CASE(test_read_from_proc_self_comm),
|
|
TEST_CASE(test_read_from_proc_self_stat),
|
|
TEST_CASE(test_read_from_proc_meminfo),
|
|
TEST_CASE(test_read_from_proc_cpuinfo),
|
|
TEST_CASE(test_read_from_proc_stat),
|
|
TEST_CASE(test_statfs),
|
|
TEST_CASE(test_readdir_root),
|
|
TEST_CASE(test_readdir_self),
|
|
TEST_CASE(test_readdir_self_fd),
|
|
TEST_CASE(test_read_from_proc_self_maps),
|
|
};
|
|
|
|
int main(int argc, const char *argv[]) {
|
|
return test_suite_run(test_cases, ARRAY_SIZE(test_cases));
|
|
}
|