2019-07-17 00:49:47 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2019 Atmosphère-NX
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "types.h"
|
2019-07-23 03:47:57 +01:00
|
|
|
#include "preprocessor.h"
|
2019-12-23 20:12:02 +00:00
|
|
|
#include "debug_log.h"
|
2019-07-17 00:49:47 +01:00
|
|
|
|
|
|
|
#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))
|
|
|
|
|
2019-07-19 00:45:56 +01:00
|
|
|
#define MAKE_REG32(a) (*(volatile u32 *)(uintptr_t)(a))
|
2019-07-17 00:49:47 +01:00
|
|
|
|
|
|
|
#define ALIGN(m) __attribute__((aligned(m)))
|
|
|
|
#define PACKED __attribute__((packed))
|
2020-01-07 02:17:05 +00:00
|
|
|
#define LIKELY(expr) __builtin_expect((expr), 1)
|
|
|
|
#define UNLIKELY(expr) __builtin_expect((expr), 0)
|
2019-07-17 00:49:47 +01:00
|
|
|
|
|
|
|
#define ALINLINE __attribute__((always_inline))
|
|
|
|
|
2019-08-06 21:26:28 +01:00
|
|
|
#define TEMPORARY __attribute__((section(".tempbss")))
|
|
|
|
|
2020-01-05 21:54:02 +00:00
|
|
|
#define PANIC(...) do { DEBUG(__VA_ARGS__); panic(); } while (false)
|
|
|
|
|
2019-12-23 20:12:02 +00:00
|
|
|
#define FOREACH_BIT(tmpmsk, var, word) for (u64 tmpmsk = (word), var = __builtin_ctzll(word); tmpmsk != 0; tmpmsk &= ~BITL(var), var = __builtin_ctzll(tmpmsk))
|
2019-08-17 23:40:47 +01:00
|
|
|
|
2020-01-05 21:54:02 +00:00
|
|
|
#define _DECLARE_ASM_ARITHMETIC_UNARY_HELPER(sz, regalloc, op)\
|
|
|
|
static inline u##sz __##op##sz(u##sz n)\
|
|
|
|
{\
|
|
|
|
u##sz res;\
|
2020-01-09 19:24:05 +00:00
|
|
|
__asm__ __volatile__ (STRINGIZE(op) " %" STRINGIZE(regalloc) "[res], %" STRINGIZE(regalloc) "[n]" : [res] "=r" (res) : [n] "r" (n));\
|
2020-01-05 21:54:02 +00:00
|
|
|
return res;\
|
|
|
|
}
|
|
|
|
|
|
|
|
#define _DECLARE_ASM_ARITHMETIC_UNARY_HELPER64(op) _DECLARE_ASM_ARITHMETIC_UNARY_HELPER(64, x, op)
|
|
|
|
#define _DECLARE_ASM_ARITHMETIC_UNARY_HELPER32(op) _DECLARE_ASM_ARITHMETIC_UNARY_HELPER(32, w, op)
|
|
|
|
|
|
|
|
_DECLARE_ASM_ARITHMETIC_UNARY_HELPER64(rbit)
|
|
|
|
_DECLARE_ASM_ARITHMETIC_UNARY_HELPER32(rbit)
|
2020-01-02 18:41:52 +00:00
|
|
|
|
2020-01-09 22:47:22 +00:00
|
|
|
static inline void __dmb_sy(void)
|
|
|
|
{
|
|
|
|
__asm__ __volatile__ ("dmb sy" ::: "memory");
|
|
|
|
}
|
|
|
|
|
2019-08-06 21:26:28 +01:00
|
|
|
static inline void __dsb_sy(void)
|
|
|
|
{
|
|
|
|
__asm__ __volatile__ ("dsb sy" ::: "memory");
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void __isb(void)
|
|
|
|
{
|
|
|
|
__asm__ __volatile__ ("isb" ::: "memory");
|
|
|
|
}
|
2019-07-17 00:49:47 +01:00
|
|
|
|
2019-08-08 20:38:13 +01:00
|
|
|
static inline void __tlb_invalidate_el1_stage12(void)
|
|
|
|
{
|
|
|
|
__asm__ __volatile__ ("tlbi alle1" ::: "memory");
|
|
|
|
}
|
|
|
|
|
2019-08-02 04:12:24 +01:00
|
|
|
bool overlaps(u64 as, u64 ae, u64 bs, u64 be);
|
2019-07-17 00:49:47 +01:00
|
|
|
|
2019-08-08 20:38:13 +01:00
|
|
|
static inline uintptr_t get_physical_address_el1_stage12(bool *valid, const uintptr_t el1_vaddr) {
|
2019-08-02 04:12:24 +01:00
|
|
|
// NOTE: interrupt must be disabled when calling this func
|
2019-07-17 00:49:47 +01:00
|
|
|
uintptr_t PAR;
|
2019-08-08 20:38:13 +01:00
|
|
|
__asm__ __volatile__ ("at s12e1r, %0" :: "r"(el1_vaddr)); // note: we don't care whether it's writable in EL1&0 translation regime
|
2019-07-17 00:49:47 +01:00
|
|
|
__asm__ __volatile__ ("mrs %0, par_el1" : "=r"(PAR));
|
2019-08-08 20:38:13 +01:00
|
|
|
*valid = (PAR & 1) == 0ull;
|
2019-08-02 04:12:24 +01:00
|
|
|
return (PAR & 1) ? 0ull : (PAR & MASK2L(40, 12)) | ((uintptr_t)el1_vaddr & MASKL(12));
|
2019-07-17 00:49:47 +01:00
|
|
|
}
|
|
|
|
|
2019-08-08 20:38:13 +01:00
|
|
|
bool readEl1Memory(void *dst, uintptr_t addr, size_t size);
|
|
|
|
bool writeEl1Memory(uintptr_t addr, const void *src, size_t size);
|
|
|
|
|
2019-07-22 23:54:35 +01:00
|
|
|
static inline void panic(void) {
|
2020-01-02 18:41:52 +00:00
|
|
|
#ifndef PLATFORM_QEMU
|
2019-07-31 01:30:17 +01:00
|
|
|
__builtin_trap();
|
2020-01-02 18:41:52 +00:00
|
|
|
#endif
|
2019-07-22 23:54:35 +01:00
|
|
|
for (;;);
|
|
|
|
}
|