Andrew Scull | 1883487 | 2018-10-12 11:48:09 +0100 | [diff] [blame] | 1 | /* |
Andrew Walbran | 692b325 | 2019-03-07 15:51:31 +0000 | [diff] [blame] | 2 | * Copyright 2018 The Hafnium Authors. |
Andrew Scull | 1883487 | 2018-10-12 11:48:09 +0100 | [diff] [blame] | 3 | * |
Andrew Walbran | e959ec1 | 2020-06-17 15:01:09 +0100 | [diff] [blame] | 4 | * Use of this source code is governed by a BSD-style |
| 5 | * license that can be found in the LICENSE file or at |
| 6 | * https://opensource.org/licenses/BSD-3-Clause. |
Andrew Scull | 1883487 | 2018-10-12 11:48:09 +0100 | [diff] [blame] | 7 | */ |
| 8 | |
Andrew Scull | fbc938a | 2018-08-20 14:09:28 +0100 | [diff] [blame] | 9 | #pragma once |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 10 | |
| 11 | #include <stddef.h> |
| 12 | #include <stdint.h> |
| 13 | |
Andrew Scull | 6386f25 | 2018-12-06 13:29:10 +0000 | [diff] [blame] | 14 | #define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0])) |
| 15 | |
Karl Meakin | 2ad6b66 | 2024-07-29 20:45:40 +0100 | [diff] [blame] | 16 | /* NOLINTNEXTLINE(readability-redundant-declaration) */ |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 17 | int memcmp(const void *a, const void *b, size_t n); |
| 18 | |
Karl Meakin | 2ad6b66 | 2024-07-29 20:45:40 +0100 | [diff] [blame] | 19 | /* NOLINTNEXTLINE(readability-redundant-declaration) */ |
Andrew Scull | 3c351e9 | 2020-01-28 11:26:05 +0000 | [diff] [blame] | 20 | int strncmp(const char *a, const char *b, size_t n); |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 21 | |
Andrew Walbran | 318f573 | 2018-11-20 16:23:42 +0000 | [diff] [blame] | 22 | #define ctz(x) __builtin_ctz(x) |
| 23 | |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 24 | /** |
| 25 | * Check whether the value `v` is aligned to the boundary `a`, |
| 26 | * with `a` power of 2. |
| 27 | */ |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 28 | #define is_aligned(v, a) __builtin_is_aligned((v), (a)) |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 29 | |
| 30 | /** |
| 31 | * Align up the value `v` to the boundary `a`, with `a` power of 2. |
| 32 | */ |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 33 | #define align_up(v, a) __builtin_align_up((v), (a)) |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 34 | |
| 35 | /** |
| 36 | * Align down the value `v` to the boundary `a`, with `a` power of 2. |
| 37 | */ |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 38 | #define align_down(v, a) __builtin_align_down((v), (a)) |
Alfredo Mazzinghi | eb1997c | 2019-02-07 18:00:01 +0000 | [diff] [blame] | 39 | |
J-Alves | 586f1fb | 2024-11-13 11:53:54 +0000 | [diff] [blame^] | 40 | /* |
| 41 | * Add operation together with checking whether the operation overflowed |
| 42 | * The result is '*res', |
| 43 | * return false on success and true on overflow |
| 44 | */ |
| 45 | #define add_overflow(a, b, res) __builtin_add_overflow((a), (b), (res)) |
| 46 | |
| 47 | /* |
| 48 | * Round up a value to align with a given size and |
| 49 | * check whether overflow happens. |
| 50 | * The rounded value is '*res', return false on success and true on overflow. |
| 51 | */ |
| 52 | #define align_up_overflow(v, size, res) \ |
| 53 | __extension__({ \ |
| 54 | typedef __typeof(v) v_t; \ |
| 55 | v_t __v = (v); \ |
| 56 | __typeof(size) __size = (size); \ |
| 57 | __typeof(res) __res = (res); \ |
| 58 | *__res = align_up(__v, __size); \ |
| 59 | __v > *__res; \ |
| 60 | }) |
| 61 | |
| 62 | /* |
| 63 | * Add a with b, then round up the result to align with a given size and |
| 64 | * check whether overflow happens. |
| 65 | * The rounded value is '*res', return false on success and true on overflow. |
| 66 | */ |
| 67 | #define add_with_round_up_overflow(a, b, size, res) \ |
| 68 | (__extension__({ \ |
| 69 | __typeof__(a) __add_res = 0; \ |
| 70 | add_overflow((a), (b), &__add_res) || \ |
| 71 | align_up_overflow(__add_res, (size), (res)); \ |
| 72 | })) |
| 73 | |
Andrew Walbran | 4a53ba6 | 2019-03-05 17:26:12 +0000 | [diff] [blame] | 74 | #ifndef be16toh |
Andrew Scull | f12b35d | 2018-07-16 12:12:59 +0100 | [diff] [blame] | 75 | #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 76 | |
Andrew Scull | f12b35d | 2018-07-16 12:12:59 +0100 | [diff] [blame] | 77 | #define be16toh(v) __builtin_bswap16(v) |
| 78 | #define be32toh(v) __builtin_bswap32(v) |
| 79 | #define be64toh(v) __builtin_bswap64(v) |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 80 | |
Andrew Scull | f12b35d | 2018-07-16 12:12:59 +0100 | [diff] [blame] | 81 | #define htobe16(v) __builtin_bswap16(v) |
| 82 | #define htobe32(v) __builtin_bswap32(v) |
| 83 | #define htobe64(v) __builtin_bswap64(v) |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 84 | |
Andrew Scull | f12b35d | 2018-07-16 12:12:59 +0100 | [diff] [blame] | 85 | #define le16toh(v) (v) |
| 86 | #define le32toh(v) (v) |
| 87 | #define le64toh(v) (v) |
Wedson Almeida Filho | 987c0ff | 2018-06-20 16:34:38 +0100 | [diff] [blame] | 88 | |
Andrew Scull | f12b35d | 2018-07-16 12:12:59 +0100 | [diff] [blame] | 89 | #define htole16(v) (v) |
| 90 | #define htole32(v) (v) |
| 91 | #define htole64(v) (v) |
| 92 | |
| 93 | #elif __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__ |
| 94 | |
| 95 | #define be16toh(v) (v) |
| 96 | #define be32toh(v) (v) |
| 97 | #define be64toh(v) (v) |
| 98 | |
| 99 | #define htobe16(v) (v) |
| 100 | #define htobe32(v) (v) |
| 101 | #define htobe64(v) (v) |
| 102 | |
| 103 | #define le16toh(v) __builtin_bswap16(v) |
| 104 | #define le32toh(v) __builtin_bswap32(v) |
| 105 | #define le64toh(v) __builtin_bswap64(v) |
| 106 | |
| 107 | #define htole16(v) __builtin_bswap16(v) |
| 108 | #define htole32(v) __builtin_bswap32(v) |
| 109 | #define htole64(v) __builtin_bswap64(v) |
| 110 | |
Andrew Scull | cbefbdb | 2019-01-11 16:36:26 +0000 | [diff] [blame] | 111 | #else |
| 112 | |
| 113 | /* |
| 114 | * __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__ && |
| 115 | * __BYTE_ORDER__ != __ORDER_BIG_ENDIAN__ |
| 116 | */ |
Andrew Scull | f12b35d | 2018-07-16 12:12:59 +0100 | [diff] [blame] | 117 | |
| 118 | #error "Unsupported byte order" |
| 119 | |
| 120 | #endif |
Andrew Walbran | 4a53ba6 | 2019-03-05 17:26:12 +0000 | [diff] [blame] | 121 | #endif |