From 880c7e641127ab858c3517ee8bf1ba0eaa2f491c Mon Sep 17 00:00:00 2001 From: Andrew Tridgell Date: Wed, 4 Jan 2012 15:40:22 +1100 Subject: [PATCH 1/4] DataFlash: cope with older dataflash chips some of the APM2 DataFlash cards use an older chip with only 4096 pages, and a density code of 6. Doug has one of these. --- libraries/DataFlash/DataFlash_APM2.cpp | 23 ++++++++++++++++++----- 1 file changed, 18 insertions(+), 5 deletions(-) diff --git a/libraries/DataFlash/DataFlash_APM2.cpp b/libraries/DataFlash/DataFlash_APM2.cpp index 415cdc062d..72e336e2ee 100644 --- a/libraries/DataFlash/DataFlash_APM2.cpp +++ b/libraries/DataFlash/DataFlash_APM2.cpp @@ -53,8 +53,6 @@ extern "C" { # error Please check the Tools/Board menu to ensure you have selected Arduino Mega as your target. #endif -#define DF_LAST_PAGE 8192 - // AT45DB321D Commands (from Datasheet) #define DF_TRANSFER_PAGE_TO_BUFFER_1 0x53 #define DF_TRANSFER_PAGE_TO_BUFFER_2 0x55 @@ -133,8 +131,23 @@ void DataFlash_APM2::Init(void) // get page size: 512 or 528 (by default: 528) df_PageSize=PageSize(); - // the last page is reserved for config information - df_NumPages = DF_LAST_PAGE - 1; + ReadManufacturerID(); + + // see page 22 of the spec for the density code + uint8_t density_code = (df_device >> 8) & 0x1F; + + // note that we set df_NumPages to one lower than the highest, as + // the last page is reserved for a config page + if (density_code == 0x7) { + // 32 Mbit + df_NumPages = 8191; + } else if (density_code == 0x6) { + // 16 Mbit + df_NumPages = 4095; + } else { + // what is this??? card not inserted perhaps? + df_NumPages = 0; + } } // This function is mainly to test the device @@ -155,7 +168,7 @@ void DataFlash_APM2::ReadManufacturerID() // This function return 1 if Card is inserted on SD slot bool DataFlash_APM2::CardInserted() { - return (digitalRead(DF_CARDDETECT) == 0); + return (df_NumPages >= 4095 && digitalRead(DF_CARDDETECT) == 0); } // Read the status register From 9fb3b13af32e6ae0aae006d78e6797e526306176 Mon Sep 17 00:00:00 2001 From: Andrew Tridgell Date: Wed, 4 Jan 2012 16:35:16 +1100 Subject: [PATCH 2/4] AP_Baro: when I2c fails, don't retry for 1s --- libraries/AP_Baro/AP_Baro_BMP085.cpp | 12 ++++++++++++ libraries/AP_Baro/AP_Baro_BMP085.h | 1 + 2 files changed, 13 insertions(+) diff --git a/libraries/AP_Baro/AP_Baro_BMP085.cpp b/libraries/AP_Baro/AP_Baro_BMP085.cpp index e88016f362..611fe58885 100644 --- a/libraries/AP_Baro/AP_Baro_BMP085.cpp +++ b/libraries/AP_Baro/AP_Baro_BMP085.cpp @@ -152,7 +152,13 @@ void AP_Baro_BMP085::ReadPress() { uint8_t buf[3]; + if (!healthy && millis() < _retry_time) { + return; + } + if (I2c.read(BMP085_ADDRESS, 0xF6, 3, buf) != 0) { + _retry_time = millis() + 1000; + I2c.setSpeed(false); healthy = false; return; } @@ -197,7 +203,13 @@ void AP_Baro_BMP085::ReadTemp() { uint8_t buf[2]; + if (!healthy && millis() < _retry_time) { + return; + } + if (I2c.read(BMP085_ADDRESS, 0xF6, 2, buf) != 0) { + _retry_time = millis() + 1000; + I2c.setSpeed(false); healthy = false; return; } diff --git a/libraries/AP_Baro/AP_Baro_BMP085.h b/libraries/AP_Baro/AP_Baro_BMP085.h index ea9e5742b0..8e08965996 100644 --- a/libraries/AP_Baro/AP_Baro_BMP085.h +++ b/libraries/AP_Baro/AP_Baro_BMP085.h @@ -50,6 +50,7 @@ class AP_Baro_BMP085 : public AP_Baro uint8_t _temp_index; uint8_t _press_index; + uint32_t _retry_time; void Command_ReadPress(); void Command_ReadTemp(); From 60185509f4a44decad09a7569fdeb06b124ed38d Mon Sep 17 00:00:00 2001 From: Andrew Tridgell Date: Wed, 4 Jan 2012 16:35:47 +1100 Subject: [PATCH 3/4] AP_Compass: when I2c fails, don't retry for 1s --- libraries/AP_Compass/AP_Compass_HMC5843.cpp | 13 +++++++++++-- libraries/AP_Compass/AP_Compass_HMC5843.h | 1 + 2 files changed, 12 insertions(+), 2 deletions(-) diff --git a/libraries/AP_Compass/AP_Compass_HMC5843.cpp b/libraries/AP_Compass/AP_Compass_HMC5843.cpp index 8096275623..2f6b9e72cc 100644 --- a/libraries/AP_Compass/AP_Compass_HMC5843.cpp +++ b/libraries/AP_Compass/AP_Compass_HMC5843.cpp @@ -246,11 +246,20 @@ AP_Compass_HMC5843::init() // Read Sensor data bool AP_Compass_HMC5843::read() { - if (!healthy && !re_initialise()) { - return false; + if (!healthy) { + if (millis() < _retry_time) { + return false; + } + if (!re_initialise()) { + _retry_time = millis() + 1000; + return false; + } } if (!read_raw()) { + // try again in 1 second, and set I2c clock speed slower + _retry_time = millis() + 1000; + I2c.setSpeed(false); return false; } diff --git a/libraries/AP_Compass/AP_Compass_HMC5843.h b/libraries/AP_Compass/AP_Compass_HMC5843.h index 3dbd747eef..2549f38a8f 100644 --- a/libraries/AP_Compass/AP_Compass_HMC5843.h +++ b/libraries/AP_Compass/AP_Compass_HMC5843.h @@ -53,6 +53,7 @@ class AP_Compass_HMC5843 : public Compass virtual bool re_initialise(void); bool read_register(uint8_t address, uint8_t *value); bool write_register(uint8_t address, byte value); + uint32_t _retry_time; // when unhealthy the millis() value to retry at public: AP_Compass_HMC5843(AP_Var::Key key = AP_Var::k_key_none) : Compass(key) {} From cdd55894986b544f8df1b1090a3d6c55f290e89c Mon Sep 17 00:00:00 2001 From: Andrew Tridgell Date: Wed, 4 Jan 2012 16:36:47 +1100 Subject: [PATCH 4/4] I2c: default to 5ms timeout, and faster bus speed we will slow down the speed when we get a failure --- ArduCopter/system.pde | 4 +++- ArduPlane/system.pde | 4 +++- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/ArduCopter/system.pde b/ArduCopter/system.pde index a82f1c7555..6cf6d95e1f 100644 --- a/ArduCopter/system.pde +++ b/ArduCopter/system.pde @@ -108,7 +108,9 @@ static void init_ardupilot() // #ifndef DESKTOP_BUILD I2c.begin(); - I2c.timeOut(20); + I2c.timeOut(5); + // initially set a fast I2c speed, and drop it on first failures + I2c.setSpeed(true); #endif SPI.begin(); SPI.setClockDivider(SPI_CLOCK_DIV16); // 1MHZ SPI rate diff --git a/ArduPlane/system.pde b/ArduPlane/system.pde index b3a30b7a5e..5b802942d6 100644 --- a/ArduPlane/system.pde +++ b/ArduPlane/system.pde @@ -108,7 +108,9 @@ static void init_ardupilot() // #ifndef DESKTOP_BUILD I2c.begin(); - I2c.timeOut(20); + I2c.timeOut(5); + // initially set a fast I2c speed, and drop it on first failures + I2c.setSpeed(true); #endif SPI.begin(); SPI.setClockDivider(SPI_CLOCK_DIV16); // 1MHZ SPI rate