1
0
mirror of https://gitlab.com/80486DX2-66/gists synced 2025-01-10 17:32:05 +05:30
gists/c-programming/io/freadln.c

181 lines
4.4 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.
*
* NOTE: Test: Declare macro NO_SIZE_T_FORMAT when compiling if your compiler
* or glibc do not support `%zu` format specifier.
2024-04-07 17:38:40 +03:00
* NOTE: Test: Declare macro POSIX when compiling if you're using a POSIX
* system.
*
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
*
* Author: Intel A80486DX2-66
* License: Creative Commons Zero 1.0 Universal
2024-02-20 01:38:45 +03:00
*/
#include "freadln.h"
int freadln(FILE* f, 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_EOF: end of file
2024-02-20 01:38:45 +03:00
* freadln_ERROR: an error occurred (see errno)
*/
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
if (f == stdin)
fflush(stdout);
2024-03-10 14:12:05 +03:00
freadln_length_type length = 0; // initial length
2024-02-20 01:38:45 +03:00
*output = malloc((length + 1) * sizeof(char));
if (*output == NULL) {
errno = EINVAL;
2024-02-20 01:38:45 +03:00
return freadln_ERROR;
}
2024-02-20 01:38:45 +03:00
int character;
2024-04-05 21:40:19 +03:00
while ((character = fgetc(f)) != EOF && character != EOT
2024-02-20 01:38:45 +03:00
/* 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 15:25:59 +03:00
freadln_epilogue;
return freadln_ERROR;
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 15:25:59 +03:00
freadln_epilogue;
return freadln_ERROR;
2024-02-20 01:38:45 +03:00
}
*output = temp;
}
errno = 0;
2024-03-10 15:25:59 +03:00
freadln_epilogue;
if (character == EOF)
return freadln_EOF;
2024-03-10 15:25:59 +03:00
return freadln_OK;
2024-02-20 01:38:45 +03:00
}
2024-03-10 14:14:38 +03:00
#ifdef TEST
# include <inttypes.h>
# include <stdint.h>
2024-04-07 17:38:40 +03:00
# if POSIX
# include <unistd.h>
# define SLEEP_FN sleep
# define DO_SLEEP 1
# elif defined(_WIN32) || defined(WIN32)
# include <windows.h>
# define SLEEP_FN(x) Sleep((DWORD) (x))
# define DO_SLEEP 1
# elif defined(__CYGWIN__) || defined(__unix__) || (defined(__APPLE__) && \
defined(__MACH__)) || defined(__linux__) || defined(__FreeBSD__) || \
2024-03-10 15:46:24 +03:00
defined(__NetBSD__) || defined(__OpenBSD__) || defined(__bsdi__) || \
defined(__DragonFly__) || defined(__MINGW32__) || defined(__MINGW64__)
# include <unistd.h>
# define SLEEP_FN(x) usleep((x) * 1000ULL)
# define DO_SLEEP 1
# else
2024-03-13 08:19:40 +03:00
# define SLEEP_FN(...)
# define DO_SLEEP 0
# endif
# if DO_SLEEP
# include <time.h>
# endif
# ifndef NO_SIZE_T_FORMAT
# if defined(__TINYC__) || (defined(__STDC_VERSION__) && \
__STDC_VERSION__ < 199901L)
# define NO_SIZE_T_FORMAT
# endif
# endif
# if defined(NO_SIZE_T_FORMAT)
# define PRIuSIZE PRIuMAX
typedef uintmax_t SIZE_T_FORMAT;
# else
# define PRIuSIZE "zu"
typedef size_t SIZE_T_FORMAT;
# endif
2024-03-10 14:14:38 +03:00
int main(void) {
// stdin test
2024-03-10 14:14:38 +03:00
printf("Type something> ");
char* line;
if (finreadln(&line, NULL) == freadln_ERROR) {
2024-03-10 14:14:38 +03:00
perror("freadln");
exit(EXIT_FAILURE);
}
printf("Input string: '%s'\n", line);
// file test
#define TEST_FILE "freadln_test.txt"
FILE* f = fopen(TEST_FILE, "w");
if (f == NULL) {
perror("fopen");
exit(EXIT_FAILURE);
}
fprintf(f, "Hello, world!\nAnother line\n\n");
FILE* new_f = freopen(TEST_FILE, "r", f);
if (new_f == NULL) {
perror("freopen");
exit(EXIT_FAILURE);
}
f = new_f;
printf("Waiting for 4 seconds...\n");
fflush(stdout);
clock_t start;
for (int i = 0; i < 4; i++) {
#if DO_SLEEP
start = clock();
#endif
size_t line_length;
int result = freadln(f, &line, &line_length);
if (result == freadln_ERROR) {
perror("freadln");
exit(EXIT_FAILURE);
} else if (result == freadln_EOF || feof(f)) {
printf("File: EOF, breaking the loop (returned by function? %d, "
"feof? %d)\n", result == freadln_EOF, !!feof(f));
break;
}
printf("File, line #%d: '%s' (%" PRIuSIZE " characters)\n", i + 1, line,
(SIZE_T_FORMAT) line_length);
SLEEP_FN(1000 - ((long double) (clock() - start) * 1000.l) /
CLOCKS_PER_SEC);
}
fclose(f);
fflush(stdout);
2024-03-10 14:14:38 +03:00
return 0;
}
#endif