gdjn  Artifact [70ed0e36b0]

Artifact 70ed0e36b0b42f09c4c5852dd955fd215fec418d143d071711849fd7eabaefa1:


/* [ʞ] util.h
 *  ~ lexi hale <lexi@hale.su>
 *  🄯 AGPLv3
 *  ? encapsulate annoying operations (read: pitiful, fragile blast
 *    shield over the most indefensibly psychotic pieces of the godot
 *    "type" "system")
 *
 *    if you want to use this outside gdjn, redefine the macro _t
 *    from gdjn.h appropriately.
 *
 *    (honestly tho you should use c-bind-gen.janet too)
 */

#pragma once
#include "gdjn.h"
#include <string.h>

static inline gd_string
gdu_string_of_stringName(const gd_stringName* const s) {
	gd_string r;
	_t(string).fromStringName(&r, (void*)&s);
	return r;
}

static inline gd_stringName
gdu_stringName_of_string(const gd_stringName* const s) {
	gd_stringName r;
	_t(stringName).fromString(&r, (void*)&s);
	return r;
}

static inline gd_stringName
gdu_intern_sz (const char* str, const size_t sz) {
	gd_stringName r = {};
	if (sz == 0) _t(stringName).newWithUtf8Chars(&r, str);
	        else _t(stringName).newWithUtf8CharsAndLen(&r, str, sz);
	return r;
}

static inline gd_stringName
gdu_intern (const char* str) {
	return gdu_intern_sz(str, 0);
}

static inline gd_stringName
gdu_sym_null(void) {
	gd_stringName n;
	_t(stringName).empty(&n, nullptr);
	return n;
}

static inline gd_string
gdu_str_null(void) {
	gd_string n;
	_t(string).empty(&n, nullptr);
	return n;
}

static inline gd_string
gdu_str_sz (const char* str, const size_t sz) {
	gd_string r = {};
	if (sz == 0) _t(string).newWithUtf8Chars(&r, str);
	        else _t(string).newWithUtf8CharsAndLen(&r, str, sz);
	return r;
}

static inline gd_string
gdu_str (const char* str) {
	return gdu_str_sz(str, 0);
}

#define _gdu_intern(x) (gdu_intern_sz((x), sizeof(x)-1))
#define  _litSz(x) (x), (sizeof (x)-1)
#define    _ref(x) typeof(typeof(x) const* const)
#define _refMut(x) typeof(typeof(x)      * const)
#define _with(T, k, v, ...) ({\
	typeof(gd_##T) k = v; \
	do { __VA_ARGS__; } while (0); \
	_t(T).dtor(&k); \
})

#define _withSym(k, v, ...) \
	_with(stringName, k, gdu_intern(v), __VA_ARGS__)
#define _withSym0(k, ...) \
	_with(stringName, k, {}, __VA_ARGS__)
#define _withStr(k, v, ...) \
	_with(string, k, gdu_str(v), __VA_ARGS__)
#define _withStr0(k, v, ...) \
	_with(string, k, {}, __VA_ARGS__)

#define _typeEq(a, b) \
	__builtin_classify_type(typeof(a)) == _builtin_classify_type(typeof(b))

#define _refVal   0
#define _refPtr   1
#define _refArray 2

#define _indirect(a) \
		__builtin_types_compatible_p(typeof(a), void*)

#define _refKind(a) \
	__builtin_choose( _typeEq(typeof_unqual(a), \
	                          typeof_unqual(a[0]) []), _refArray \
		/* false */ __builtin_choose(_typeEq(a, (typeof_unqual(a[0]) *)), _refPtr ))

#define _szElse(a, zero) \
	__builtin_choose(_refKind(a) == _refArray, \
		/* true */ _sz(a), \
		/* false */ __builtin_choose(_refKind(a) == _refPtr, \
			/* true */ (__builtin_counted_by(ptr) != nullptr ? \
			            *__builtin_counted_by(ptr) : (zero)) \
			/* false */ (void)0 /* bad type */ ))
#define _sz0(a)   _szElse(a,0)
#define _szStr(a) \
	__builtin_choose(_typeEq((a), pstr), (a).sz, _szElse(a, strlen(a)))

#define _array(t) struct {t* v; size_t sz;}
typedef _array(char) pstr;

#define _strWithSz(a) (a), _strLen(a)

static inline bool
gdu_symIs
(	gd_stringName const* const a,
	gd_stringName const* const b
) {
	bool res;
	_t(stringName).equal(a, b, &res);
	return res;
}

static inline bool
gdu_symEq_sz
(	_ref(gd_stringName) a,
	_ref(char)          b,
	size_t const        sz
) {
	auto bSym = gdu_intern_sz(b,sz);
	bool res  = gdu_symIs(a, &bSym);
	_t(stringName).dtor(&bSym);
	return res;
}

static inline bool
gdu_symEq
(	_ref(gd_stringName) a,
	_ref(char)          b
) { return gdu_symEq_sz(a, b, 0); }

#define _gdu_symEq(a,b) (gdu_symEq_sz(a, _litSz(b)))

static inline bool
gdu_objIs
(	GDExtensionObjectPtr obj,
	_ref(char)           id,
	size_t const         sz
) {
	bool res = false;
	_withSym0(name, ({
		if (!_t(object).getClassName(obj, gdjn_ctx -> gd.lib, &name))
			break;
		res = gdu_symEq_sz(&name, id, sz);
	}));
	return res;
}

static inline bool
gdu_strIs
(	gd_string const* const a,
	gd_string const* const b
) {
	bool res;
	_t(string).equal(a, b, &res);
	return res;
}

static inline bool
gdu_strEq_sz
(	_ref(gd_string) a,
	_ref(char)      b,
	size_t const    sz
) {
	auto bSym = gdu_str_sz(b,sz);
	bool res  = gdu_strIs(a, &bSym);
	_t(string).dtor(&bSym);
	return res;
}

static inline bool
gdu_strEq
(	_ref(gd_string) a,
	_ref(char)      b
) { return gdu_strEq_sz(a, b, 0); }

#define _gdu_symEq(a,b) (gdu_symEq_sz(a, _litSz(b)))
#define _gdu_strEq(a,b) (gdu_strEq_sz(a, _litSz(b)))
#define _gdu_objIs(obj, id) \
	(gdu_objIs(obj, _litSz(#id)))

#define _gdu_string_emit(s, tgt) \
	(gdu_string_emit(&(s), (tgt), sizeof (tgt)-1))

static inline size_t
gdu_string_emit
(	size_t sz;

	const gd_string* const s,
	char                   target[static sz],
	size_t                 sz
) {
	/* docs lie btw, this returns a count of bytes,
	 * not "characters" (??)
	 * (thank the gods for small mercies) */
	size_t len = _t(string).toUtf8Chars(s, target, sz);
	target[len] = 0;
	return len;
}

#define _gdu_stringName_emit(s, tgt) \
	(gdu_stringName_emit(&(s), (tgt), sizeof (tgt) - 1))

static inline pstr
gdu_string_pdup (_ref(gd_string) s) {
	size_t len = gd_string_length(s) + 1;
	char* buf = _alloc(char, len);
	gdu_string_emit(s, buf, len - 1);
	return (pstr){buf,len};
}

static inline gd_string
gdu_string_dup (_ref(gd_string) s) {
	gd_string cp;
	_t(string).copy(&cp, (void const*[]) {s});
	return cp;
}

#define _cat(x,y) x##y
#define __cat(x,y) _cat(x,y)
#define _gensym __cat(_sym_,__COUNTER__)

#define __gdu_string_auto(szid, name, str) \
	size_t szid = gd_string_length(str) + 1; \
	char[szid] name; \
	gdu_string_emit(str, name, szid-1); 

#define _gdu_string_auto(...) __gdu_string_auto(_gensym, __VA_ARGS__)

#if __has_builtin(__builtin_alloca_with_align)
#	define _stalloc(ty, n) \
		(__builtin_alloca_with_align(sizeof(ty)*(n), alignof(ty)*8))
#else
#	define _stalloc(ty, n) \
		(__builtin_alloca(sizeof(ty)*(n)))
#endif

#define _gdu_gstr_stack(ty, str) ({ \
	size_t sz = gd_##ty##_length(str) + 1; \
	char* buf = _stalloc(char, sz);        \
	gdu_##ty##_emit(str, buf, sz - 1);     \
	buf; \
})
#define _gdu_gstr_stackp(ty, str) ({ \
	size_t sz = gd_##ty##_length(str) + 1; \
	char* buf = _stalloc(char, sz);        \
	gdu_##ty##_emit(str, buf, sz - 1);     \
	(pstr) {.v = buf, .sz = sz}; \
})

#define _gdu_string_stack(str)      _gdu_gstr_stack(string, str)
#define _gdu_stringName_stack(str)  _gdu_gstr_stack(stringName, str)
#define _gdu_string_stackp(str)     _gdu_gstr_stackp(string, str)
#define _gdu_stringName_stackp(str) _gdu_gstr_stackp(stringName, str)


static inline size_t
gdu_stringName_emit
(	size_t sz;

	_ref(gd_stringName) s,
	char                target[static sz],
	size_t              sz
) {
	gd_string r;
	_t(string).fromStringName(&r, (void*)&s);
	const auto len = gdu_string_emit(&r, target, sz);
	_t(string).dtor(&r);
	return len;
}

#define _gdu_packedArray_push_def(name, T, input, fn) \
	static inline bool \
	gdu_array_##name   \
	(	gd_packed##T##Array*         self,\
		const typeof(input)* const   arg  \
	) {\
		bool ret;\
		auto c = &gdjn_ctx -> gd.t; \
		(c -> gd_packed##T##Array.fn) ( \
			self,\
			(GDExtensionConstTypePtr[]) { \
				arg,\
			}, &ret, 1\
		);\
		return ret;\
	}

#define _gdu_packedArrayTypes \
	_(Byte,    byte,    uint8_t   ) \
	_(Int32,   int32,   int32_t   ) \
	_(Int64,   int64,   int64_t   ) \
	_(Float32, float32, int32_t   ) \
	_(Float64, float64, int64_t   ) \
	_(String,  string,  gd_string ) \
	_(Vector2, vector2, gd_vector2) \
	_(Vector3, vector3, gd_vector3) \
	_(Vector4, vector4, gd_vector4) \

#define _gdu_packedArray_defs(maj, min, input) \
	_gdu_packedArray_push_def(min##_##push,   maj, input, append) \
	_gdu_packedArray_push_def(min##_##concat, maj, gd_packed##maj##Array, append_array)
/* bool gdu_array_(type)_push(self, type)
 * bool gdu_array_(type)_concat(self, packedArray[type])
 */

#define _(...) _gdu_packedArray_defs(__VA_ARGS__)
	_gdu_packedArrayTypes
#undef _

/* obnoxious special case */
static inline bool
gdu_array_string_pushPtr
(	gd_packedStringArray* self,
	const char* const     str,
	size_t                sz
) {
	gd_string tmp;
	if (sz == 0) _t(string).newWithUtf8Chars      (&tmp, str);
	else         _t(string).newWithUtf8CharsAndLen(&tmp, str, sz);
	bool ret = gdu_array_string_push(self, &tmp);
	_t(string).dtor(&tmp);
	return ret;
}
#define _gdu_array_string_pushLit(self, str) \
	(gdu_array_string_pushPtr((self), (str), sizeof (str) - 1))

static inline bool
gdu_string_suffix
(	_ref(gd_string) self,
	_ref(char)      affix,
	size_t          affsz
) {
	auto ch = _gdu_string_stackp(self);
	if (affsz == 0) affsz = strlen(affix);
	if (ch.sz < affsz) return false;
	for (size_t i = 0; i < affsz; ++i) {
		auto a =  ch.v[ch.sz - 2 - i];
		auto b = affix[affsz - 1 - i];
		if (a != b) return false;
	}
	return true;
}

static inline void*
gdu_classTag(_ref(char) name) {
	void* tag = nullptr;
	_withSym(sName, name,
		tag = _t(classdb).getClassTag(&sName);
	);
	return tag;
}
static inline GDExtensionObjectPtr
gdu_cast
(	GDExtensionConstObjectPtr what,
	_ref(char)                to
) {
	return _t(object).castTo(what, gdu_classTag(to));
}