/* file.c * This file is part of the base16384 distribution (https://github.com/fumiama/base16384). * Copyright (c) 2022-2023 Fumiama Minamoto. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef __cosmopolitan #include #include #include #include #include #include #ifdef _WIN32 #include #include #include #else #include #include #endif #endif #include "base16384.h" #ifdef __cosmopolitan #define get_file_size(filepath) ((off_t)GetFileSize(filepath)) #else static inline off_t get_file_size(const char* filepath) { struct stat statbuf; return stat(filepath, &statbuf)?-1:statbuf.st_size; } #endif #define is_standard_io(filename) (*(uint16_t*)(filename) == *(uint16_t*)"-") base16384_err_t base16384_encode_file(const char* input, const char* output, char* encbuf, char* decbuf) { off_t inputsize; FILE* fp = NULL; FILE* fpo; if(!input || !output || strlen(input) <= 0 || strlen(output) <= 0) return base16384_err_invalid_file_name; if(is_standard_io(input)) { // read from stdin inputsize = 0; fp = stdin; } else inputsize = get_file_size(input); if(inputsize < 0) { return base16384_err_get_file_size; } fpo = is_standard_io(output)?stdout:fopen(output, "wb"); if(!fpo) { return base16384_err_fopen_output_file; } if(!inputsize || inputsize > BASE16384_ENCBUFSZ) { // stdin or big file, use encbuf & fread inputsize = BASE16384_ENCBUFSZ-7; #if defined _WIN32 || defined __cosmopolitan } #endif if(!fp) fp = fopen(input, "rb"); if(!fp) { return base16384_err_fopen_input_file; } size_t cnt = 0; fputc(0xFE, fpo); fputc(0xFF, fpo); while((cnt = fread(encbuf, sizeof(char), inputsize, fp)) > 0) { int n = base16384_encode(encbuf, cnt, decbuf); if(fwrite(decbuf, n, 1, fpo) <= 0) { return base16384_err_write_file; } } if(!is_standard_io(output)) fclose(fpo); if(!is_standard_io(input)) fclose(fp); #if !defined _WIN32 && !defined __cosmopolitan } else { // small file, use mmap & fwrite int fd = open(input, O_RDONLY); if(fd < 0) { return base16384_err_open_input_file; } char *input_file = mmap(NULL, (size_t)inputsize+16, PROT_READ, MAP_PRIVATE, fd, 0); if(input_file == MAP_FAILED) { return base16384_err_map_input_file; } fputc(0xFE, fpo); fputc(0xFF, fpo); int n = base16384_encode(input_file, (int)inputsize, decbuf); if(fwrite(decbuf, n, 1, fpo) <= 0) { return base16384_err_write_file; } munmap(input_file, (size_t)inputsize); if(!is_standard_io(output)) fclose(fpo); close(fd); } #endif return base16384_err_ok; } base16384_err_t base16384_encode_fp(FILE* input, FILE* output, char* encbuf, char* decbuf) { if(!input) { return base16384_err_fopen_input_file; } if(!output) { return base16384_err_fopen_output_file; } off_t inputsize = BASE16384_ENCBUFSZ-7; size_t cnt = 0; fputc(0xFE, output); fputc(0xFF, output); while((cnt = fread(encbuf, sizeof(char), inputsize, input)) > 0) { int n = base16384_encode(encbuf, cnt, decbuf); if(fwrite(decbuf, n, 1, output) <= 0) { return base16384_err_write_file; } } return base16384_err_ok; } base16384_err_t base16384_encode_fd(int input, int output, char* encbuf, char* decbuf) { if(input < 0) { return base16384_err_fopen_input_file; } if(output < 0) { return base16384_err_fopen_output_file; } off_t inputsize = BASE16384_ENCBUFSZ-7; size_t cnt = 0; write(output, "\xfe\xff", 2); while((cnt = read(input, encbuf, inputsize)) > 0) { int n = base16384_encode(encbuf, cnt, decbuf); if(write(output, decbuf, n) < n) { return base16384_err_write_file; } } return base16384_err_ok; } #define rm_head(fp) {\ int ch = fgetc(fp);\ if(ch == 0xFE) fgetc(fp);\ else ungetc(ch, fp);\ } #define skip_offset(input_file) ((input_file[0]==(char)0xFE)?2:0) static inline int is_next_end(FILE* fp) { int ch = fgetc(fp); if(ch == EOF) return 0; if(ch == '=') return fgetc(fp); ungetc(ch, fp); return 0; } base16384_err_t base16384_decode_file(const char* input, const char* output, char* encbuf, char* decbuf) { off_t inputsize; FILE* fp = NULL; FILE* fpo; if(!input || !output || strlen(input) <= 0 || strlen(output) <= 0) return base16384_err_invalid_file_name; if(is_standard_io(input)) { // read from stdin inputsize = 0; fp = stdin; } else inputsize = get_file_size(input); if(inputsize < 0) { return base16384_err_get_file_size; } fpo = is_standard_io(output)?stdout:fopen(output, "wb"); if(!fpo) { return base16384_err_fopen_output_file; } if(!inputsize || inputsize > BASE16384_DECBUFSZ) { // stdin or big file, use decbuf & fread inputsize = BASE16384_DECBUFSZ/8*8; #if defined _WIN32 || defined __cosmopolitan } #endif if(!fp) fp = fopen(input, "rb"); if(!fp) { return base16384_err_fopen_input_file; } int cnt = 0; int end = 0; rm_head(fp); while((cnt = fread(decbuf, sizeof(char), inputsize, fp)) > 0) { if((end = is_next_end(fp))) { decbuf[cnt++] = '='; decbuf[cnt++] = end; } if(fwrite(encbuf, base16384_decode(decbuf, cnt, encbuf), 1, fpo) <= 0) { return base16384_err_write_file; } } if(!is_standard_io(output)) fclose(fpo); if(!is_standard_io(input)) fclose(fp); #if !defined _WIN32 && !defined __cosmopolitan } else { // small file, use mmap & fwrite int fd = open(input, O_RDONLY); if(fd < 0) { return base16384_err_open_input_file; } char *input_file = mmap(NULL, (size_t)inputsize+16, PROT_READ, MAP_PRIVATE, fd, 0); if(input_file == MAP_FAILED) { return base16384_err_map_input_file; } int off = skip_offset(input_file); if(fwrite(encbuf, base16384_decode(input_file+off, inputsize-off, encbuf), 1, fpo) <= 0) { return base16384_err_write_file; } munmap(input_file, (size_t)inputsize); if(!is_standard_io(output)) fclose(fpo); close(fd); } #endif return base16384_err_ok; } base16384_err_t base16384_decode_fp(FILE* input, FILE* output, char* encbuf, char* decbuf) { if(!input) { return base16384_err_fopen_input_file; } if(!output) { return base16384_err_fopen_output_file; } off_t inputsize = BASE16384_DECBUFSZ/8*8; int cnt = 0; int end = 0; rm_head(input); while((cnt = fread(decbuf, sizeof(char), inputsize, input)) > 0) { if((end = is_next_end(input))) { decbuf[cnt++] = '='; decbuf[cnt++] = end; } if(fwrite(encbuf, base16384_decode(decbuf, cnt, encbuf), 1, output) <= 0) { return base16384_err_write_file; } } return base16384_err_ok; } static inline int is_next_end_fd(int fd) { char ch = 0; read(fd, &ch, 1); if(ch == '=') { read(fd, &ch, 1); } return (int)ch; } base16384_err_t base16384_decode_fd(int input, int output, char* encbuf, char* decbuf) { if(input < 0) { return base16384_err_fopen_input_file; } if(output < 0) { return base16384_err_fopen_output_file; } off_t inputsize = BASE16384_DECBUFSZ/8*8; int cnt = 0; int end = 0; decbuf[0] = 0; if(read(input, decbuf, 2) < 2) { return base16384_err_read_file; } if(decbuf[0] != (char)(0xfe)) cnt = 2; while((end = read(input, decbuf+cnt, inputsize-cnt)) > 0 || cnt > 0) { if(end > 0) { cnt += end; if((end = is_next_end_fd(input))) { decbuf[cnt++] = '='; decbuf[cnt++] = end; end = 0; } else end = 1; } else end = 0; cnt = base16384_decode(decbuf, cnt, encbuf); if(write(output, encbuf, cnt) < cnt) { return base16384_err_write_file; } cnt = end; } return base16384_err_ok; }