`extern "C"` is not a valid language linkage for declaring `int main(...)`, as per the ISO C++ Standard Specification. This fixes the violations of -Wmain, and brings Zephyr closer to valid C++. See the C++ standard wording here: https://eel.is/c++draft/basic.start.main#3.sentence-5 See also the clang warning -Wmain: https://clang.llvm.org/docs/DiagnosticsReference.html#wmain However, for freestanding code (i.e. with -ffreestanding), main has no special meaning, and we need to prevent name mangling. So allow this forward linkage when not hosted. This only applies to C++ as these linkage declarations only exist inside __cplusplus guards. Signed-off-by: Jordan R Abrahams-Whitehead <ajordanr@google.com>
74 lines
2.3 KiB
C
74 lines
2.3 KiB
C
/*
|
|
* Copyright (c) 2017 Linaro Limited
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#ifndef ZEPHYR_INCLUDE_ZEPHYR_TYPES_H_
|
|
#define ZEPHYR_INCLUDE_ZEPHYR_TYPES_H_
|
|
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/*
|
|
* A type with strong alignment requirements, similar to C11 max_align_t. It can
|
|
* be used to force alignment of data structures allocated on the stack or as
|
|
* return * type for heap allocators.
|
|
*/
|
|
typedef union {
|
|
long long thelonglong;
|
|
long double thelongdouble;
|
|
uintmax_t theuintmax_t;
|
|
size_t thesize_t;
|
|
uintptr_t theuintptr_t;
|
|
void *thepvoid;
|
|
void (*thepfunc)(void);
|
|
} z_max_align_t;
|
|
|
|
/*
|
|
* Thread local variables are declared with different keywords depending on
|
|
* which C/C++ standard that is used. C++11 and C23 uses "thread_local" whilst
|
|
* C11 uses "_Thread_local". Previously the GNU "__thread" keyword was used
|
|
* which is the same in both gcc and g++.
|
|
*/
|
|
#ifndef Z_THREAD_LOCAL
|
|
#if defined(__cplusplus) && (__cplusplus) >= 201103L /* C++11 */
|
|
#define Z_THREAD_LOCAL thread_local
|
|
#elif defined(__STDC_VERSION__) && (__STDC_VERSION__) >= 202311L /* C23 */
|
|
#define Z_THREAD_LOCAL thread_local
|
|
#elif defined(__STDC_VERSION__) && (__STDC_VERSION__) >= 201112L /* C11 */
|
|
#define Z_THREAD_LOCAL _Thread_local
|
|
#else /* Default back to old behavior which used the GNU keyword. */
|
|
#define Z_THREAD_LOCAL __thread
|
|
#endif
|
|
#endif /* Z_THREAD_LOCAL */
|
|
|
|
#ifdef __cplusplus
|
|
|
|
#if (!__STDC_HOSTED__)
|
|
/*
|
|
* Zephyr requires an int main(void) signature with C linkage for the
|
|
* application main if present. gcc, and clang when building in 'hosted' mode
|
|
* will correctly assume this. But, when building freestanding, clang does not
|
|
* treat main() specially, and by default name mangles its symbol, which
|
|
* results in the linker not linking from the kernel init code into this
|
|
* name mangled app main().
|
|
*
|
|
* At the same time, according to the C++ standard Section 6.9.3.1 of
|
|
* ISO/IEC 14882:2024, main cannot be explicitly declared to have "C" linkage.
|
|
* This restriction is relaxed for freestanding code, as main is not treated
|
|
* specially in these circumstances.
|
|
* Therefore, let's include the prototype when we are not building the code as
|
|
* freestanding/not-hosted.
|
|
*/
|
|
extern int main(void);
|
|
#endif
|
|
|
|
}
|
|
#endif
|
|
|
|
#endif /* ZEPHYR_INCLUDE_ZEPHYR_TYPES_H_ */
|