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