zephyr/arch/common/CMakeLists.txt

100 lines
3.1 KiB
CMake
Raw Normal View History

# SPDX-License-Identifier: Apache-2.0
zephyr_library()
zephyr_library_include_directories(include)
# Library may be empty due to kconfigs
zephyr_library_property(ALLOW_EMPTY TRUE)
if(CONFIG_GEN_ISR_TABLES)
zephyr_library_sources(
sw_isr_common.c
)
zephyr_library_sources_ifdef(
CONFIG_DYNAMIC_INTERRUPTS
dynamic_isr.c
)
endif()
zephyr_library_sources_ifdef(
CONFIG_MULTI_LEVEL_INTERRUPTS
multilevel_irq.c
)
zephyr_library_sources_ifdef(CONFIG_SHARED_INTERRUPTS shared_irq.c)
if(NOT CONFIG_ARCH_HAS_TIMING_FUNCTIONS AND
NOT CONFIG_SOC_HAS_TIMING_FUNCTIONS AND
NOT CONFIG_BOARD_HAS_TIMING_FUNCTIONS)
zephyr_library_sources_ifdef(CONFIG_TIMING_FUNCTIONS timing.c)
endif()
# Put functions and data in their own binary sections so that ld can
# garbage collect them
zephyr_cc_option(-ffunction-sections -fdata-sections)
zephyr_linker_sources_ifdef(CONFIG_GEN_ISR_TABLES
SECTIONS
${ZEPHYR_BASE}/include/zephyr/linker/intlist.ld
)
irq: Fix IRQ vector table relocation The generation of the software ISR table and the IRQ vector table (respectively generated by CONFIG_GEN_SW_ISR_TABLE and CONFIG_GEN_IRQ_VECTOR_TABLE) should (in theory) go through three stages: 1. A placeholder table is generated in arch/common/isr_tables.c and placed in an orphaned .gnu.linkonce.{irq_vector_table, sw_isr_table} section 2. The real table is generated by arch/common/gen_isr_tables.py (creating the build/zephyr/isr_tables.c file) 3. The real table is un-orphaned by moving it in a proper section with a proper alignment While all the steps are done automatically for the software ISR table, for the IRQ vector table each architectures must take care of modiying its own linker script to place somewhere the generated IRQ vector table (basically step 3 is missing). This is currently only done for 2 architectures: Cortex-M (ARMv7) and ARC. But when another architecture tries to use the IRQ vector table, the linker complains about that. For example: Linking C executable zephyr/zephyr.elf riscv64-zephyr-elf/bin/ld.bfd: warning: orphan section `.gnu.linkonce.irq_vector_table' from `zephyr/CMakeFiles/zephyr_final.dir/isr_tables.c.obj' being placed in section `.gnu.linkonce.irq_vector_table' In this patch we introduce a new CONFIG_ARCH_IRQ_VECTOR_TABLE_ALIGN to support the architectures requiring a special alignment for the IRQ vector table and we also introduce a way to automatically place the IRQ vector table in place in the same way it is done for the ISR software table. Signed-off-by: Carlo Caione <ccaione@baylibre.com>
2022-06-21 14:26:03 +02:00
zephyr_linker_sources_ifdef(CONFIG_GEN_IRQ_VECTOR_TABLE
ROM_START
SORT_KEY 0x0vectors
${ZEPHYR_BASE}/include/zephyr/linker/irq-vector-table-section.ld
)
if(CONFIG_GEN_ISR_TABLES)
zephyr_linker_section(NAME .intList VMA IDT_LIST LMA IDT_LIST NOINPUT PASS NOT LINKER_ZEPHYR_FINAL)
zephyr_linker_section_configure(SECTION .intList KEEP INPUT ".irq_info" FIRST)
zephyr_linker_section_configure(SECTION .intList KEEP INPUT ".intList")
zephyr_linker_section_configure(SECTION /DISCARD/ KEEP INPUT ".irq_info" PASS LINKER_ZEPHYR_FINAL)
zephyr_linker_section_configure(SECTION /DISCARD/ KEEP INPUT ".intList" PASS LINKER_ZEPHYR_FINAL)
endif()
zephyr_linker_sources_ifdef(CONFIG_ARCH_HAS_RAMFUNC_SUPPORT
RAM_SECTIONS
ramfunc.ld
)
zephyr_linker_sources_ifdef(CONFIG_NOCACHE_MEMORY
RAM_SECTIONS
nocache.ld
)
# Only ARM, X86 and OPENISA_RV32M1_RISCV32 use ROM_START_OFFSET.
if (DEFINED CONFIG_ARM OR DEFINED CONFIG_X86 OR DEFINED CONFIG_ARM64
OR DEFINED CONFIG_SOC_OPENISA_RV32M1_RISCV32)
arch/common: Fix moving location counter backwards when using LLD In GNU LD, the location counter (the 'dot' variable) always refers to the byte offset from the start of current object as mentioned in documentation[1]: ``` '.' actually refers to the byte offset from the start of the current containing object. Normally this is the SECTIONS statement, whose start address is 0, hence '.' can be used as an absolute address. If '.' is used inside a section description however, it refers to the byte offset from the start of that section, not an absolute address. ``` For example, if the section 'rom_start': rom_start : { . = 0x400; _vector_start = ABSOLUTE(.); } > FLASH has a starting address of 0x8000000, then _vector_start will be 0x8000400 However, behavior of LLVM LLD is quite different, the value of the location counter is always absolute (see discussion [2]), so in the example above, the linker will return error, because it will interpret '. = 0x400' as an attempt to move the location counter backwards. It could be fixed by changing line to '. += 0x400' (#54796) which will move the location counter by 0x400 for both linkers, but it would work only when we are at the beginning of section. Consider the following example: rom_start : { . = 0x400; KEEP(*(.boot_hdr.conf)) . = 0x1000; KEEP(*(.boot_hdr.ivt)) KEEP(*(.boot_hdr.data)) KEEP(*(.boot_hdr.dcd_data)) . = 0x2000; _vector_start = .; } > FLASH In this case, _vector_start will be 0x2000, but if we change '. = 0x2000' to '. += 0x2000', then the value of _vector_start depends on size of data in input sections (but it's 0x3000 at least). Actually, this example comes from final linker script when compiling firmware for mimxrt1170_evk_cm7 board. This board failed to boot (#55296) after #54796 was merged. This patch introduces method compatible with both linkers. We calculate relative offset from the beginning of the section and use that value to calculate number of bytes by which we should move the location counter to get CONFIG_ROM_START_OFFSET. [1] https://sourceware.org/binutils/docs/ld/Location-Counter.html [2] https://discourse.llvm.org/t/lld-location-counter-inside-objects Signed-off-by: Patryk Duda <pdk@semihalf.com>
2023-05-26 13:47:08 +02:00
# Exclamation mark is printable character with lowest number in ASCII table.
# We are sure that this file will be included as a first.
zephyr_linker_sources(ROM_START SORT_KEY ! rom_start_address.ld)
# Some linkers fill unspecified region with pattern other than 0x00. Include
# fill_with_zeros.ld file which forces the linker to use 0x00 pattern. Please
# note that the pattern will affect empty spaces created after FILL(0x00).
zephyr_linker_sources(ROM_START SORT_KEY $ fill_with_zeros.ld)
zephyr_linker_sources(ROM_START SORT_KEY 0x0 rom_start_offset.ld)
# Handled in ld.cmake
endif()
# isr_tables is a normal CMake library and not a zephyr_library because it
# should not be --whole-archive'd
if (CONFIG_GEN_ISR_TABLES)
add_library(isr_tables
isr_tables.c
)
add_dependencies(isr_tables zephyr_generated_headers)
target_link_libraries(isr_tables zephyr_interface)
zephyr_library_link_libraries(isr_tables)
endif()
if(CONFIG_COVERAGE)
zephyr_compile_options($<TARGET_PROPERTY:compiler,coverage>)
zephyr_link_libraries_ifndef(CONFIG_NATIVE_LIBRARY $<TARGET_PROPERTY:linker,coverage>)
endif()
zephyr_library_sources_ifdef(CONFIG_SEMIHOST semihost.c)