关于arm-none-eabi-gcc printf float问题

arm-none-eabi-gcc 的 printf 好像并不能支持double的输出。

使用double f = va_arg(ap, double);并将f内存打印出来时发现,

1. 当第一次出现浮点参数时,会出现4个字节的不明数据,从而导致后面的数据顺序全乱了

2. 还一点就是参数是8字节对齐的,va_arg一次,不管什么类型都会强制偏移8个字节。

3. 现在的办法是用个8字节的缓冲,先把拿数据再出来再重新拼接

usartPrintf("ft1:\n %f %f %f\n",f[0],f[1],f[2]);
	usartPrintf("ft1:\n %d\n %f %f\n",in[0],f[1],f[2]);
	usartPrintf("ft2:\n %f %d\n %f\n",f[0],in[1],f[2]);
	usartPrintf("ft3:\n %f %f %d\n",f[0],f[1],in[2]);

	extern void hexFloatTest(double f);
	hexFloatTest(f[0]);
	hexFloatTest(f[1]);
	hexFloatTest(f[2]);
ft1:
 Hex:[E8]	[3]	[0]	[0]	[0]	[0]	[0]	[20]	
 Hex:[DB]	[49]	[93]	[40]	[0]	[0]	[0]	[0]	
 Hex:[0]	[80]	[4B]	[40]	[0]	[0]	[0]	[A0]	

ft1:
 1334
 Hex:[0]	[80]	[4B]	[40]	[0]	[0]	[0]	[A0]	
 Hex:[1]	[BC]	[DD]	[3F]	[0]	[0]	[0]	[A0]	

ft2:
 Hex:[E8]	[3]	[0]	[0]	[0]	[0]	[0]	[20]	
 1083394523
 Hex:[1]	[BC]	[DD]	[3F]	[0]	[0]	[0]	[A0]	

ft3:
 Hex:[E8]	[3]	[0]	[0]	[0]	[0]	[0]	[20]	
 Hex:[DB]	[49]	[93]	[40]	[0]	[0]	[0]	[0]	
 1078689792
Hex:[0]	[0]	[0]	[20]	[DB]	[49]	[93]	[40]	
Hex:[0]	[0]	[0]	[0]	[0]	[80]	[4B]	[40]	
Hex:[0]	[0]	[0]	[A0]	[1]	[BC]	[DD]	[3F]

解决办法:

void uartPrintfloat(const char* format, ...)
{
	char str[30];
	char c = 0;
	char ch = 0;
	bool catchFloat = 0;
	union _floatHex
	{
		double _f;
		uint8_t _u8;
		int8_t _i8;
		uint16_t _u16;
		int16_t _i16;
		uint32_t _u32;
		int32_t _i32;
		uint64_t _u64;
		int64_t _i64;
		uint8_t hex[8];
	}ftmp;
	va_list ap;
	va_start(ap, format);
	char *p = 0;
	while ((c = *format))
	{
		switch (c)
		{
		case '%':
			ch = *++format;
			switch (ch)
			{
			case 'd':
			{
				if(catchFloat)
				{
					p = va_arg(ap, char[8]);
					memcpy(&ftmp.hex[4],p,4);
				}
				else
				{
					p = va_arg(ap, char[8]);
					memcpy(&ftmp.hex[0],p,8);
				}
				myitoa(ftmp._i64, str, 10);
				Serial_puts(&serialHandle[serial1], str);
				break;
			}
			case 'f':
				
				if (!catchFloat)
				{
					catchFloat = 1;
					/**FIXME 第一次遇到浮点数据时,前面有4个不明字节数据, 跳过它! */
					/**NOTE 不管什么类型都是8字节对齐的! */
					p = va_arg(ap, char[8]);
					memcpy(ftmp.hex,&p[4],4);
				}
				p = va_arg(ap, char[8]);
				memcpy(&ftmp.hex[4],p,4);

				// hexFloatTest(f);
				int32_t n;
				n = ftmp._f;
				myitoa(n, str, 10);
				Serial_puts(&serialHandle[serial1], str);
				Serial_putc(&serialHandle[serial1], '.');
				n = (ftmp._f - n) * 1000000;
				if (n < 0)
					n = (unsigned)-n;
				myitoa(n, str, 10);
				Serial_puts(&serialHandle[serial1], str);
				memcpy(ftmp.hex,&p[4],4);
				break;
			case '%':
				Serial_putc(&serialHandle[serial1], '%');
				break;
			default:
				Serial_putc(&serialHandle[serial1], '%');
				Serial_putc(&serialHandle[serial1], ch);
				break;
			}
			break;
		default:
			Serial_putc(&serialHandle[serial1], c);
			break;
		}
		format++;
	}
	va_end(ap);
	Serial_flush(&serialHandle[serial1]);
}

重新测试:

double f[3]={1234.464f,55.0f,0.4646f};
	uint32_t in[3]={1334,56463,846};
	// usartPrintf("float test1:%f--%f---%f##\n",1234.0154f,12.0f,34.464f);
	
	// usartPrintf("ft1:\n %f %f %f\n",f[0],f[1],f[2]);
	// usartPrintf("ft1:\n %d\n %f %f\n",in[0],f[1],f[2]);
	// usartPrintf("ft2:\n %f %d\n %f\n",f[0],in[1],f[2]);
	// usartPrintf("ft3:\n %f %f %d\n",f[0],f[1],in[2]);

	// extern void hexFloatTest(double f);
	// hexFloatTest(f[0]);
	// hexFloatTest(f[1]);
	// hexFloatTest(f[2]);
	extern void uartPrintfloat(const char* format, ...);
	uartPrintfloat("ft1:\n %f %f %f\n",f[0],f[1],f[2]);
	uartPrintfloat("ft1:\n %d\n %f %f\n",in[0],f[1],f[2]);
	uartPrintfloat("ft2:\n %f %d\n %f %d\n",f[0],in[1],f[2], in[2]);
	uartPrintfloat("ft3:\n %f %f %d\n",f[0],f[1],in[2]);
ft1:
 1234.463989 55.0 0.464599
ft1:
 1334
 0.464599 0.464599
ft2:
 1234.463989 56463
 0.464599 846
ft3:
 1234.463989 55.0 846

貌似可以用了,字符串参数类型没有测试,估计后面还有坑

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: gcc-arm-none-eabi是一款ARM Cortex-M处理器的编译器工具链,可以用于开发嵌入式系统。安装gcc-arm-none-eabi需要以下步骤: 1. 下载gcc-arm-none-eabi的安装包,可以从官网或者其他镜像站点下载。 2. 解压安装包到指定目录,例如/opt/gcc-arm-none-eabi。 3. 配置环境变量,将gcc-arm-none-eabi的bin目录添加到PATH环境变量中,例如在~/.bashrc文件中添加以下行: export PATH=/opt/gcc-arm-none-eabi/bin:$PATH 4. 测试安装是否成功,可以在终端中输入arm-none-eabi-gcc --version命令,如果输出gcc-arm-none-eabi的版本信息,则安装成功。 注意:在安装过程中可能会遇到一些依赖库缺失的问题,需要根据提示安装相应的依赖库。 ### 回答2: gcc-arm-none-eabi是一款在嵌入式系统开发中广泛使用的开源工具链。它可以用于开发ARM Cortex-M微控制器,并支持多种平台和操作系统。下面介绍如何在Ubuntu Linux上安装gcc-arm-none-eabi。 第一步是安装依赖项。在终端运行以下命令: ``` sudo apt-get install build-essential git lib32z1 lib32ncurses5 libxml2-dev ``` 第二步是从官网下载gcc-arm-none-eabi的tar包。可以通过以下URL访问官方网站:https://developer.arm.com/tools-and-software/open-source-software/developer-tools/gnu-toolchain/gnu-rm/downloads 在下载页面,找到最新版本的tar包并下载。下载完成后,将tar包解压到任意目录下: ``` tar -xf gcc-arm-none-eabi-9-2020-q2-update-linux.tar.bz2 ``` 然后将解压的目录添加到环境变量中: ``` export PATH=$PATH:/path/to/gcc-arm-none-eabi-9-2020-q2-update/bin ``` 完成后可以输入以下命令检查是否成功: ``` arm-none-eabi-gcc --version ``` 如果可以正常打印出版本号,说明gcc-arm-none-eabi已经安装成功。 值得注意的是,上面的步骤仅适用于在Ubuntu上安装gcc-arm-none-eabi。如果在其他操作系统上使用,可能需要修改一些步骤或注明其他操作系统的安装说明。 ### 回答3: gcc-arm-none-eabi是一款开源的ARM嵌入式交叉编译器,是ARM架构下的编程必备工具之一。下面是gcc-arm-none-eabi安装过程的详细介绍: 1. 下载gcc-arm-none-eabi 在官方网站上下载最新的gcc-arm-none-eabi的版本。这些版本可能是以tar或zip文件的形式出现,根据自己的操作系统选择相应的版本。 2. 解压文件 在终端中进入下载目录,解压文件,完成后进入解压得到的文件夹。 3. 配置环境变量 打开终端,输入以下命令: export PATH=$PATH:/path/to/gcc-arm-none-eabi/bin 其中“/path/to/”是指gcc-arm-none-eabi文件夹的路径。此时,gcc-arm-none-eabi就已经可以在任何目录下使用了。 4. 测试工具链 通过编译一个简单的程序来测试gcc-arm-none-eabi是否已成功安装。 在终端中输入以下代码: #include <stdio.h> int main(void) { printf("Hello, World!\n"); return 0; } 将文件保存为hello.c,并在终端中进入该文件所在的目录。 输入以下命令来编译程序: arm-none-eabi-gcc -g -O0 -o hello hello.c 输入以下命令来运行程序: arm-none-eabi-run hello 控制台会输出“Hello, World!”,说明gcc-arm-none-eabi已经安装成功。 综上所述,gcc-arm-none-eabi的安装过程需要下载最新的版本、解压文件、配置环境变量、测试工具链。这个过程需要仔细进行,以确保编程时能够正常地使用这个工具。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值