Update getline implementation for non-POSIX systems

This commit is contained in:
ado 2024-02-22 23:59:58 +01:00
parent 42d618ed64
commit 230da6a3f2
2 changed files with 73 additions and 62 deletions

View File

@ -1,7 +1,9 @@
#pragma once #pragma once
#include <cerrno>
#include <cstdint> #include <cstdint>
#include <cstdio> #include <cstdio>
#include <cstdlib> #include <cstdlib>
#include <cstring>
#include <vector> #include <vector>
namespace ss { namespace ss {
@ -12,6 +14,7 @@ using string_range = std::pair<const char*, const char*>;
using split_data = std::vector<string_range>; using split_data = std::vector<string_range>;
constexpr inline auto default_delimiter = ","; constexpr inline auto default_delimiter = ",";
constexpr inline auto get_line_initial_buffer_size = 128;
template <bool StringError> template <bool StringError>
inline void assert_string_error_defined() { inline void assert_string_error_defined() {
@ -30,55 +33,58 @@ inline ssize_t get_line(char** lineptr, size_t* n, FILE* stream) {
return getline(lineptr, n, stream); return getline(lineptr, n, stream);
} }
#else #else
using ssize_t = int64_t;
inline ssize_t get_line(char** lineptr, size_t* n, FILE* stream) {
size_t pos;
int c;
if (lineptr == nullptr || stream == nullptr || n == nullptr) { using ssize_t = int64_t;
ssize_t get_line(char** lineptr, size_t* n, FILE* fp) {
if (lineptr == nullptr || n == nullptr || fp == nullptr) {
errno = EINVAL; errno = EINVAL;
return -1; return -1;
} }
c = getc(stream); char buff[get_line_initial_buffer_size];
if (c == EOF) {
return -1;
}
if (*lineptr == nullptr) { if (*lineptr == nullptr || *n < sizeof(buff)) {
*lineptr = static_cast<char*>(malloc(128)); *n = sizeof(buff);
if (*lineptr == nullptr) { auto new_lineptr = static_cast<char*>(realloc(*lineptr, *n));
if (new_lineptr == nullptr) {
errno = ENOMEM;
return -1; return -1;
} }
*n = 128;
*lineptr = new_lineptr;
} }
pos = 0; (*lineptr)[0] = '\0';
while (c != EOF) {
if (pos + 1 >= *n) { while (fgets(buff, sizeof(buff), fp) != nullptr) {
size_t new_size = *n + (*n >> 2); size_t line_used = strlen(*lineptr);
if (new_size < 128) { size_t buff_used = strlen(buff);
new_size = 128;
} if (*n < buff_used + line_used) {
char* new_ptr = static_cast<char*>( *n *= 2;
realloc(static_cast<void*>(*lineptr), new_size));
if (new_ptr == nullptr) { auto new_lineptr = static_cast<char*>(realloc(*lineptr, *n));
if (new_lineptr == nullptr) {
errno = ENOMEM;
return -1; return -1;
} }
*n = new_size;
*lineptr = new_ptr; *lineptr = new_lineptr;
} }
(*lineptr)[pos++] = c; memcpy(*lineptr + line_used, buff, buff_used);
if (c == '\n') { line_used += buff_used;
break; (*lineptr)[line_used] = '\0';
if ((*lineptr)[line_used - 1] == '\n') {
return line_used;
} }
c = getc(stream);
} }
(*lineptr)[pos] = '\0'; return -1;
return pos;
} }
#endif #endif
} /* ss */ } /* ss */

67
ssp.hpp
View File

@ -1,5 +1,6 @@
#include <algorithm> #include <algorithm>
#include <array> #include <array>
#include <cerrno>
#include <charconv> #include <charconv>
#include <cstdint> #include <cstdint>
#include <cstdio> #include <cstdio>
@ -625,6 +626,7 @@ using string_range = std::pair<const char*, const char*>;
using split_data = std::vector<string_range>; using split_data = std::vector<string_range>;
constexpr inline auto default_delimiter = ","; constexpr inline auto default_delimiter = ",";
constexpr inline auto get_line_initial_buffer_size = 128;
template <bool StringError> template <bool StringError>
inline void assert_string_error_defined() { inline void assert_string_error_defined() {
@ -643,55 +645,58 @@ inline ssize_t get_line(char** lineptr, size_t* n, FILE* stream) {
return getline(lineptr, n, stream); return getline(lineptr, n, stream);
} }
#else #else
using ssize_t = int64_t;
inline ssize_t get_line(char** lineptr, size_t* n, FILE* stream) {
size_t pos;
int c;
if (lineptr == nullptr || stream == nullptr || n == nullptr) { using ssize_t = int64_t;
ssize_t get_line(char** lineptr, size_t* n, FILE* fp) {
if (lineptr == nullptr || n == nullptr || fp == nullptr) {
errno = EINVAL; errno = EINVAL;
return -1; return -1;
} }
c = getc(stream); char buff[get_line_initial_buffer_size];
if (c == EOF) {
return -1;
}
if (*lineptr == nullptr) { if (*lineptr == nullptr || *n < sizeof(buff)) {
*lineptr = static_cast<char*>(malloc(128)); *n = sizeof(buff);
if (*lineptr == nullptr) { auto new_lineptr = static_cast<char*>(realloc(*lineptr, *n));
if (new_lineptr == nullptr) {
errno = ENOMEM;
return -1; return -1;
} }
*n = 128;
*lineptr = new_lineptr;
} }
pos = 0; (*lineptr)[0] = '\0';
while (c != EOF) {
if (pos + 1 >= *n) { while (fgets(buff, sizeof(buff), fp) != nullptr) {
size_t new_size = *n + (*n >> 2); size_t line_used = strlen(*lineptr);
if (new_size < 128) { size_t buff_used = strlen(buff);
new_size = 128;
} if (*n < buff_used + line_used) {
char* new_ptr = static_cast<char*>( *n *= 2;
realloc(static_cast<void*>(*lineptr), new_size));
if (new_ptr == nullptr) { auto new_lineptr = static_cast<char*>(realloc(*lineptr, *n));
if (new_lineptr == nullptr) {
errno = ENOMEM;
return -1; return -1;
} }
*n = new_size;
*lineptr = new_ptr; *lineptr = new_lineptr;
} }
(*lineptr)[pos++] = c; memcpy(*lineptr + line_used, buff, buff_used);
if (c == '\n') { line_used += buff_used;
break; (*lineptr)[line_used] = '\0';
if ((*lineptr)[line_used - 1] == '\n') {
return line_used;
} }
c = getc(stream);
} }
(*lineptr)[pos] = '\0'; return -1;
return pos;
} }
#endif #endif
} /* ss */ } /* ss */