X-Git-Url: https://plomlompom.com/repos/?a=blobdiff_plain;f=src%2Fcommon%2Freadwrite.c;h=912e70dfa7e443071a07220b7b5c663f258d062d;hb=27f8cdc988b4bbd92d9d0daee7019461c690008e;hp=3dcd2a2d90ec307d86a6c23fe99a36048725e143;hpb=1008f26c7c6855f59222bcbe2cd1505ba8a6f28f;p=plomrogue diff --git a/src/common/readwrite.c b/src/common/readwrite.c index 3dcd2a2..912e70d 100644 --- a/src/common/readwrite.c +++ b/src/common/readwrite.c @@ -4,26 +4,31 @@ #include /* size_t */ #include /* uint8_t, uint16_t, uint32_t, UINT32_MAX */ #include /* FILE, fseek(), sprintf(), fgets(), fgetc(), ferror(), - * fputc(), fwrite(), fclose(), fopen() + * fputc(), fwrite(), fclose(), fopen(), clearerr() */ +#include /* free() */ #include /* strlen() */ -#include /* for access(), unlink() */ +#include /* access(), unlink() */ #include "rexit.h" /* exit_err(), exit_trouble() */ +#include "try_malloc.h" /* try_malloc() */ extern FILE * try_fopen(char * path, char * mode, char * f) { + char * f_name = "try_fopen()"; char * msg1 = "Trouble in "; char * msg2 = " with fopen() (mode '"; char * msg3 = "') on path '"; char * msg4 = "'."; uint16_t size = strlen(msg1) + strlen(msg2) + strlen(msg3) + strlen(msg4) + strlen(f) + strlen(path) + strlen(mode) + 1; - char msg[size]; - sprintf(msg, "%s%s%s%s%s%s%s", msg1, f, msg2, mode, msg3, path, msg4); + char * msg = try_malloc(size, f_name); + int test = sprintf(msg, "%s%s%s%s%s%s%s", msg1,f,msg2,mode,msg3,path,msg4); + exit_trouble(test < 0, f_name, "sprintf()"); FILE * file_p = fopen(path, mode); exit_err(NULL == file_p, msg); + free(msg); return file_p; } @@ -53,6 +58,7 @@ extern void try_fputc(uint8_t c, FILE * file, char * f) extern int try_fgetc(FILE * file, char * f) { + clearerr(file); /* OSX' (BSD?) fgetc() needs this to undo previous EOFs. */ int test = fgetc(file); exit_trouble(EOF == test && ferror(file), f, "fgetc()"); return test; @@ -69,28 +75,48 @@ extern char * try_fgets(char * line, int linemax, FILE * file, char * f) -extern void try_fclose_unlink_rename(FILE * file, char * p1, char * p2, - char * f) +extern FILE * atomic_write_start(char * path, char ** path_tmp) { - try_fclose(file, f); + char * f_name = "atomic_write_start()"; + char * suffix_tmp = "_tmp"; + uint16_t size = strlen(path) + strlen(suffix_tmp) + 1; + *path_tmp = try_malloc(size, f_name); + int test = sprintf(*path_tmp, "%s%s", path, suffix_tmp); + exit_trouble(test < 0, f_name, "sprintf()"); + return try_fopen(*path_tmp, "w", f_name); +} + + + +extern void atomic_write_finish(FILE * file, char * path, char * path_tmp) +{ + char * f_name = "atomic_write_finish()"; + try_fclose(file, f_name); char * msg1 = "Trouble in "; char * msg4 = "'."; - if (!access(p2, F_OK)) + if (!access(path, F_OK)) { char * msg2 = " with unlink() on path '"; uint16_t size = strlen(msg1) + strlen(msg2) + strlen(msg4) - + strlen(f) + strlen(p2) + 1; - char msg[size]; - sprintf(msg, "%s%s%s%s%s", msg1, f, msg2, p2, msg4); - exit_err(unlink(p2), msg); + + strlen(f_name) + strlen(path) + 1; + char * msg = try_malloc(size, f_name); + int test = sprintf(msg, "%s%s%s%s%s", msg1, f_name, msg2, path, msg4); + exit_trouble(test < 0, f_name, "sprintf()"); + exit_err(unlink(path), msg); + free(msg); } char * msg2 = " with rename() from '"; char * msg3 = "' to '"; - uint16_t size = strlen(msg1) + strlen(f) + strlen(msg2) + strlen(p1) - + strlen(msg3) + strlen(p2) + strlen(msg4) + 1; - char msg[size]; - sprintf(msg, "%s%s%s%s%s%s%s", msg1, f, msg2, p1, msg3, p2, msg4); - exit_err(rename(p1, p2), msg); + uint16_t size = strlen(msg1) + strlen(f_name) + strlen(msg2) + + + strlen(path_tmp) + strlen(msg3) + strlen(path) + + strlen(msg4) + 1; + char * msg = try_malloc(size, f_name); + int test = sprintf(msg, "%s%s%s%s%s%s%s", + msg1, f_name, msg2, path_tmp, msg3, path, msg4); + exit_trouble(test < 0, f_name, "sprintf()"); + exit_err(rename(path_tmp, path), msg); + free(msg); + free(path_tmp); } @@ -108,8 +134,8 @@ extern uint32_t textfile_width(FILE * file) { break; } - exit_trouble(UINT32_MAX == c_count, f_name, "too large text file line"); c_count++; + exit_trouble(UINT32_MAX == c_count, f_name, "too large text file line"); if ('\n' == c) { if (c_count > linemax)