2012-08-31 17:12:39 -03:00
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
#include <AP_HAL.h>
|
2012-08-31 17:12:39 -03:00
|
|
|
#include "Dataflash.h"
|
|
|
|
using namespace AP_HAL_AVR;
|
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
|
|
|
|
/* flash size */
|
|
|
|
#define DF_LAST_PAGE 4096
|
|
|
|
|
2012-11-28 21:57:20 -04:00
|
|
|
#define DF_RESET_PIN 31 /* (PC6) */
|
2012-08-31 21:07:03 -03:00
|
|
|
|
|
|
|
/* AT45DB161D commands (from datasheet) */
|
|
|
|
#define DF_TRANSFER_PAGE_TO_BUFFER_1 0x53
|
|
|
|
#define DF_TRANSFER_PAGE_TO_BUFFER_2 0x55
|
|
|
|
#define DF_STATUS_REGISTER_READ 0xD7
|
|
|
|
#define DF_READ_MANUFACTURER_AND_DEVICE_ID 0x9F
|
|
|
|
#define DF_PAGE_READ 0xD2
|
|
|
|
#define DF_BUFFER_1_READ 0xD4
|
|
|
|
#define DF_BUFFER_2_READ 0xD6
|
|
|
|
#define DF_BUFFER_1_WRITE 0x84
|
|
|
|
#define DF_BUFFER_2_WRITE 0x87
|
|
|
|
#define DF_BUFFER_1_TO_PAGE_WITH_ERASE 0x83
|
|
|
|
#define DF_BUFFER_2_TO_PAGE_WITH_ERASE 0x86
|
|
|
|
#define DF_PAGE_ERASE 0x81
|
|
|
|
#define DF_BLOCK_ERASE 0x50
|
|
|
|
#define DF_SECTOR_ERASE 0x7C
|
|
|
|
#define DF_CHIP_ERASE_0 0xC7
|
|
|
|
#define DF_CHIP_ERASE_1 0x94
|
|
|
|
#define DF_CHIP_ERASE_2 0x80
|
|
|
|
#define DF_CHIP_ERASE_3 0x9A
|
|
|
|
|
2012-08-31 17:12:39 -03:00
|
|
|
|
|
|
|
void APM1Dataflash::init(void* machtnichts) {
|
2012-11-28 21:57:20 -04:00
|
|
|
hal.gpio->pinMode(DF_RESET_PIN, GPIO_OUTPUT);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
|
|
|
/* Reset the dataflash chip */
|
2012-11-28 21:57:20 -04:00
|
|
|
hal.gpio->write(DF_RESET_PIN, 0);
|
2012-08-31 21:07:03 -03:00
|
|
|
hal.scheduler->delay(1);
|
2012-11-28 21:57:20 -04:00
|
|
|
hal.gpio->write(DF_RESET_PIN, 1);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi = hal.spi->device(AP_HAL::SPIDevice_Dataflash);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
|
|
|
_num_pages = DF_LAST_PAGE - 1;
|
|
|
|
uint8_t status = _read_status_reg();
|
|
|
|
_page_size = (status & 0x01) ? 512 : 528;
|
2012-08-31 17:12:39 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void APM1Dataflash::read_mfg_id() {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
|
|
|
_spi->transfer(DF_READ_MANUFACTURER_AND_DEVICE_ID);
|
|
|
|
_mfg = _spi->transfer(0xFF);
|
|
|
|
_device = _spi->transfer(0xFF);
|
|
|
|
_device = (_device << 8) | _spi->transfer(0xFF);
|
2012-08-31 21:07:03 -03:00
|
|
|
/* fourth byte is dont care */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0xFF);
|
|
|
|
_spi->cs_release();
|
2012-08-31 17:12:39 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool APM1Dataflash::media_present() {
|
2012-08-31 21:07:03 -03:00
|
|
|
return true;
|
|
|
|
}
|
2012-08-31 17:12:39 -03:00
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
void APM1Dataflash::_wait_ready() {
|
|
|
|
while(!_read_status());
|
2012-08-31 17:12:39 -03:00
|
|
|
}
|
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
void APM1Dataflash::_page_to_buffer(uint8_t buffer_num, uint16_t page_addr) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
2012-08-31 21:07:03 -03:00
|
|
|
if (_buffer_num == 1) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_TRANSFER_PAGE_TO_BUFFER_1);
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_TRANSFER_PAGE_TO_BUFFER_2);
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_page_size == 512) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( page_addr >> 7 );
|
|
|
|
_spi->transfer( page_addr << 1 );
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( page_addr >> 6 );
|
|
|
|
_spi->transfer( page_addr << 2 );
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
/* finally send one dont care byte */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0x00);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
_wait_ready();
|
|
|
|
}
|
2012-08-31 17:12:39 -03:00
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
void APM1Dataflash::_buffer_to_page(uint8_t buffer_num, uint16_t page_addr, bool wait) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
2012-08-31 21:07:03 -03:00
|
|
|
if (_buffer_num == 1) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_BUFFER_1_TO_PAGE_WITH_ERASE);
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_BUFFER_2_TO_PAGE_WITH_ERASE);
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_page_size == 512) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( page_addr >> 7 );
|
|
|
|
_spi->transfer( page_addr << 1 );
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( page_addr >> 6 );
|
|
|
|
_spi->transfer( page_addr << 2 );
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
/* finally send one dont care byte */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0x00);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
if (wait) {
|
|
|
|
_wait_ready();
|
|
|
|
}
|
2012-08-31 17:12:39 -03:00
|
|
|
}
|
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
void APM1Dataflash::_page_erase(uint16_t page_addr) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
|
|
|
_spi->transfer(DF_PAGE_ERASE);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
|
|
|
if (_page_size == 512) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( page_addr >> 7 );
|
|
|
|
_spi->transfer( page_addr << 1 );
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( page_addr >> 6 );
|
|
|
|
_spi->transfer( page_addr << 2 );
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* finally send one dont care byte */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0x00);
|
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
_wait_ready();
|
|
|
|
}
|
2012-08-31 17:12:39 -03:00
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
void APM1Dataflash::_block_erase(uint16_t block_addr) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
|
|
|
_spi->transfer(DF_BLOCK_ERASE);
|
2012-08-31 21:07:03 -03:00
|
|
|
|
|
|
|
if (_page_size == 512) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( block_addr >> 7 );
|
|
|
|
_spi->transfer( block_addr << 1 );
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer( block_addr >> 6 );
|
|
|
|
_spi->transfer( block_addr << 2 );
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
/* finally send one dont care byte */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0x00);
|
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
_wait_ready();
|
2012-08-31 17:12:39 -03:00
|
|
|
}
|
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
void APM1Dataflash::_chip_erase() {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
|
|
|
_spi->transfer(DF_CHIP_ERASE_0);
|
|
|
|
_spi->transfer(DF_CHIP_ERASE_1);
|
|
|
|
_spi->transfer(DF_CHIP_ERASE_2);
|
|
|
|
_spi->transfer(DF_CHIP_ERASE_3);
|
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
|
|
|
|
while(!_read_status()) {
|
|
|
|
hal.scheduler->delay(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void APM1Dataflash::_buffer_write(uint8_t buffer_num, uint16_t page_addr, uint8_t data) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
2012-08-31 21:07:03 -03:00
|
|
|
if (buffer_num == 1) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_BUFFER_1_WRITE);
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_BUFFER_2_WRITE);
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
/* Don't care */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0);
|
2012-08-31 21:07:03 -03:00
|
|
|
/* Internal buffer address */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer((uint8_t)(page_addr >> 8));
|
|
|
|
_spi->transfer((uint8_t)(page_addr & 0xFF));
|
2012-08-31 21:07:03 -03:00
|
|
|
/* Byte to write */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(data);
|
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t APM1Dataflash::_buffer_read(uint8_t buffer_num, uint16_t page_addr) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
2012-08-31 21:07:03 -03:00
|
|
|
if (buffer_num == 1) {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_BUFFER_1_READ);
|
2012-08-31 21:07:03 -03:00
|
|
|
} else {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(DF_BUFFER_2_READ);
|
2012-08-31 21:07:03 -03:00
|
|
|
}
|
|
|
|
/* Don't care */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0);
|
2012-08-31 21:07:03 -03:00
|
|
|
/* Internal buffer address */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer((uint8_t)(page_addr >> 8));
|
|
|
|
_spi->transfer((uint8_t)(page_addr & 0xFF));
|
2012-08-31 21:07:03 -03:00
|
|
|
/* Don't care */
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->transfer(0);
|
2012-08-31 21:07:03 -03:00
|
|
|
/* Read data byte */
|
2012-11-28 21:57:20 -04:00
|
|
|
uint8_t res = _spi->transfer(0);
|
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
return res;
|
|
|
|
}
|
2012-08-31 17:12:39 -03:00
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
inline uint8_t APM1Dataflash::_read_status_reg() {
|
2012-11-28 21:57:20 -04:00
|
|
|
_spi->cs_assert();
|
|
|
|
_spi->transfer(DF_STATUS_REGISTER_READ);
|
2012-08-31 21:07:03 -03:00
|
|
|
/* Read the first byte of the result */
|
2012-11-28 21:57:20 -04:00
|
|
|
uint8_t res = _spi->transfer(0);
|
|
|
|
_spi->cs_release();
|
2012-08-31 21:07:03 -03:00
|
|
|
return res;
|
2012-08-31 17:12:39 -03:00
|
|
|
}
|
|
|
|
|
2012-08-31 21:07:03 -03:00
|
|
|
inline uint8_t APM1Dataflash::_read_status() {
|
|
|
|
/* Busy status is the top bit of the status register */
|
|
|
|
return _read_status_reg() & 0x80;
|
|
|
|
}
|
2012-08-31 17:12:39 -03:00
|
|
|
|