本帖最后由 jinglixixi 于 2023-7-19 12:04 编辑
#申请原创#
BMP085是一种可对环境温度及大气压进行检测的传感器,它体积小易于同0.96寸的LCD一起构成一个便携式的环境状态检测装置。 该装置的整体电路构成及连接如图1所示:
整体构成
BMP085是一种采用I2C接口工作的器件,故占用的引脚资源极少,在使用时它与开发板的连接关系为: SCL----PA1 SDA----PA0
所用引脚输出高低电平的定义为: #defineSDA_Set1() GPIO_SetPinLevel(GPIOA,GPIO_PIN0, GPIO_LEVEL_HIGH) #defineSDA_Clr1() GPIO_SetPinLevel(GPIOA,GPIO_PIN0, GPIO_LEVEL_LOW)
#define SCL_Set1() GPIO_SetPinLevel(GPIOA, GPIO_PIN1,GPIO_LEVEL_HIGH) #defineSCL_Clr1() GPIO_SetPinLevel(GPIOA,GPIO_PIN1, GPIO_LEVEL_LOW)
为读取数据所作的相应定义为: #defineSDA_IN GPIO_GetPinLevel(GPIOA,GPIO_PIN0)
BMP085的初始化函数为: - void Init_BMP085()
- {
- 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);
- }
在以GPIO口模拟I2C收发字节数据的函数为: - void BMP085_Send_Byte(char txd)
- {
- char t;
- OUTPUT_MODE_SET();
- SCL_Clr1();
- for(t=0;t<8;t++)
- {
- if((txd&0x80)>>7)
- SDA_Set1();
- else
- SDA_Clr1();
- txd<<=1;
- udelay(1);
- SCL_Set1();
- udelay(1);
- SCL_Clr1();
- udelay(1);
- }
- }
- char BMP085_Read_Byte(unsigned char ack)
- {
- unsigned char i,receive=0;
- INPUT_MODE_SET();
- for(i=0;i<8;i++)
- {
- SCL_Clr1();
- udelay(1);
- SCL_Set1();
- receive<<=1;
- if(SDA_IN) receive++;
- udelay(1);
- }
- if(!ack)
- BMP085_NAck();
- else
- BMP085_Ack();
- return receive;
- }
BMP085读取温度和大气压数据的函数为: - long bmp085ReadTemp(void)
- {
- BMP085_Start();
- BMP085_Send_Byte(BMP085_SlaveAddress);
- while(BMP085_Wait_Ack());
- BMP085_Send_Byte(0xF4);
- while(BMP085_Wait_Ack());
- BMP085_Send_Byte(0x2E);
- while(BMP085_Wait_Ack());
- BMP085_Stop();
- mdelay(10);
- return (long) Multiple_read(0xF6);
- }
- long bmp085ReadPressure(void)
- {
- long pressure = 0;
- BMP085_Start();
- BMP085_Send_Byte(BMP085_SlaveAddress);
- while(BMP085_Wait_Ack()){}
- BMP085_Send_Byte(0xF4);
- while(BMP085_Wait_Ack()){}
- BMP085_Send_Byte(0x34);
- while(BMP085_Wait_Ack()){}
- BMP085_Stop();
- mdelay(10);
- pressure = Multiple_read(0xF6);
- pressure &= 0x0000FFFF;
- return pressure;
- }
BMP085数据变换处理的函数为: - void bmp085Convert()
- {
- unsigned int ut;
- unsigned long up;
- long x1, x2, b5, b6, x3, b3, p;
- unsigned long b4, b7;
- ut = bmp085ReadTemp();
- up = bmp085ReadPressure();
- x1 = (((long)ut - (long)ac6)*(long)ac5) >> 15;
- x2 = ((long) mc << 11) / (x1 + md);
- b5 = x1 + x2;
- temperature1 = ((b5 + 8) >> 4);
- b6 = b5 - 4000;
- x1 = (b2 * (b6 * b6)>>12)>>11;
- x2 = (ac2 * b6)>>11;
- x3 = x1 + x2;
- b3 = (((((long)ac1)*4 + x3)<<OSS) + 2)>>2;
- 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;
- pressure = p+((x1 + x2 + 3791)>>4);
- }
实现测试效果的主程序为: - int main(void)
- {
- InitDelay();
- InitDebug();
- GPIO_LedInit();
- TIMER_PrdInit();
- GPIO_OLEDInit();
- OLED_Init();
- OLED_Clear();
- OLED_ShowString(10,0,"AC7802x Test",16);
- OLED_ShowString(10,2,"BMP085",16);
- BMP085_Init();
- Init_BMP085();
- OLED_ShowString(0,4,"t= C",16);
- OLED_ShowString(0,6,"p= KPa",16);
- mdelay(500);
- while (1)
- {
- bmp085Convert();
- OLED_ShowNum(24,4,temperature1/10,3,16);
- OLED_ShowNum(24,6,pressure/100,5,16);
- OLED_ShowNum(24,4,temperature1/10,3,16);
- OLED_ShowNum(24,6,pressure/100,5,16);
- mdelay(500);
- }
- }
|