mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-07 00:18:29 -04:00
c9eff28c44
By opening with O_CLOEXEC we make sure we don't leak the file descriptor when we are exec'ing or calling out subprograms. Right now we currently don't do it so there's no harm, but it's good practice in Linux to have it.
163 lines
3.6 KiB
C++
163 lines
3.6 KiB
C++
/*
|
|
hacked up DataFlash library for Desktop support
|
|
*/
|
|
|
|
#include "DataFlash_SITL.h"
|
|
|
|
#include <AP_HAL/AP_HAL.h>
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <stdint.h>
|
|
#include <assert.h>
|
|
|
|
#pragma GCC diagnostic ignored "-Wunused-result"
|
|
|
|
#define DF_PAGE_SIZE 512
|
|
#define DF_NUM_PAGES 16384
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
static int flash_fd;
|
|
static uint8_t buffer[2][DF_PAGE_SIZE];
|
|
|
|
// Public Methods //////////////////////////////////////////////////////////////
|
|
void DataFlash_SITL::Init()
|
|
{
|
|
DataFlash_Backend::Init();
|
|
if (flash_fd == 0) {
|
|
flash_fd = open("dataflash.bin", O_RDWR|O_CLOEXEC, 0777);
|
|
if (flash_fd == -1) {
|
|
uint8_t *fill;
|
|
fill = (uint8_t *)malloc(DF_PAGE_SIZE*DF_NUM_PAGES);
|
|
flash_fd = open("dataflash.bin", O_RDWR | O_CREAT | O_CLOEXEC, 0777);
|
|
memset(fill, 0xFF, DF_PAGE_SIZE*DF_NUM_PAGES);
|
|
write(flash_fd, fill, DF_PAGE_SIZE*DF_NUM_PAGES);
|
|
free(fill);
|
|
}
|
|
ftruncate(flash_fd, DF_PAGE_SIZE*DF_NUM_PAGES);
|
|
}
|
|
df_PageSize = DF_PAGE_SIZE;
|
|
|
|
// reserve last page for config information
|
|
df_NumPages = DF_NUM_PAGES - 1;
|
|
}
|
|
|
|
// This function is mainly to test the device
|
|
void DataFlash_SITL::ReadManufacturerID()
|
|
{
|
|
df_manufacturer = 1;
|
|
df_device = 0x0203;
|
|
}
|
|
|
|
bool DataFlash_SITL::CardInserted(void)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
// Read the status register
|
|
uint8_t DataFlash_SITL::ReadStatusReg()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// Read the status of the DataFlash
|
|
inline
|
|
uint8_t DataFlash_SITL::ReadStatus()
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline
|
|
uint16_t DataFlash_SITL::PageSize()
|
|
{
|
|
return df_PageSize;
|
|
}
|
|
|
|
|
|
// Wait until DataFlash is in ready state...
|
|
void DataFlash_SITL::WaitReady()
|
|
{
|
|
while(!ReadStatus());
|
|
}
|
|
|
|
void DataFlash_SITL::PageToBuffer(unsigned char BufferNum, uint16_t PageAdr)
|
|
{
|
|
assert(PageAdr>=1);
|
|
pread(flash_fd, buffer[BufferNum], DF_PAGE_SIZE, (PageAdr-1)*DF_PAGE_SIZE);
|
|
}
|
|
|
|
void DataFlash_SITL::BufferToPage (unsigned char BufferNum, uint16_t PageAdr, unsigned char wait)
|
|
{
|
|
assert(PageAdr>=1);
|
|
pwrite(flash_fd, buffer[BufferNum], DF_PAGE_SIZE, (PageAdr-1)*(uint32_t)DF_PAGE_SIZE);
|
|
}
|
|
|
|
void DataFlash_SITL::BufferWrite (unsigned char BufferNum, uint16_t IntPageAdr, unsigned char Data)
|
|
{
|
|
buffer[BufferNum][IntPageAdr] = (uint8_t)Data;
|
|
}
|
|
|
|
void DataFlash_SITL::BlockWrite(uint8_t BufferNum, uint16_t IntPageAdr,
|
|
const void *pHeader, uint8_t hdr_size,
|
|
const void *pBuffer, uint16_t size)
|
|
{
|
|
if (!_writes_enabled) {
|
|
return;
|
|
}
|
|
if (hdr_size) {
|
|
memcpy(&buffer[BufferNum][IntPageAdr],
|
|
pHeader,
|
|
hdr_size);
|
|
}
|
|
memcpy(&buffer[BufferNum][IntPageAdr+hdr_size],
|
|
pBuffer,
|
|
size);
|
|
}
|
|
|
|
// read size bytes of data to a page. The caller must ensure that
|
|
// the data fits within the page, otherwise it will wrap to the
|
|
// start of the page
|
|
bool DataFlash_SITL::BlockRead(uint8_t BufferNum, uint16_t IntPageAdr, void *pBuffer, uint16_t size)
|
|
{
|
|
memcpy(pBuffer, &buffer[BufferNum][IntPageAdr], size);
|
|
return true;
|
|
}
|
|
|
|
|
|
// *** END OF INTERNAL FUNCTIONS ***
|
|
|
|
void DataFlash_SITL::PageErase (uint16_t PageAdr)
|
|
{
|
|
uint8_t fill[DF_PAGE_SIZE];
|
|
memset(fill, 0xFF, sizeof(fill));
|
|
assert(PageAdr>=1);
|
|
pwrite(flash_fd, fill, DF_PAGE_SIZE, (PageAdr-1)*DF_PAGE_SIZE);
|
|
}
|
|
|
|
void DataFlash_SITL::BlockErase (uint16_t BlockAdr)
|
|
{
|
|
uint8_t fill[DF_PAGE_SIZE*8];
|
|
memset(fill, 0xFF, sizeof(fill));
|
|
assert(BlockAdr>=1);
|
|
pwrite(flash_fd, fill, DF_PAGE_SIZE*8, (BlockAdr-1)*DF_PAGE_SIZE*8);
|
|
}
|
|
|
|
|
|
void DataFlash_SITL::ChipErase()
|
|
{
|
|
for (int i=0; i<DF_NUM_PAGES; i++) {
|
|
PageErase(i);
|
|
hal.scheduler->delay(1);
|
|
}
|
|
}
|
|
|
|
|
|
#endif
|