mirror of
https://github.com/thepigeongenerator/mcaselector-lite.git
synced 2025-12-19 12:55:45 +01:00
enable a column limit.
This commit is contained in:
@@ -11,8 +11,9 @@
|
||||
#error platform not supported!
|
||||
#endif
|
||||
|
||||
/* tests a files access with F_OK, X_OK, R_OK, W_OK OR'd together
|
||||
returns 0 upon success. -1 when errno is set and anything else when one or more of the permissions isn't set */
|
||||
/* Tests a files access with F_OK, X_OK, R_OK, W_OK OR'd together.
|
||||
Returns 0 upon success, -1 when errno is set,
|
||||
anything else when one or more of the permissions isn't set */
|
||||
static inline int faccess(const char *restrict fname, int perms);
|
||||
|
||||
// define the constants if they haven't been
|
||||
@@ -30,16 +31,24 @@ static inline int faccess(const char *restrict fname, int perms);
|
||||
#endif
|
||||
|
||||
#if defined __unix__ && _POSIX_C_SOURCE >= 200809L
|
||||
#define PATH_SEP '/' // contains the path separator as a character. Yes it is extremely annoying that this has to exist.
|
||||
#define PATH_SEP_STR "/" // contains the path separator as a string, useful for concatenation. Yes it is extremely annoying that this has to exist.
|
||||
#define PATH_SEP \
|
||||
'/' // contains the path separator as a character. Yes it is extremely annoying that this
|
||||
// has to exist.
|
||||
#define PATH_SEP_STR \
|
||||
"/" // contains the path separator as a string, useful for concatenation. Yes it is
|
||||
// extremely annoying that this has to exist.
|
||||
|
||||
int faccess(char const *restrict fname, int perms)
|
||||
{
|
||||
return access(fname, perms);
|
||||
}
|
||||
#elif defined(_WIN32)
|
||||
#define PATH_SEP '\\' // contains the path separator as a character. Yes it is extremely annoying that this has to exist.
|
||||
#define PATH_SEP_STR "\\" // contains the path separator as a string, useful for concatenation. Yes it is extremely annoying that this has to exist.
|
||||
#define PATH_SEP \
|
||||
'\\' // contains the path separator as a character. Yes it is extremely annoying that this
|
||||
// has to exist.
|
||||
#define PATH_SEP_STR \
|
||||
"\\" // contains the path separator as a string, useful for concatenation. Yes it is
|
||||
// extremely annoying that this has to exist.
|
||||
|
||||
int faccess(char const *restrict fname, int perms)
|
||||
{
|
||||
|
||||
@@ -4,9 +4,9 @@
|
||||
|
||||
#if defined(__unix__)
|
||||
#define unixonly(_exec) _exec // executes inline code when __unix__ is defined, otherwise is no-op
|
||||
#define winonly(_exec) // (no-op) executes inline code when _WIN32 is defined, otherwise is no-op
|
||||
#define winonly(_exec) // (no-op) executes inline code when _WIN32 is defined, otherwise is no-op
|
||||
#elif defined(_WIN32)
|
||||
#define unixonly(_exec) // (no-op) executes inline code when __unix__ is defined, otherwise is no-op
|
||||
#define unixonly(_exec) // (no-op) executes inline code when __unix__ is defined, otherwise is no-op
|
||||
#define winonly(_exec) _exec // executes inline code when _WIN32 is defined, otherwise is no-op
|
||||
#else
|
||||
#error platform unsupported!!
|
||||
|
||||
@@ -29,9 +29,7 @@ int conf_procbuf(const char *restrict buf, char *restrict kout, char *restrict v
|
||||
case '\n':
|
||||
case '\r':
|
||||
case '\0':
|
||||
case '#':
|
||||
brk = true;
|
||||
break;
|
||||
case '#': brk = true; break;
|
||||
}
|
||||
if (brk)
|
||||
break;
|
||||
@@ -46,8 +44,8 @@ int conf_procbuf(const char *restrict buf, char *restrict kout, char *restrict v
|
||||
*pos = buf[i]; // copy over the buffer's data
|
||||
pos++; // increment the position pointer
|
||||
}
|
||||
// null-terminate what we've got now (yes, there should be enough space for this since \0 isn't stored)
|
||||
// this also ensures the value is valid, even if none is given
|
||||
// null-terminate what we've got now (yes, there should be enough space for this since \0
|
||||
// isn't stored) this also ensures the value is valid, even if none is given
|
||||
*pos = '\0';
|
||||
|
||||
// no data if we didn't move from the key position
|
||||
@@ -55,7 +53,8 @@ int conf_procbuf(const char *restrict buf, char *restrict kout, char *restrict v
|
||||
return (pos == kout) ? CONF_ENODAT : (!feq ? CONF_ESYNTAX : 0);
|
||||
}
|
||||
|
||||
struct conf_entry const *conf_matchopt(struct conf_entry const *opts, usize optc, const char *restrict key)
|
||||
struct conf_entry const *conf_matchopt(struct conf_entry const *opts, usize optc,
|
||||
const char *restrict key)
|
||||
{
|
||||
// find a match for the current key
|
||||
usize i = 0;
|
||||
@@ -96,7 +95,8 @@ int conf_procval(struct conf_entry const *opt, const char *restrict val)
|
||||
// string data parsing
|
||||
case CONF_STR:
|
||||
if (*(char **)opt->out) {
|
||||
free(*(char **)opt->out); // if the same key is given multiple times, free the memory so we don't leak.
|
||||
free(*(char **)opt->out); // if the same key is given multiple times, free
|
||||
// the memory so we don't leak.
|
||||
warn("encountered a dynamic string multiple times, this is sub-optimal.");
|
||||
}
|
||||
*(char **)opt->out = strdup(val);
|
||||
@@ -116,25 +116,53 @@ int conf_procval(struct conf_entry const *opt, const char *restrict val)
|
||||
}
|
||||
|
||||
switch (opt->type) {
|
||||
case CONF_U8: *(u8 *)opt->out = *(u64 *)dat >= UINT8_MAX ? UINT8_MAX : *(u64 *)dat; return 0;
|
||||
case CONF_U16: *(u16 *)opt->out = *(u64 *)dat >= UINT16_MAX ? UINT16_MAX : *(u64 *)dat; return 0;
|
||||
case CONF_U32: *(u32 *)opt->out = *(u64 *)dat >= UINT32_MAX ? UINT32_MAX : *(u64 *)dat; return 0;
|
||||
case CONF_U64: *(u64 *)opt->out = *(u64 *)dat >= UINT64_MAX ? UINT64_MAX : *(u64 *)dat; return 0;
|
||||
case CONF_I8: *(i8 *)opt->out = *(i64 *)dat >= INT8_MAX ? INT8_MAX : (*(i64 *)dat <= INT8_MIN ? INT8_MIN : *(i64 *)dat); return 0;
|
||||
case CONF_I16: *(i16 *)opt->out = *(i64 *)dat >= INT16_MAX ? INT16_MAX : (*(i64 *)dat <= INT16_MIN ? INT16_MIN : *(i64 *)dat); return 0;
|
||||
case CONF_I32: *(i32 *)opt->out = *(i64 *)dat >= INT32_MAX ? INT32_MAX : (*(i64 *)dat <= INT32_MIN ? INT32_MIN : *(i64 *)dat); return 0;
|
||||
case CONF_I64: *(i64 *)opt->out = *(i64 *)dat >= INT64_MAX ? INT64_MAX : (*(i64 *)dat <= INT64_MIN ? INT64_MIN : *(i64 *)dat); return 0;
|
||||
case CONF_U8:
|
||||
*(u8 *)opt->out = *(u64 *)dat >= UINT8_MAX ? UINT8_MAX : *(u64 *)dat;
|
||||
return 0;
|
||||
case CONF_U16:
|
||||
*(u16 *)opt->out = *(u64 *)dat >= UINT16_MAX ? UINT16_MAX : *(u64 *)dat;
|
||||
return 0;
|
||||
case CONF_U32:
|
||||
*(u32 *)opt->out = *(u64 *)dat >= UINT32_MAX ? UINT32_MAX : *(u64 *)dat;
|
||||
return 0;
|
||||
case CONF_U64:
|
||||
*(u64 *)opt->out = *(u64 *)dat >= UINT64_MAX ? UINT64_MAX : *(u64 *)dat;
|
||||
return 0;
|
||||
case CONF_I8:
|
||||
*(i8 *)opt->out = *(i64 *)dat >= INT8_MAX ?
|
||||
INT8_MAX :
|
||||
(*(i64 *)dat <= INT8_MIN ? INT8_MIN : *(i64 *)dat);
|
||||
return 0;
|
||||
case CONF_I16:
|
||||
*(i16 *)opt->out = *(i64 *)dat >= INT16_MAX ?
|
||||
INT16_MAX :
|
||||
(*(i64 *)dat <= INT16_MIN ? INT16_MIN : *(i64 *)dat);
|
||||
return 0;
|
||||
case CONF_I32:
|
||||
*(i32 *)opt->out = *(i64 *)dat >= INT32_MAX ?
|
||||
INT32_MAX :
|
||||
(*(i64 *)dat <= INT32_MIN ? INT32_MIN : *(i64 *)dat);
|
||||
return 0;
|
||||
case CONF_I64:
|
||||
*(i64 *)opt->out = *(i64 *)dat >= INT64_MAX ?
|
||||
INT64_MAX :
|
||||
(*(i64 *)dat <= INT64_MIN ? INT64_MIN : *(i64 *)dat);
|
||||
return 0;
|
||||
case CONF_F32: *(f32 *)opt->out = *(f32 *)dat; return 0;
|
||||
case CONF_F64: *(f64 *)opt->out = *(f64 *)dat; return 0;
|
||||
default: fatal("invalid switch state, all cases should be handled already"); // abort; this shouldn't be possible, so I blame the programmer
|
||||
default:
|
||||
/* abort; this shouldn't be possible so I blame the programmer. */
|
||||
fatal("invalid switch state, all cases should be handled already");
|
||||
}
|
||||
}
|
||||
|
||||
/* utility function for conf_getpat to concatenate 3 strings, where we already know the size */
|
||||
NONNULL((1, 3))
|
||||
static char *conf_getpat_concat(const char *restrict s1, const char *restrict s2, const char *restrict s3, usize s1len, usize s2len, usize s3len)
|
||||
static char *conf_getpat_concat(const char *restrict s1, const char *restrict s2,
|
||||
const char *restrict s3, usize s1len, usize s2len, usize s3len)
|
||||
{
|
||||
assert(s2 || (!s2 && !s2len)); // ensuring the programmer passes both s2 and s2len as 0, if they intend to
|
||||
assert(s2 || (!s2 && !s2len)); // ensuring the programmer passes both s2 and s2len as 0, if
|
||||
// they intend to
|
||||
char *buf, *ptr;
|
||||
|
||||
// allocate enough data for all three to the buffer
|
||||
@@ -146,8 +174,9 @@ static char *conf_getpat_concat(const char *restrict s1, const char *restrict s2
|
||||
// copy data to the buffer
|
||||
ptr = memcpy(ptr, s1, s1len) + s1len; // copy s1 data to the buffer
|
||||
if (s2len)
|
||||
ptr = memcpy(ptr, s2, s2len) + s2len; // copy s2 data to the buffer (excluding null-terminator)
|
||||
(void)strcpy(ptr, s3); // copy s3 as a string, thus including null-terminator
|
||||
ptr = memcpy(ptr, s2, s2len) + s2len; // copy s2 data to the buffer (excluding
|
||||
// null-terminator)
|
||||
(void)strcpy(ptr, s3); // copy s3 as a string, thus including null-terminator
|
||||
|
||||
// return the buffer
|
||||
return buf;
|
||||
|
||||
@@ -19,18 +19,18 @@ enum conf_err {
|
||||
|
||||
/* defines the primitive types available in the config file */
|
||||
enum conf_primitive {
|
||||
CONF_STR = 0, // expects: `char**`, will output malloc'd data !!must be freed!!
|
||||
CONF_I8 = 1, // expects: `int8_t*`, will point to a location in memory where an i8 is stored.
|
||||
CONF_I16 = 2, // expects: `int16_t*`, will point to a location in memory where an i16 is stored.
|
||||
CONF_I32 = 4, // expects: `int32_t*`, will point to a location in memory where an i32 is stored.
|
||||
CONF_I64 = 8, // expects: `int64_t*`, will point to a location in memory where an i64 is stored.
|
||||
CONF_U8 = CONF_I8 | 0x80, // expects: `uint8_t*`, will point to a location in memory where an u8 is stored.
|
||||
CONF_U16 = CONF_I16 | 0x80, // expects: `uint16_t*`, will point to a location in memory where an u16 is stored.
|
||||
CONF_U32 = CONF_I32 | 0x80, // expects: `uint32_t*`, will point to a location in memory where an u32 is stored.
|
||||
CONF_U64 = CONF_I64 | 0x80, // expects: `uint64_t*`, will point to a location in memory where an u64 is stored.
|
||||
CONF_F32 = CONF_I32 | 0x40, // expects: `float*`, will point to a location in memory where an f32 is stored.
|
||||
CONF_F64 = CONF_I64 | 0x40, // expects: `double*`, will point to a location in memory where an f64 is stored.
|
||||
CONF_FSTR = 0x40, // expects: `struct conf_fstr*`, which contains the data for a fixed-width string
|
||||
CONF_STR = 0, // expects `char**`, will output malloc'd data
|
||||
CONF_I8 = 1, // expects `int8_t*`,
|
||||
CONF_I16 = 2, // expects `int16_t*`,
|
||||
CONF_I32 = 4, // expects `int32_t*`,
|
||||
CONF_I64 = 8, // expects `int64_t*`,
|
||||
CONF_U8 = CONF_I8 | 0x80, // expects `uint8_t*`,
|
||||
CONF_U16 = CONF_I16 | 0x80, // expects `uint16_t*`,
|
||||
CONF_U32 = CONF_I32 | 0x80, // expects `uint32_t*`,
|
||||
CONF_U64 = CONF_I64 | 0x80, // expects `uint64_t*`,
|
||||
CONF_F32 = CONF_I32 | 0x40, // expects `float*`,
|
||||
CONF_F64 = CONF_I64 | 0x40, // expects `double*`,
|
||||
CONF_FSTR = 0x40, // expects `struct conf_fstr*`
|
||||
};
|
||||
|
||||
/* for outputting a fixed string as this config field */
|
||||
@@ -42,29 +42,33 @@ struct conf_fstr {
|
||||
/* defines the structure of a config file entry */
|
||||
struct conf_entry {
|
||||
const char *key; // the key of this entry
|
||||
void *out; // the pointer to which the data is written value is read if the given option is incorrect or missing
|
||||
void *out; // where data will be written, is read when key is missing or invalid
|
||||
u8 type; // the primitive type which we are querying for
|
||||
};
|
||||
|
||||
/* processes an incoming buffer.
|
||||
* `buf`, `kout` and `vout` mustn't overlap, and must be (at least) `len` bytes long!
|
||||
* `kout` and `vout` will contain a null-terminated string if the function returned successfully.
|
||||
* returns `0` on success, `<0` when no data was found. `>0` when data was invalid but something went wrong.
|
||||
* see `CONF_E*` or `enum conf_err` */
|
||||
* returns `0` on success, `<0` when no data was found. `>0` when data was invalid but something
|
||||
* went wrong. see `CONF_E*` or `enum conf_err` */
|
||||
int conf_procbuf(const char *restrict buf, char *restrict kout, char *restrict vout, usize len);
|
||||
|
||||
/* matches the key with one of the options and returns the pointer. Returns NULL if none could be found. */
|
||||
struct conf_entry const *conf_matchopt(struct conf_entry const *opts, usize optc, const char *restrict key);
|
||||
/* matches the key with one of the options and returns the pointer. Returns NULL if none could be
|
||||
* found. */
|
||||
struct conf_entry const *conf_matchopt(struct conf_entry const *opts, usize optc,
|
||||
const char *restrict key);
|
||||
|
||||
/* processes the value belonging to the key and outputs the result to opts.
|
||||
* - `val` points to a null-terminated string which contains the key and value.
|
||||
* returns `0` upon success, non-zero upon failure. For information about specific error codes, see `enum conf_err` */
|
||||
/* Processes the value belonging to the key and outputs the result to opts.
|
||||
* `val` points to a null-terminated string which contains the key and value.
|
||||
* returns `0` upon success, non-zero upon failure.
|
||||
* For information about specific error codes, see`enum conf_err` */
|
||||
int conf_procval(struct conf_entry const *opts, const char *restrict val);
|
||||
|
||||
/* acquires the config file path, appending str to the end (you need to handle path separators yourself)
|
||||
* expecting str to be null-terminated
|
||||
* - linux: reads $XDG_CONFIG_HOME, if empty $HOME/.config is used, if $HOME is empty NULL is returned.
|
||||
* - windows: reads %APPDATA%, if empty %USERPROFILE%\AppData\Roaming is used, if both are empty NULL is returned.
|
||||
* - osx: reads $HOME, uses $HOME/Library/Application Support, if $HOME is empty NULL is returned.
|
||||
* !! A malloc'd null-terminated string is returned !! */
|
||||
/* acquires the config file path, appending str to the end expecting str to be null-terminated
|
||||
* note: you must handle path separators yourself.
|
||||
* checks the following environment variables for each platform in order:
|
||||
* LINUX: `$XDG_CONFIG_HOME`, `$HOME/.config`.
|
||||
* WINDOWS: `%APPDATA%`, `%USERPROFILE%\AppData\Roaming`.
|
||||
* MACOSX: `$HOME/Library/Application Support`.
|
||||
* `NULL` is returned if the path couldn't be found. */
|
||||
char *conf_getpat(const char *) MALLOC NONNULL((1));
|
||||
|
||||
@@ -7,7 +7,8 @@
|
||||
|
||||
#include "../types.h"
|
||||
|
||||
static void error_log(FILE *restrict stream, const char *restrict pfx, uint ln, const char *restrict file, const char *restrict fmt, va_list ap)
|
||||
static void error_log(FILE *restrict stream, const char *restrict pfx, uint ln,
|
||||
const char *restrict file, const char *restrict fmt, va_list ap)
|
||||
{
|
||||
fprintf(stream, "(%s:%u) [%s] '", file, ln, pfx);
|
||||
vfprintf(stream, fmt, ap);
|
||||
|
||||
Reference in New Issue
Block a user