STM32实现气压传感器测量(BMP180)

news2025/1/23 7:54:06

目录

0.接线设计

1.功能描述

2.四种方式实现大气压采集

3.模块选择

4.编程环境

5.模块主要参数

6.代码实现

        1)标准库模拟IIC实现气压值采集

         2)标准库硬件IIC实现气压值采集

         3)HAL库模拟IIC实现气压值采集

        4)HAL库硬件IIC实现气压值采集


0.接线设计

硬件接口:IIC_SCL -- PB6
                  IIC_SDA -- PB7

串口使用串口1:PA9和PA10

1.功能描述

        通过STM32实现BMP180大气压传感器模块数据采集,并将采集的数据在串口显示。

        本传感器支持IIC通信,实物图如下:

         网上建议在SCL和SDA引脚上,上拉两个4.7K电阻,保证通信稳定,但我实现没有增加这两个电阻,通信正常,可能也有不稳定情况,大家自行判断。我没有上拉电阻,信息采集正常。

2.四种方式实现大气压采集

        1)标准库模拟IIC实现气压值采集;

        2)标准库硬件IIC实现气压值采集;

        3)HAL库模拟IIC实现气压值采集;

        4)HAL库硬件IIC实现气压值采集。

3.模块选择

        ·STM32F103C8T6(其它型号也是一样)

        ·BMP180气压传感器模块

        ·USB-TTL串口调试助手

4.编程环境

        STM32CubeMX(工程可视化配置环境IDE,HAL库使用)

        keil5(编程环境)

5.模块主要参数

        BMP180有一个产品文档,大家可以参考文档实现程序。

         测量流程图。

        温度和大气压测量:

        海拔测量:

6.代码实现

        1)标准库模拟IIC实现气压值采集

//********************************************************************
u16 bmp180ReadTemp(void)
{
	int cnt = 0;
	IIC_Start();                  //起始信号
	IIC_Send_Byte(BMP180_SlaveAddress);   //发送设备地址+写信号   BMP180_SlaveAddress  0xee	气压传感器器件地址
	cnt = 0;
	while(IIC_Wait_Ack())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		delay_ms(1);
		if(cnt>=100)return 1;
	} 
	IIC_Send_Byte(0xF4);	          // write register address
	cnt = 0;
	while(IIC_Wait_Ack())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		delay_ms(1);
		if(cnt>=100)return 1;
	} 
	IIC_Send_Byte(0x2E);       	// write register data for temp
	cnt = 0;
	while(IIC_Wait_Ack())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		delay_ms(1);
		if(cnt>=100)return 1;
	} 
	IIC_Stop();                   //发送停止信号
	delay_ms(10);	// max time is 4.5ms
	return Multiple_read(0xF6);
}
//*************************************************************
u16 bmp180ReadPressure(void)
{
	int cnt = 0;
	IIC_Start();                   //起始信号
	IIC_Send_Byte(BMP180_SlaveAddress);   //发送设备地址+写信号
	cnt = 0;
	while(IIC_Wait_Ack())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		delay_ms(1);
		if(cnt>=100)return 1;
	} 
	IIC_Send_Byte(0xF4);	          // write register address
	cnt = 0;
	while(IIC_Wait_Ack())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		delay_ms(1);
		if(cnt>=100)return 1;
	} 
	IIC_Send_Byte(0x34);       	  // write register data for pressure
	cnt = 0;
	while(IIC_Wait_Ack())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		delay_ms(1);
		if(cnt>=100)return 1;
	} 
	IIC_Stop();                    //发送停止信号
	delay_ms(20);    	                  // max time is 4.5ms
	return Multiple_read(0xF6);//pressure;	
}
 
//**************************************************************
 
//初始化BMP085,根据需要请参考pdf进行修改**************
void Init_BMP180()
{
	IIC_Init();
	ac1 = Multiple_read(0xAA);
	ac2 = Multiple_read(0xAC);
	ac3 = Multiple_read(0xAE);
	ac4 = Multiple_read(0xB0);
	ac5 = Multiple_read(0xB2);
	ac6 = Multiple_read(0xB4);
	b1 =  Multiple_read(0xB6);
	b2 =  Multiple_read(0xB8);
	mb =  Multiple_read(0xBA);
	mc =  Multiple_read(0xBC);
	md =  Multiple_read(0xBE);
}
//***********************************************************************
void bmp180Convert()
{
	unsigned int ut;
	unsigned long up;
	long x1, x2, b5, b6, x3, b3, p;
	unsigned long b4, b7;
 
	ut = bmp180ReadTemp();	   // 读取温度
	up = bmp180ReadPressure();  // 读取压强    return pressure;	
	//*************
	x1 = (((long)ut - (long)ac6)*(long)ac5) >> 15;
	x2 = ((long) mc << 11) / (x1 + md);
	b5 = x1 + x2;
	result_UT = ((b5 + 8) >> 4);
	//*************		
	b6 = b5 - 4000;
	                         // Calculate B3
	x1 = (b2 * (b6 * b6)>>12)>>11;
	x2 = (ac2 * b6)>>11;
	x3 = x1 + x2;
	b3 = (((((long)ac1)*4 + x3)<<OSS) + 2)>>2;	
	                         // Calculate B4
	x1 = (ac3 * b6)>>13;
	x2 = (b1 * ((b6 * b6)>>12))>>16;
	x3 = ((x1 + x2) + 2)>>2;
	b4 = (ac4 * (unsigned long)(x3 + 32768))>>15;
	
	b7 = ((unsigned long)(up - b3) * (50000>>OSS));
	if (b7 < 0x80000000)
	p = (b7<<1)/b4;
	else
	p = (b7/b4)<<1;
	
	x1 = (p>>8) * (p>>8);
	x1 = (x1 * 3038)>>16;
	x2 = (-7357 * p)>>16;
	result_UP = p+((x1 + x2 + 3791)>>4);
}
 
 
/****************************************************************
   函数名称:BMP085_Get_Altitude			    
   函数功能:获取海拔高度值
   入口参数:无
   出口参数:altitude //int型  2字节,当前海拔高度值
   备 注:	 返回的高度值单位为厘米,调用时再换算成带小数的以米为单位的高度值
*****************************************************************/
double BMP085_Get_Altitude()
{
	double altitude;
	//pressure=(float)pressure;				//获取气压值
	altitude=44330.0*(1-pow((double)result_UP/101325,1/5.255));  	//根据芯片手册提供的公式计算海拔高度
	//altitude*=100;	 		//转换成厘米单位的高度值,调用时再换算成带小数的高度值,提高精度
	return altitude;
}
 

         实现效果:

         2)标准库硬件IIC实现气压值采集

//向EEPROM写入一个字节
int  BMP180_Byte_Write(uint8_t addr,uint8_t data) 
{
	I2CTimeout = I2CT_LONG_TIMEOUT;
	while (I2C_GetFlagStatus(I2C1,  I2C_FLAG_BUSY))  //检查I2C总线是否繁忙
	{
    if((I2CTimeout--) == 0) return 13;
  } 
	//发送起始信号
	I2C_GenerateSTART(BMP180_I2C,ENABLE);
	//检测EV5事件
	I2CTimeout = I2CT_LONG_TIMEOUT;
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_MODE_SELECT)==ERROR)
	{
    if((I2CTimeout--) == 0) return 14;
  } 
	//发送设备写地址
	I2C_Send7bitAddress(BMP180_I2C,BMP180_I2C_Address,I2C_Direction_Transmitter);
	//检测EV6事件
	I2CTimeout = I2CT_LONG_TIMEOUT;
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)==ERROR)
	{
    if((I2CTimeout--) == 0) return 15;
  } 
	//发送要操作设备内部的地址
	I2C_SendData(BMP180_I2C,addr);
	I2CTimeout = I2CT_LONG_TIMEOUT;
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING )==ERROR)
	{
    if((I2CTimeout--) == 0) return 16;
  } 
  I2C_SendData(BMP180_I2C,data);
	//检测EV8_2事件
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTED )==ERROR);
	//发送停止信号
	I2C_GenerateSTOP(BMP180_I2C,ENABLE);
	return 0;
}


//向EEPROM读取多个字节
uint32_t BMP180_Read(uint8_t *data,uint8_t addr,uint16_t Num_ByteToRead)
{
	 I2CTimeout = I2CT_LONG_TIMEOUT;
  //判断IIC总线是否忙碌
  while(I2C_GetFlagStatus(BMP180_I2C, I2C_FLAG_BUSY))   
  {
    if((I2CTimeout--) == 0) return 1;
  } 
	
	I2CTimeout = I2CT_FLAG_TIMEOUT;
	//发送起始信号
	I2C_GenerateSTART(BMP180_I2C,ENABLE);
	//检测EV5事件
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_MODE_SELECT )==ERROR)
  {
        if((I2CTimeout--) == 0) return 7;
   } 
	
	I2CTimeout = I2CT_FLAG_TIMEOUT;
	//发送设备写地址
	I2C_Send7bitAddress(BMP180_I2C,BMP180_I2C_Address,I2C_Direction_Transmitter);
	//检测EV6事件等待从机应答
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED )==ERROR)
 {
        if((I2CTimeout--) == 0) return 8;
  }
  
	I2CTimeout = I2CT_FLAG_TIMEOUT;
	//发送要操作设备内部存储器的地址
	I2C_SendData(BMP180_I2C,addr);
	//检测EV8事件
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING )==ERROR)
 {
        if((I2CTimeout--) == 0) return 9;
  }
	I2CTimeout = I2CT_FLAG_TIMEOUT;
	//发送起始信号
	I2C_GenerateSTART(BMP180_I2C,ENABLE);
	//检测EV5事件
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_MODE_SELECT )==ERROR)
	{
        if((I2CTimeout--) == 0) return 10;
   }
	I2CTimeout = I2CT_FLAG_TIMEOUT;	 
	//发送设备读地址
	I2C_Send7bitAddress(BMP180_I2C,BMP180_I2C_Address+1,I2C_Direction_Receiver);
	//检测EV6事件
	while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED )==ERROR)
	{
       if((I2CTimeout--) == 0) return 11;
   }
	 
	while(Num_ByteToRead--)
	{
		//是否是最后一个字节,若是则发送非应答信号
		if( Num_ByteToRead==0)
	 {
		 //发送非应答信号
		 I2C_AcknowledgeConfig(BMP180_I2C,DISABLE);
		 //发送停止信号
	   I2C_GenerateSTOP(BMP180_I2C,ENABLE);
	 }
	 
	 I2CTimeout = I2CT_FLAG_TIMEOUT;	 
	 //检测EV7事件
   while( I2C_CheckEvent(BMP180_I2C,I2C_EVENT_MASTER_BYTE_RECEIVED )==ERROR)
   {
       if((I2CTimeout--) == 0) return 12;
   }
	 
    *data=I2C_ReceiveData(BMP180_I2C);
	  data++; 
	 
	}
	
	//重新开启应答信号
	I2C_AcknowledgeConfig(BMP180_I2C,ENABLE);
  return 0;
}


void BMP180_Init(void)
{
	u8 msb, lsb;
  delay_ms(200);     // 1s,这里的延时很重要,上电后延时,没有错误的冗余设计
	
	BMP180_Read(&msb,0xAA,1);  
	BMP180_Read(&lsb,0xAB,1); 
	ac1 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xAC,1);  
	BMP180_Read(&lsb,0xAD,1); 
	ac2 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xAE,1);  
	BMP180_Read(&lsb,0xAF,1); 
	ac3 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xB0,1);  
	BMP180_Read(&lsb,0xB1,1); 
	ac4 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xB2,1);  
	BMP180_Read(&lsb,0xB3,1); 
	ac5 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xB4,1);  
	BMP180_Read(&lsb,0xB5,1); 
	ac6 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xB6,1);  
	BMP180_Read(&lsb,0xB7,1); 
	b1 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xB8,1);  
	BMP180_Read(&lsb,0xB9,1); 
	b2 = (msb << 8)|lsb;
	BMP180_Read(&msb,0xBA,1);  
	BMP180_Read(&lsb,0xBB,1); 
	mb = (msb << 8)|lsb;
	BMP180_Read(&msb,0xBC,1);  
	BMP180_Read(&lsb,0xBD,1); 
	mc = (msb << 8)|lsb;
	BMP180_Read(&msb,0xBE,1);  
	BMP180_Read(&lsb,0xBF,1); 
	md = (msb << 8)|lsb;
}

//********************************************************************

u16 bmp180ReadTemp(void)
{
	u8 msb, lsb;
	//BMP180_Byte_Write(BMP180_SlaveAddress);   //发送设备地址+写信号   BMP180_SlaveAddress  0xee	气压传感器器件地址
	BMP180_Byte_Write(0xF4,0x2E);
	//BMP180_Byte_Write(0x2E);
	
	delay_ms(20);	// max time is 4.5ms
	BMP180_Read(&msb,0xF6,1);  
	BMP180_Read(&lsb,0xF7,1); 
	
	return (msb << 8)|lsb;
}
//*************************************************************

u16 bmp180ReadPressure(void)
{
	u8 msb, lsb;
	//BMP180_Byte_Write(BMP180_SlaveAddress);   //发送设备地址+写信号
	BMP180_Byte_Write(0xF4,0x34);	          // write register address
	//BMP180_Byte_Write(0x34);       	  	// write register data for pressure
	
	delay_ms(20);    	                  // max time is 4.5ms
	BMP180_Read(&msb,0xF6,1);  
	BMP180_Read(&lsb,0xF7,1); 
	return (msb << 8)|lsb;
}

实现效果:

         3)HAL库模拟IIC实现气压值采集

//********************************************************************
uint16_t bmp180ReadTemp(void)
{
	int cnt = 0;
	BMP180_Start();                  //起始信号
	BMP180_SendByte(BMP180_SlaveAddress);   //发送设备地址+写信号   BMP180_SlaveAddress  0xee	气压传感器器件地址
	cnt = 0;
	while(BMP180_RecvACK())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		Delay_mms(1); 
		if(cnt>=100)return 1;
	} 
	BMP180_SendByte(0xF4);	          // write register address
	cnt = 0;
	while(BMP180_RecvACK())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		Delay_mms(1);
		if(cnt>=100)return 1;
	} 
	BMP180_SendByte(0x2E);       	// write register data for temp
	cnt = 0;
	while(BMP180_RecvACK())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		Delay_mms(1);
		if(cnt>=100)return 1;
	} 
	BMP180_Stop();                   //发送停止信号
	Delay_mms(10);	// max time is 4.5ms
	return Multiple_read(0xF6);
}
//*************************************************************
uint16_t bmp180ReadPressure(void)
{
	int cnt = 0;
	BMP180_Start();                   //起始信号
	BMP180_SendByte(BMP180_SlaveAddress);   //发送设备地址+写信号
	cnt = 0;
	while(BMP180_RecvACK())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		Delay_mms(1);
		if(cnt>=100)return 1;
	} 
	BMP180_SendByte(0xF4);	          // write register address
	cnt = 0;
	while(BMP180_RecvACK())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		Delay_mms(1);
		if(cnt>=100)return 1;
	} 
	BMP180_SendByte(0x34);       	  // write register data for pressure
	cnt = 0;
	while(BMP180_RecvACK())								//这边需要增加超时退出,避免卡死
	{
		cnt++;
		Delay_mms(1);
		if(cnt>=100)return 1;
	} 
	BMP180_Stop();                    //发送停止信号
	Delay_mms(20);    	                  // max time is 4.5ms
	return Multiple_read(0xF6);//pressure;	
}


 
//初始化BMP180,根据需要请参考pdf进行修改**************
void Init_BMP180()
{
	char str[100] = "";
	ac1 = Multiple_read(0xAA);
	ac2 = Multiple_read(0xAC);
	ac3 = Multiple_read(0xAE);
	ac4 = Multiple_read(0xB0);
	ac5 = Multiple_read(0xB2);
	ac6 = Multiple_read(0xB4);
	b1 =  Multiple_read(0xB6);
	b2 =  Multiple_read(0xB8);
	mb =  Multiple_read(0xBA);
	mc =  Multiple_read(0xBC);
	md =  Multiple_read(0xBE);
	
	sprintf(str,"ac1=%d ac2=%d ac3=%d ac4=%d ac5=%d ac6=%d b1=%d b2=%d mb=%d mc=%d md=%d\r\n",(int)ac1,(int)ac2,(int)ac3,(int)ac4,(int)ac5,(int)ac6,(int)b1,(int)b2,(int)mb,(int)mc,(int)md);
  HAL_UART_Transmit(&huart1, (uint8_t *)str, (uint8_t)strlen(str), 5000);
}
//***********************************************************************
void bmp180Convert()
{
	unsigned int ut;
	unsigned long up;
	long x1, x2, b5, b6, x3, b3, p;
	unsigned long b4, b7;
 
	ut = bmp180ReadTemp();	   // 读取温度
	up = bmp180ReadPressure();  // 读取压强    return pressure;	
	//*************
	x1 = (((long)ut - (long)ac6)*(long)ac5) >> 15;
	x2 = ((long) mc << 11) / (x1 + md);
	b5 = x1 + x2;
	result_UT = ((b5 + 8) >> 4);
	//*************		
	b6 = b5 - 4000;
	                         // Calculate B3
	x1 = (b2 * (b6 * b6)>>12)>>11;
	x2 = (ac2 * b6)>>11;
	x3 = x1 + x2;
	b3 = (((((long)ac1)*4 + x3)<<OSS) + 2)>>2;	
	                         // Calculate B4
	x1 = (ac3 * b6)>>13;
	x2 = (b1 * ((b6 * b6)>>12))>>16;
	x3 = ((x1 + x2) + 2)>>2;
	b4 = (ac4 * (unsigned long)(x3 + 32768))>>15;
	
	b7 = ((unsigned long)(up - b3) * (50000>>OSS));
	if (b7 < 0x80000000)
	p = (b7<<1)/b4;
	else
	p = (b7/b4)<<1;
	
	x1 = (p>>8) * (p>>8);
	x1 = (x1 * 3038)>>16;
	x2 = (-7357 * p)>>16;
	result_UP = p+((x1 + x2 + 3791)>>4);
}
 
 
/****************************************************************
   函数名称:BMP085_Get_Altitude			    
   函数功能:获取海拔高度值
   入口参数:无
   出口参数:altitude //int型  2字节,当前海拔高度值
   备 注:	 返回的高度值单位为厘米,调用时再换算成带小数的以米为单位的高度值
*****************************************************************/
//double BMP085_Get_Altitude()
//{
//	double altitude;
//	//pressure=(float)pressure;				//获取气压值
//	altitude=44330.0*(1-pow((double)pressure/101325,1/5.255));  	//根据芯片手册提供的公式计算海拔高度
//	//altitude*=100;	 		//转换成厘米单位的高度值,调用时再换算成带小数的高度值,提高精度
//	return altitude;
//}

 实现效果:

        4)HAL库硬件IIC实现气压值采集

 

//	在从设备指定内存中写数据
void BMP180_WR_CMD(uint8_t cmd,uint8_t data)
{
	//HAL_I2C_Master_Transmit(&hi2c1, BMP180_I2C_Address, &cmd,1, 0x100);
	HAL_I2C_Mem_Write(&hi2c1, BMP180_I2C_Address, cmd, I2C_MEMADD_SIZE_8BIT, &data, 1, 0x100);
	//HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size, uint32_t Timeout);
	//HAL_I2C_Mem_Write(&hi2c1 ,0x46,0x00,I2C_MEMADD_SIZE_8BIT,&cmd,1,0x100);
}

//读数据
void BMP180_READ_DATA(uint8_t cmd,uint8_t *data,uint16_t size)
{
	HAL_I2C_Master_Transmit(&hi2c1, BMP180_I2C_Address, &cmd,size, 0x100);
	HAL_I2C_Master_Receive(&hi2c1, BMP180_I2C_Address+1, data,size, 0x100);
}


/**
 @brief BH1750初始化函数
 @param 无
 @return 无
*/
void BMP180_Init(void)
{
	char str[100] = "";
	uint8_t msb, lsb;
  HAL_Delay(200);     // 1s,这里的延时很重要,上电后延时,没有错误的冗余设计
	
	BMP180_READ_DATA(0xAA,&msb,1);
	BMP180_READ_DATA(0xAB,&lsb,1);
	ac1 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xAC,&msb,1);
	BMP180_READ_DATA(0xAD,&lsb,1); 
	ac2 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xAE,&msb,1);
	BMP180_READ_DATA(0xAF,&lsb,1); 
	ac3 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xB0,&msb,1);
	BMP180_READ_DATA(0xB1,&lsb,1); 
	ac4 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xB2,&msb,1);
	BMP180_READ_DATA(0xB3,&lsb,1); 
	ac5 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xB4,&msb,1);
	BMP180_READ_DATA(0xB5,&lsb,1); 
	ac6 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xB6,&msb,1);
	BMP180_READ_DATA(0xB7,&lsb,1); 
	b1 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xB8,&msb,1);
	BMP180_READ_DATA(0xB9,&lsb,1); 
	b2 = (msb << 8)|lsb;
	BMP180_READ_DATA(0xBA,&msb,1);
	BMP180_READ_DATA(0xBB,&lsb,1);
	mb = (msb << 8)|lsb;
	BMP180_READ_DATA(0xBC,&msb,1);
	BMP180_READ_DATA(0xBD,&lsb,1);
	mc = (msb << 8)|lsb;
	BMP180_READ_DATA(0xBE,&msb,1);
	BMP180_READ_DATA(0xBF,&lsb,1);
	md = (msb << 8)|lsb;
	
	sprintf(str,"ac1=%d ac2=%d ac3=%d ac4=%d ac5=%d ac6=%d b1=%d b2=%d mb=%d mc=%d md=%d\r\n",(int)ac1,(int)ac2,(int)ac3,(int)ac4,(int)ac5,(int)ac6,(int)b1,(int)b2,(int)mb,(int)mc,(int)md);
  HAL_UART_Transmit(&huart1, (uint8_t *)str, (uint8_t)strlen(str), 5000);
}


uint16_t bmp180ReadTemp(void)
{
	uint8_t msb, lsb;
	//发送设备地址+写信号   BMP180_SlaveAddress  0xee	气压传感器器件地址
	BMP180_WR_CMD(0xF4,0x2E);
	
	HAL_Delay(20);	// max time is 4.5ms
	BMP180_READ_DATA(0xF6,&msb,1);  
	BMP180_READ_DATA(0xF7,&lsb,1); 
	
	return (msb << 8)|lsb;
}
//*************************************************************

uint16_t bmp180ReadPressure(void)
{
	uint8_t msb, lsb;
	//发送设备地址+写信号
	BMP180_WR_CMD(0xF4,0x34);
	
	HAL_Delay(20);    	                  // max time is 4.5ms
	BMP180_READ_DATA(0xF6,&msb,1);  
	BMP180_READ_DATA(0xF7,&lsb,1); 
	return (msb << 8)|lsb;
}


//***********************************************************************

void bmp180Convert()
{
	unsigned int ut;
	unsigned long up;
	long x1, x2, b5, b6, x3, b3, p;
	unsigned long b4, b7;
 
	ut = bmp180ReadTemp();	   // 读取温度
	up = bmp180ReadPressure();  // 读取压强    return pressure;	
	//
	x1 = (((long)ut - (long)ac6)*(long)ac5) >> 15;
	x2 = ((long) mc << 11) / (x1 + md);
	b5 = x1 + x2;
	result_UT = ((b5 + 8) >> 4);
	//	
	b6 = b5 - 4000;
	                         // Calculate B3
	x1 = (b2 * (b6 * b6)>>12)>>11;
	x2 = (ac2 * b6)>>11;
	x3 = x1 + x2;
	b3 = (((((long)ac1)*4 + x3)<<OSS) + 2)>>2;	
	                         // Calculate B4
	x1 = (ac3 * b6)>>13;
	x2 = (b1 * ((b6 * b6)>>12))>>16;
	x3 = ((x1 + x2) + 2)>>2;
	b4 = (ac4 * (unsigned long)(x3 + 32768))>>15;
	
	b7 = ((unsigned long)(up - b3) * (50000>>OSS));
	if (b7 < 0x80000000)
	p = (b7<<1)/b4;
	else
	p = (b7/b4)<<1;
	
	x1 = (p>>8) * (p>>8);
	x1 = (x1 * 3038)>>16;
	x2 = (-7357 * p)>>16;
	result_UP = p+((x1 + x2 + 3791)>>4);
}

实现效果:

资源下载链接:(20条消息) STM32实现气压传感器测量(BMP180)资源-CSDN文库

吾芯电子工作室

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/737522.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

分享一次腾讯云轻量应用服务器被攻击

腾讯云轻量应用服务器&#xff0c;centOS。在上面装了redis、rabbit mq等服务&#xff0c;开着端口&#xff0c;结果被入侵了。 发现问题是通过腾讯云发来的邮件&#xff0c; 首先进到主机安全控制台&#xff0c;左侧这里进主机列表 然后可以看到自己的主机情况&#xff0c;防…

ROS:机器人系统仿真

目录 一、概念二、作用2.1仿真优势:2.2仿真缺陷: 三、组件3.1URDF3.2rviz3.3gazebo 一、概念 通过计算机对实体机器人系统进行模拟的技术&#xff0c;在 ROS 中&#xff0c;仿真实现涉及的内容主要有三:对机器人建模(URDF)、创建仿真环境(Gazebo)以及感知环境(Rviz)等系统性实…

HTML特性(attribute)和DOM属性(property)

文章目录 定义位置不同attributeproperty 范围不同属性映射行为区别数据类型不同大小写敏感区别相同属性返回值可能不同DOM 属性具有写保护 定义位置不同 attribute 是 HTML 标签上的某个属性&#xff0c;如 id、class、value 等以及自定义属性,定义后会呈现在标签上 proper…

赛效:如何在线编辑图片

1&#xff1a;点击导航栏里的“图片编辑”。 2&#xff1a;点击打开图片或者拖放打开图片。 3&#xff1a;左侧几十种工具&#xff0c;你可以用来在线编辑图片。 4&#xff1a;编辑完成后点击页面右上角的“下载”按钮&#xff0c;根据提示登录账号下载图片就可以了。 如果你想…

2023广州建博会:鸿雁总裁王米成详解全屋智能的发展脉络

全屋智能落地的模式有很多&#xff0c;但鸿雁依托其智能面板优势&#xff0c;逐渐探索出一条属于鸿雁的全屋智能发展路径和商业模式。 智哪儿创始人、总编彭安军&#xff08;左&#xff09;&#xff0c;鸿雁电器总裁王米成&#xff08;右&#xff09; 在2023年的广州建博会上&a…

【Linux从入门到放弃】进程状态的理解以及什么是僵尸进程和孤儿进程?

&#x1f9d1;‍&#x1f4bb;作者&#xff1a; 情话0.0 &#x1f4dd;专栏&#xff1a;《Linux从入门到放弃》 &#x1f466;个人简介&#xff1a;一名双非编程菜鸟&#xff0c;在这里分享自己的编程学习笔记&#xff0c;欢迎大家的指正与点赞&#xff0c;谢谢&#xff01; 文…

你知道什么是基于StyleNeRF的conditional GAN模型吗

随着深度学习技术的不断发展&#xff0c;生成对抗网络&#xff08;GAN&#xff09;已经成为了人工智能研究和应用中的重要组成部分。其中&#xff0c;GAN可以被用来生成高质量的图像、视频等内容&#xff0c;这为娱乐产业和数字化制作带来了新的机遇和挑战。本文将介绍一种基于…

numpy 笔记 pad

1 基本介绍 对原本的矩阵进行填充 numpy.pad(array, pad_width, modeconstant, **kwargs) array待填充的矩阵pad_width要填充的位置mode填充方式 2 参数举例 2.0 使用的数据 import numpy as np anp.arange(12).reshape(3,4) aarray([[ 0, 1, 2, 3],[ 4, 5, 6, 7],…

【分布式 】 ELK 企业级日志分析系统

目录 一、ELK概述1.1 ELK简介1.2 为什么要使用ELK1.3 完整日志系统基本特征1.4 ELK的工作原理 二、搭建ELK2.1 ELK Elasticsearch 集群部署&#xff08;在Node1、Node2节点上操作&#xff09;node1 操作node2 同node1操作 2.2 安装 Elasticsearch-head 插件ELK Logstash 部署&a…

作为一个测试工程师,你选择python还是java?

问&#xff1a;“你平时工作中&#xff0c;用java多还是用python多”&#xff1f; 答&#xff1a;“都还可以&#xff0c;根据具体的场景选择不同的语言”。 问&#xff1a;“比如说呢”&#xff1f; 答&#xff1a;“开发自己的测试平台&#xff0c;肯定会选择java&#xf…

Stable Diffusion - ControlNet 插件中扩展局部重绘 InpaintOnly + LaMa 的算法与应用

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/131643131 LaMa: https://github.com/advimman/lama Paper: Resolution-robust Large Mask Inpainting with Fourier Convolutions LaMa: Large…

ActiveMQ详细入门教程系列

一、什么是消息中间件 两个系统或两个客户端之间进行消息传送&#xff0c;利用高效可靠的消息传递机制进行平台无关的数据交流&#xff0c;并基于数据通信来进行分布式系统的集成。通过提供消息传递和消息排队模型&#xff0c;它可以在分布式环境下扩展进程间的通信。 消息中…

ABeam中国2023社招 | ABeam旗下德硕管理咨询(深圳)招贤纳士

岗位需求 SAP Basis顾问 岗位职责 ■ 参与公司的SAP售前项目&#xff0c;负责Basis相关工作的方案制定 ■ 参与公司既有SAP运维项目&#xff0c;负责Basis相关的课题对应&#xff0c;系统改善等 ■ 负责SAP系统的Basis实施&#xff0c;SAP产品系统安装、升级、迁移、数据归档…

spring-java面向切面拦截器

切面&#xff0c;就是可以在代码执行的时候&#xff0c;在它执行的前面添加一个东西&#xff0c;一般我们用来做登陆拦截器验证以及敏感词的过滤。 他就3个东西&#xff0c;指定切点&#xff08;要执行的代码&#xff09;&#xff0c;before代码执行前面加东西。after代码后加东…

AC220V转负压5V芯片方案-220v转-5vic

AC220V到负压5V的电路转换 问题描述 --------- 如何将AC220V电压转换为负压5V输出&#xff0c;并且在输入电压范围为45V至265V的情况下工作&#xff1f;还要求该电路能够提供可调的电流范围&#xff0c;从100mA至2A。同时&#xff0c;所需的芯片为AH8699&#xff08;在700mA以…

BaGet做了一个Nuget私有服务器,Nginx代理之后还是会请求被代理得地址

Nuget搭建和使用可以参考官网得文档 https://loic-sharma.github.io/BaGet/installation/docker/ 这是我用Nginx代理之后出现得问题&#xff0c;观察请求url和响应回来得配置。配置中得ip地址得url是我被代理得下游地址&#xff0c;所以是无法访问的。 我原本以为是要去server…

怎么扫描二维码看视频?视频转成二维码的技巧

通过扫码来查看视频&#xff0c;是现在很多人都会选择一种视频展现方式&#xff0c;那么怎么制作视频二维码呢&#xff1f;下面给大家分享一个在线二维码生成器&#xff0c;支持多种二维码制作&#xff08;免费在线二维码生成器-二维码在线制作-音视频二维码在线生成工具-机智熊…

手机上有好用的时间管理工具吗?

在日常工作中&#xff0c;时间管理是非常重要的一项技能。这样做不仅可以更好地安排工作计划&#xff0c;还能有效减轻工作压力&#xff0c;减少时间的浪费&#xff0c;从而达到提高工作效率的目的。那么&#xff0c;随着手机的普及和智能化移动应用的不断涌现&#xff0c;我们…

【ARM Coresight 系列文章 4 - ARM Coresight APB-AP 介绍】

文章目录 APB-AP 介绍及使用APB-MuxROM TableROM Table entries bit 分配 APB-AP 介绍及使用 下图是一个 常见的DAP 结构图&#xff0c;其AP使用的是APB-AP。 图 1-1 SWJ-DP 通过监控电路选择外部串行总线SWD或者 外部JTAG&#xff1b;通过 DPACC中的select register 选着AP(…

润和软件HopeStage与易捷行云EasyStack ECF×86云基础设施平台完成产品兼容性互认证

近日&#xff0c;润和软件HopeStage操作系统与北京易捷思达科技发展有限公司&#xff08;以下简称“易捷行云EasyStack”&#xff09;ECF86云基础设施平台完成产品兼容性测试。 测试结果表明&#xff0c;企业级通用操作系统HopeStage V1.0产品与ECF86云基础设施平台可以顺利适…