diff options
| author | kr.angelov <kr.angelov@gmail.com> | 2012-01-20 13:41:10 +0000 |
|---|---|---|
| committer | kr.angelov <kr.angelov@gmail.com> | 2012-01-20 13:41:10 +0000 |
| commit | 2eee382a62a909d5a3f2f5eda94f30fe68fd5335 (patch) | |
| tree | b0b0d513535895f244214aebf6358e172b8dce6d /src/runtime/c/gu/variant.h | |
| parent | b9728357126f8b9a6311cca17d9f0dcc2a7bfb9b (diff) | |
initial import of the C runtime
Diffstat (limited to 'src/runtime/c/gu/variant.h')
| -rw-r--r-- | src/runtime/c/gu/variant.h | 167 |
1 files changed, 167 insertions, 0 deletions
diff --git a/src/runtime/c/gu/variant.h b/src/runtime/c/gu/variant.h new file mode 100644 index 000000000..a9bb10a4a --- /dev/null +++ b/src/runtime/c/gu/variant.h @@ -0,0 +1,167 @@ +/* + * Copyright 2010 University of Helsinki. + * + * This file is part of libgu. + * + * Libgu is free software: you can redistribute it and/or modify it under + * the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, either version 3 of the License, or (at your + * option) any later version. + * + * Libgu is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public + * License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with libgu. If not, see <http://www.gnu.org/licenses/>. + */ + +/** @file + * + * Lightweight tagged data. + */ + +#ifndef GU_VARIANT_H_ +#define GU_VARIANT_H_ + +#include <gu/defs.h> +#include <gu/mem.h> +#include <gu/type.h> + +/** @name Variants + * @{ + */ + +typedef struct GuVariant GuVariant; + + +void* gu_alloc_variant(uint8_t tag, + size_t size, size_t align, + GuVariant* variant_out, GuPool* pool); + +GuVariant gu_make_variant(uint8_t tag, + size_t size, size_t align, + const void* init, GuPool* pool); + +#define gu_new_variant(tag, type, variant_out, pool) \ + ((type*)gu_alloc_variant(tag, sizeof(type), \ + gu_alignof(type), variant_out, pool)) + +/**< + * @hideinitializer */ + +#define gu_new_variant_i(POOL, TAG, T, ...) \ + gu_make_variant(TAG, sizeof(T), gu_alignof(T), \ + &(T){ __VA_ARGS__ }, POOL) + + + +#define gu_new_flex_variant(tag, type, flex_mem, n_elems, variant_out, pool) \ + ((type*)gu_alloc_variant(tag, \ + GU_FLEX_SIZE(type, flex_mem, n_elems), \ + gu_flex_alignof(type), \ + variant_out, pool)) +/**< + * @hideinitializer */ + +enum { + GU_VARIANT_NULL = -1 +}; + +int gu_variant_tag(GuVariant variant); + +void* gu_variant_data(GuVariant variant); + + +typedef struct GuVariantInfo GuVariantInfo; + +struct GuVariantInfo { + int tag; + void* data; +}; + +GuVariantInfo gu_variant_open(GuVariant variant); + +/** @privatesection */ +struct GuVariant { + uintptr_t p; + /**< @private */ +}; + +/** @} */ + +static inline void* +gu_variant_to_ptr(GuVariant variant) +{ + return (void*)variant.p; +} + +static inline GuVariant +gu_variant_from_ptr(const void* p) +{ + GuVariant v = { (uintptr_t)p }; + return v; +} + +extern const GuVariant gu_null_variant; + +static inline bool +gu_variant_is_null(GuVariant v) { + return ((void*)v.p == NULL); +} + + +// variant + +typedef const struct GuConstructor GuConstructor; + +struct GuConstructor { + int c_tag; + const char* c_name; + const GuType* type; +}; + +#define GU_CONSTRUCTOR_V(ctag, c_type) { \ + .c_tag = ctag, \ + .c_name = #ctag, \ + .type = c_type \ +} + +#define GU_CONSTRUCTOR(ctag, t_) \ + GU_CONSTRUCTOR_V(ctag, gu_type(t_)) + +#define GU_CONSTRUCTOR_P(ctag, t_) \ + GU_CONSTRUCTOR_V(ctag, gu_ptr_type(t_)) + +#define GU_CONSTRUCTOR_S(ctag, t_, ...) \ + GU_CONSTRUCTOR_V(ctag, GU_TYPE_LIT(struct, t_, __VA_ARGS__)) + +#define GU_CONSTRUCTOR_S1(ctag, t_, mem1_, type1_) \ + GU_CONSTRUCTOR_S(ctag, t_, \ + GU_MEMBER(t_, mem1_, type1_)) + +#define GU_CONSTRUCTOR_S2(ctag, t_, mem1_, type1_, mem2_, type2_) \ + GU_CONSTRUCTOR_S(ctag, t_, \ + GU_MEMBER(t_, mem1_, type1_), \ + GU_MEMBER(t_, mem2_, type2_)) + + + +typedef GuSList(GuConstructor) GuConstructors; + +typedef const struct GuVariantType GuVariantType, GuType_GuVariant; + +struct GuVariantType { + GuType_repr repr_base; + GuConstructors ctors; +}; + +#define GU_TYPE_INIT_GuVariant(k_, t_, ...) { \ + .repr_base = GU_TYPE_INIT_repr(k_, GuVariant, _), \ + .ctors = GU_SLIST(GuConstructor, __VA_ARGS__) \ +} + +extern GU_DECLARE_KIND(GuVariant); + +#endif // GU_VARIANT_H_ |
