Initial working version of generic cord func
This commit is contained in:
parent
8631776f2f
commit
05f23c2507
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,3 +1,4 @@
|
||||
*.o
|
||||
*.so
|
||||
nextlang
|
||||
tags
|
||||
|
5
Makefile
5
Makefile
@ -26,10 +26,13 @@ LDLIBS=-lgc -lgccjit -lcord -lm -lunistring
|
||||
BUILTIN_OBJS=builtins/array.o builtins/bool.o builtins/builtins.o builtins/char.o builtins/floats.o builtins/functions.o builtins/integers.o \
|
||||
builtins/memory.o builtins/string.o builtins/table.o builtins/types.o
|
||||
|
||||
all: nextlang
|
||||
all: nextlang libnext.so
|
||||
|
||||
nextlang: nextlang.c parse.o files.o util.o ast.o compile.o types.o SipHash/halfsiphash.o $(BUILTIN_OBJS)
|
||||
|
||||
libnext.so: metamethods/cord.o util.o SipHash/halfsiphash.o
|
||||
$(CC) $^ $(CFLAGS) $(EXTRA) $(CWARN) $(G) $(O) $(OSFLAGS) $(LDLIBS) -Wl,-soname,libnext.so -shared -o $@
|
||||
|
||||
SipHash/halfsiphash.c:
|
||||
git submodule update --init --recursive
|
||||
|
||||
|
100
metamethods/cord.c
Normal file
100
metamethods/cord.c
Normal file
@ -0,0 +1,100 @@
|
||||
#include <err.h>
|
||||
#include <gc/cord.h>
|
||||
#include <stdarg.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
|
||||
#include "../util.h"
|
||||
|
||||
typedef CORD (*custom_cord_func)(void *x, void *userdata);
|
||||
|
||||
typedef struct {
|
||||
void *data;
|
||||
int64_t length:42;
|
||||
uint8_t free:4;
|
||||
bool copy_on_write:1, atomic:1;
|
||||
int16_t stride:16;
|
||||
} generic_array_t;
|
||||
|
||||
static CORD vas_cord(void *x, const char **fmt, va_list args)
|
||||
{
|
||||
char c = **fmt;
|
||||
++(*fmt);
|
||||
switch (c) {
|
||||
case '@': case '?': {
|
||||
if (!x)
|
||||
return CORD_asprintf("%c%r", vas_cord(NULL, fmt, args));
|
||||
void *ptr = *(void**)x;
|
||||
return CORD_cat(ptr ? (c == '@' ? "@" : "?") : "!", vas_cord(ptr, fmt, args));
|
||||
}
|
||||
case 'B': {
|
||||
if (!x) return "Bool";
|
||||
return *(bool*)x ? "yes" : "no";
|
||||
}
|
||||
case 'I': {
|
||||
size_t bits = va_arg(args, size_t);
|
||||
switch (bits) {
|
||||
case 64:
|
||||
return x ? CORD_asprintf("%ld", *(int64_t*)x) : "Int64";
|
||||
case 32:
|
||||
return x ? CORD_asprintf("%d", *(int32_t*)x) : "Int32";
|
||||
case 16:
|
||||
return x ? CORD_asprintf("%d", *(int16_t*)x) : "Int16";
|
||||
case 8:
|
||||
return x ? CORD_asprintf("%d", *(int8_t*)x) : "Int8";
|
||||
default: errx(1, "Unsupported Int precision: %ld", bits);
|
||||
}
|
||||
}
|
||||
case 'N': {
|
||||
size_t bits = va_arg(args, size_t);
|
||||
switch (bits) {
|
||||
case 64:
|
||||
return x ? CORD_asprintf("%g", *(double*)x) : "Num64";
|
||||
case 32:
|
||||
return x ? CORD_asprintf("%g", *(double*)x) : "Num32";
|
||||
default: errx(1, "Unsupported Num precision: %ld", bits);
|
||||
}
|
||||
}
|
||||
case 'S': {
|
||||
return x ? *(CORD*)x : "Str";
|
||||
}
|
||||
case '[': {
|
||||
if (!x) {
|
||||
CORD cord = CORD_asprintf("[%r]", vas_cord(NULL, fmt, args));
|
||||
if (**fmt == ']')
|
||||
++(*fmt);
|
||||
return cord;
|
||||
}
|
||||
CORD cord = "[";
|
||||
generic_array_t *arr = x;
|
||||
for (int64_t i = 0; i < arr->length; i++) {
|
||||
const char *item_fmt = *fmt;
|
||||
va_list args_copy;
|
||||
va_copy(args_copy, args);
|
||||
if (i > 0) cord = CORD_cat(cord, ", ");
|
||||
CORD item_cord = vas_cord(arr->data + i*arr->stride, &item_fmt, args_copy);
|
||||
cord = CORD_cat(cord, item_cord);
|
||||
va_end(args_copy);
|
||||
}
|
||||
(void)vas_cord(NULL, fmt, args);
|
||||
if (**fmt == ']') ++(*fmt);
|
||||
return CORD_cat(cord, "]");
|
||||
}
|
||||
case '_': {
|
||||
custom_cord_func fn = va_arg(args, custom_cord_func);
|
||||
void *arg = va_arg(args, void*);
|
||||
return fn(x, arg);
|
||||
}
|
||||
default: errx(1, "Unsupported format specifier: '%c'", c);
|
||||
}
|
||||
errx(1, "Unreachable");
|
||||
}
|
||||
|
||||
public CORD as_cord(void *x, const char *fmt, ...)
|
||||
{
|
||||
va_list args;
|
||||
va_start(args, fmt);
|
||||
CORD ret = vas_cord(x, &fmt, args);
|
||||
va_end(args);
|
||||
return ret;
|
||||
}
|
@ -116,7 +116,7 @@ int main(int argc, char *argv[])
|
||||
if (!cflags)
|
||||
cflags = "-std=c11";
|
||||
|
||||
const char *ldlibs = "-lgc -lcord -lm";
|
||||
const char *ldlibs = "-lgc -lcord -lm -L. -lnext";
|
||||
if (getenv("LDLIBS"))
|
||||
ldlibs = heap_strf("%s %s", ldlibs, getenv("LDLIBS"));
|
||||
|
||||
|
@ -30,6 +30,8 @@
|
||||
|
||||
#define __Array(t) array_t
|
||||
|
||||
CORD as_cord(void *x, const char *fmt, ...);
|
||||
|
||||
#define CORD_asprintf(...) ({ CORD __c; CORD_sprintf(&__c, __VA_ARGS__); __c; })
|
||||
#define __declare(var, val) __typeof(val) var = val
|
||||
#define __cord(x) _Generic(x, bool: x ? "yes" : "no", \
|
||||
|
Loading…
Reference in New Issue
Block a user