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