2014-02-12 14:58:59 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
*
|
2014-08-23 15:43:13 +02:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
2014-02-12 14:58:59 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @addtogroup core_internal
|
|
|
|
* @{
|
|
|
|
*
|
2015-05-22 07:34:41 +02:00
|
|
|
* @file
|
2016-03-09 01:27:23 +01:00
|
|
|
* @brief Common macros and compiler attributes/pragmas configuration
|
2014-02-12 14:58:59 +01:00
|
|
|
*
|
|
|
|
* @author René Kijewski <rene.kijewski@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
2016-03-09 01:27:23 +01:00
|
|
|
#ifndef KERNEL_DEFINES_H_
|
|
|
|
#define KERNEL_DEFINES_H_
|
2014-02-12 14:58:59 +01:00
|
|
|
|
2014-06-03 21:53:15 +02:00
|
|
|
#include <stddef.h>
|
|
|
|
|
2014-10-09 01:18:16 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2016-03-09 01:27:23 +01:00
|
|
|
/**
|
|
|
|
* @def container_of(PTR, TYPE, MEMBER)
|
|
|
|
* @brief Returns the container of a pointer to a member.
|
|
|
|
* @details For a struct `TYPE` with a member `MEMBER`,
|
|
|
|
* given a pointer `PTR` to `TYPE::MEMBER` this function returns a pointer
|
|
|
|
* to the instance of `TYPE`.
|
|
|
|
* @details E.g. for `struct my_struct_t { ...; something_t n; ... } my_struct;`,
|
|
|
|
* `&my_struct == container_of(&my_struct.n, struct my_struct_t, n)`.
|
|
|
|
* @param[in] PTR pointer to a member
|
|
|
|
* @param[in] TYPE a type name (a struct or union), container of PTR
|
|
|
|
* @param[in] MEMBER name of the member of TYPE which PTR points to
|
|
|
|
* @return Pointer to the container of PTR.
|
|
|
|
*/
|
|
|
|
#if __STDC_VERSION__ >= 201112L
|
|
|
|
# define container_of(PTR, TYPE, MEMBER) \
|
|
|
|
(_Generic((PTR), \
|
|
|
|
const __typeof__ (((TYPE *) 0)->MEMBER) *: \
|
|
|
|
((TYPE *) ((char *) (PTR) - offsetof(TYPE, MEMBER))), \
|
|
|
|
__typeof__ (((TYPE *) 0)->MEMBER) *: \
|
|
|
|
((TYPE *) ((char *) (PTR) - offsetof(TYPE, MEMBER))) \
|
|
|
|
))
|
|
|
|
#elif defined __GNUC__
|
|
|
|
# define container_of(PTR, TYPE, MEMBER) \
|
|
|
|
(__extension__ ({ \
|
|
|
|
__extension__ const __typeof__ (((TYPE *) 0)->MEMBER) *__m____ = (PTR); \
|
|
|
|
((TYPE *) ((char *) __m____ - offsetof(TYPE, MEMBER))); \
|
|
|
|
}))
|
|
|
|
#else
|
|
|
|
# define container_of(PTR, TYPE, MEMBER) \
|
|
|
|
((TYPE *) ((char *) (PTR) - offsetof(TYPE, MEMBER)))
|
|
|
|
#endif
|
|
|
|
|
2014-04-07 11:16:32 +02:00
|
|
|
/**
|
|
|
|
* @def NORETURN
|
|
|
|
* @brief The *NORETURN* keyword tells the compiler to assume that the function
|
|
|
|
* cannot return.
|
|
|
|
*/
|
2014-02-12 14:58:59 +01:00
|
|
|
#ifdef __GNUC__
|
|
|
|
#define NORETURN __attribute__((noreturn))
|
|
|
|
#else
|
|
|
|
#define NORETURN
|
|
|
|
#endif
|
|
|
|
|
2014-04-07 11:16:32 +02:00
|
|
|
/**
|
|
|
|
* @def CONST
|
2014-04-30 19:16:20 +02:00
|
|
|
* @brief A function declared as *CONST* is #PURE and also not allowed to
|
|
|
|
* examine global memory. I.e. a *CONST* function cannot even
|
|
|
|
* dereference a pointer parameter.
|
2014-04-07 11:16:32 +02:00
|
|
|
*/
|
2014-02-12 14:58:59 +01:00
|
|
|
#ifdef __GNUC__
|
|
|
|
#define CONST __attribute__((const))
|
|
|
|
#else
|
|
|
|
#define CONST
|
|
|
|
#endif
|
|
|
|
|
2014-04-07 11:16:32 +02:00
|
|
|
/**
|
|
|
|
* @def PURE
|
|
|
|
* @brief The function has no effects except the return value and its return
|
|
|
|
* value depends only on the parameters and/or global variables. Such a
|
|
|
|
* function can be subject to common subexpression elimination and loop
|
|
|
|
* optimization just as an arithmetic operator would be.
|
|
|
|
*/
|
2014-02-12 14:58:59 +01:00
|
|
|
#ifdef __GNUC__
|
|
|
|
#define PURE __attribute__((pure))
|
|
|
|
#else
|
|
|
|
#define PURE
|
|
|
|
#endif
|
|
|
|
|
2014-05-06 17:43:22 +02:00
|
|
|
/**
|
|
|
|
* @def UNREACHABLE()
|
|
|
|
* @brief Tell the compiler that this line of code cannot be reached.
|
|
|
|
* @details Most useful in junction with #NORETURN.
|
|
|
|
* Use this if the compiler cannot tell that e.g.
|
|
|
|
* an assembler instruction causes a longjmp, or a write causes a reboot.
|
|
|
|
*/
|
|
|
|
#if ((__GNUC__ == 4) && (__GNUC_MINOR__ >= 5)) || (__GNUC__ >= 5)
|
|
|
|
#define UNREACHABLE() __builtin_unreachable()
|
|
|
|
#else
|
|
|
|
#define UNREACHABLE() do { /* nothing */ } while (1)
|
|
|
|
#endif
|
|
|
|
|
2014-06-03 21:53:15 +02:00
|
|
|
/**
|
|
|
|
* @def ALIGN_OF(T)
|
|
|
|
* @brief Calculate the minimal alignment for type T.
|
|
|
|
* @param[in] T Type to examine
|
|
|
|
* @returns The minimal alignment of T.
|
|
|
|
*/
|
|
|
|
#define ALIGN_OF(T) (offsetof(struct { char c; T t; }, t))
|
|
|
|
|
2014-10-09 01:18:16 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-03-09 01:27:23 +01:00
|
|
|
#endif /* KERNEL_DEFINES_H_ */
|
2014-04-07 11:16:32 +02:00
|
|
|
/** @} */
|