1 /* read-file.c -- read file contents into a string
2 Copyright (C) 2006, 2009-2024 Free Software Foundation, Inc.
3 Written by Simon Josefsson and Bruno Haible.
5 This file is free software: you can redistribute it and/or modify
6 it under the terms of the GNU Lesser General Public License as
7 published by the Free Software Foundation; either version 2.1 of the
8 License, or (at your option) any later version.
10 This file is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public License
16 along with this program. If not, see <https://www.gnu.org/licenses/>. */
20 #include "read-file.h"
28 /* Get PTRDIFF_MAX. */
31 /* Get malloc, realloc, free. */
34 /* Get memcpy, memset_explicit. */
40 /* Read a STREAM and return a newly allocated string with the content,
41 and set *LENGTH to the length of the string. The string is
42 zero-terminated, but the terminating zero byte is not counted in
43 *LENGTH. On errors, *LENGTH is undefined, errno preserves the
44 values set by system functions (if any), and NULL is returned.
46 If the RF_SENSITIVE flag is set in FLAGS:
47 - You should control the buffering of STREAM using 'setvbuf'. Either
48 clear the buffer of STREAM after closing it, or disable buffering of
49 STREAM before calling this function.
50 - The memory buffer internally allocated will be cleared upon failure. */
52 fread_file (FILE *stream
, int flags
, size_t *length
)
55 size_t alloc
= BUFSIZ
;
57 /* For a regular file, allocate a buffer that has exactly the right
58 size. This avoids the need to do dynamic reallocations later. */
62 if (fstat (fileno (stream
), &st
) >= 0 && S_ISREG (st
.st_mode
))
64 off_t pos
= ftello (stream
);
66 if (pos
>= 0 && pos
< st
.st_size
)
68 off_t alloc_off
= st
.st_size
- pos
;
70 /* '1' below, accounts for the trailing NUL. */
71 if (PTRDIFF_MAX
- 1 < alloc_off
)
77 alloc
= alloc_off
+ 1;
82 if (!(buf
= malloc (alloc
)))
83 return NULL
; /* errno is ENOMEM. */
86 size_t size
= 0; /* number of bytes read so far */
91 /* This reads 1 more than the size of a regular file
92 so that we get eof immediately. */
93 size_t requested
= alloc
- size
;
94 size_t count
= fread (buf
+ size
, 1, requested
, stream
);
97 if (count
!= requested
)
103 /* Shrink the allocated memory if possible. */
104 if (size
< alloc
- 1)
106 if (flags
& RF_SENSITIVE
)
108 char *smaller_buf
= malloc (size
+ 1);
109 if (smaller_buf
== NULL
)
110 memset_explicit (buf
+ size
, 0, alloc
- size
);
113 memcpy (smaller_buf
, buf
, size
);
114 memset_explicit (buf
, 0, alloc
);
121 char *smaller_buf
= realloc (buf
, size
+ 1);
122 if (smaller_buf
!= NULL
)
134 size_t saved_alloc
= alloc
;
136 if (alloc
== PTRDIFF_MAX
)
138 saved_errno
= ENOMEM
;
142 if (alloc
< PTRDIFF_MAX
- alloc
/ 2)
143 alloc
= alloc
+ alloc
/ 2;
147 if (flags
& RF_SENSITIVE
)
149 new_buf
= malloc (alloc
);
152 /* BUF should be cleared below after the loop. */
156 memcpy (new_buf
, buf
, saved_alloc
);
157 memset_explicit (buf
, 0, saved_alloc
);
160 else if (!(new_buf
= realloc (buf
, alloc
)))
170 if (flags
& RF_SENSITIVE
)
171 memset_explicit (buf
, 0, alloc
);
179 /* Open and read the contents of FILENAME, and return a newly
180 allocated string with the content, and set *LENGTH to the length of
181 the string. The string is zero-terminated, but the terminating
182 zero byte is not counted in *LENGTH. On errors, *LENGTH is
183 undefined, errno preserves the values set by system functions (if
184 any), and NULL is returned.
186 If the RF_BINARY flag is set in FLAGS, the file is opened in binary
187 mode. If the RF_SENSITIVE flag is set in FLAGS, the memory buffer
188 internally allocated will be cleared upon failure. */
190 read_file (const char *filename
, int flags
, size_t *length
)
192 const char *mode
= (flags
& RF_BINARY
) ? "rbe" : "re";
193 FILE *stream
= fopen (filename
, mode
);
199 if (flags
& RF_SENSITIVE
)
200 setvbuf (stream
, NULL
, _IONBF
, 0);
202 out
= fread_file (stream
, flags
, length
);
204 if (fclose (stream
) != 0)
208 if (flags
& RF_SENSITIVE
)
209 memset_explicit (out
, 0, *length
);