This commit is contained in:
2026-03-29 18:40:26 +02:00
parent e5331e8e81
commit e41ef17d6b
+63 -50
View File
@@ -42,6 +42,8 @@ scan_error_t scan_error(const scan_t *s, const char *message) {
} }
#define scan_value_result(VAL) ((scan_result_t){.ok=true, .value=(scan_value_t){VAL}})
scan_result_t scan_error_result(const scan_t *s, const char *message) { scan_result_t scan_error_result(const scan_t *s, const char *message) {
return (scan_result_t) { return (scan_result_t) {
.ok=false, .ok=false,
@@ -63,9 +65,7 @@ scan_result_t scan_literal(scan_t *s, const char *lit) {
s->cur++; s->cur++;
} }
if (*lit == '\0') { if (*lit == '\0') {
return (scan_result_t) { return scan_value_result();
.ok=true,
};
} }
s->cur = save; s->cur = save;
return scan_error_result(s, "invalid literal"); return scan_error_result(s, "invalid literal");
@@ -95,12 +95,7 @@ scan_result_t scan_digit(scan_t *s) {
return scan_error_result(s, "expected digit"); return scan_error_result(s, "expected digit");
} }
s->cur++; s->cur++;
return (scan_result_t){ return scan_value_result(.digit=(*s->cur) - '0');
.ok=true,
.value=(scan_value_t){
.digit=(*s->cur) - '0',
},
};
} }
scan_result_t scan_i64(scan_t *s) { scan_result_t scan_i64(scan_t *s) {
@@ -153,7 +148,7 @@ scan_result_t scan_i16(scan_t *s) {
if (!result_i64.ok) { if (!result_i64.ok) {
return result_i64; return result_i64;
} }
int16_t val = (int32_t)result_i64.value.i64; int16_t val = (int16_t)result_i64.value.i64;
int64_t back = (int64_t)val; int64_t back = (int64_t)val;
if (back != result_i64.value.i64) { if (back != result_i64.value.i64) {
return scan_error_result(s, "integer does not fit in i16 value"); return scan_error_result(s, "integer does not fit in i16 value");
@@ -171,7 +166,7 @@ scan_result_t scan_i8(scan_t *s) {
if (!result_i64.ok) { if (!result_i64.ok) {
return result_i64; return result_i64;
} }
int8_t val = (int32_t)result_i64.value.i64; int8_t val = (int8_t)result_i64.value.i64;
int64_t back = (int64_t)val; int64_t back = (int64_t)val;
if (back != result_i64.value.i64) { if (back != result_i64.value.i64) {
return scan_error_result(s, "integer does not fit in i8 value"); return scan_error_result(s, "integer does not fit in i8 value");
@@ -192,7 +187,7 @@ scan_result_t scan_u64(scan_t *s) {
if (*s->cur == '+') s->cur++; if (*s->cur == '+') s->cur++;
if (!isdigit((unsigned char)*s->cur)) { if (!isdigit((unsigned char)*s->cur)) {
s->cur = save; s->cur = save;
return false; return scan_error_result(s, "not an integer");
} }
s->cur = save; s->cur = save;
char *end; char *end;
@@ -200,75 +195,93 @@ scan_result_t scan_u64(scan_t *s) {
uint64_t val = strtoull(s->cur, &end, 10); uint64_t val = strtoull(s->cur, &end, 10);
if (end == s->cur) { if (end == s->cur) {
s->cur = save; s->cur = save;
return false; return scan_error_result(s, "not an integer");
} }
if (errno == ERANGE) { if (errno == ERANGE) {
scan_error(s, "integer does not fit in u64 value"); return scan_error_result(s, "integer does not fit in i64 value");
return false;
} }
s->cur = end; s->cur = end;
s->value.u64 = val; return (scan_result_t){
return true; .ok=true,
.value=(scan_value_t){
.u64=val,
},
};
} }
bool scan_u32(scan_t *s) { scan_result_t scan_u32(scan_t *s) {
if (!scan_u64(s)) { scan_result_t result_u64 = scan_u64(s);
return false; if (!result_u64.ok) {
return result_u64;
} }
uint32_t val = (uint32_t)s->value.u64; uint32_t val = (uint32_t)result_u64.value.u64;
uint64_t back = (uint64_t)val; uint64_t back = (uint64_t)val;
if (back != s->value.u64) { if (back != result_u64.value.u64) {
scan_error(s, "int does not fit in u32 value"); return scan_error_result(s, "integer does not fit in u32 values");
return false;
} }
s->value.u32 = val; return (scan_result_t) {
return true; .ok=true,
.value=(scan_value_t){
.u32=val,
},
};
} }
bool scan_u16(scan_t *s) { scan_result_t scan_u16(scan_t *s) {
if (!scan_u64(s)) { scan_result_t result_u64 = scan_u64(s);
return false; if (!result_u64.ok) {
return result_u64;
} }
uint16_t val = (uint16_t)s->value.u64; uint16_t val = (uint16_t)result_u64.value.u64;
uint64_t back = (uint64_t)val; uint64_t back = (uint64_t)val;
if (back != s->value.u64) { if (back != result_u64.value.u64) {
scan_error(s, "int does not fit in u16 value"); return scan_error_result(s, "integer does not fit in u16 value");
return false;
} }
s->value.u16 = val; return (scan_result_t) {
return true; .ok=true,
.value=(scan_value_t){
.u16=val,
},
};
} }
bool scan_u8(scan_t *s) { scan_result_t scan_u8(scan_t *s) {
if (!scan_u64(s)) { scan_result_t result_u64 = scan_u64(s);
return false; if (!result_u64.ok) {
return result_u64;
} }
uint8_t val = (uint8_t)s->value.u64; uint8_t val = (uint8_t)result_u64.value.u64;
uint64_t back = (uint64_t)val; uint64_t back = (uint64_t)val;
if (back != s->value.u64) { if (back != result_u64.value.u64) {
scan_error(s, "int does not fit in u8 value"); return scan_error_result(s, "integer does not fit in u8 value");
return false;
} }
s->value.u8 = val; return (scan_result_t) {
return true; .ok=true,
.value=(scan_value_t){
.u8=val,
},
};
} }
bool scan_f64(scan_t *s) { scan_result_t scan_f64(scan_t *s) {
const char *save = s->cur; const char *save = s->cur;
char *end; char *end;
errno = 0; errno = 0;
double val = strtod(s->cur, &end); double val = strtod(s->cur, &end);
if (end == s->cur) { if (end == s->cur) {
s->cur = save; s->cur = save;
return false; return scan_error_result(s, "not a float");
} }
if (errno == ERANGE) { if (errno == ERANGE) {
scan_error(s, "float does not fit in f64 value"); return scan_error_result(s, "float does not fit in f64 value");
return false;
} }
s->cur = end; s->cur = end;
s->value.f64 = val; return (scan_result_t) {
return true; .ok=true,
.value=(scan_value_t){
.f64=val,
},
};
} }
bool scan_f32(scan_t *s) { bool scan_f32(scan_t *s) {