2018-03-15 15:14:41 +00:00
|
|
|
#ifndef FUSEE_UTILS_H
|
|
|
|
#define FUSEE_UTILS_H
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#define BIT(n) (1u << (n))
|
|
|
|
#define BITL(n) (1ull << (n))
|
|
|
|
#define MASK(n) (BIT(n) - 1)
|
|
|
|
#define MASKL(n) (BITL(n) - 1)
|
|
|
|
#define MASK2(a,b) (MASK(a) & ~MASK(b))
|
|
|
|
#define MASK2L(a,b) (MASKL(a) & ~MASKL(b))
|
|
|
|
|
|
|
|
#define MAKE_REG32(a) (*(volatile uint32_t *)(a))
|
|
|
|
|
|
|
|
#define ALIGN(m) __attribute__((aligned(m)))
|
|
|
|
#define PACKED __attribute__((packed))
|
|
|
|
|
|
|
|
#define ALINLINE __attribute__((always_inline))
|
2018-03-31 22:36:54 +01:00
|
|
|
#define NOINLINE __attribute__((noinline))
|
2018-03-15 15:14:41 +00:00
|
|
|
|
|
|
|
#define SET_SYSREG(reg, val) do { temp_reg = (val); __asm__ __volatile__ ("msr " #reg ", %0" :: "r"(temp_reg) : "memory"); } while(false)
|
|
|
|
|
|
|
|
static inline uintptr_t get_physical_address(const void *addr) {
|
|
|
|
return (uintptr_t)addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t read32le(const volatile void *dword, size_t offset) {
|
|
|
|
uintptr_t addr = (uintptr_t)dword + offset;
|
|
|
|
uint32_t dst;
|
|
|
|
memcpy(&dst, (void *)addr, 4);
|
|
|
|
return dst;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t read32be(const volatile void *dword, size_t offset) {
|
|
|
|
return __builtin_bswap32(read32le(dword, offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint64_t read64le(const volatile void *qword, size_t offset) {
|
|
|
|
uintptr_t addr = (uintptr_t)qword + offset;
|
|
|
|
uint64_t dst;
|
|
|
|
memcpy(&dst, (void *)addr, 8);
|
|
|
|
return dst;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint64_t read64be(const volatile void *qword, size_t offset) {
|
|
|
|
return __builtin_bswap64(read64le(qword, offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void write32le(volatile void *dword, size_t offset, uint32_t value) {
|
|
|
|
memcpy((void *)((uintptr_t)dword + offset), &value, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void write32be(volatile void *dword, size_t offset, uint32_t value) {
|
|
|
|
write32le(dword, offset, __builtin_bswap32(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void write64le(volatile void *qword, size_t offset, uint64_t value) {
|
|
|
|
memcpy((void *)((uintptr_t)qword + offset), &value, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void write64be(volatile void *qword, size_t offset, uint64_t value) {
|
|
|
|
write64le(qword, offset, __builtin_bswap64(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool check_32bit_additive_overflow(uint32_t a, uint32_t b) {
|
|
|
|
return __builtin_add_overflow_p(a, b, (uint32_t)0);
|
|
|
|
}
|
|
|
|
|
2018-04-08 20:51:44 +01:00
|
|
|
__attribute__ ((noreturn)) void panic(uint32_t code);
|
|
|
|
__attribute__ ((noreturn)) void generic_panic(void);
|
|
|
|
__attribute__ ((noreturn)) void panic_predefined(uint32_t which);
|
2018-03-15 15:14:41 +00:00
|
|
|
bool overlaps(uint64_t as, uint64_t ae, uint64_t bs, uint64_t be);
|
|
|
|
|
|
|
|
#endif
|