Рекомендации по созданию и проверке скрипта компоновщика - PullRequest
7 голосов
/ 22 января 2012

Короче говоря.Я хотел бы узнать, как создать хороший скрипт компоновщика, чтобы при смене платформ / архитектур / поставщиков я не застрял на первом месте, не зная, что делать.Меня не беспокоит сложность задачи, а скорее ее понимание.

Я начал своего рода проект , чтобысоздать базу или каркас для программирования и разработки на 32-битных чипах Cortex-M3 от STM.С помощью jsiei97 Начиная с STM32F103RB (у меня также есть TI Stellaris LM3S828, но это еще одна проблема), без необходимости в лицензированной IDE.Поскольку я студент, и большинство студентов не могут себе этого позволить.

Я понимаю, что есть ODev и Eclipse Plugins, а что нет, и читал различные блоги, вики, документы / справочные страницы и большинствопроекты предоставляют вам скрипт компоновщика с небольшим количеством информации для объяснения того, почему и где что-то было определено.

Я собрал набор инструментов arm-none-eabi для STM32, но где я завис,скрипт компоновщика.CodeSourcery также требует один также.У меня есть общее представление о том, как их создавать и их синтаксис после прочтения справочных страниц gnu, но я просто не представляю, с чего начать с добавления дополнительных разделов, кроме очевидных .text, .bss и .data,

Я создал элементарную версию , но я получаю ошибки связывания, запрашивающие определения разделов, и вот где я застреваю.Я знаю, как определить их, но проблема в том, что то, что я делаю, близко к правде.

Ответы [ 2 ]

8 голосов
/ 23 января 2012

У меня есть простой скрипт компоновщика, который я регулярно использую на разных платформах, просто меняйте некоторые адреса по мере необходимости.

http://github.com/dwelch67/

Существует множество сэмплов с gcc и большинствому них есть скрипты компоновщика.

MEMORY
{
   rom : ORIGIN = 0x00000000, LENGTH = 0x40000
   ram : ORIGIN = 0x10000000, LENGTH = 30K
}

SECTIONS
{
   .text : { *(.text*) } > rom
   .bss  : { *(.bss*) } > ram
}
2 голосов
/ 23 января 2012

Вот рабочий скрипт компоновщика для STM32F105RB (есть также версии для R8 и RC):

https://github.com/anrope/stm-arp/blob/github/arp.rb.ld (текст ниже)

Мое предположение в том, что вам не нужно ничего менять. Может быть, происхождение регионов определено в операторе MEMORY {}. Надеемся, что комментарии будут полезны для вас.

Я использовал это с кросс-компилятором GNU / GCC, который я сам катал. После компиляции полезно запустить nm в вашем коде, чтобы убедиться, что разделы размещаются по правильным адресам.

Edit: Я собрал этот скрипт компоновщика, используя документацию GNU ld:

http://sourceware.org/binutils/docs/ld/

и путем проверки результатов кросс-компиляции GCC со стандартным сценарием компоновщика, используя nm. Я в основном идентифицировал все разделы, которые выводились, и выяснил, какие из них действительно полезны, и где в памяти они должны идти для STM32F105.

Я сделал примечания в скрипте компоновщика назначения каждого раздела.

/*
arp.{r8,rb,rc}.ld :
These linker scripts (one for each memory density of the stm32f105) are used by
the linker to arrange program symbols and sections in memory. This is especially
important for sections like the interrupt vector, which must be placed where the
processor is hard-coded to look for it.
*/

/*stm32f105 dev board linker script*/

/*
OUTPUT_FORMAT() defines the BFD (binary file descriptor) format
OUTPUT_FORMAT(default, big, little)
*/
OUTPUT_FORMAT ("elf32-littlearm", "elf32-bigarm", "elf32-littlearm")
/* ENTRY() defines the symbol at which to begin executing code */
ENTRY(_start)
/* tell ld where to look for archive libraries */
/*SEARCH_DIR("/home/arp/stm/ctc/arm-eabi/lib")*/
/*SEARCH_DIR("/home/arp/stm/ccbuild/method2/install/arm-eabi/lib")*/
SEARCH_DIR("/home/arp/stm32dev-root/usrlol/arm-eabi/lib")

/*
MEMORY{} defines the memory regions of the target device,
and gives them an alias for use later in the linker script.
*/

/* stm32f105rb */
MEMORY
{
  ram (rwx) : ORIGIN = 0x20000000, LENGTH = 32k
  flash (rx) : ORIGIN = 0x08000000, LENGTH = 128k
  option_bytes_rom (rx) : ORIGIN = 0x1FFFF800, LENGTH = 16
}

_sheap = _ebss + 4;
_sstack = _ebss + 4;
/*placed __stack_base__ trying to figure out
global variable overwrite issue
__stack_base__ = _ebss + 4;*/

_eheap = ORIGIN(ram) + LENGTH(ram) - 1;
_estack = ORIGIN(ram) + LENGTH(ram) - 1;

/* SECTIONS{} defines all the ELF sections we want to create */
SECTIONS
{
    /*
    set . to an initial value (0 here).
    . (dot) is the location counter. New sections are placed at the
    location pointed to by the location counter, and the location counter
    is automatically moved ahead the length of the new section. It is important
    to maintain alignment (not handled automatically by the location counter).
    */
    . = SEGMENT_START("text-segment", 0);



    /*isr_vector contains the interrupt vector.

    isr_vector is read only (could be write too?).

    isr_vector must appear at start of flash (USR),
    address 0x0800 0000*/
    .isr_vector :
    {
      . = ALIGN(4);
      _sisr_vector = .;

      *(.isr_vector)

      _eisr_vector = .;
    } >flash

    /*text contains executable code.

    text is read and execute.*/
    .text :
    {
      . = ALIGN(4);
      *(.text)
      . = ALIGN(4);
      *(.text.*)
    } >flash

    /*init contains constructor functions
    called before entering main. used by crt (?).*/
    .init :
    {
      . = ALIGN(4);
      KEEP(*(.init))
    } >flash

    /*fini contains destructor functions
    called after leaving main. used by crt (?).*/
    .fini :
    {
      . = ALIGN(4);
      KEEP(*(.fini))
    } >flash

    /* rodata contains read only data.*/
    .rodata :
    {
      . = ALIGN(4);
      *(.rodata)

      /* sidata contains the initial values
      for variables in the data section.

      sidata is read only.*/
      . = ALIGN(4);
      _sidata = .;
    } >flash

    /*data contains all initalized variables.

    data is read and write. 
    .data (NOLOAD) : AT(_sidata)*/
    .data : AT(_sidata)
    {
      . = ALIGN(4);
      _sdata = .;

      *(.data)

      _edata = .;
    } >ram

    /*bss contains unintialized variables.

    bss is read and write.
    .bss (NOLOAD) :*/
    .bss :
    {
      . = ALIGN(4);
      _sbss = .;
      __bss_start__ = .;

      *(.bss)
      . = ALIGN(4);

      /*COMMON is a special section containing
      uninitialized data.

      Example: (declared globally)
      int temp; //this will appear in COMMON */
      *(COMMON)

      _ebss = .;
      __bss_end__ = .;
    } >ram AT>flash

    . = ALIGN(4);
    end = .;

        /* remove the debugging information from the standard libraries */
    DISCARD :
    {
     libc.a ( * )
     libm.a ( * )
     libgcc.a ( * )
     }

    /* Stabs debugging sections.  */
    .stab          0 : { *(.stab) }
    .stabstr       0 : { *(.stabstr) }
    .stab.excl     0 : { *(.stab.excl) }
    .stab.exclstr  0 : { *(.stab.exclstr) }
    .stab.index    0 : { *(.stab.index) }
    .stab.indexstr 0 : { *(.stab.indexstr) }
    .comment       0 : { *(.comment) }
    /* DWARF debug sections.
       Symbols in the DWARF debugging sections are relative to the beginning
       of the section so we begin them at 0.  */
    /* DWARF 1 */
    .debug          0 : { *(.debug) }
    .line           0 : { *(.line) }
    /* GNU DWARF 1 extensions */
    .debug_srcinfo  0 : { *(.debug_srcinfo) }
    .debug_sfnames  0 : { *(.debug_sfnames) }
    /* DWARF 1.1 and DWARF 2 */
    .debug_aranges  0 : { *(.debug_aranges) }
    .debug_pubnames 0 : { *(.debug_pubnames) }
    /* DWARF 2 */
    .debug_info     0 : { *(.debug_info .gnu.linkonce.wi.*) }
    .debug_abbrev   0 : { *(.debug_abbrev) }
    .debug_line     0 : { *(.debug_line) }
    .debug_frame    0 : { *(.debug_frame) }
    .debug_str      0 : { *(.debug_str) }
    .debug_loc      0 : { *(.debug_loc) }
    .debug_macinfo  0 : { *(.debug_macinfo) }
    /* SGI/MIPS DWARF 2 extensions */
    .debug_weaknames 0 : { *(.debug_weaknames) }
    .debug_funcnames 0 : { *(.debug_funcnames) }
    .debug_typenames 0 : { *(.debug_typenames) }
    .debug_varnames  0 : { *(.debug_varnames) }
}
...