我有以下代码:
#include <ctype.h>
void myfn(void)
{
uint8_t any0 = 'c';
char any1 = 'c';
if (isprint(any0))
{
return;
}
if (isprint(any1))
{
return;
}
}
当我使用 gcc for arm 编译它时,出现以下错误:
error: array subscript has type 'char' [-Werror=char-subscripts]
if (isprint(any1))
^
如果我将unit8_t
传递给isprint
,编译器会很高兴,但如果我将其传递给char
则不会。
isprint
的原型是:
int isprint(int c);
它期望一个int
作为参数,我给它一个char
. 我希望它会抱怨参数的类型,而不是与"数组下标"无关的东西。
如果我将调用更改为:
if (isprint((uint8_t)any1))
我忽略了什么吗?
我使用的编译器是:
GNU C (15:4.9.3+svn231177-1) version 4.9.3 20150529 (prerelease) (arm-none-eabi)
compiled by GNU C version 5.2.1 20151129, GMP version 6.1.0, MPFR version 3.1.3, MPC version 1.0.3
warning: MPFR header version 3.1.3 differs from library version 3.1.4.
命令行选项:
'-v' '-fshort-enums' '-specs=nosys.specs' '-specs=nano.specs'
'-mfloat-abi=soft' '-save-temps' '-Werror' '-Wpedantic' '-pedantic-errors' '-mthumb' '-fno-builtin' '-mcpu=cortex-m0' '-Wall' '-std=gnu99' '-ffunction-sections'
'-fdata-sections' '-fomit-frame-pointer' '-mabi=aapcs' '-fno-unroll-loops' '-ffast-math' '-ftree-vectorize' '-Og' '-g'
如果我使用 gcc 编译器为 AVR/usr/bin/avr-gcc
编译相同的代码(使用类似的命令行选项并且我故意添加了选项-Werror=char-subscripts
(,它不会抱怨isprint
.所以它似乎与ARM编译器有关。
事实证明,ARM的isprint
实现是一个宏,更重要的是,头文件ctype.h包含以下注释:
These macros are intentionally written in a manner that will trigger a gcc -Wall warning if the user mistakenly passes a 'char' instead of an int containing an 'unsigned char'.
对于 AVR,头文件 ctype.h 包含:
extern int isprint(int __c) __ATTR_CONST__;
这解释了针对 AVR 或 ARM 进行编译时的行为差异。