mirror of
https://github.com/seL4/seL4.git
synced 2026-03-27 10:29:57 +00:00
Adding a spec with precondition False means verification has to show that the function is not called. Signed-off-by: Gerwin Klein <gerwin.klein@proofcraft.systems>
328 lines
11 KiB
C
328 lines
11 KiB
C
/*
|
|
* Copyright 2014, General Dynamics C4 Systems
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-only
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#define PASTE(a, b) a ## b
|
|
#define _STRINGIFY(a) #a
|
|
#define STRINGIFY(a) _STRINGIFY(a)
|
|
|
|
#ifdef __ASSEMBLER__
|
|
|
|
/* Provide a helper macro to define integer constants that are not of the
|
|
* default type 'int', but 'unsigned long [long]'. When such constants are
|
|
* shared between assembly and C code, some assemblers will fail because they
|
|
* don't support C-style integer suffixes like 'ul'. Using a macro works around
|
|
* this, as the suffix is only applied when the C compiler is used and dropped
|
|
* when the assembler runs.
|
|
*/
|
|
#define UL_CONST(x) x
|
|
#define ULL_CONST(x) x
|
|
#define NULL 0
|
|
|
|
#else /* not __ASSEMBLER__ */
|
|
|
|
/* There is no difference between using 'ul' or 'lu' as suffix for numbers to
|
|
* enforce a specific type besides the default 'int'. Just when it comes to the
|
|
* printf() format specifiers, '%lu' is the only form that is supported. Thus
|
|
* 'ul' is the preferred suffix to avoid confusion.
|
|
*/
|
|
#define UL_CONST(x) PASTE(x, ul)
|
|
#define ULL_CONST(x) PASTE(x, llu)
|
|
#define NULL ((void *)0)
|
|
|
|
#endif /* [not] __ASSEMBLER__ */
|
|
|
|
#define BIT(n) (UL_CONST(1) << (n))
|
|
#define MASK(n) (BIT(n) - UL_CONST(1))
|
|
#define IS_ALIGNED(n, b) (!((n) & MASK(b)))
|
|
#define ROUND_DOWN(n, b) (((n) >> (b)) << (b))
|
|
#define ROUND_UP(n, b) (((((n) - UL_CONST(1)) >> (b)) + UL_CONST(1)) << (b))
|
|
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
|
|
#define MIN(a,b) (((a)<(b))?(a):(b))
|
|
#define MAX(a,b) (((a)>(b))?(a):(b))
|
|
|
|
/* Time constants are defined to use the 'unsigned long long'. Rationale is,
|
|
* that the C rules define the calculation result is determined by largest type
|
|
* involved. Enforcing the largest possible type C provides avoids pitfalls with
|
|
* 32-bit overflows when values are getting quite large. Keep in mind that even
|
|
* 2^32 milli-seconds roll over within 50 days, which is an uptime that embedded
|
|
* systems will reach easily and it resembles not even two months in a calendar
|
|
* calculation. In addition, using the largest integer type C currently defines
|
|
* enforces that all calculations results need a cast back to a 32-bit type
|
|
* explicitly. This might feel annoying, but practically it makes code more
|
|
* robust and enforces thinking about potential overflows.
|
|
* Note that at this stage of the includes, we do not have defined the type
|
|
* uint64_t yet, so we can't use any definitions around it, but have to stick to
|
|
* plain C types. Neither moving the time constant definitions behind the
|
|
* uint64_t type definitions nor including the header with the uint64_t
|
|
* definitions here is currently a feasible option.
|
|
*/
|
|
#define MS_IN_S ULL_CONST(1000)
|
|
#define US_IN_MS ULL_CONST(1000)
|
|
#define HZ_IN_KHZ ULL_CONST(1000)
|
|
#define KHZ_IN_MHZ ULL_CONST(1000)
|
|
#define HZ_IN_MHZ ULL_CONST(1000000)
|
|
|
|
#ifndef __ASSEMBLER__
|
|
|
|
#define PACKED __attribute__((packed))
|
|
#define NORETURN __attribute__((__noreturn__))
|
|
#define CONST __attribute__((__const__))
|
|
#define PURE __attribute__((__pure__))
|
|
#define ALIGN(n) __attribute__((__aligned__(n)))
|
|
#define FASTCALL __attribute__((fastcall))
|
|
#ifdef __clang__
|
|
#define VISIBLE /* nothing */
|
|
#else
|
|
#define VISIBLE __attribute__((externally_visible))
|
|
#endif
|
|
#define NO_INLINE __attribute__((noinline))
|
|
#define FORCE_INLINE __attribute__((always_inline))
|
|
#define SECTION(sec) __attribute__((__section__(sec)))
|
|
#define UNUSED __attribute__((unused))
|
|
#define USED __attribute__((used))
|
|
#ifdef __clang__
|
|
#define FORCE_O2 /* nothing */
|
|
#else
|
|
#define FORCE_O2 __attribute__((optimize("O2")))
|
|
#endif
|
|
/** MODIFIES: */
|
|
/** FNSPEC
|
|
StrictC'__unreachable_spec: "\<Gamma> \<turnstile> {} Call StrictC'__unreachable_'proc UNIV"
|
|
*/
|
|
void __builtin_unreachable(void);
|
|
#define UNREACHABLE() __builtin_unreachable()
|
|
#define MAY_ALIAS __attribute__((may_alias))
|
|
|
|
#define OFFSETOF(type, member) __builtin_offsetof(type, member)
|
|
|
|
#ifdef __GNUC__
|
|
/* Borrowed from linux/include/linux/compiler.h */
|
|
#define likely(x) __builtin_expect(!!(x), 1)
|
|
#define unlikely(x) __builtin_expect(!!(x), 0)
|
|
#else
|
|
#define likely(x) (!!(x))
|
|
#define unlikely(x) (!!(x))
|
|
#endif
|
|
|
|
/* need that for compiling with c99 instead of gnu99 */
|
|
#define asm __asm__
|
|
|
|
/* Wrapping an int literal such as 0 or 1 in a function to prevent it from being
|
|
simplified away in verification while still allowing the compiler to optimise
|
|
it out and do code elimination on the result. */
|
|
static inline int wrap_config_set(int x)
|
|
{
|
|
return x;
|
|
}
|
|
|
|
/* Evaluate a CMake configuration setting at compile-time. */
|
|
#define config_set(macro) wrap_config_set(_is_set_(macro))
|
|
#define _macrotest_1 ,
|
|
#define _is_set_(value) _is_set__(_macrotest_##value)
|
|
#define _is_set__(comma) _is_set___(comma 1, 0)
|
|
#define _is_set___(_, v, ...) v
|
|
|
|
/* Check the existence of a configuration setting, returning one value if it
|
|
* exists and a different one if it does not */
|
|
#define config_ternary(macro, true, false) _config_ternary(macro, true, false)
|
|
#define _config_ternary(value, true, false) _config_ternary_(_macrotest_##value, true, false)
|
|
#define _config_ternary_(comma, true, false) _config_ternary__(comma true, false)
|
|
#define _config_ternary__(_, v, ...) v
|
|
|
|
/** MODIFIES:
|
|
FNSPEC
|
|
halt_spec: "\<Gamma> \<turnstile> {} Call halt_'proc {}"
|
|
*/
|
|
void halt(void) NORETURN;
|
|
void memzero(void *s, unsigned long n);
|
|
void *memset(void *s, unsigned long c, unsigned long n) VISIBLE;
|
|
void *memcpy(void *ptr_dst, const void *ptr_src, unsigned long n) VISIBLE;
|
|
int PURE strncmp(const char *s1, const char *s2, int n);
|
|
long CONST char_to_long(char c);
|
|
long PURE str_to_long(const char *str);
|
|
|
|
/* Library functions for counting leading/trailing zeros.
|
|
*
|
|
* GCC/LLVM provides builtin function like __builtin_clzl() for this, which
|
|
* either get translated to machine specific instructions or calls helper
|
|
* functions like __clzsi2() that a compiler library is expected to implement.
|
|
* At the time of writing this comment, the GCC documentation about the compiler
|
|
* library (https://gcc.gnu.org/onlinedocs/gccint/Integer-library-routines.html)
|
|
* is not very detailed and the signatures given for these helper functions
|
|
* appear incorrect. For example, is says "int __clzsi2(unsigned int a)", but
|
|
* both the GCC and LLVM libraries implement it in a way that is independent of
|
|
* the implementation choices for the sizes of `unsigned int`. Instead, it
|
|
* appears that `si` always signifies a 32-bit argument and `di` always
|
|
* signifies a 64-bit argument. Tests with __builtin_clzl() on RISC-V have shown
|
|
* that if 'unsigned long' is 32 bits __builtin_clzl() uses __clzsi2() and if
|
|
* the type is 64 bits __builtin_clzl() uses __clzdi2(). Thus using the types
|
|
* uint32_t and uint64_t from stdint.h in the signatures below is considered the
|
|
* semantically correct way.
|
|
* Note that we only emit actual function implementations for these functions if
|
|
* CONFIG_CLZ_32 etc. are set. Otherwise, the compiler's internal implementation
|
|
* may get used or compilation fails if there is no machine instruction.
|
|
*/
|
|
#include <stdint.h>
|
|
CONST int __clzsi2(uint32_t x);
|
|
CONST int __clzdi2(uint64_t x);
|
|
CONST int __ctzsi2(uint32_t x);
|
|
CONST int __ctzdi2(uint64_t x);
|
|
|
|
// Used for compile-time constants, so should always use the builtin.
|
|
#define CTZL(x) __builtin_ctzl(x)
|
|
|
|
// Count leading zeros.
|
|
// The CONFIG_CLZ_NO_BUILTIN macro may be used to expose the library function
|
|
// to the C parser for verification.
|
|
#ifndef CONFIG_CLZ_NO_BUILTIN
|
|
// If we use a compiler builtin, we cannot verify it, so we use the following
|
|
// annotations to hide the function body from the proofs, and axiomatise its
|
|
// behaviour.
|
|
// On the other hand, if we use our own implementation instead of the builtin,
|
|
// then we want to expose that implementation to the proofs, and therefore hide
|
|
// these annotations.
|
|
/** MODIFIES: */
|
|
/** DONT_TRANSLATE */
|
|
/** FNSPEC clzl_spec:
|
|
"\<forall>s. \<Gamma> \<turnstile>
|
|
{\<sigma>. s = \<sigma> \<and> x___unsigned_long_' s \<noteq> 0 }
|
|
\<acute>ret__long :== PROC clzl(\<acute>x)
|
|
\<lbrace> \<acute>ret__long = of_nat (word_clz (x___unsigned_long_' s)) \<rbrace>"
|
|
*/
|
|
#endif
|
|
static inline long
|
|
CONST clzl(unsigned long x)
|
|
{
|
|
#ifdef CONFIG_CLZ_NO_BUILTIN
|
|
#if CONFIG_WORD_SIZE == 32
|
|
return __clzsi2(x);
|
|
#else
|
|
return __clzdi2(x);
|
|
#endif
|
|
#else
|
|
return __builtin_clzl(x);
|
|
#endif
|
|
}
|
|
|
|
#ifndef CONFIG_CLZ_NO_BUILTIN
|
|
// See comments on clzl.
|
|
/** MODIFIES: */
|
|
/** DONT_TRANSLATE */
|
|
/** FNSPEC clzll_spec:
|
|
"\<forall>s. \<Gamma> \<turnstile>
|
|
{\<sigma>. s = \<sigma> \<and> x___unsigned_longlong_' s \<noteq> 0 }
|
|
\<acute>ret__longlong :== PROC clzll(\<acute>x)
|
|
\<lbrace> \<acute>ret__longlong = of_nat (word_clz (x___unsigned_longlong_' s)) \<rbrace>"
|
|
*/
|
|
#endif
|
|
static inline long long
|
|
CONST clzll(unsigned long long x)
|
|
{
|
|
#ifdef CONFIG_CLZ_NO_BUILTIN
|
|
return __clzdi2(x);
|
|
#else
|
|
return __builtin_clzll(x);
|
|
#endif
|
|
}
|
|
|
|
// Count trailing zeros.
|
|
#ifndef CONFIG_CTZ_NO_BUILTIN
|
|
// See comments on clzl.
|
|
/** MODIFIES: */
|
|
/** DONT_TRANSLATE */
|
|
/** FNSPEC ctzl_spec:
|
|
"\<forall>s. \<Gamma> \<turnstile>
|
|
{\<sigma>. s = \<sigma> \<and> x___unsigned_long_' s \<noteq> 0 }
|
|
\<acute>ret__long :== PROC ctzl(\<acute>x)
|
|
\<lbrace> \<acute>ret__long = of_nat (word_ctz (x___unsigned_long_' s)) \<rbrace>"
|
|
*/
|
|
#endif
|
|
static inline long
|
|
CONST ctzl(unsigned long x)
|
|
{
|
|
#ifdef CONFIG_CTZ_NO_BUILTIN
|
|
// If there is a builtin CLZ, but no builtin CTZ, then CTZ will be implemented
|
|
// using the builtin CLZ, rather than the long-form implementation.
|
|
// This is typically the fastest way to calculate ctzl on such platforms.
|
|
#ifdef CONFIG_CLZ_NO_BUILTIN
|
|
// Here, there are no builtins we can use, so call the library function.
|
|
#if CONFIG_WORD_SIZE == 32
|
|
return __ctzsi2(x);
|
|
#else
|
|
return __ctzdi2(x);
|
|
#endif
|
|
#else
|
|
// Here, we have __builtin_clzl, but no __builtin_ctzl.
|
|
if (unlikely(x == 0)) {
|
|
return 8 * sizeof(unsigned long);
|
|
}
|
|
// -x = ~x + 1, so (x & -x) isolates the least significant 1-bit of x,
|
|
// allowing ctzl to be calculated from clzl and the word size.
|
|
return 8 * sizeof(unsigned long) - 1 - __builtin_clzl(x & -x);
|
|
#endif
|
|
#else
|
|
// Here, we have __builtin_ctzl.
|
|
return __builtin_ctzl(x);
|
|
#endif
|
|
}
|
|
|
|
#ifndef CONFIG_CTZ_NO_BUILTIN
|
|
// See comments on clzl.
|
|
/** MODIFIES: */
|
|
/** DONT_TRANSLATE */
|
|
/** FNSPEC ctzll_spec:
|
|
"\<forall>s. \<Gamma> \<turnstile>
|
|
{\<sigma>. s = \<sigma> \<and> x___unsigned_longlong_' s \<noteq> 0 }
|
|
\<acute>ret__longlong :== PROC ctzll(\<acute>x)
|
|
\<lbrace> \<acute>ret__longlong = of_nat (word_ctz (x___unsigned_longlong_' s)) \<rbrace>"
|
|
*/
|
|
#endif
|
|
static inline long long
|
|
CONST ctzll(unsigned long long x)
|
|
{
|
|
#ifdef CONFIG_CTZ_NO_BUILTIN
|
|
// See comments on ctzl.
|
|
#ifdef CONFIG_CLZ_NO_BUILTIN
|
|
return __ctzdi2(x);
|
|
#else
|
|
if (unlikely(x == 0)) {
|
|
return 8 * sizeof(unsigned long long);
|
|
}
|
|
// See comments on ctzl.
|
|
return 8 * sizeof(unsigned long long) - 1 - __builtin_clzll(x & -x);
|
|
#endif
|
|
#else
|
|
return __builtin_ctzll(x);
|
|
#endif
|
|
}
|
|
|
|
int __builtin_popcountl(unsigned long x);
|
|
|
|
/** DONT_TRANSLATE */
|
|
static inline long
|
|
CONST popcountl(unsigned long mask)
|
|
{
|
|
#ifndef __POPCNT__
|
|
unsigned int count; // c accumulates the total bits set in v
|
|
for (count = 0; mask; count++) {
|
|
mask &= mask - 1; // clear the least significant bit set
|
|
}
|
|
|
|
return count;
|
|
#else
|
|
return __builtin_popcountl(mask);
|
|
#endif
|
|
}
|
|
|
|
#define POPCOUNTL(x) popcountl(x)
|
|
|
|
/* Can be used to insert padding to the next L1 cache line boundary */
|
|
#define PAD_TO_NEXT_CACHE_LN(used) char padding[L1_CACHE_LINE_SIZE - ((used) % L1_CACHE_LINE_SIZE)]
|
|
|
|
#endif /* !__ASSEMBLER__ */
|