我正在尝试使用IAR EWARM编译以下C代码但是我得到三个编译错误(错误[Pe028]:表达式必须具有常量值)。见下文:
#include<stdio.h>
#include<stdlib.h>
#include<stdint.h>
typedef uint8_t I2C_BusIdentifier;
typedef uint8_t I2C_SlaveAddress;
typedef enum {
I2C_BUS_STATE_UNINITIALIZED = 0,
I2C_BUS_STATE_GPIO_HARDWARE_READY,
I2C_BUS_STATE_READY_TO_OPERATE,
} I2C_BusState;
typedef struct BUS_I2C_BUS_INSTANCE_TYPE {
I2C_BusIdentifier BusIdentifer; // 0 for I2C0, 1 for I2C1
I2C_BusState CurrentState; // bus status
} I2C_Bus; // I²C Bus Instance Type, I2C_BusInstanceType
typedef struct DEVICE_I2C_GENERIC {
I2C_Bus* DeviceBusPointer;
I2C_SlaveAddress DeviceAddress;
} I2C_Device;
// inherits from I2C_Device
typedef struct DEVICE_ADC123 {
I2C_Device Device;
} ADC123_Device;
#define NUMBER_OF_I2C_PORTS 2
static I2C_Bus g_I2C_Bus[NUMBER_OF_I2C_PORTS] = {
{ 0, I2C_BUS_STATE_UNINITIALIZED, },
{ 1, I2C_BUS_STATE_UNINITIALIZED, },
};
I2C_Bus* const g_I2C_BusPtr_Port0 = &(g_I2C_Bus[0]);
I2C_Bus* const g_I2C_BusPtr_Port1 = &(g_I2C_Bus[1]);
const ADC123_Device g_Device_ADC123_U14 = {
{ g_I2C_BusPtr_Port0, 0xAE, }, // <--- Error[Pe028]: expression must have a constant value
};
const ADC123_Device g_Device_ADC123_U15 = {
{ g_I2C_BusPtr_Port1, 0x8A, }, // <--- Error[Pe028]: expression must have a constant value
};
const ADC123_Device g_Device_ADC123_U9 = {
{ g_I2C_BusPtr_Port1, 0xAA, }, // <--- Error[Pe028]: expression must have a constant value
};
#define NUMBER_OF_ADC123_DEVICES 3
const ADC123_Device* g_ADC123_Array[NUMBER_OF_ADC123_DEVICES] = {
&g_Device_ADC123_U14,
&g_Device_ADC123_U15,
&g_Device_ADC123_U9,
};
int main(void)
{
while(1);
}
但是,如果我直接使用g_I2C_Bus地址而不是通过g_I2C_BusPtr_PortX指针,那么一切都可以正常编译:
const ADC123_Device g_Device_ADC123_U14 = {
{ &(g_I2C_Bus[0]), 0xAE, },
};
const ADC123_Device g_Device_ADC123_U15 = {
{ &(g_I2C_Bus[1]), 0x8A, },
};
const ADC123_Device g_Device_ADC123_U9 = {
{ &(g_I2C_Bus[1]), 0xAA, },
};
我想使用const指针(g_I2C_BusPtr_Port0,g_I2C_BusPtr_Port1),因为它们在.h文件中是extern'd,而数组(g_I2C_Bus [])不会全局公开,而是在特定的.c文件中静态显示。 / p>
当定义/值应该是等价的时,为什么计算器对此不满意,因为它们引用相同的东西?
答案 0 :(得分:4)
这是C语言的限制。变量的值,如
int const a = 1;
不能用于常量表达式,如初始化器:
int b = a; /* Will not work */
甚至没有const
限定符。推理是编译器无法知道变量的值,即使它看起来完全无关紧要。 const
中的变量在C中不是常量,它只是你无法更改的变量。
全局变量的地址是另一回事。链接器完全控制这些变量所在的位置,并且可以为初始化器使用相同的信息。
解决方法是使用预处理器:
#define g_I2C_BusPtr_Port0 (&(g_I2C_Bus[0]))
答案 1 :(得分:1)
如果已经不清楚其他答案;正确的代码组织是:
// stuff.h
//
extern I2C_Bus *const g_I2C_BusPtr_Port0;
extern I2C_Bus *const g_I2C_BusPtr_Port1;
然后:
// stuff.c
//
#include "stuff.h"
I2C_Bus* const g_I2C_BusPtr_Port0 = &g_I2C_Bus[0];
I2C_Bus* const g_I2C_BusPtr_Port1 = &g_I2C_Bus[1];
const ADC123_Device g_Device_ADC123_U14 =
{ { &g_I2C_Bus[0], 0xAE } };
等等。
答案 2 :(得分:0)
静态对象的初始化器必须是常量表达式或字符串文字,如果例如ADC123_Device g_Device_ADC123_U14
是自动变量,则不会出现错误,例如,如果您在main
中声明它。我们可以通过转到draft C99 standard部分6.7.8
初始化来说明这一点:
具有静态存储持续时间的对象的初始值设定项中的所有表达式都应为 常量表达式或字符串文字。
第二种情况的原因是允许使用常量地址,我们可以从6.6
常量表达式中看到这一点:
初始值设定项中的常量表达式允许更多纬度。这样的常数 表达式应为或评估为以下之一:
并包含以下项目符号:
- 地址常量,或