2019-09-04 15:11:45 +10:00
|
|
|
/*
|
|
|
|
|
* Copyright © 2008-2011 Kristian Høgsberg
|
|
|
|
|
* Copyright © 2011 Intel Corporation
|
|
|
|
|
* Copyright © 2013-2015 Red Hat, Inc.
|
|
|
|
|
*
|
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
|
|
|
* to deal in the Software without restriction, including without limitation
|
|
|
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
|
|
|
*
|
|
|
|
|
* The above copyright notice and this permission notice (including the next
|
|
|
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
|
|
|
* Software.
|
|
|
|
|
*
|
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
|
* DEALINGS IN THE SOFTWARE.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
2025-04-07 07:35:25 +10:00
|
|
|
#ifndef HAVE_C23_AUTO
|
|
|
|
|
#define auto __auto_type
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-09-04 15:11:45 +10:00
|
|
|
#define ARRAY_LENGTH(a) (sizeof (a) / sizeof (a)[0])
|
2022-03-07 14:44:21 +10:00
|
|
|
/**
|
|
|
|
|
* Iterate through the array _arr, assigning the variable elem to each
|
|
|
|
|
* element. elem only exists within the loop.
|
|
|
|
|
*/
|
2019-09-04 15:11:45 +10:00
|
|
|
#define ARRAY_FOR_EACH(_arr, _elem) \
|
2022-03-07 14:44:21 +10:00
|
|
|
for (__typeof__((_arr)[0]) *_elem = _arr; \
|
|
|
|
|
_elem < (_arr) + ARRAY_LENGTH(_arr); \
|
|
|
|
|
_elem++)
|
2019-09-04 15:11:45 +10:00
|
|
|
|
|
|
|
|
#define min(a, b) (((a) < (b)) ? (a) : (b))
|
|
|
|
|
#define max(a, b) (((a) > (b)) ? (a) : (b))
|
|
|
|
|
|
|
|
|
|
#define ANSI_HIGHLIGHT "\x1B[0;1;39m"
|
|
|
|
|
#define ANSI_RED "\x1B[0;31m"
|
|
|
|
|
#define ANSI_GREEN "\x1B[0;32m"
|
|
|
|
|
#define ANSI_YELLOW "\x1B[0;33m"
|
|
|
|
|
#define ANSI_BLUE "\x1B[0;34m"
|
|
|
|
|
#define ANSI_MAGENTA "\x1B[0;35m"
|
|
|
|
|
#define ANSI_CYAN "\x1B[0;36m"
|
2025-06-18 13:14:35 +10:00
|
|
|
#define ANSI_WHITE "\x1B[0;37m"
|
2019-09-04 15:11:45 +10:00
|
|
|
#define ANSI_BRIGHT_RED "\x1B[0;31;1m"
|
|
|
|
|
#define ANSI_BRIGHT_GREEN "\x1B[0;32;1m"
|
|
|
|
|
#define ANSI_BRIGHT_YELLOW "\x1B[0;33;1m"
|
|
|
|
|
#define ANSI_BRIGHT_BLUE "\x1B[0;34;1m"
|
|
|
|
|
#define ANSI_BRIGHT_MAGENTA "\x1B[0;35;1m"
|
|
|
|
|
#define ANSI_BRIGHT_CYAN "\x1B[0;36;1m"
|
2025-06-18 13:14:35 +10:00
|
|
|
#define ANSI_BRIGHT_WHITE "\x1B[0;37;1m"
|
2019-09-04 15:11:45 +10:00
|
|
|
#define ANSI_NORMAL "\x1B[0m"
|
|
|
|
|
|
2019-11-05 16:17:01 +10:00
|
|
|
#define ANSI_UP "\x1B[%dA"
|
|
|
|
|
#define ANSI_DOWN "\x1B[%dB"
|
|
|
|
|
#define ANSI_RIGHT "\x1B[%dC"
|
|
|
|
|
#define ANSI_LEFT "\x1B[%dD"
|
|
|
|
|
|
2025-06-11 15:05:44 +10:00
|
|
|
#define ANSI_RGB(r, g, b) "\x1B[38;2;" #r ";" #g ";" #b "m"
|
|
|
|
|
#define ANSI_RGB_BG(r, g, b) "\x1B[48;2;" #r ";" #g ";" #b "m"
|
|
|
|
|
|
2019-09-04 15:11:45 +10:00
|
|
|
#define CASE_RETURN_STRING(a) case a: return #a
|
2021-07-22 15:00:32 +10:00
|
|
|
|
2024-10-17 19:25:39 +10:00
|
|
|
/**
|
|
|
|
|
* Concatenate two macro args into one, e.g.:
|
|
|
|
|
* int CONCAT(foo_, __LINE__);
|
|
|
|
|
* will produce:
|
|
|
|
|
* int foo_123;
|
|
|
|
|
*/
|
|
|
|
|
#define CONCAT2(X,Y) X##Y
|
|
|
|
|
#define CONCAT(X,Y) CONCAT2(X,Y)
|
|
|
|
|
|
2025-04-07 15:18:55 +10:00
|
|
|
#define _unused_ __attribute__((unused))
|
2021-07-22 15:00:32 +10:00
|
|
|
#define _fallthrough_ __attribute__((fallthrough))
|
2025-02-19 13:44:00 +10:00
|
|
|
|
|
|
|
|
/* Returns the number of macro arguments, this expands
|
|
|
|
|
* _VARIABLE_MACRO_NARGS(a, b, c) to NTH_ARG(a, b, c, 15, 14, 13, .... 4, 3, 2, 1).
|
|
|
|
|
* _VARIABLE_MACRO_NTH_ARG always returns the 16th argument which in our case is 3.
|
|
|
|
|
*
|
|
|
|
|
* If we want more than 16 values _VARIABLE_MACRO_COUNTDOWN and
|
|
|
|
|
* _VARIABLE_MACRO_NTH_ARG both need to be updated.
|
|
|
|
|
*/
|
|
|
|
|
#define _VARIABLE_MACRO_NARGS(...) _VARIABLE_MACRO_NARGS1(__VA_ARGS__, _VARIABLE_MACRO_COUNTDOWN)
|
|
|
|
|
#define _VARIABLE_MACRO_NARGS1(...) _VARIABLE_MACRO_NTH_ARG(__VA_ARGS__)
|
|
|
|
|
|
|
|
|
|
/* Add to this if we need more than 16 args */
|
|
|
|
|
#define _VARIABLE_MACRO_COUNTDOWN \
|
|
|
|
|
15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0
|
|
|
|
|
|
|
|
|
|
/* Return the 16th argument passed in. See _VARIABLE_MACRO_NARGS above for usage.
|
|
|
|
|
* Note this is 1-indexed.
|
|
|
|
|
*/
|
|
|
|
|
#define _VARIABLE_MACRO_NTH_ARG( \
|
|
|
|
|
_1, _2, _3, _4, _5, _6, _7, _8, \
|
|
|
|
|
_9, _10, _11, _12, _13, _14, _15,\
|
|
|
|
|
N, ...) N
|
|
|
|
|
|
|
|
|
|
/* Defines a different expansion of macros depending on the
|
|
|
|
|
* number of arguments, e.g. it turns
|
|
|
|
|
* VARIABLE_MACRO(_ARG, a, b, c) into _ARG3(a, b, c)
|
|
|
|
|
*
|
|
|
|
|
* This can be used to have custom macros that expand to different things
|
|
|
|
|
* depending on the number of arguments. This example converts a
|
|
|
|
|
* single macro argument into value + stringify, two arguments into
|
|
|
|
|
* first and second argument.
|
|
|
|
|
*
|
|
|
|
|
* #define _ARG1(_1) _1, #_1,
|
|
|
|
|
* #define _ARG2(_1, _2) _1, _2,
|
|
|
|
|
*
|
|
|
|
|
* #define MYMACRO(...) _VARIABLE_MACRO(_ARG, __VA_ARGS__)
|
|
|
|
|
*
|
|
|
|
|
* static void foo(int value, char *name) { printf("%d: %s\n", value, name); }
|
|
|
|
|
*
|
|
|
|
|
* int main(void) {
|
|
|
|
|
* foo(MYMACRO(0)); // prints "0: 0"
|
|
|
|
|
* foo(MYMACRO(0, "zero")); // prints "0: zero"
|
|
|
|
|
* return 0;
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* The first argument to VARIABLE_MACRO defines the prefix of the
|
|
|
|
|
* expander macros (here _ARG -> _ARG0, _ARG1, ...). These need to be defined
|
|
|
|
|
* in the caller for the number of arguments accepted
|
|
|
|
|
* (up to _VARIABLE_MACRO_COUNTDOWN args).
|
|
|
|
|
*/
|
|
|
|
|
#define _VARIABLE_MACRO(func, ...) CONCAT(func, _VARIABLE_MACRO_NARGS(__VA_ARGS__)) (__VA_ARGS__)
|