diff options
| author | Bruce Hill <bruce@bruce-hill.com> | 2024-02-18 14:28:35 -0500 |
|---|---|---|
| committer | Bruce Hill <bruce@bruce-hill.com> | 2024-02-18 14:28:35 -0500 |
| commit | 095970170ca3ef4a91970e02494f8056ca47c2c9 (patch) | |
| tree | 93d9de38f0610da4e80b1b2338cbe38f9505ec96 /builtins/array.h | |
| parent | 88261bcbbd92cd78a4c068ae58906e36e26fd02a (diff) | |
Implement indexing
Diffstat (limited to 'builtins/array.h')
| -rw-r--r-- | builtins/array.h | 10 |
1 files changed, 8 insertions, 2 deletions
diff --git a/builtins/array.h b/builtins/array.h index 7c83e17d..8884f794 100644 --- a/builtins/array.h +++ b/builtins/array.h @@ -8,8 +8,14 @@ #include "types.h" // Convert negative indices to back-indexed without branching: index0 = index + (index < 0)*(len+1)) - 1 -#define $index(x, i) _Generic(x, array_t: ({ __typeof(x) $obj; int64_t $offset = i; $offset += ($offset < 0) * ($obj.length + 1) - 1; assert($offset >= 0 && offset < $obj.length); $obj.data + $obj.stride * $offset;})) -#define $safe_index(x, i) _Generic(x, array_t: ({ __typeof(x) $obj; int64_t $offset = i - 1; $obj.data + $obj.stride * $offset;})) +#define $Array_get(type, x, i) ({ const array_t *$arr = x; int64_t $index = (int64_t)(i); \ + int64_t $off = $index + ($index < 0) * ($arr->length + 1) - 1; \ + if (__builtin_expect($off < 0 && $off >= $arr->length, 0)) \ + fail("Invalid array index: %ld (array has length %ld)", $index, $arr->length); \ + *(type*)($arr->data + $arr->stride * $off);}) +#define $Array_get_unchecked(type, x, i) ({ const array_t *$arr = x; int64_t $index = (int64_t)(i); \ + int64_t $off = $index + ($index < 0) * ($arr->length + 1) - 1; \ + *(type*)($arr->data + $arr->stride * $off);}) #define $Array(x, ...) ({ __typeof(x) $items[] = {x, __VA_ARGS__}; \ (array_t){.length=sizeof($items)/sizeof($items[0]), \ .stride=(int64_t)&$items[1] - (int64_t)&$items[0], \ |
