首先,这类似于:How are integer types converted implicitly?,但有一个不同的MISRA警告。
编译器不会生成MISRA错误,但静态分析工具会生成错误。我和工具制造商签了一张票。
假设:
#include <stdio.h>
enum Color {RED, VIOLET, BLUE, GREEN, YELLOW, ORANGE};
int main(void)
{
enum Color my_color;
my_color = BLUE;
if (my_color == YELLOW) // Generates MISRA violation, see below.
{
printf("Color is yellow.\n");
}
else
{
printf("Color is not yellow.\n");
}
return 0;
}
静态分析工具正在为if
语句生成MISRA违规:
MISRA-2004 Rule 10.1 violation: implicitly changing the signedness of an expression.
Converting "4", with underlying type "char" (8 bits, signed),
to type "unsigned int" (32 bits, unsigned) with different signedness.
编译器是正确的(不是识别缺陷)还是静态分析工具?
答案 0 :(得分:6)
根据C语言规范,表达式的类型:
typedef enum Colors {RED, VIOLET, BLUE, GREEN, YELLOW, ORANGE} Colors_t;
是signed int
。
同样根据语言,枚举项的值是可以包含整个枚举的最小单位。因此,在上面的枚举中,BLUE
的类型为signed char
。
当Colors_t
的变量与BLUE
进行比较时,静态分析工具会报告MISRA违规行为:
Colors_t my_color;
if (my_color == BLUE) // This generates a MISRA violation.
与signed int
相比,违规行为为signed char
。
此外,枚举项可以与其他枚举类型混合而不会出现错误,因为枚举不是唯一类型:
typedef enum Tree_Species {REDWOOD, CYPRUS, PALM, OAK, JUNIPER, SEGUARO} Tree_Species_t;
Tree_Species_t my_tree;
my_tree = BLUE;
答案 1 :(得分:0)
似乎有人(您的编译器或静态工具)认为您的枚举与 int 的大小和/或签名不同。
我的Green Hills编译器有一个选项--short-enum(使用可能最小的Enum类型),它将为上面的示例选择 char 类型。你的编译器有这样的选择吗?它启用了吗?编译器默认枚举为“非标准”值吗?
根据我的经验,默认情况下,静态工具默认遵循语言规范,这意味着它应该期望枚举为int-size。 (参见此链接以供参考:What is the size of an enum in C?)。由于大多数静态工具都会窥探您的编译器命令行用法,因此编译时选项可能会使静态分析器确信您的枚举小于 ints 。
我建议彻底检查您的编译器和静态分析器文档以解决冲突。密切关注你的构建过程(编译时选项,默认值等)。
答案 2 :(得分:-1)
可能是my_color
enum Color my_color;
尝试从中删除“枚举”:
Color my_color;
静态分析器可能会认为您正在声明一个单独的枚举,因此违反了10.1。 由于它的类型不同,MISRA规则禁止将一种类型分配给另一种类型。