home · contact · privacy
Make try_fgetc() prepend clearerr() for fgetc() versions that need it.
[plomrogue] / src / common / readwrite.c
1 /* src/common/readwrite.c */
2
3 #include "readwrite.h"
4 #include <stddef.h> /* size_t */
5 #include <stdint.h> /* uint8_t, uint16_t, uint32_t, UINT32_MAX */
6 #include <stdio.h> /* FILE, fseek(), sprintf(), fgets(), fgetc(), ferror(),
7                     * fputc(), fwrite(), fclose(), fopen(), clearerr()
8                     */
9 #include <string.h> /* strlen() */
10 #include <unistd.h> /* for access(), unlink() */
11 #include "rexit.h" /* exit_err(), exit_trouble() */
12
13
14
15 extern FILE * try_fopen(char * path, char * mode, char * f)
16 {
17     char * msg1 = "Trouble in ";
18     char * msg2 = " with fopen() (mode '";
19     char * msg3 = "') on path '";
20     char * msg4 = "'.";
21     uint16_t size = strlen(msg1) + strlen(msg2) + strlen(msg3) + strlen(msg4)
22                     + strlen(f) + strlen(path) + strlen(mode) + 1;
23     char msg[size];
24     sprintf(msg, "%s%s%s%s%s%s%s", msg1, f, msg2, mode, msg3, path, msg4);
25     FILE * file_p = fopen(path, mode);
26     exit_err(NULL == file_p, msg);
27     return file_p;
28 }
29
30
31
32 extern void try_fclose(FILE * file, char * f)
33 {
34     exit_trouble(fclose(file), f, "fclose()");
35 }
36
37
38
39 extern void try_fwrite(void * ptr, size_t size, size_t nmemb, FILE * stream,
40                        char * f)
41 {
42     exit_trouble(0 == fwrite(ptr, size, nmemb, stream), f, "fwrite()");
43 }
44
45
46
47 extern void try_fputc(uint8_t c, FILE * file, char * f)
48 {
49     exit_trouble(EOF == fputc(c, file), f, "fputc()");
50 }
51
52
53
54 extern int try_fgetc(FILE * file, char * f)
55 {
56     clearerr(file); /* OSX' (BSD?) fgetc() needs this to undo previous EOFs. */
57     int test = fgetc(file);
58     exit_trouble(EOF == test && ferror(file), f, "fgetc()");
59     return test;
60 }
61
62
63
64 extern char * try_fgets(char * line, int linemax, FILE * file, char * f)
65 {
66     char * test = fgets(line, linemax, file);
67     exit_trouble(NULL == test && ferror(file), f, "fgets()");
68     return test;
69 }
70
71
72
73 extern void try_fclose_unlink_rename(FILE * file, char * p1, char * p2,
74                                      char * f)
75 {
76     try_fclose(file, f);
77     char * msg1 = "Trouble in ";
78     char * msg4 = "'.";
79     if (!access(p2, F_OK))
80     {
81         char * msg2 = " with unlink() on path '";
82         uint16_t size = strlen(msg1) + strlen(msg2) + strlen(msg4)
83                         + strlen(f) + strlen(p2) + 1;
84         char msg[size];
85         sprintf(msg, "%s%s%s%s%s", msg1, f, msg2, p2, msg4);
86         exit_err(unlink(p2), msg);
87     }
88     char * msg2 = " with rename() from '";
89     char * msg3 = "' to '";
90     uint16_t size = strlen(msg1) + strlen(f) + strlen(msg2) + strlen(p1)
91                     + strlen(msg3) + strlen(p2) + strlen(msg4) + 1;
92     char msg[size];
93     sprintf(msg, "%s%s%s%s%s%s%s", msg1, f, msg2, p1, msg3, p2, msg4);
94     exit_err(rename(p1, p2), msg);
95 }
96
97
98
99 extern uint32_t textfile_width(FILE * file)
100 {
101     char * f_name = "textfile_width()";
102     int c = 0;
103     uint32_t c_count = 0;
104     uint32_t linemax = 0;
105     while (1)
106     {
107         c = try_fgetc(file, f_name);
108         if (EOF == c)
109         {
110             break;
111         }
112         c_count++;
113         exit_trouble(UINT32_MAX == c_count, f_name, "too large text file line");
114         if ('\n' == c)
115         {
116             if (c_count > linemax)
117             {
118                 linemax = c_count;
119             }
120             c_count = 0;
121         }
122     }
123   if (0 == linemax && 0 < c_count) /* Handle files that consist of only one */
124   {                                /* line / lack newline chars.            */
125       linemax = c_count;
126    }
127    exit_trouble(-1 == fseek(file, 0, SEEK_SET), f_name, "fseek()");
128    return linemax;
129 }