Электронный барометр

pryschik
Offline
Зарегистрирован: 14.01.2013

Ну вот чтоб я не делал, как его не мучил,

какие-бы скетчи не заливал,

он (BMP085) показывает чепуху,

т.е. показания изменяются в районе одного метра,

хотя сам датчик жестко закреплен,

Это оно у всех так или я чего-то не догоняю ?

конструктивная критика приветствуется.

 

Andrey_Y_Ostanovsky
Offline
Зарегистрирован: 03.12.2012

Шум?

26rus
26rus аватар
Offline
Зарегистрирован: 03.05.2011

может брак?

вроде должен ровно показывать и довольно точно...

 

pryschik
Offline
Зарегистрирован: 14.01.2013

температуру показывает стабильно,

а давление и высоту все время перепады в районе 1 метра

это много, может конечно брак, попробую второй датчик.

если у кого есть опыт с BMP085 подскажите, плиз .

step962
Offline
Зарегистрирован: 23.05.2011

Вам аббревиатура RTFM не знакома?

Из даташита на сей элемент (на второй странице - там, где фичами меряются):

low noice: 0.06 hPa (0.5 m) in ultra low power mode
           0.03 hPa (0.25 m) in ultra high resolution mode
           < 0.1 m possible with software aweraging algorithm

А ведь есть же еще и другие режимы питания, кроме ultra low.

И другие точности, кроме ultra high.

И алгоритмы усреднения у вас не упомянуты.

И вообще - а чего вы ожидали? Миллиметровой точности? А знаете ли вы, какому перепаду высоты соответствует изменение давления в 1 мм.рт.ст. (т.н. барометрическая ступень)? 11.3 м. Так что ваш элемент шумит в пределах 0.1 мм.рт.ст., и, прочитав таки даташит, вы вполне можете снизить этот шум до 0.02 мм.рт.ст. 

 

pryschik
Offline
Зарегистрирован: 14.01.2013

Усреднение использую конечно,

подскажите как переключить на ultra high точность ?

 

step962
Offline
Зарегистрирован: 23.05.2011

RTFM

pryschik
Offline
Зарегистрирован: 14.01.2013

Всё равно, что гугл послал.

Big thanks.

Но переключение на ultra high

ничего пока не дало, высота все равно пляшет,

даже усреднее по 10 значениям сильно гуляет.

прогресс не пошел, короче.

ustas
Offline
Зарегистрирован: 12.03.2012

вот код, который работает без библиотек:
 

//Arduino 1.0+ Only
//Arduino 1.0+ Only

/*Based largely on code by  Jim Lindblom

 Get pressure, altitude, and temperature from the BMP085.
 Serial.print it out at 9600 baud to serial monitor.
 */

#include <Wire.h>

#define BMP085_ADDRESS 0x77  // I2C address of BMP085

const unsigned char OSS = 0;  // Oversampling Setting

// Calibration values
int ac1;
int ac2;
int ac3;
unsigned int ac4;
unsigned int ac5;
unsigned int ac6;
int b1;
int b2;
int mb;
int mc;
int md;

// b5 is calculated in bmp085GetTemperature(...), this variable is also used in bmp085GetPressure(...)
// so ...Temperature(...) must be called before ...Pressure(...).
long b5; 

void setup(){
  Serial.begin(9600);
  Wire.begin();
Serial.print("Temperaturet: ");
  bmp085Calibration();
  Serial.print("Temperaturet2: ");
}

void loop()
{
  float temperature = bmp085GetTemperature(bmp085ReadUT()); //MUST be called first
  float pressure = bmp085GetPressure(bmp085ReadUP());
  float atm = pressure / 101325; // "standard atmosphere"
  float altitude = calcAltitude(pressure); //Uncompensated caculation - in Meters 

  Serial.print("Temperature: ");
  Serial.print(temperature, 2); //display 2 decimal places
  Serial.println("deg C");

  Serial.print("Pressure: ");
  Serial.print(pressure*0.0075006375541921, 0); //whole number only.
  Serial.println(" Pa");

  Serial.print("Standard Atmosphere: ");
  Serial.println(atm, 4); //display 4 decimal places

  Serial.print("Altitude: ");
  Serial.print(altitude, 2); //display 2 decimal places
  Serial.println(" M");

  Serial.println();//line break

  delay(1000); //wait a second and get values again.
}

// Stores all of the bmp085's calibration values into global variables
// Calibration values are required to calculate temp and pressure
// This function should be called at the beginning of the program
void bmp085Calibration()
{
  //Serial.println("test0");
  ac1 = bmp085ReadInt(0xAA);
  //Serial.println("test1");
  ac2 = bmp085ReadInt(0xAC);
  ac3 = bmp085ReadInt(0xAE);
  ac4 = bmp085ReadInt(0xB0);
  ac5 = bmp085ReadInt(0xB2);
  ac6 = bmp085ReadInt(0xB4);
  b1 = bmp085ReadInt(0xB6);
  b2 = bmp085ReadInt(0xB8);
  mb = bmp085ReadInt(0xBA);
  mc = bmp085ReadInt(0xBC);
  md = bmp085ReadInt(0xBE);
  //Serial.println("test2");
}

// Calculate temperature in deg C
float bmp085GetTemperature(unsigned int ut){
  long x1, x2;

  x1 = (((long)ut - (long)ac6)*(long)ac5) >> 15;
  x2 = ((long)mc << 11)/(x1 + md);
  b5 = x1 + x2;

  float temp = ((b5 + 8)>>4);
  temp = temp /10;

  return temp;
}

// Calculate pressure given up
// calibration values must be known
// b5 is also required so bmp085GetTemperature(...) must be called first.
// Value returned will be pressure in units of Pa.
long bmp085GetPressure(unsigned long up){
  long x1, x2, x3, b3, b6, p;
  unsigned long b4, b7;

  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;
  p += (x1 + x2 + 3791)>>4;

  long temp = p;
  return temp;
}

// Read 1 byte from the BMP085 at 'address'
char bmp085Read(unsigned char address)
{
  unsigned char data;

  Wire.beginTransmission(BMP085_ADDRESS);
  Wire.write(address);
  Wire.endTransmission();

  Wire.requestFrom(BMP085_ADDRESS, 1);
  while(!Wire.available())
    ;

  return Wire.read();
}

// Read 2 bytes from the BMP085
// First byte will be from 'address'
// Second byte will be from 'address'+1
int bmp085ReadInt(unsigned char address)
{
  unsigned char msb, lsb;
  //Serial.println("test3");
  Wire.beginTransmission(BMP085_ADDRESS);
  Wire.write(address);
  Wire.endTransmission();
//Serial.println("test4");
  Wire.requestFrom(BMP085_ADDRESS, 2);
  while(Wire.available()<2);
  msb = Wire.read();
  lsb = Wire.read();
//Serial.println("test5");
  return (int) msb<<8 | lsb;
}

// Read the uncompensated temperature value
unsigned int bmp085ReadUT(){
  unsigned int ut;

  // Write 0x2E into Register 0xF4
  // This requests a temperature reading
  Wire.beginTransmission(BMP085_ADDRESS);
  Wire.write(0xF4);
  Wire.write(0x2E);
  Wire.endTransmission();

  // Wait at least 4.5ms
  delay(5);

  // Read two bytes from registers 0xF6 and 0xF7
  ut = bmp085ReadInt(0xF6);
  return ut;
}

// Read the uncompensated pressure value
unsigned long bmp085ReadUP(){

  unsigned char msb, lsb, xlsb;
  unsigned long up = 0;

  // Write 0x34+(OSS<<6) into register 0xF4
  // Request a pressure reading w/ oversampling setting
  Wire.beginTransmission(BMP085_ADDRESS);
  Wire.write(0xF4);
  Wire.write(0x34 + (OSS<<6));
  Wire.endTransmission();

  // Wait for conversion, delay time dependent on OSS
  delay(2 + (3<<OSS));

  // Read register 0xF6 (MSB), 0xF7 (LSB), and 0xF8 (XLSB)
  msb = bmp085Read(0xF6);
  lsb = bmp085Read(0xF7);
  xlsb = bmp085Read(0xF8);

  up = (((unsigned long) msb << 16) | ((unsigned long) lsb << 8) | (unsigned long) xlsb) >> (8-OSS);

  return up;
}

void writeRegister(int deviceAddress, byte address, byte val) {
  Wire.beginTransmission(deviceAddress); // start transmission to device 
  Wire.write(address);       // send register address
  Wire.write(val);         // send value to write
  Wire.endTransmission();     // end transmission
}

int readRegister(int deviceAddress, byte address){

  int v;
  Wire.beginTransmission(deviceAddress);
  Wire.write(address); // register to read
  Wire.endTransmission();

  Wire.requestFrom(deviceAddress, 1); // read a byte

  while(!Wire.available()) {
    // waiting
  }

  v = Wire.read();
  return v;
}

float calcAltitude(float pressure){

  float A = pressure/101325;
  float B = 1/5.25588;
  float C = pow(A,B);
  C = 1 - C;
  C = C /0.0000225577;

  return C;
}

еще обратите внимание на значение OSS - включение "oversampling" для повышения точности.

я высоту над уровнем не мерял (без надобности), но, имхо, надо включить повышенную точность и усреднять.. 

 

pryschik
Offline
Зарегистрирован: 14.01.2013

Какие будут предложения по OSS ?

изначально смотрю ноль стоит.

ustas
Offline
Зарегистрирован: 12.03.2012

открываем даташит на него (в этой теме чуть выше) и Ctrl+F по строке OSS