15#if !defined(CSV_ENABLE_THREADS) || CSV_ENABLE_THREADS
20# ifndef WIN32_LEAN_AND_MEAN
21# define WIN32_LEAN_AND_MEAN
26#elif defined(__linux__)
36#if defined(__EMSCRIPTEN__)
37#undef CSV_ENABLE_THREADS
38#define CSV_ENABLE_THREADS 0
39#elif !defined(CSV_ENABLE_THREADS)
40#define CSV_ENABLE_THREADS 1
44#if defined(CSV_CODE_COVERAGE)
47 #define CSV_FORCE_INLINE inline
49 #define CSV_NON_NULL(...)
50#elif defined(__clang__) || defined(__GNUC__)
51 #define CSV_CONST __attribute__((__const__))
52 #define CSV_PURE __attribute__((__pure__))
53 #define CSV_FORCE_INLINE inline __attribute__((__always_inline__))
57 #define CSV_PRIVATE __attribute__((__visibility__("hidden")))
59 #define CSV_NON_NULL(...) __attribute__((__nonnull__(__VA_ARGS__)))
60#elif defined(_MSC_VER)
63 #define CSV_FORCE_INLINE __forceinline
65 #define CSV_NON_NULL(...)
69 #define CSV_FORCE_INLINE inline
71 #define CSV_NON_NULL(...)
77# define CSV_MSVC_PUSH_DISABLE(w) __pragma(warning(push)) __pragma(warning(disable: w))
78# define CSV_MSVC_POP __pragma(warning(pop))
80# define CSV_MSVC_PUSH_DISABLE(w)
85#if defined(__cpp_exceptions) || defined(_CPPUNWIND) || defined(__EXCEPTIONS)
86 #define CSV_EXCEPTIONS_ENABLED 1
88 #define CSV_EXCEPTIONS_ENABLED 0
91#if !CSV_EXCEPTIONS_ENABLED
92 #error "csv-parser requires C++ exceptions. Enable exception handling (for example, remove -fno-exceptions or use /EHsc)."
97#if defined(_MSVC_LANG) && _MSVC_LANG > __cplusplus
98# define CSV_CPLUSPLUS _MSVC_LANG
100# define CSV_CPLUSPLUS __cplusplus
103#if CSV_CPLUSPLUS >= 202302L
107#if CSV_CPLUSPLUS >= 202002L
111#if CSV_CPLUSPLUS >= 201703L
115#if CSV_CPLUSPLUS >= 201402L
121#if defined(CSV_HAS_CXX17)
122#define CSV_FALLTHROUGH [[fallthrough]]
123#elif defined(__clang__) && defined(__has_cpp_attribute)
124#if __has_cpp_attribute(clang::fallthrough)
125#define CSV_FALLTHROUGH [[clang::fallthrough]]
127#define CSV_FALLTHROUGH ((void)0)
129#elif defined(__GNUC__) && __GNUC__ >= 7
130#define CSV_FALLTHROUGH __attribute__((fallthrough))
132#define CSV_FALLTHROUGH ((void)0)
137#include <string_view>
139#include "../external/string_view.hpp"
148#pragma region Compatibility Macros
164#define STATIC_ASSERT(x) static_assert(x, "Assertion failed")
167 #define CSV_DEBUG_ASSERT(x) ((void)sizeof(x), (void)0)
169 #define CSV_DEBUG_ASSERT(x) assert(x)
185 #define IF_CONSTEXPR if constexpr
186 #define CONSTEXPR_VALUE constexpr
188 #define CONSTEXPR_17 constexpr
190 #define IF_CONSTEXPR if
191 #define CONSTEXPR_VALUE const
193 #define CONSTEXPR_17 inline
197 template<
bool B,
class T =
void>
198 using enable_if_t = std::enable_if_t<B, T>;
200 #define CONSTEXPR_14 constexpr
201 #define CONSTEXPR_VALUE_14 constexpr
203 template<
bool B,
class T =
void>
204 using enable_if_t =
typename std::enable_if<B, T>::type;
206 #define CONSTEXPR_14 inline
207 #define CONSTEXPR_VALUE_14 const
210 namespace internals {
211 template<
bool B,
class T =
void>
212 using enable_if_t = csv::enable_if_t<B, T>;
216 template<
typename F,
typename... Args>
217 using invoke_result_t =
typename std::invoke_result<F, Args...>::type;
219 template<
typename F,
typename... Args>
220 using invoke_result_t =
typename std::result_of<F(Args...)>::type;
223 template<
typename... Ts>
226 template<
typename F,
typename ReturnType,
typename Enable,
typename... Args>
229 template<
typename F,
typename ReturnType,
typename... Args>
233 void_t<invoke_result_t<F, Args...>>,
235 > : std::integral_constant<
237 std::is_convertible<invoke_result_t<F, Args...>, ReturnType>::value
240 template<
typename F,
typename ReturnType,
typename... Args>
247#if defined(__GNUC__) && !defined(__clang__)
248 #if defined(CSV_HAS_CXX17) && (((__GNUC__ == 7) && (__GNUC_MINOR__ >= 2)) || (__GNUC__ >= 8))
249 #define CONSTEXPR constexpr
253 #define CONSTEXPR constexpr
258#define CONSTEXPR inline
265 namespace internals {
270 static auto test(
int) ->
decltype(
271 std::hash<U>{}(std::declval<const U&>()),
276 static std::false_type test(...);
279 static constexpr bool value =
decltype(test<T>(0))::value;
286 static auto test(
int) ->
decltype(
287 std::declval<const U&>() == std::declval<const U&>(),
292 static std::false_type test(...);
295 static constexpr bool value =
decltype(test<T>(0))::value;
308 if (
this != &other) {
309 value_ = std::move(other.value_);
315 template<
typename Factory>
316 T& get_or_create(Factory&& factory)
const {
317#if CSV_ENABLE_THREADS
318 std::call_once(init_once_, [
this, &factory]() {
329 T* get()
const noexcept {
334 mutable std::shared_ptr<T> value_ =
nullptr;
335#if CSV_ENABLE_THREADS
336 mutable std::once_flag init_once_;
342 #pragma region CXX20 Concepts
347 std::ranges::input_range<std::remove_reference_t<T>>
348 && std::convertible_to<
349 std::ranges::range_reference_t<std::remove_reference_t<T>>,
355 { value.to_sv_range() } -> std::ranges::input_range;
356 requires std::convertible_to<
357 std::ranges::range_reference_t<
decltype(value.to_sv_range())>,
367 std::ranges::input_range<std::remove_reference_t<T>>
369 std::ranges::range_reference_t<std::remove_reference_t<T>>
378#if defined(PAGE_SIZE)
384 inline int getpagesize() {
385 _SYSTEM_INFO sys_info = {};
386 GetSystemInfo(&sys_info);
387 return std::max(sys_info.dwPageSize, sys_info.dwAllocationGranularity);
391#elif defined(__linux__)
415 CONSTEXPR_VALUE_14
CSVChunkIndex CSV_CHUNK_INDEX_MAX = (std::numeric_limits<CSVChunkIndex>::max)();
417 CONSTEXPR_VALUE_14
size_t CSV_CHUNK_SIZE_MAX = CSV_CHUNK_INDEX_MAX;
430 inline bool is_equal(T a, T b, T epsilon = 0.001) {
432 static_assert(std::is_floating_point<T>::value,
"T must be a floating point type.");
433 return std::abs(a - b) < epsilon;
454 return (
ParseFlags)((int)flag & ~((
int)ParseFlags::QUOTE * quote_escape));
459 STATIC_ASSERT(ParseFlags::DELIMITER < ParseFlags::CARRIAGE_RETURN);
460 STATIC_ASSERT(ParseFlags::DELIMITER < ParseFlags::NEWLINE);
461 STATIC_ASSERT(ParseFlags::CARRIAGE_RETURN < ParseFlags::NEWLINE);
470 STATIC_ASSERT(
quote_escape_flag(ParseFlags::DELIMITER,
false) == ParseFlags::DELIMITER);
471 STATIC_ASSERT(
quote_escape_flag(ParseFlags::CARRIAGE_RETURN,
false) == ParseFlags::CARRIAGE_RETURN);
472 STATIC_ASSERT(
quote_escape_flag(ParseFlags::NEWLINE,
false) == ParseFlags::NEWLINE);
474 STATIC_ASSERT(
quote_escape_flag(ParseFlags::NOT_SPECIAL,
true) == ParseFlags::NOT_SPECIAL);
475 STATIC_ASSERT(
quote_escape_flag(ParseFlags::QUOTE,
true) == ParseFlags::QUOTE_ESCAPE_QUOTE);
476 STATIC_ASSERT(
quote_escape_flag(ParseFlags::DELIMITER,
true) == ParseFlags::NOT_SPECIAL);
477 STATIC_ASSERT(
quote_escape_flag(ParseFlags::CARRIAGE_RETURN,
true) == ParseFlags::NOT_SPECIAL);
478 STATIC_ASSERT(
quote_escape_flag(ParseFlags::NEWLINE,
true) == ParseFlags::NOT_SPECIAL);
491 constexpr unsigned CHAR_OFFSET = std::numeric_limits<char>::is_signed ? 128 : 0;
std::array< ParseFlags, 256 > ParseFlagMap
An array which maps ASCII chars to a parsing flag.
std::array< bool, 256 > WhitespaceMap
An array which maps ASCII chars to a flag indicating if it is whitespace.
bool is_equal(T a, T b, T epsilon=0.001)
const int PAGE_SIZE
Size of a memory page in bytes.
ParseFlags
An enum used for describing the significance of each character with respect to CSV parsing.
@ QUOTE_ESCAPE_QUOTE
A quote inside or terminating a quote_escaped field.
@ NOT_SPECIAL
Characters with no special meaning or escaped delimiters and newlines.
@ NEWLINE
Characters which signify a new row.
@ CARRIAGE_RETURN
Characters which signify a carriage return.
@ QUOTE
Characters which may signify a quote escape.
@ DELIMITER
Characters which signify a new field.
constexpr size_t CSV_CHUNK_SIZE_FLOOR
Minimum supported custom chunk size for CSVFormat::chunk_size().
constexpr ParseFlags quote_escape_flag(ParseFlags flag, bool quote_escape) noexcept
Transform the ParseFlags given the context of whether or not the current field is quote escaped.
std::uint32_t CSVChunkIndex
Type used to represent the location of a CSV byte within a larger chunk.
constexpr size_t CSV_SPECULATIVE_PARALLEL_MIN_BYTES
Default minimum source size before speculative parallel parsing is considered.
constexpr size_t CSV_CHUNK_SIZE_DEFAULT
Default chunk size for lazy-loading large CSV files.
The all encompassing namespace.
constexpr int CSV_NOT_FOUND
Integer indicating a requested column wasn't found.
constexpr unsigned CHAR_OFFSET
Offset to convert char into array index.
std::string_view string_view
The string_view class used by this library.