Userspace: implement basic stat
I tried to format the output pretty much how linux does, except for uid and git -> name resolution which is not implemented
This commit is contained in:
parent
9cb50cba33
commit
77ff585248
|
@ -9,6 +9,7 @@ set(USERSPACE_PROJECTS
|
|||
init
|
||||
ls
|
||||
Shell
|
||||
stat
|
||||
tee
|
||||
test
|
||||
touch
|
||||
|
|
|
@ -0,0 +1,16 @@
|
|||
cmake_minimum_required(VERSION 3.26)
|
||||
|
||||
project(stat CXX)
|
||||
|
||||
set(SOURCES
|
||||
main.cpp
|
||||
)
|
||||
|
||||
add_executable(stat ${SOURCES})
|
||||
target_compile_options(stat PUBLIC -O2 -g)
|
||||
target_link_libraries(stat PUBLIC libc ban)
|
||||
|
||||
add_custom_target(stat-install
|
||||
COMMAND cp ${CMAKE_CURRENT_BINARY_DIR}/stat ${BANAN_BIN}/
|
||||
DEPENDS stat
|
||||
)
|
|
@ -0,0 +1,75 @@
|
|||
#include <stdio.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/sysmacros.h>
|
||||
|
||||
#include <BAN/Time.h>
|
||||
|
||||
void print_timestamp(timespec ts)
|
||||
{
|
||||
auto time = BAN::from_unix_time(ts.tv_sec);
|
||||
printf("%04d-%02d-%02d %02d:%02d:%02d.%09d",
|
||||
time.year, time.month, time.day,
|
||||
time.hour, time.minute, time.second,
|
||||
ts.tv_nsec
|
||||
);
|
||||
}
|
||||
|
||||
int main(int argc, char** argv)
|
||||
{
|
||||
for (int i = 1; i < argc; i++)
|
||||
{
|
||||
struct stat st;
|
||||
if (stat(argv[i], &st) == -1)
|
||||
{
|
||||
perror("stat");
|
||||
continue;
|
||||
}
|
||||
|
||||
char access[11];
|
||||
const char* type = nullptr;
|
||||
if (S_ISBLK(st.st_mode)) {
|
||||
access[0] = 'b';
|
||||
type = "block special file";
|
||||
} else if (S_ISCHR(st.st_mode)) {
|
||||
access[0] = 'c';
|
||||
type = "character special file";
|
||||
} else if (S_ISDIR(st.st_mode)) {
|
||||
access[0] = 'd';
|
||||
type = "directory";
|
||||
} else if (S_ISFIFO(st.st_mode)) {
|
||||
access[0] = 'f';
|
||||
type = "fifo";
|
||||
} else if (S_ISREG(st.st_mode)) {
|
||||
access[0] = '-';
|
||||
type = "regular file";
|
||||
} else if (S_ISLNK(st.st_mode)) {
|
||||
access[0] = 'l';
|
||||
type = "symbolic link";
|
||||
} else if (S_ISSOCK(st.st_mode)) {
|
||||
access[0] = 's';
|
||||
type = "socket";
|
||||
} else {
|
||||
access[0] = '-';
|
||||
type = "unknown";
|
||||
}
|
||||
|
||||
access[1] = (st.st_mode & S_IRUSR) ? 'r' : '-';
|
||||
access[2] = (st.st_mode & S_IWUSR) ? 'w' : '-';
|
||||
access[3] = (st.st_mode & S_IXUSR) ? 'x' : '-';
|
||||
access[4] = (st.st_mode & S_IRGRP) ? 'r' : '-';
|
||||
access[5] = (st.st_mode & S_IWGRP) ? 'w' : '-';
|
||||
access[6] = (st.st_mode & S_IXGRP) ? 'x' : '-';
|
||||
access[7] = (st.st_mode & S_IROTH) ? 'r' : '-';
|
||||
access[8] = (st.st_mode & S_IWOTH) ? 'w' : '-';
|
||||
access[9] = (st.st_mode & S_IXOTH) ? 'x' : '-';
|
||||
access[10] = '\0';
|
||||
|
||||
printf(" File: %s\n", argv[i]);
|
||||
printf(" Size: %-15d Blocks: %-10d IO Block: %-6d %s\n", (int)st.st_size, (int)st.st_blocks, (int)st.st_blksize, type);
|
||||
printf("Device: %d,%-5d Inode: %-11d Links: %-5d Device type: %d,%d\n", (int)major(st.st_dev), (int)minor(st.st_dev), (int)st.st_ino, (int)st.st_nlink, (int)major(st.st_rdev), (int)minor(st.st_rdev));
|
||||
printf("Access: (%04o/%s) Uid: %5d Gid: %5d\n", (int)(st.st_mode & S_IRWXMASK), access, (int)st.st_uid, (int)st.st_gid);
|
||||
printf("Access: "); print_timestamp(st.st_atim); printf("\n");
|
||||
printf("Modify: "); print_timestamp(st.st_mtim); printf("\n");
|
||||
printf("Change: "); print_timestamp(st.st_ctim); printf("\n");
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue