182 lines
4.5 KiB
C
182 lines
4.5 KiB
C
#include "util.h"
|
|
|
|
#include <math.h>
|
|
#include <stdarg.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
|
|
#ifndef __has_builtin
|
|
#define __has_builtin(_) 0
|
|
#endif
|
|
|
|
void panicf(const char *fmt, ...) {
|
|
va_list args;
|
|
va_start(args, fmt);
|
|
vprintf(fmt, args);
|
|
va_end(args);
|
|
exit(1);
|
|
}
|
|
|
|
uint16_t parse_port(const char *str) {
|
|
long long n = atoll(str);
|
|
if (n <= 0 || n > UINT16_MAX)
|
|
panicf("Invalid port: Expected a number in the range 1..%d, got '%s'\n", UINT16_MAX, str);
|
|
return n;
|
|
}
|
|
|
|
uint8_t parse_hex_digit(char h) {
|
|
if (h >= '0' && h <= '9')
|
|
return h - '0';
|
|
else if (h >= 'a' && h <= 'f')
|
|
return h - 'a' + 10;
|
|
else if (h >= 'A' && h <= 'F')
|
|
return h - 'A' + 10;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
// Defaults for json parsing
|
|
void default_to_null(void *ptr) { *(void **)ptr = NULL; }
|
|
void default_to_false(void *ptr) { *(bool *)ptr = false; }
|
|
void default_to_zero_u8(void *ptr) { *(uint8_t *)ptr = 0; }
|
|
void default_to_zero_u32(void *ptr) { *(uint32_t *)ptr = 0; }
|
|
void default_to_zero_u64(void *ptr) { *(uint64_t *)ptr = 0; }
|
|
void default_to_zero_size(void *ptr) { *(size_t *)ptr = 0; }
|
|
void default_to_zero_double(void *ptr) { *(double *)ptr = 0.0; }
|
|
void default_to_one_size(void *ptr) { *(size_t *)ptr = 1; }
|
|
void default_to_negative_one_i32(void *ptr) { *(int32_t *)ptr = -1; }
|
|
|
|
// Transformers for json parsing
|
|
void tsf_numsec_to_timespec(void *arg, void *ptr) {
|
|
double seconds = *(double *)arg;
|
|
|
|
struct timespec ts;
|
|
ts.tv_sec = floor(seconds);
|
|
ts.tv_nsec = (seconds - floor(seconds)) * 1000000000;
|
|
|
|
*(struct timespec *)ptr = ts;
|
|
}
|
|
|
|
void tsf_numsec_to_intms(void *arg, void *ptr) {
|
|
double seconds = *(double *)arg;
|
|
*(uint32_t *)ptr = seconds * 1000;
|
|
}
|
|
|
|
void tsf_uniq_to_u64(void *arg, void *ptr) {
|
|
char *s = *(char **)arg;
|
|
if (strnlen(s, 18) != 17) {
|
|
printf("JSON: wrong length for uniq, expected 'xx:xx:xx:xx:xx:xx'\n");
|
|
free(s);
|
|
return;
|
|
}
|
|
uint64_t mac = 0;
|
|
for (int i = 0; i < 17; i++) {
|
|
char c = s[i];
|
|
uint8_t digit = 0;
|
|
|
|
if (c >= '0' && c <= '9')
|
|
digit = c - '0';
|
|
else if (c >= 'a' && c <= 'f')
|
|
digit = c - 'a' + 10;
|
|
else if (c >= 'A' && c <= 'F')
|
|
digit = c - 'A' + 10;
|
|
else if (c == ':')
|
|
continue;
|
|
else {
|
|
printf("JSON: unexpected character '%c' in uniq at position %i (%s)\n", c, i, s);
|
|
free(s);
|
|
return;
|
|
}
|
|
|
|
mac <<= 4;
|
|
mac |= digit;
|
|
}
|
|
free(s);
|
|
*(uint64_t *)ptr = mac;
|
|
}
|
|
|
|
void tsf_hex_to_i32(void *arg, void *ptr) {
|
|
char *s = *(char **)arg;
|
|
char *f = s;
|
|
char c;
|
|
int32_t res = 0;
|
|
while ((c = *s++) != '\0') {
|
|
uint8_t digit = 0;
|
|
|
|
if (c >= '0' && c <= '9')
|
|
digit = c - '0';
|
|
else if (c >= 'a' && c <= 'f')
|
|
digit = c - 'a' + 10;
|
|
else if (c >= 'A' && c <= 'F')
|
|
digit = c - 'A' + 10;
|
|
else {
|
|
printf("JSON: unexpected character '%c' in hex string\n", c);
|
|
free(f);
|
|
return;
|
|
}
|
|
res <<= 4;
|
|
res |= digit;
|
|
}
|
|
free(f);
|
|
*(int32_t *)ptr = res;
|
|
}
|
|
|
|
void tsf_double_to_size(void *arg, void *ptr) {
|
|
double d = *(double *)arg;
|
|
*(size_t *)ptr = d;
|
|
}
|
|
|
|
static uint8_t hex_digit(char c) {
|
|
if (c >= '0' && c <= '9')
|
|
return c - '0';
|
|
else if (c >= 'a' && c <= 'f')
|
|
return c - 'a' + 10;
|
|
else if (c >= 'A' && c <= 'F')
|
|
return c - 'A' + 10;
|
|
else
|
|
return 16;
|
|
}
|
|
|
|
void tsf_hex_to_color(void *arg, void *ptr) {
|
|
char *s = *(char **)arg;
|
|
int len = strnlen(s, 8);
|
|
if (len != 7 || s[0] != '#') {
|
|
printf("JSON: bad hex color format expected '#RRGGBB' or '#rrggbb', got '%s'\n", s);
|
|
free(s);
|
|
return;
|
|
}
|
|
|
|
uint8_t digit[6] = {0};
|
|
uint8_t *color = ptr;
|
|
|
|
for (int i = 1; i < 7; i++) {
|
|
digit[i] = hex_digit(s[i]);
|
|
if (digit[i] == 16) {
|
|
printf("JSON: illegal character in hex color: '%c'\n", s[i]);
|
|
free(s);
|
|
return;
|
|
}
|
|
|
|
if (i % 2 == 0) {
|
|
uint8_t c = digit[i];
|
|
uint8_t p = digit[i - 1];
|
|
color[i / 2] = (p << 4) | c;
|
|
}
|
|
}
|
|
|
|
free(s);
|
|
}
|
|
|
|
void tsf_num_to_u8_clamp(void *arg, void *ptr) {
|
|
double n = *(double *)arg;
|
|
*(uint8_t *)ptr = n > 255.0 ? 255.0 : n < 0.0 ? 0.0 : n;
|
|
}
|
|
|
|
void tsf_num_to_int(void *arg, void *ptr) {
|
|
double n = *(double *)arg;
|
|
*(int *)ptr = n;
|
|
}
|