#ifndef INTERNAL_STRUCT_H /*-*-C-*-vi:se ft=c:*/ #define INTERNAL_STRUCT_H /** * @author Ruby developers * @copyright This file is a part of the programming language Ruby. * Permission is hereby granted, to either redistribute and/or * modify this file, provided that the conditions mentioned in the * file COPYING are met. Consult the file for details. * @brief Internal header for Struct. */ #include "ruby/internal/stdbool.h" /* for bool */ #include "ruby/ruby.h" /* for struct RBasic */ enum { RSTRUCT_EMBED_LEN_MASK = RUBY_FL_USER7 | RUBY_FL_USER6 | RUBY_FL_USER5 | RUBY_FL_USER4 | RUBY_FL_USER3 | RUBY_FL_USER2 | RUBY_FL_USER1, RSTRUCT_EMBED_LEN_SHIFT = (RUBY_FL_USHIFT+1), }; struct RStruct { struct RBasic basic; union { struct { long len; const VALUE *ptr; } heap; /* This is a length 1 array because: * 1. GCC has a bug that does not optimize C flexible array members * (https://gcc.gnu.org/bugzilla/show_bug.cgi?id=102452) * 2. Zero length arrays are not supported by all compilers */ const VALUE ary[1]; } as; }; #define RSTRUCT(obj) ((struct RStruct *)(obj)) #ifdef RSTRUCT_LEN # undef RSTRUCT_LEN #endif #ifdef RSTRUCT_PTR # undef RSTRUCT_PTR #endif #ifdef RSTRUCT_SET # undef RSTRUCT_SET #endif #ifdef RSTRUCT_GET # undef RSTRUCT_GET #endif #define RSTRUCT_LEN internal_RSTRUCT_LEN #define RSTRUCT_SET internal_RSTRUCT_SET #define RSTRUCT_GET internal_RSTRUCT_GET /* struct.c */ VALUE rb_struct_init_copy(VALUE copy, VALUE s); VALUE rb_struct_lookup(VALUE s, VALUE idx); VALUE rb_struct_s_keyword_init(VALUE klass); static inline const VALUE *rb_struct_const_heap_ptr(VALUE st); static inline long RSTRUCT_EMBED_LEN(VALUE st); static inline long RSTRUCT_LEN(VALUE st); static inline int RSTRUCT_LENINT(VALUE st); static inline const VALUE *RSTRUCT_CONST_PTR(VALUE st); static inline void RSTRUCT_SET(VALUE st, long k, VALUE v); static inline VALUE RSTRUCT_GET(VALUE st, long k); static inline long RSTRUCT_EMBED_LEN(VALUE st) { long ret = FL_TEST_RAW(st, RSTRUCT_EMBED_LEN_MASK); ret >>= RSTRUCT_EMBED_LEN_SHIFT; return ret; } static inline long RSTRUCT_LEN(VALUE st) { if (FL_TEST_RAW(st, RSTRUCT_EMBED_LEN_MASK)) { return RSTRUCT_EMBED_LEN(st); } else { return RSTRUCT(st)->as.heap.len; } } static inline int RSTRUCT_LENINT(VALUE st) { return rb_long2int(RSTRUCT_LEN(st)); } static inline const VALUE * RSTRUCT_CONST_PTR(VALUE st) { const struct RStruct *p = RSTRUCT(st); if (FL_TEST_RAW(st, RSTRUCT_EMBED_LEN_MASK)) { return p->as.ary; } else { return p->as.heap.ptr; } } static inline void RSTRUCT_SET(VALUE st, long k, VALUE v) { RB_OBJ_WRITE(st, &RSTRUCT_CONST_PTR(st)[k], v); } static inline VALUE RSTRUCT_GET(VALUE st, long k) { return RSTRUCT_CONST_PTR(st)[k]; } static inline const VALUE * rb_struct_const_heap_ptr(VALUE st) { assert(!FL_TEST_RAW(st, RSTRUCT_EMBED_LEN_MASK)); return RSTRUCT(st)->as.heap.ptr; } #endif /* INTERNAL_STRUCT_H */