/* * Copyright (c) 2024 Nordic Semiconductor ASA. * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #ifdef CONFIG_LOG_FRONTEND_STMESP #include #endif #include #include #include #include #include #include #include LOG_MODULE_REGISTER(soc, CONFIG_SOC_LOG_LEVEL); #if defined(NRF_APPLICATION) #define HSFLL_NODE DT_NODELABEL(cpuapp_hsfll) #elif defined(NRF_RADIOCORE) #define HSFLL_NODE DT_NODELABEL(cpurad_hsfll) #endif sys_snode_t soc_node; #define FICR_ADDR_GET(node_id, name) \ DT_REG_ADDR(DT_PHANDLE_BY_NAME(node_id, nordic_ficrs, name)) + \ DT_PHA_BY_NAME(node_id, nordic_ficrs, name, offset) #define SPU_INSTANCE_GET(p_addr) \ ((NRF_SPU_Type *)((p_addr) & (ADDRESS_REGION_Msk | \ ADDRESS_SECURITY_Msk | \ ADDRESS_DOMAIN_Msk | \ ADDRESS_BUS_Msk))) static void power_domain_init(void) { /* * Set: * - LRCCONF010.POWERON.MAIN: 1 * - LRCCONF010.POWERON.ACT: 1 * - LRCCONF010.RETAIN.MAIN: 1 * - LRCCONF010.RETAIN.ACT: 1 * * This is done here at boot so that when the idle routine will hit * WFI the power domain will be correctly retained. */ soc_lrcconf_poweron_request(&soc_node, NRF_LRCCONF_POWER_DOMAIN_0); nrf_lrcconf_poweron_force_set(NRF_LRCCONF010, NRF_LRCCONF_POWER_MAIN, false); nrf_memconf_ramblock_ret_enable_set(NRF_MEMCONF, 0, RAMBLOCK_RET_BIT_ICACHE, false); nrf_memconf_ramblock_ret_enable_set(NRF_MEMCONF, 0, RAMBLOCK_RET_BIT_DCACHE, false); nrf_memconf_ramblock_ret_enable_set(NRF_MEMCONF, 1, RAMBLOCK_RET_BIT_ICACHE, false); nrf_memconf_ramblock_ret_enable_set(NRF_MEMCONF, 1, RAMBLOCK_RET_BIT_DCACHE, false); #if defined(RAMBLOCK_RET2_BIT_ICACHE) nrf_memconf_ramblock_ret2_enable_set(NRF_MEMCONF, 0, RAMBLOCK_RET2_BIT_ICACHE, false); nrf_memconf_ramblock_ret2_enable_set(NRF_MEMCONF, 1, RAMBLOCK_RET2_BIT_ICACHE, false); #endif #if defined(RAMBLOCK_RET2_BIT_DCACHE) nrf_memconf_ramblock_ret2_enable_set(NRF_MEMCONF, 0, RAMBLOCK_RET2_BIT_DCACHE, false); nrf_memconf_ramblock_ret2_enable_set(NRF_MEMCONF, 1, RAMBLOCK_RET2_BIT_DCACHE, false); #endif nrf_memconf_ramblock_ret_mask_enable_set(NRF_MEMCONF, 0, RAMBLOCK_RET_MASK, true); nrf_memconf_ramblock_ret_mask_enable_set(NRF_MEMCONF, 1, RAMBLOCK_RET_MASK, true); #if defined(RAMBLOCK_RET2_MASK) /* * TODO: Use nrf_memconf_ramblock_ret2_mask_enable_set() function * when will be provided by HAL. */ NRF_MEMCONF->POWER[0].RET2 = RAMBLOCK_RET2_MASK; NRF_MEMCONF->POWER[1].RET2 = RAMBLOCK_RET2_MASK; #endif } static int trim_hsfll(void) { #if defined(HSFLL_NODE) NRF_HSFLL_Type *hsfll = (NRF_HSFLL_Type *)DT_REG_ADDR(HSFLL_NODE); nrf_hsfll_trim_t trim = { .vsup = sys_read32(FICR_ADDR_GET(HSFLL_NODE, vsup)), .coarse = sys_read32(FICR_ADDR_GET(HSFLL_NODE, coarse)), .fine = sys_read32(FICR_ADDR_GET(HSFLL_NODE, fine)) }; LOG_DBG("Trim: HSFLL VSUP: 0x%.8x", trim.vsup); LOG_DBG("Trim: HSFLL COARSE: 0x%.8x", trim.coarse); LOG_DBG("Trim: HSFLL FINE: 0x%.8x", trim.fine); nrf_hsfll_clkctrl_mult_set(hsfll, DT_PROP(HSFLL_NODE, clock_frequency) / DT_PROP(DT_CLOCKS_CTLR(HSFLL_NODE), clock_frequency)); nrf_hsfll_trim_set(hsfll, &trim); nrf_hsfll_task_trigger(hsfll, NRF_HSFLL_TASK_FREQ_CHANGE); /* HSFLL task frequency change needs to be triggered twice to take effect.*/ nrf_hsfll_task_trigger(hsfll, NRF_HSFLL_TASK_FREQ_CHANGE); LOG_DBG("NRF_HSFLL->TRIM.VSUP = %d", hsfll->TRIM.VSUP); LOG_DBG("NRF_HSFLL->TRIM.COARSE = %d", hsfll->TRIM.COARSE); LOG_DBG("NRF_HSFLL->TRIM.FINE = %d", hsfll->TRIM.FINE); #endif /* defined(HSFLL_NODE) */ return 0; } #if defined(CONFIG_ARM_ON_ENTER_CPU_IDLE_HOOK) bool z_arm_on_enter_cpu_idle(void) { #ifdef CONFIG_LOG_FRONTEND_STMESP log_frontend_stmesp_pre_sleep(); #endif return true; } #endif static int nordicsemi_nrf54h_init(void) { int err; sys_cache_instr_enable(); sys_cache_data_enable(); power_domain_init(); trim_hsfll(); err = dmm_init(); if (err < 0) { return err; } #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(ccm030)) /* DMASEC is set to non-secure by default, which prevents CCM from * accessing secure memory. Change DMASEC to secure. */ uint32_t ccm030_addr = DT_REG_ADDR(DT_NODELABEL(ccm030)); NRF_SPU_Type *spu = SPU_INSTANCE_GET(ccm030_addr); nrf_spu_periph_perm_dmasec_set(spu, nrf_address_slave_get(ccm030_addr), true); #endif return 0; } void arch_busy_wait(uint32_t time_us) { nrfx_coredep_delay_us(time_us); } SYS_INIT(nordicsemi_nrf54h_init, PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);