【问题标题】:huge binary files with objcopy带有 objcopy 的巨大二进制文件
【发布时间】:2011-07-11 05:42:34
【问题描述】:

当我在 ARM9 处理器的基本 C 程序中定义全局变量时遇到问题。我正在使用 EABI GNU 编译器,从 12KB 精灵生成的二进制文件是 4GB!我认为问题出在我的分散文件上,但我无法理解它。

我有 256KB 的 ROM(基地址 0xFFFF0000)和 32KB 的 RAM(基地址 0x01000000)

SECTIONS {
  . = 0xFFFF0000;
  .text : {
    * (vectors);
    * (.text);
  }
  .rodata : { *(.rodata) }
  . = 0x01000000;
  sbss = .;
  .data : { *(.data) }
  .bss  : { *(.bss) }
  ebss = .;
  bssSize = ebss - sbss;
}

而我的程序如下:

int a=10;
int main() {
  int b=5;
  b = (a>b)? a : b;  
  return b;
};

如果我将 a 声明为局部变量,即没有 .data 部分,那么一切正常。 美好的。非常感谢任何帮助。

--2011 年 3 月 16 日--
任何人都可以帮助解决这个问题,我无处可去并阅读了手册、论坛等...
我的引导、编译命令和objcopy命令粘贴在下面

     .section "vectors"
reset:  b   start
undef:  b   undef
swi:    b   swi
pabt:   b   pabt
dabt:   b   dabt
    nop
irq:    b   irq
fiq:    b   fiq
  .text
start:
        ldr   sp, =0x01006000
        bl    main

stop:   b     stop

arm-none-eabi-gcc -mcpu=arm926ej-s -Wall -nostartfiles -Wall main.c boot.s -o main.elf -T \ scatter_file
arm-none-eabi-objcopy ./main.elf --output-target=binary ./main.bin
arm-none-eabi-objdump ./main.elf --disassemble-all > ./main.dis

【问题讨论】:

  • 如果删除行',文件有多大。 = 0x01000000;' ?
  • 如果我删除 ram 地址 (0x01000000),它的 364 字节

标签: c gcc linker arm objcopy


【解决方案1】:

我发现了问题。 objcopy 命令将尝试创建链接描述文件中描述的整个地址空间,从最低地址到最高地址,包括其间的所有地址。您可以告诉它只生成 ROM 代码,如下所示:

objcopy ./main.elf -j ROM --output-target=binary ./main.bin

我还稍微更改了链接描述文件

MEMORY {
  ram(WXAIL) : ORIGIN = 0x01000000, LENGTH = 32K    
  rom(RX)    : ORIGIN = 0xFFFF0000, LENGTH = 32K                    
}

SECTIONS {
  ROM : { 
    *(vectors);
    *(.text);
    *(.rodata);
  } > rom 

  RAM : {
    *(.data); 
    *(.bss);
  } > ram 
}

【讨论】:

  • 但是.data 段不是也很重要吗?我不能把它扔掉,如果你只是复制ROM部分,那里有初始化的数据会丢失。
  • 关于@user1273684 评论的任何想法?
【解决方案2】:

您正在创建一个文件,该文件将从地址 0x01000000 开始,并且至少包含地址 0xFFFF0000。难怪它将近4GB。你喜欢什么?如果您不想要数据段,请尝试使用选项 -R 删除它们(如果您正在准备 ROM 初始化文件,可能就是这种情况)。

【讨论】:

  • 我不明白。我认为链接器脚本定义了一个从 0xFFFF0000 开始的部分,其中包含文本、向量和只读数据(即 ROM),所以从 ffff0000 到所有这些的大小。然后从地址 0x01000000 开始为数据和 bss 定义另一个部分。这些部分之间的内容没有定义,为什么要填满它?如果我有一个全局变量,它只会填满它。我尝试使用 -R 但如何删除未定义的内存部分?
  • elf 文件好像没问题,只是二进制不正确。[link](readelf -l main.elf) Elf 文件类型为 EXEC(可执行文件)入口点 0xffff0000 有 2 个程序标头,从偏移量开始52个程序标题:类型偏移VirtAddr PhysAddr FileSiz MemSiz FLG对齐LOAD 0x008000为0x01000000为0x01000000 0x00004 0x00004 RW为0x8000 LOAD 0x010000 0xFFFF0000地址0xFFFF0000地址0x00148 0x00148 RE为0x8000到节段的映射:区段部00 ... 01。数据的.text [ \code]
【解决方案3】:

添加 (NOLOAD) 参数对我有用。例如

MEMORY {
  ram(WXAIL) : ORIGIN = 0x01000000, LENGTH = 32K    
  rom(RX)    : ORIGIN = 0xFFFF0000, LENGTH = 32K                    
}

SECTIONS {
  ROM : { 
    *(vectors);
    *(.text);
    *(.rodata);
  } > rom 

  RAM (NOLOAD) : {
    *(.data); 
    *(.bss);
  } > ram 
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2013-10-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-10-03
    • 1970-01-01
    • 1970-01-01
    • 2011-11-28
    相关资源
    最近更新 更多