This is an automated email from the ASF dual-hosted git repository.

xiaoxiang pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/nuttx.git

commit d250808c1c088d665e456feef92e35f37e658f5b
Author: Tiago Medicci Serrano <tiago.medi...@espressif.com>
AuthorDate: Thu Aug 14 14:33:03 2025 -0300

    esp32s3/elf: Fix ELF loader on ESP32-S3 when using external PSRAM
    
    Prior to this commit, it wasn't possible to load ELF modules from
    the external PSRAM. There were two main issues about it: 1) copying
    data using the instruction bus was being used instead of the data
    bus (this, per si, isn't a problem, but requires special attention
    regarding data alignment), and 2) the cache was not being properly
    cleaned and flushed to properly access the loaded data using the
    instruction bus.
    
    Signed-off-by: Tiago Medicci Serrano <tiago.medi...@espressif.com>
---
 arch/xtensa/Kconfig                                |   2 +
 arch/xtensa/src/esp32s3/Make.defs                  |   2 +-
 arch/xtensa/src/esp32s3/esp32s3_cache.c            | 572 +++++++++++++++++++++
 arch/xtensa/src/esp32s3/esp32s3_textheap.c         |   6 +-
 arch/xtensa/src/esp32s3/hal.mk                     |   4 +-
 .../esp32s3/common/scripts/esp32s3_rom_aliases.ld  |   9 +
 libs/libc/elf/elf_bind.c                           |   1 +
 libs/libc/machine/xtensa/arch_elf.c                |   2 +-
 8 files changed, 591 insertions(+), 7 deletions(-)

diff --git a/arch/xtensa/Kconfig b/arch/xtensa/Kconfig
index d7cb000b154..bc148aec073 100644
--- a/arch/xtensa/Kconfig
+++ b/arch/xtensa/Kconfig
@@ -89,6 +89,8 @@ config ARCH_CHIP_ESP32S3
        select ARCH_HAVE_TEXT_HEAP_SEPARATE_DATA_ADDRESS
        select ARCH_HAVE_TEXT_HEAP_WORD_ALIGNED_READ
        select ARCH_HAVE_TESTSET
+       select ARCH_DCACHE
+       select ARCH_ICACHE
        select ARCH_VECNOTIRQ
        select LIBC_PREVENT_STRING_KERNEL
        select LIBC_ARCH_MEMCPY if BUILD_FLAT
diff --git a/arch/xtensa/src/esp32s3/Make.defs 
b/arch/xtensa/src/esp32s3/Make.defs
index a145f48bb0a..e6a140d31cd 100644
--- a/arch/xtensa/src/esp32s3/Make.defs
+++ b/arch/xtensa/src/esp32s3/Make.defs
@@ -28,7 +28,7 @@ HEAD_CSRC  = esp32s3_start.c
 
 # Required ESP32-S3 files (arch/xtensa/src/esp32s3)
 
-CHIP_CSRCS  = esp32s3_irq.c esp32s3_clockconfig.c esp32s3_region.c
+CHIP_CSRCS  = esp32s3_cache.c esp32s3_irq.c esp32s3_clockconfig.c 
esp32s3_region.c
 CHIP_CSRCS += esp32s3_systemreset.c esp32s3_user.c esp32s3_allocateheap.c 
esp32s3_reset_reasons.c
 CHIP_CSRCS += esp32s3_wdt.c esp32s3_gpio.c esp32s3_lowputc.c esp32s3_serial.c
 CHIP_CSRCS += esp32s3_rtc_gpio.c esp32s3_libc_stubs.c esp32s3_spi_timing.c
diff --git a/arch/xtensa/src/esp32s3/esp32s3_cache.c 
b/arch/xtensa/src/esp32s3/esp32s3_cache.c
new file mode 100644
index 00000000000..78426ebf97d
--- /dev/null
+++ b/arch/xtensa/src/esp32s3/esp32s3_cache.c
@@ -0,0 +1,572 @@
+/****************************************************************************
+ * arch/xtensa/src/esp32s3/esp32s3_cache.c
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements.  See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.  The
+ * ASF licenses this file to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance with the
+ * License.  You may obtain a copy of the License at
+ *
+ *   http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
+ * License for the specific language governing permissions and limitations
+ * under the License.
+ *
+ ****************************************************************************/
+
+/****************************************************************************
+ * Included Files
+ ****************************************************************************/
+
+#include <nuttx/config.h>
+#include <nuttx/cache.h>
+
+/****************************************************************************
+ * Pre-processor Definitions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Inline Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * ROM Function Prototypes
+ ****************************************************************************/
+
+extern uint32_t cache_get_icache_line_size(void);
+extern void cache_resume_icache(uint32_t autoload);
+extern uint32_t cache_suspend_icache(void);
+extern void cache_invalidate_icache_items(uint32_t addr, uint32_t items);
+extern void cache_invalidate_icache_all(void);
+extern void cache_lock_icache_items(uint32_t addr, uint32_t items);
+extern void cache_unlock_icache_items(uint32_t addr, uint32_t items);
+
+extern uint32_t cache_get_dcache_line_size(void);
+extern void cache_resume_dcache(uint32_t autoload);
+extern uint32_t cache_suspend_dcache(void);
+extern void cache_invalidate_dcache_items(uint32_t addr, uint32_t items);
+extern void cache_invalidate_dcache_all(void);
+extern int cache_writeback_addr(uint32_t addr, uint32_t size);
+extern void cache_writeback_all(void);
+extern void cache_lock_dcache_items(uint32_t addr, uint32_t items);
+extern void cache_unlock_dcache_items(uint32_t addr, uint32_t items);
+
+/****************************************************************************
+ * Public Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: up_get_icache_linesize
+ *
+ * Description:
+ *   Get icache linesize
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   Cache line size
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE
+size_t up_get_icache_linesize(void)
+{
+  return (size_t)cache_get_icache_line_size();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_get_icache_size
+ *
+ * Description:
+ *   Get icache size
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   Cache size
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE
+size_t up_get_icache_size(void)
+{
+  return (size_t)CONFIG_ESP32S3_INSTRUCTION_CACHE_SIZE;
+}
+#endif
+
+/****************************************************************************
+ * Name: up_enable_icache
+ *
+ * Description:
+ *   Enable the I-Cache
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ * Caution:
+ *   The writable global variables aren't initialized yet.
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE
+void up_enable_icache(void)
+{
+  cache_resume_icache(0);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_disable_icache
+ *
+ * Description:
+ *   Disable the I-Cache
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE
+void up_disable_icache(void)
+{
+  cache_suspend_icache();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_invalidate_icache
+ *
+ * Description:
+ *   Invalidate the instruction cache within the specified region.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE
+void up_invalidate_icache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_icache_linesize();
+
+  cache_invalidate_icache_items((uint32_t)start, items);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_invalidate_icache_all
+ *
+ * Description:
+ *   Invalidate the entire contents of I cache.
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE
+void up_invalidate_icache_all(void)
+{
+  cache_invalidate_icache_all();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_lock_icache
+ *
+ * Description:
+ *   Prefetch and lock the instruction cache within the specified region.
+ *   If the specified address if not present in the instruction cache,
+ *   some architectures transfer the line from memory, others wait the
+ *   address be read from memory, and then lock.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE_LOCK
+void up_lock_icache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_icache_linesize();
+
+  cache_lock_icache_items((uint32_t)start, items);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_unlock_icache
+ *
+ * Description:
+ *   Unlock the instruction cache within the specified region.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_ICACHE_LOCK
+void up_unlock_icache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_icache_linesize();
+
+  cache_unlock_icache_items((uint32_t)start, items);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_get_dcache_linesize
+ *
+ * Description:
+ *   Get dcache linesize
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   Cache line size
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+size_t up_get_dcache_linesize(void)
+{
+  return (size_t)cache_get_dcache_line_size();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_get_dcache_size
+ *
+ * Description:
+ *   Get dcache size
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   Cache size
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+size_t up_get_dcache_size(void)
+{
+  return (size_t)CONFIG_ESP32S3_DATA_CACHE_SIZE;
+}
+#endif
+
+/****************************************************************************
+ * Name: up_enable_dcache
+ *
+ * Description:
+ *   Enable the D-Cache
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ * Caution:
+ *   The writable global variables aren't initialized yet.
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_enable_dcache(void)
+{
+  cache_resume_dcache(0);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_disable_dcache
+ *
+ * Description:
+ *   Disable the D-Cache
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_disable_dcache(void)
+{
+  cache_suspend_dcache();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_invalidate_dcache
+ *
+ * Description:
+ *   Invalidate the data cache within the specified region; we will be
+ *   performing a DMA operation in this region and we want to purge old data
+ *   in the cache.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_invalidate_dcache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_dcache_linesize();
+
+  cache_invalidate_dcache_items((uint32_t)start, end);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_invalidate_dcache_all
+ *
+ * Description:
+ *   Invalidate the entire contents of D cache.
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_invalidate_dcache_all(void)
+{
+  cache_invalidate_dcache_all();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_clean_dcache
+ *
+ * Description:
+ *   Clean the data cache within the specified region by flushing the
+ *   contents of the data cache to memory.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_clean_dcache(uintptr_t start, uintptr_t end)
+{
+  /* Please note that, according to ESP32-S3 TRM, the clean operation just
+   * clears the dirty bits in the dirty block, without updating data to the
+   * external memory. After the clean operation finishes, there will still be
+   * old data stored in the external memory, while the cache keeps the new
+   * one (but the cache does not know about this). That's why - according to
+   * this function's description - we need to call the writeback operation.
+   * The writeback operation will clear the dirty bits and updates the data
+   * to the external memory.
+   */
+
+  cache_writeback_addr((uint32_t)start, (uint32_t)(end - start));
+}
+#endif
+
+/****************************************************************************
+ * Name: up_clean_dcache_all
+ *
+ * Description:
+ *   Clean the entire data cache within the specified region by flushing the
+ *   contents of the data cache to memory.
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_clean_dcache_all(void)
+{
+  cache_writeback_all();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_flush_dcache
+ *
+ * Description:
+ *   Flush the data cache within the specified region by cleaning and
+ *   invalidating the D cache.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_flush_dcache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_dcache_linesize();
+
+  /* Please note that, according to ESP32-S3 TRM, the clean operation just
+   * clears the dirty bits in the dirty block, without updating data to the
+   * external memory. After the clean operation finishes, there will still be
+   * old data stored in the external memory, while the cache keeps the new
+   * one (but the cache does not know about this). That's why - according to
+   * this function's description - we need to call the writeback operation.
+   * The writeback operation will clear the dirty bits and updates the data
+   * to the external memory.
+   */
+
+  cache_writeback_addr((uint32_t)start, (uint32_t)(end - start));
+
+  cache_invalidate_dcache_items((uint32_t)start, end);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_flush_dcache_all
+ *
+ * Description:
+ *   Flush the entire data cache by cleaning and invalidating the D cache.
+ *
+ * Input Parameters:
+ *   None
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE
+void up_flush_dcache_all(void)
+{
+  cache_writeback_all();
+  cache_invalidate_dcache_all();
+}
+#endif
+
+/****************************************************************************
+ * Name: up_lock_dcache
+ *
+ * Description:
+ *   Prefetch and lock the data cache within the specified region.
+ *   If the specified address is not present in the data cache,
+ *   some architectures transfer the line from memory, others wait the
+ *   address be read from memory, and then lock.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE_LOCK
+void up_lock_dcache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_dcache_linesize();
+
+  cache_lock_dcache_items((uint32_t)start, items);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_unlock_dcache
+ *
+ * Description:
+ *   Unlock the data cache within the specified region.
+ *
+ * Input Parameters:
+ *   start - virtual start address of region
+ *   end   - virtual end address of region + 1
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_ARCH_DCACHE_LOCK
+void up_unlock_dcache(uintptr_t start, uintptr_t end)
+{
+  uint32_t items = (end - start) / up_get_dcache_linesize();
+
+  cache_unlock_dcache_items((uint32_t)start, items);
+}
+#endif
+
+/****************************************************************************
+ * Name: up_coherent_dcache
+ *
+ * Description:
+ *   Ensure that the I and D caches are coherent within specified region
+ *   by cleaning the D cache (i.e., flushing the D cache contents to memory)
+ *   and invalidating the I cache. This is typically used when code has been
+ *   written to a memory region, and will be executed.
+ *
+ * Input Parameters:
+ *   addr - virtual start address of region
+ *   len  - Size of the address region in bytes
+ *
+ * Returned Value:
+ *   None
+ *
+ ****************************************************************************/
+
+#if defined(CONFIG_ARCH_ICACHE) && defined(CONFIG_ARCH_DCACHE)
+void up_coherent_dcache(uintptr_t addr, size_t len)
+{
+  uint32_t items = (len) / up_get_dcache_linesize();
+
+  cache_writeback_addr((uint32_t)addr, (uint32_t)len);
+  cache_invalidate_icache_items((uint32_t)addr, items);
+}
+#endif
diff --git a/arch/xtensa/src/esp32s3/esp32s3_textheap.c 
b/arch/xtensa/src/esp32s3/esp32s3_textheap.c
index a0695200f7f..3d365d808e4 100644
--- a/arch/xtensa/src/esp32s3/esp32s3_textheap.c
+++ b/arch/xtensa/src/esp32s3/esp32s3_textheap.c
@@ -87,7 +87,7 @@ void *up_textheap_memalign(size_t align, size_t size)
 
   if (ret == NULL)
     {
-      ret = kmm_memalign(align, size);
+      ret = memalign(align, size);
       if (ret)
         {
           /* kmm_memalign buffer is at the Data bus offset.  Adjust it so we
@@ -137,7 +137,7 @@ void up_textheap_free(void *p)
 #endif
         {
           p = up_textheap_data_address(p);
-          kmm_free(p);
+          free(p);
         }
     }
 }
@@ -171,7 +171,7 @@ bool up_textheap_heapmember(void *p)
 #endif
 
   p = up_textheap_data_address(p);
-  return kmm_heapmember(p);
+  return umm_heapmember(p);
 }
 
 /****************************************************************************
diff --git a/arch/xtensa/src/esp32s3/hal.mk b/arch/xtensa/src/esp32s3/hal.mk
index bf5be01d755..f4bc0892831 100644
--- a/arch/xtensa/src/esp32s3/hal.mk
+++ b/arch/xtensa/src/esp32s3/hal.mk
@@ -186,6 +186,8 @@ CHIP_CSRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)components$(DELIM)spi_
 
 CHIP_CSRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)nuttx$(DELIM)src$(DELIM)components$(DELIM)esp_driver_gpio$(DELIM)src$(DELIM)rtc_io.c
 
+CHIP_ASRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)components$(DELIM)esp_rom$(DELIM)patches$(DELIM)esp_rom_cache_writeback_esp32s3.S
+
 ifeq ($(CONFIG_ESPRESSIF_SIMPLE_BOOT),y)
   CHIP_CSRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)nuttx$(DELIM)src$(DELIM)bootloader_banner_wrap.c
   CHIP_CSRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)components$(DELIM)bootloader_support$(DELIM)src$(DELIM)bootloader_console.c
@@ -213,8 +215,6 @@ ifeq ($(CONFIG_ESPRESSIF_SIMPLE_BOOT),y)
   CHIP_CSRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)components$(DELIM)efuse$(DELIM)src$(DELIM)esp_efuse_fields.c
   CHIP_CSRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)components$(DELIM)efuse$(DELIM)src$(DELIM)efuse_controller$(DELIM)keys$(DELIM)with_key_purposes$(DELIM)esp_efuse_api_key.c
 
-  CHIP_ASRCS += 
chip$(DELIM)$(ESP_HAL_3RDPARTY_REPO)$(DELIM)components$(DELIM)esp_rom$(DELIM)patches$(DELIM)esp_rom_cache_writeback_esp32s3.S
-
   LDFLAGS += --wrap=bootloader_print_banner
 endif
 
diff --git a/boards/xtensa/esp32s3/common/scripts/esp32s3_rom_aliases.ld 
b/boards/xtensa/esp32s3/common/scripts/esp32s3_rom_aliases.ld
index 46c451c1f7b..fb5d7d3880d 100644
--- a/boards/xtensa/esp32s3/common/scripts/esp32s3_rom_aliases.ld
+++ b/boards/xtensa/esp32s3/common/scripts/esp32s3_rom_aliases.ld
@@ -24,11 +24,20 @@
 
 /* Lower-case aliases for symbols not compliant to nxstyle */
 
+PROVIDE( cache_get_dcache_line_size = Cache_Get_DCache_Line_Size );
+PROVIDE( cache_get_icache_line_size = Cache_Get_ICache_Line_Size );
 PROVIDE( cache_dbus_mmu_set = Cache_Dbus_MMU_Set );
 PROVIDE( cache_ibus_mmu_set = Cache_Ibus_MMU_Set );
+PROVIDE( cache_clean_addr = Cache_Clean_Addr );
 PROVIDE( cache_invalidate_addr = Cache_Invalidate_Addr );
 PROVIDE( cache_invalidate_dcache_all = Cache_Invalidate_DCache_All );
 PROVIDE( cache_invalidate_icache_all = Cache_Invalidate_ICache_All );
+PROVIDE( cache_invalidate_icache_items = Cache_Invalidate_ICache_Items );
+PROVIDE( cache_invalidate_dcache_items = Cache_Invalidate_DCache_Items );
+PROVIDE( cache_lock_icache_items = Cache_Lock_ICache_Items );
+PROVIDE( cache_lock_dcache_items = Cache_Lock_DCache_Items );
+PROVIDE( cache_unlock_icache_items = Cache_Unlock_ICache_Items );
+PROVIDE( cache_unlock_dcache_items = Cache_Unlock_DCache_Items );
 PROVIDE( cache_occupy_addr = Cache_Occupy_Addr );
 PROVIDE( cache_resume_dcache = Cache_Resume_DCache );
 PROVIDE( cache_resume_icache = Cache_Resume_ICache );
diff --git a/libs/libc/elf/elf_bind.c b/libs/libc/elf/elf_bind.c
index faaf087e63c..33ffb869bcc 100644
--- a/libs/libc/elf/elf_bind.c
+++ b/libs/libc/elf/elf_bind.c
@@ -32,6 +32,7 @@
 #include <assert.h>
 #include <debug.h>
 
+#include <nuttx/arch.h>
 #include <nuttx/cache.h>
 #include <nuttx/elf.h>
 #include <nuttx/lib/elf.h>
diff --git a/libs/libc/machine/xtensa/arch_elf.c 
b/libs/libc/machine/xtensa/arch_elf.c
index c9911dc7f1b..2ae521d129c 100644
--- a/libs/libc/machine/xtensa/arch_elf.c
+++ b/libs/libc/machine/xtensa/arch_elf.c
@@ -194,7 +194,7 @@ int up_relocateadd(const Elf32_Rela *rel, const Elf32_Sym 
*sym,
       break;
 
     case R_XTENSA_32:
-      (*(uint32_t *)addr) += value;
+      (*(uint32_t *)(up_textheap_data_address((void *)addr))) += value;
       break;
 
     case R_XTENSA_ASM_EXPAND:

Reply via email to