#include "shared.h" #include #include #include #include #include #include #include #include #include #include extern char* cmd; void error_s(const char *format, ...) { va_list list; va_start(list, format); fprintf(stderr, "%s: ", cmd); vfprintf(stderr, format, list); fprintf(stderr, "\n"); } void error(const char *format, ...) { va_list list; va_start(list, format); fprintf(stderr, "%s: ", cmd); vfprintf(stderr, format, list); fprintf(stderr, "\n"); exit(EXIT_FAILURE); } void output(const char *format, ...) { va_list list; va_start(list, format); fprintf(stdout, "%s: ", cmd); vfprintf(stdout, format, list); fprintf(stdout, "\n"); } FILE* get_file_s(const char* path, const char* type) { struct stat s; if (streql("-", path) && type[0] == 'r') { clearerr(stdin); fflush(stdin); return stdin; } if (lstat(path, &s) < 0) { if (type[0] != 'r') goto read; error_s("failed to read %s: %s", path, strerror(errno)); return NULL; } if (S_ISDIR(s.st_mode)) { error_s("%s is a directory", path); return NULL; } FILE* file; read: file = fopen(path, type); if (file == NULL) { error_s("failed to %s file %s: %s", type[0] == 'r' ? "read" : "write", path, strerror(errno)); } return file; } FILE* get_file(const char* path, const char* type) { FILE* file = get_file_s(path, type); if (file == NULL) exit(EXIT_FAILURE); return file; } long int get_number(const char* text) { char* end; long int n = strtol(text, &end, 10); if (text == end) { error("%s is not a valid number", text); } return n; } long int get_blkm(const char* text) { char* end; long int n = strtol(text, &end, 10); if (text == end) { error("%s is not a valid bkm", text); } if (*end == '\0') return n; switch (*end) { case 'K': case 'k': return n * 1024; case 'B': case 'b': return n * 512; case 'M': case 'm': return n * 1024 * 1204; default: error("invalid bkm type %c", *end); } // shouldnt get here anyways return 0; } mode_t get_mode(const char* next) { char* end = NULL; mode_t mode = (mode_t)strtol(next, &end, 8); if (!end) return 0; while(isspace(*end)) end++; if (*end != '\0' || (unsigned) mode < 010000) { error("invalid file mode: `%s`", next); } return mode; } bool streql(const char* a, const char* b) { if (*a != *b) return false; int n = 0; while (true) { if (*(a+n) != *(b+n)) return false; if (*(a+n) == '\0') return true; ++n; } } bool prefix(const char* pre, const char* str) { return strncmp(pre, str, strlen(pre)) == 0; } static char fs_types[5] = {'K','M','G','T','P'}; void print_file_size(size_t bytes, char buf[5]) { int index = 0; float next = bytes; while (true) { if (next < 1000) { break; } if (index == 5) { printf("999P"); return; }; next /= 1024; index++; } int n = snprintf(buf, 4, "%u", (int)(next+.5)); if (index > 0) { buf[n] = (fs_types[index - 1]); n++; } buf[n] = '\0'; } static char* months[12] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}; void print_date_time(time_t mills, char buf[13]) { struct tm* info; info = localtime(&mills); int n = snprintf(buf, 5, "%s ", months[info->tm_mon]); if (info->tm_mday < 10) { buf[n] = ' '; n++; } snprintf(buf + n, 13 - n, "%d %02d:%02d ", info->tm_mday, info->tm_hour, info->tm_sec); } #ifndef MAJOR #define MAJOR 0 #endif #ifndef MINOR #define MINOR 0 #endif #ifndef PATCH #define PATCH 0 #endif void check_arg (char* arg) { if (arg == NULL) { error("expected another argument after option"); } } void global_help(void (*help)(void)) { printf("LazySphere v%d.%d.%d multi-call binary.\n\n", MAJOR, MINOR, PATCH); help(); exit(EXIT_SUCCESS); } void parse_help(int argc, char** argv, void (*help)(void)) { if (argc < 1) return; for (int i = 0; i < argc; i++) { if (!prefix("-", argv[i]) || streql("-", argv[i])) break; if (help != NULL && streql("--help", argv[i])) global_help(help); } } int parse_args(int argc, char** argv, void (*help)(void), int (*short_arg)(char, char*), int (*long_arg)(char*, char*)) { if (argc < 1) return 0; int start = 0; for (int i = 0; i < argc; i++) { if (!prefix("-", argv[i]) || streql("-", argv[i])) break; if (help != NULL && streql("--help", argv[i])) global_help(help); char* next_arg; if (i + 1 == argc) { next_arg = NULL; } else { next_arg = argv[i+1]; } int current = i; if (prefix("--", argv[i])) { if (long_arg == NULL) { goto exit; } int r = long_arg(argv[current], next_arg); if (r == ARG_USED) { i++; start++; } else if (r == ARG_IGNORE) { goto exit; } else if (r == ARG_INVALID) { error("invalid argument %s", argv[current]); } } else { if (short_arg == NULL) { goto exit; } for (size_t j = 1; j < strlen(argv[current]); j++) { int r = short_arg(argv[current][j], next_arg); if (r == ARG_USED) { i++; start++; } else if (r == ARG_IGNORE) { goto exit; } else if (r == ARG_INVALID) { error("invalid argument -%c", argv[current][j]); } } } start++; } exit: return start; } int get_tty() { int fd = open(_PATH_TTY, O_RDONLY); if (fd < 0) error("failed to get tty: %s", strerror(errno)); return fd; } FILE* get_tty_stream(char* type) { int fd = get_tty(); FILE* file = fdopen(fd, type); if (file == NULL) { error("failed to open tty stream: %s", strerror(errno)); } return file; } static int push_path_buffer_b(char* buf, int* index, const char* string) { int save = *index; if (*index > 1 || (*index == 1 && buf[0] != '/')) { buf[(*index)++] = '/'; } int string_len = strlen(string); memcpy(buf + *index, string, string_len + 1); *index += string_len; return save; } static void pop_path_buffer_b(char* buf, int* index, int i) { *index = i; buf[*index] = '\0'; } static char path_buffer[PATH_MAX + 1]; static int path_buffer_index = 0; char* get_path_buffer() { return path_buffer; } int push_path_buffer(const char* string) { return push_path_buffer_b(path_buffer, &path_buffer_index, string); } void pop_path_buffer(int i) { pop_path_buffer_b(path_buffer, &path_buffer_index, i); } static char path_buffer_2[PATH_MAX + 1]; static int path_buffer_index_2 = 0; char* get_path_buffer_2() { return path_buffer_2; } int push_path_buffer_2(const char* string) { return push_path_buffer_b(path_buffer_2, &path_buffer_index_2, string); } void pop_path_buffer_2(int i) { pop_path_buffer_b(path_buffer_2, &path_buffer_index_2, i); } bool is_dot_dir(const char* path) { return streql(path, ".") || streql(path, ".."); }