GCC错误“导致节类型冲突”

为什么在尝试将两个(RAM)变量(只是它们的初始化值不同)放入同一部分时会出现编译器错误?

问题

C源:

int __attribute__((section(".in_my_data"))) _foo = 1; int __attribute__((section(".in_my_data"))) _bar = 0; 

(相关)GCC编译器输出:

 mcve/main.c:75:45: error: _bar causes a section type conflict 

链接描述文件在SECTIONS定义中包含以下行,但(致命)错误来自编译器,而不是链接器。

 .my_data : { *(.in_my_data) } > data 

更多的信息

更改C源以允许编译器使用两个部分允许编译通过,但是如果两个输入节被映射到相同的输出节,则链接器会生成错误。

C源:

 int __attribute__((section(".in_my_data_nonzero"))) _foo = 1; int __attribute__((section(".in_my_data_zero"))) _bar = 0; 

链接器脚本:

 .my_data : { *(.in_my_data*) } > data 

(相关)链接器输出:

 Link Error: attributes for input section '.in_my_data_nonzero' conflict with output section '.my_data' 

交换C源代码行的顺序只会改变哪个部分(第二个出现在C源代码中)是错误的。

对于初始化为零的变量,GCC编译器需要哪个属性,而不是对于初始化为非零的变量,反之亦然?

编译器是否尝试将初始化的变量放在.bss部分中,而不是初始化数据的.data部分? 或者是否有另一部分数据初始化为零?

相关问题

类似的问题似乎涵盖了内存类型(ROM与RAM)之间冲突的问题:

  • “部分类型冲突”在arm嵌入,是什么?
  • 如何解决“部分类型冲突”编译错误和使用gcc的section属性的最佳实践
  • 使用M2tklib和glcd获取“部分类型冲突”

…或将初始化的const数据放入NOLOAD输出节:

  • 由于GCC 4.8.2中的宏定义,“段类型冲突”

…或者对于原因仍然是一个谜,可能是相关的:

  • 对于相同定义的变量,段类型冲突

据我所知,上述所有内容似乎都没有我可以在这个问题上申请的答案。

警告:此答案可能仅适用于Microchip XC16编译器。

研究

编译器将属性分配给C变量,以便按如下方式将它们分配给特定部分(有关XC16特定信息,请参阅下面的注释1 )。

  • int a = 1; – 被分配到.data
  • int b = 0; – 被分配到.bss
  • int c; – 被分配到.bss

这些中的第一个和最后一个是有意义的: .data用于初始化数据, .bss用于未初始化数据。 但是,ANSI C启动时.bss数据也设置为零(参见注释2 )。

回答

看起来编译器包含初始化的变量,但其值等于.bss部分中的所有位0以及所有未初始化的位。

根据维基百科 :

实现还可以向BSS部分分配静态分配的变量和用仅由零值比特组成的值初始化的常量。

解决方法

GCC有一个选项-fno-zero-initialized-in-bss ,它可用于强制所有初始化为零的变量进入.data部分,如本答案中所述 。 这可以基于每个源文件应用,但不能应用于单个变量。

妄想

如果有一个__attribute__((**doload**))可以应用于强制编译器在.data而不是.bss放置零初始化变量,那将是很好的。

笔记

注1:XC16编译器可以使用.ndata.nbss来指示地址0x8000以下的近数据。

注2:使用__attribute__((noload))标记变量将导致变量从.bss部分中排除。 XC16生成一个特定的输出节,每个变量标记为(GUID?)唯一的名称。

默认情况下,GCC将对象类放在不同的部分,可执行代码转到.text,初始化数据到.data,静态数据转到.bss,还有一些比较模糊的。

如果您尝试强制不同类的两个对象属于同一节,GCC将引发此节错误。

解决方案是将它们放在不同的部分中,最后告诉链接器最终在同一部分中合并这些对象,例如:

 .boot : { // target ELF section will contain.... *(.boot) // executable code *(.boot_rodata) // constant data *(.boot_bss) // static data } > BOOT // to finally be place in this area