Стирание и запись во флэш дает ошибку при сборке - PullRequest
0 голосов
/ 05 июня 2018

Я пишу программу на C для NXP, свободно распространяемого микроконтроллера Kinetis KEA128.Я изменил файл описания компоновщика и приступил к написанию простой программы на C на C. Но когда я собираю проект, он выдает следующее сообщение об ошибке

build\src\flash.o:(.rodata.flashLocat+0x0): multiple definition of `flashLocat'
build\src\main.o:(.rodata.flashLocat+0x0): first defined here
collect2.exe: error: ld returned 1 exit status

Здесь я прилагаю свой файл программы на C и основной файлэто вызывает функции для стирания и записи во флэш и мой файл описания компоновщика.Ниже приведены мои файлы flash.c и flash.h

#include "flash.h"

int flashEraseSector(long int locat)
{
        int temp1,temp2,temp3;

            temp1 = (locat << 8);
            temp1 = temp1 >> 24;
            temp2 = (locat << 16);
            temp2 = temp2 >> 24;
            temp3 = (locat << 24);
            temp3 = temp3 >> 24;


            while(((FTMRE_FSTAT)&(1UL << 7))==0x00);

            if(!((FTMRE_FSTAT)==0x80))
                    {FTMRE_FSTAT = 0x30;}

            int IX = 0;
            FTMRE_FCCOBIX = temp1;
            IX = 1;
            FTMRE_FCCOBIX = temp2;
            IX = 2;
            FTMRE_FCCOBIX = temp3;
            IX = 3;
            FTMRE_FCCOBIX = 0x0A;   //erase a sector

            FTMRE_FSTAT = 0x80;

            while(((FTMRE_FSTAT)&(1UL << 7))==0x00);
        return 1;
}

/*------------------------------------------------------------------------------
  Write a long word to an erased flash block
 *------------------------------------------------------------------------------*/

int flashWriteLongWord(long int locat,long int value)
{
        int temp1,temp2,temp3,temp4,temp5,temp6,temp7;

            temp1 = (locat << 8);
            temp1 = temp1 >> 24;
            temp2 = (locat << 16);
            temp2 = temp2 >> 24;
            temp3 = (locat << 24);
            temp3 = temp3 >> 24;
            temp4 = value >> 24;
            temp5 = (value << 8);
            temp5 = temp5 >> 24;
            temp6 = (value << 16);
            temp6 = temp6 >> 24;
            temp7 = (value << 24);
            temp7 = temp7 >> 24;

            while(((FTMRE_FSTAT)&(1UL << 7))==0x00);

            if(!((FTMRE_FSTAT)==0x80))
                    {FTMRE_FSTAT = 0x30;}
            int IX = 0;
            FTMRE_FCCOBIX = 0x06;    //0x06 is instruction code for write long word.
            IX = 1;
            FTMRE_FCCOBIX = temp1;
            IX = 2;
            FTMRE_FCCOBIX = temp2;
            IX = 3;
            FTMRE_FCCOBIX = temp3;
            IX = 4;
            FTMRE_FCCOBIX = temp4;
            IX = 5;
            FTMRE_FCCOBIX = temp5;
            IX = 6;
            FTMRE_FCCOBIX = temp6;
            IX = 7;
            FTMRE_FCCOBIX = temp7;

            FTMRE_FSTAT = 0x80;

            while(((FTMRE_FSTAT)&(1UL << 7))==0x00);
        return 1;
}

Вот мой файл flash.h

#ifndef SRC_FLASH_H_
#define SRC_FLASH_H_

#include "derivative.h" /* include peripheral declarations */
const long int flashLocat = 0x00000800;
int flashEraseSector(long int locat);
int flashWriteLongWord(long int locat,long int value);

#endif /* SRC_FLASH_H_ */

Вот мой основной файл int main (void) {long int value;uint8_t flashStatus;значение = 2147483648;

    NVIC_ISER |= ((unsigned int)0x1<<22); // Interrupt Set-Enable Register, IRQ22 (PIT_CH0)

    PIT_TFLG0 = 0x1; // clear TIF
    PIT_TCTRL0 = 0x03; // CHN = 0 -> not chained, TIE = 1 -> interrupt enabled, TEN = 1 -> timer enabled

    CpuComm_Init();

    flashStatus = flashEraseSector(flashLocat);
    flashWriteLongWord(flashLocat,value);

    for(;;)
    {
    }

    return 0;
}

Вот мой файл описания компоновщика:

/ *


**
**  File        : KEAZ128M4_flash.ld
**
**  Default linker command file for Flash targets
**
*****************************************************************************
*/
/* Entry Point */
/*ENTRY(__init_hardware)*/
ENTRY(RESET_handler)

/* Highest address of the user mode stack */
_estack = 0x20003000;    /* end of SRAM */
__SP_INIT = _estack;

/* Generate a link error if heap and stack don't fit into RAM */
/*__heap_size = 0x100;  */          /* required amount of heap is 256 Bytes */
/*__stack_size = 0x100; */              /* required amount of stack is 256 Bytes*/

/* Specify the memory areas */
MEMORY
{
  m_interrupts  (rx) : ORIGIN = 0x00000000, LENGTH = 0xC0   /*192 Bytes*/
  m_cfmprotrom  (rx) : ORIGIN = 0x00000400, LENGTH = 0x10   /*16 Bytes*/
  m_text        (rx) : ORIGIN = 0x00000800, LENGTH = 128K - 0x800
  /*reserving the last 2KB of flash by subtracting those bytes (0x800) from m_text segment*/
  m_data       (rwx) : ORIGIN = 0x1FFFF000, LENGTH = 16K            /* SRAM */
}


/* Define output sections */
SECTIONS
{
  /* The startup code goes first into Flash */

  /*'.' means the location coounter, at the start of SECTIONS 
   *  command the location counter is set to the value '0'
   */

  .interrupts :
  {
    __vector_table = .;             
    . = ALIGN(4);
    KEEP(*(.vectortable)) /* Startup code */
    . = ALIGN(4);
  } > m_interrupts

  /* Above lines of code mean an output section '.interrupts'
   * has input sections called vector_table and *(.vectortable)
   * means all '.vectortable' input sections in all input files.
   */
  .cfmprotect :
  {
    . = ALIGN(4);
    KEEP(*(.cfmconfig)) /* Flash Configuration Field (FCF) */
    . = ALIGN(4);
  } > m_cfmprotrom

  /* The program code and other data goes into Flash */
  .text :
  {
    . = ALIGN(4);
    *(.text)           /* .text sections (code) */
    *(.text*)          /* .text* sections (code) */
    *(.rodata)         /* .rodata sections (constants, strings, etc.) */
    *(.rodata*)        /* .rodata* sections (constants, strings, etc.) */
    *(.glue_7)         /* glue arm to thumb code */
    *(.glue_7t)        /* glue thumb to arm code */
    *(.eh_frame)

    KEEP (*(.init))
    KEEP (*(.fini))

    . = ALIGN(4);
    _etext = .;        /* define a global symbols at end of code */
  } > m_text

  .ARM.extab   : { *(.ARM.extab* .gnu.linkonce.armextab.*) } > m_text
  .ARM : {
    __exidx_start = .;
      *(.ARM.exidx*)
      __exidx_end = .;
  } > m_text

  .ctors :
  {
    __CTOR_LIST__ = .;
    /* gcc uses crtbegin.o to find the start of
       the constructors, so we make sure it is
       first.  Because this is a wildcard, it
       doesn't matter if the user does not
       actually link against crtbegin.o; the
       linker won't look for a file to match a
       wildcard.  The wildcard also means that it
       doesn't matter which directory crtbegin.o
       is in.  */
    KEEP (*crtbegin.o(.ctors))
    /* We don't want to include the .ctor section from
       from the crtend.o file until after the sorted ctors.
       The .ctor section from the crtend file contains the
       end of ctors marker and it must be last */
    KEEP (*(EXCLUDE_FILE (*crtend.o ) .ctors))
    KEEP (*(SORT(.ctors.*)))
    KEEP (*(.ctors))
    __CTOR_END__ = .;
  } > m_text  
  .dtors :
  {
    __DTOR_LIST__ = .;
    KEEP (*crtbegin.o(.dtors))
    KEEP (*(EXCLUDE_FILE (*crtend.o ) .dtors))
    KEEP (*(SORT(.dtors.*)))
    KEEP (*(.dtors))
    __DTOR_END__ = .;
  } > m_text  

  .preinit_array :
  {
    PROVIDE_HIDDEN (__preinit_array_start = .);
    KEEP (*(.preinit_array*))
    PROVIDE_HIDDEN (__preinit_array_end = .);
  } > m_text
  .init_array :
  {
    PROVIDE_HIDDEN (__init_array_start = .);
    KEEP (*(SORT(.init_array.*)))
    KEEP (*(.init_array*))
    PROVIDE_HIDDEN (__init_array_end = .);
  } > m_text
  .fini_array :
  {
    PROVIDE_HIDDEN (__fini_array_start = .);
    KEEP (*(SORT(.fini_array.*)))
    KEEP (*(.fini_array*))
    PROVIDE_HIDDEN (__fini_array_end = .);

    ___ROM_AT = .;
  } > m_text

  /* reserve MTB memory at the beginning of m_data */
  .mtb : /* MTB buffer address as defined by the hardware */
  {
    . = ALIGN(8);
    _mtb_start = .;
    KEEP(*(.mtb_buf)) /* need to KEEP Micro Trace Buffer as not referenced by application */
    . = ALIGN(8);
    _mtb_end = .;
  } > m_data  

  /* Initialized data sections goes into RAM, load LMA copy after code */

   __stacktop = ORIGIN(m_data) + LENGTH(m_data);
    __data_load = LOADADDR(.data);
    . = ORIGIN(m_data);
  .data : AT(___ROM_AT)
  {
    . = ALIGN(4);
    _sdata = .;        /* create a global symbol at data start */
    *(.data)           /* .data sections */
    *(.data*)          /* .data* sections */

    . = ALIGN(4);
    _edata = .;        /* define a global symbol at data end */
  } > m_data

  ___data_size = _edata - _sdata;

  /* Uninitialized data section */
  . = ALIGN(4);
  .bss :
  {
    /* This is used by the startup in order to initialize the .bss section */
    __START_BSS = .;
    PROVIDE ( __bss_start__ = __START_BSS );
    *(.bss)
    *(.bss*)
    *(COMMON)

    . = ALIGN(4);
    __END_BSS = .;
    PROVIDE ( __bss_end__ = __END_BSS );
  } > m_data

  _romp_at = ___ROM_AT + SIZEOF(.data);
  .romp : AT(_romp_at)
  {
    __S_romp = _romp_at;
    LONG(___ROM_AT);
    LONG(_sdata);
    LONG(___data_size);
    LONG(0);
    LONG(0);
    LONG(0);
  } > m_data

  /* User_heap_stack section, used to check that there is enough RAM left */
 /* ._user_heap_stack :
  {
    . = ALIGN(4);
    PROVIDE ( end = . );
    PROVIDE ( _end = . );
    __heap_addr = .;
    . = . + __heap_size;
    . = . + __stack_size;
    . = ALIGN(4);
  } > m_data
 */
  .ARM.attributes 0 : { *(.ARM.attributes) }
}

Ответы [ 2 ]

0 голосов
/ 05 июня 2018

В заголовочном файле вы должны объявить константу:

extern const long int flashLocat;

Затем в один единственный исходный файл в глобальной области видимости вне какой-либо функции,Вы определяете константу:

const long int flashLocat = 0x00000800;
0 голосов
/ 05 июня 2018

Поскольку вы включаете flash.h в flash.c и main.c, переменная flashLocat определяется дважды.Вот почему вы получаете ошибку.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...