#include "hdr/stdint_proxy.h"
#include "src/__support/macros/config.h"
#include "src/stdlib/atexit.h"
#include "src/stdlib/exit.h"
#include "src/string/memcpy.h"
#include "src/string/memset.h"
#include "startup/baremetal/fini.h"
#include "startup/baremetal/init.h"
#include <arm_acle.h>
extern "C" {
int main(int argc, char **argv);
void _start();
[[gnu::weak]] void _platform_init() {}
extern uintptr_t __stack;
extern uintptr_t __data_source[];
extern uintptr_t __data_start[];
extern uintptr_t __data_size[];
extern uintptr_t __bss_start[];
extern uintptr_t __bss_size[];
}
namespace {
#if __ARM_ARCH_PROFILE == 'M'
void NMI_Handler() {}
void HardFault_Handler() { LIBC_NAMESPACE::exit(1); }
void MemManage_Handler() { LIBC_NAMESPACE::exit(1); }
void BusFault_Handler() { LIBC_NAMESPACE::exit(1); }
void UsageFault_Handler() { LIBC_NAMESPACE::exit(1); }
void SVC_Handler() {}
void DebugMon_Handler() {}
void PendSV_Handler() {}
void SysTick_Handler() {}
using HandlerType = void (*)(void);
[[gnu::section(".vectors"), gnu::aligned(1024), gnu::used]]
const HandlerType vector_table[] = {
reinterpret_cast<HandlerType>(&__stack), _start, NMI_Handler, HardFault_Handler, MemManage_Handler, BusFault_Handler, UsageFault_Handler, 0, 0, 0, 0, SVC_Handler, DebugMon_Handler, 0, PendSV_Handler, SysTick_Handler, };
#else
void Reset_Handler() { LIBC_NAMESPACE::exit(1); }
void Undefined_Handler() { LIBC_NAMESPACE::exit(1); }
void SWI_Handler() { LIBC_NAMESPACE::exit(1); }
void PrefetchAbort_Handler() { LIBC_NAMESPACE::exit(1); }
void DataAbort_Handler() { LIBC_NAMESPACE::exit(1); }
void IRQ_Handler() { LIBC_NAMESPACE::exit(1); }
void FIQ_Handler() { LIBC_NAMESPACE::exit(1); }
[[gnu::section(".vectors"), gnu::aligned(32), gnu::used, gnu::naked,
gnu::target("arm")]]
void vector_table() {
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm("LDR pc, [pc, #24]");
asm(".word %c0" : : "X"(Reset_Handler));
asm(".word %c0" : : "X"(Undefined_Handler));
asm(".word %c0" : : "X"(SWI_Handler));
asm(".word %c0" : : "X"(PrefetchAbort_Handler));
asm(".word %c0" : : "X"(DataAbort_Handler));
asm(".word %c0" : : "X"(0));
asm(".word %c0" : : "X"(IRQ_Handler));
asm(".word %c0" : : "X"(FIQ_Handler));
}
#endif
}
namespace LIBC_NAMESPACE_DECL {
[[noreturn]] void do_start() {
#if __ARM_ARCH_PROFILE == 'A' || __ARM_ARCH_PROFILE == 'R'
__arm_wsr("CPSR_c", 0x11); asm volatile("mov sp, %0" : : "r"(__builtin_frame_address(0)));
__arm_wsr("CPSR_c", 0x12); asm volatile("mov sp, %0" : : "r"(__builtin_frame_address(0)));
__arm_wsr("CPSR_c", 0x17); asm volatile("mov sp, %0" : : "r"(__builtin_frame_address(0)));
__arm_wsr("CPSR_c", 0x1B); asm volatile("mov sp, %0" : : "r"(__builtin_frame_address(0)));
__arm_wsr("CPSR_c", 0x1F); asm volatile("mov sp, %0" : : "r"(__builtin_frame_address(0)));
__arm_wsr("CPSR_c", 0x13); #endif
#if __ARM_ARCH_PROFILE == 'M' && \
(defined(__ARM_FP) || defined(__ARM_FEATURE_MVE))
auto cpacr = reinterpret_cast<volatile uint32_t *const>(0xE000ED88);
*cpacr |= (0xF << 20);
__dsb(0xF);
__isb(0xF);
#if defined(__ARM_FEATURE_MVE)
uint32_t fpscr;
__asm__ __volatile__("vmrs %0, FPSCR" : "=r"(fpscr) : :);
fpscr |= (0x4 << 16);
__asm__ __volatile__("vmsr FPSCR, %0" : : "r"(fpscr) :);
#endif
#elif (__ARM_ARCH_PROFILE == 'A' || __ARM_ARCH_PROFILE == 'R') && \
defined(__ARM_FP)
uint32_t cpacr = __arm_rsr("p15:0:c1:c0:2");
cpacr |= (0xF << 20);
__arm_wsr("p15:0:c1:c0:2", cpacr);
__isb(0xF);
uint32_t fpexc;
__asm__ __volatile__("vmrs %0, FPEXC" : "=r"(fpexc) : :);
fpexc |= (0x1 << 30);
__asm__ __volatile__("vmsr FPEXC, %0" : : "r"(fpexc) :);
#endif
LIBC_NAMESPACE::memcpy(__data_start, __data_source,
reinterpret_cast<uintptr_t>(__data_size));
LIBC_NAMESPACE::memset(__bss_start, '\0',
reinterpret_cast<uintptr_t>(__bss_size));
__libc_init_array();
_platform_init();
LIBC_NAMESPACE::atexit(&__libc_fini_array);
LIBC_NAMESPACE::exit(main(0, 0));
}
}
extern "C" {
#ifdef __ARM_ARCH_ISA_ARM
[[gnu::naked, gnu::target("arm")]]
#endif
void _start() {
asm volatile("mov sp, %0" : : "r"(&__stack));
asm volatile("bl %0" : : "X"(LIBC_NAMESPACE::do_start));
}
}