1
0
mirror of https://gitlab.com/80486DX2-66/gists synced 2025-01-13 18:52:03 +05:30
gists/c-programming/io/freadln.c

84 lines
2.0 KiB
C
Raw Normal View History

2024-02-20 01:38:45 +03:00
/*
* freadln.c
*
* The `freadln` function reads a line from STDIN into a string, allocating
* memory for it.
*
* Author: Intel A80486DX2-66
* License: Creative Commons Zero 1.0 Universal
*
2024-03-10 11:50:09 +03:00
* TODO: Figure out potential problems
2024-03-10 14:08:28 +03:00
* TODO: Add 'flushing' of STDIN (while there are characters, read them) before
* the input reading loop to avoid input queues
* TODO: Add the feature of reading from arbitrary file, then extract a macro:
* `#define finreadln(output, length_out) freadln(stdin, output, length_out)`
2024-02-20 01:38:45 +03:00
*/
#include "freadln.h"
2024-03-10 14:14:02 +03:00
int freadln(char** output, size_t* length_out) {
2024-02-20 01:38:45 +03:00
/*
* The length of STDIN line is counted without any terminating characters.
*
* return value:
* freadln_OK: no errors, the length of STDIN line has been stored in
* `length_out`
* freadln_ERROR: an error occurred (see errno)
* >= 0: length of stdin line
*/
if (output == NULL)
return freadln_ERROR;
2024-03-10 14:12:05 +03:00
// NOTE: if the file is STDIN, flush STDOUT before waiting for input to
// make sure a prompt is displayed
fflush(stdout);
2024-02-20 01:38:45 +03:00
freadln_length_type length = 0;
*output = malloc((length + 1) * sizeof(char));
if (*output == NULL)
return freadln_ERROR;
int character;
while ((character = fgetc(stdin)) != EOF
/* stop on a newline character: */ && character != '\n') {
(*output)[length] = (char) character;
// integer overflow and integer limit check, to keep array boundaries
if ((freadln_length_type) (length + 2) <= (freadln_length_type) length)
{
errno = ERANGE;
2024-03-10 14:12:51 +03:00
freadln_success_epilogue;
2024-02-20 01:38:45 +03:00
} else
length++;
char* temp = realloc(*output, (length + 1) * sizeof(char));
// If the function fails to allocate new memory, return the string that
// has already been accumulated.
if (temp == NULL) {
// keep errno;
2024-03-10 14:12:51 +03:00
freadln_success_epilogue;
2024-02-20 01:38:45 +03:00
}
*output = temp;
}
errno = 0;
2024-03-10 14:12:51 +03:00
freadln_success_epilogue;
2024-02-20 01:38:45 +03:00
}
2024-03-10 14:14:38 +03:00
#ifdef TEST
int main(void) {
printf("Type something> ");
char* line;
if (freadln(&line, NULL) != freadln_OK) {
perror("freadln");
exit(EXIT_FAILURE);
}
printf("Input string: '%s'\n", line);
return 0;
}
#endif