/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- /* hacked up DataFlash library for Desktop support */ #include #include #include #include #include #include "DataFlash.h" #include #define DF_PAGE_SIZE 512 #define DF_NUM_PAGES 4096 static int flash_fd; static uint8_t buffer[2][DF_PAGE_SIZE]; #define OVERWRITE_DATA 1 // 0: When reach the end page stop, 1: Start overwritten from page 1 // Public Methods ////////////////////////////////////////////////////////////// void DataFlash_APM1::Init(void) { if (flash_fd == 0) { flash_fd = open("dataflash.bin", O_RDWR, 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, 0777); memset(fill, 0xFF, DF_PAGE_SIZE*DF_NUM_PAGES); write(flash_fd, fill, DF_PAGE_SIZE*DF_NUM_PAGES); free(fill); } } df_PageSize = DF_PAGE_SIZE; df_BufferNum = 1; df_BufferIdx = 0; df_PageAdr = 0; df_Stop_Write = 0; } // This function is mainly to test the device void DataFlash_APM1::ReadManufacturerID() { df_manufacturer = 1; df_device_0 = 2; df_device_1 = 3; } // Read the status register byte DataFlash_APM1::ReadStatusReg() { return 0; } // Read the status of the DataFlash inline byte DataFlash_APM1::ReadStatus() { return 1; } inline uint16_t DataFlash_APM1::PageSize() { return df_PageSize; } // Wait until DataFlash is in ready state... void DataFlash_APM1::WaitReady() { while(!ReadStatus()); } void DataFlash_APM1::PageToBuffer(unsigned char BufferNum, uint16_t PageAdr) { pread(flash_fd, buffer[BufferNum-1], DF_PAGE_SIZE, PageAdr*DF_PAGE_SIZE); } void DataFlash_APM1::BufferToPage (unsigned char BufferNum, uint16_t PageAdr, unsigned char wait) { pwrite(flash_fd, buffer[BufferNum-1], DF_PAGE_SIZE, PageAdr*DF_PAGE_SIZE); } void DataFlash_APM1::BufferWrite (unsigned char BufferNum, uint16_t IntPageAdr, unsigned char Data) { buffer[BufferNum-1][IntPageAdr] = (uint8_t)Data; } unsigned char DataFlash_APM1::BufferRead (unsigned char BufferNum, uint16_t IntPageAdr) { return (unsigned char)buffer[BufferNum-1][IntPageAdr]; } // *** END OF INTERNAL FUNCTIONS *** void DataFlash_APM1::PageErase (uint16_t PageAdr) { uint8_t fill[DF_PAGE_SIZE]; memset(fill, 0xFF, sizeof(fill)); pwrite(flash_fd, fill, DF_PAGE_SIZE, PageAdr*DF_PAGE_SIZE); } void DataFlash_APM1::ChipErase () { for (int i=0; DF_NUM_PAGES; i++) { PageErase(i); } } // *** DATAFLASH PUBLIC FUNCTIONS *** void DataFlash_APM1::StartWrite(int16_t PageAdr) { df_BufferNum=1; df_BufferIdx=4; df_PageAdr=PageAdr; df_Stop_Write=0; WaitReady(); // We are starting a new page - write FileNumber and FilePage BufferWrite(df_BufferNum,0,df_FileNumber>>8); // High byte BufferWrite(df_BufferNum,1,df_FileNumber&0xFF); // Low byte BufferWrite(df_BufferNum,2,df_FilePage>>8); // High byte BufferWrite(df_BufferNum,3,df_FilePage&0xFF); // Low byte } void DataFlash_APM1::FinishWrite(void) { df_BufferIdx=0; BufferToPage(df_BufferNum,df_PageAdr,0); // Write Buffer to memory, NO WAIT df_PageAdr++; if (OVERWRITE_DATA==1) { if (df_PageAdr>DF_LAST_PAGE) // If we reach the end of the memory, start from the begining df_PageAdr = 1; } else { if (df_PageAdr>DF_LAST_PAGE) // If we reach the end of the memory, stop here df_Stop_Write=1; } if (df_BufferNum==1) // Change buffer to continue writing... df_BufferNum=2; else df_BufferNum=1; } void DataFlash_APM1::WriteByte(byte data) { if (!df_Stop_Write) { BufferWrite(df_BufferNum,df_BufferIdx,data); df_BufferIdx++; if (df_BufferIdx >= df_PageSize) // End of buffer? { df_BufferIdx=4; //(4 bytes for FileNumber, FilePage) BufferToPage(df_BufferNum,df_PageAdr,0); // Write Buffer to memory, NO WAIT df_PageAdr++; if (OVERWRITE_DATA==1) { if (df_PageAdr>DF_LAST_PAGE) // If we reach the end of the memory, start from the begining df_PageAdr = 1; } else { if (df_PageAdr>DF_LAST_PAGE) // If we reach the end of the memory, stop here df_Stop_Write=1; } if (df_BufferNum==1) // Change buffer to continue writing... df_BufferNum=2; else df_BufferNum=1; // We are starting a new page - write FileNumber and FilePage BufferWrite(df_BufferNum,0,df_FileNumber>>8); // High byte BufferWrite(df_BufferNum,1,df_FileNumber&0xFF); // Low byte df_FilePage++; BufferWrite(df_BufferNum,2,df_FilePage>>8); // High byte BufferWrite(df_BufferNum,3,df_FilePage&0xFF); // Low byte } } } void DataFlash_APM1::WriteInt(int16_t data) { WriteByte(data>>8); // High byte WriteByte(data&0xFF); // Low byte } void DataFlash_APM1::WriteLong(int32_t data) { WriteByte(data>>24); // First byte WriteByte(data>>16); WriteByte(data>>8); WriteByte(data&0xFF); // Last byte } // Get the last page written to int16_t DataFlash_APM1::GetWritePage() { return(df_PageAdr); } // Get the last page read int16_t DataFlash_APM1::GetPage() { return(df_Read_PageAdr-1); } void DataFlash_APM1::StartRead(int16_t PageAdr) { df_Read_BufferNum=1; df_Read_BufferIdx=4; df_Read_PageAdr=PageAdr; WaitReady(); PageToBuffer(df_Read_BufferNum,df_Read_PageAdr); // Write Memory page to buffer //Serial.print(df_Read_PageAdr, DEC); Serial.print("\t"); df_Read_PageAdr++; // We are starting a new page - read FileNumber and FilePage df_FileNumber = BufferRead(df_Read_BufferNum,0); // High byte //Serial.print(df_FileNumber, DEC); Serial.print("\t"); df_FileNumber = (df_FileNumber<<8) | BufferRead(df_Read_BufferNum,1); // Low byte //Serial.println(df_FileNumber, DEC); Serial.print("\t"); df_FilePage = BufferRead(df_Read_BufferNum,2); // High byte df_FilePage = (df_FilePage<<8) | BufferRead(df_Read_BufferNum,3); // Low byte } byte DataFlash_APM1::ReadByte() { byte result; WaitReady(); result = BufferRead(df_Read_BufferNum,df_Read_BufferIdx); df_Read_BufferIdx++; if (df_Read_BufferIdx >= df_PageSize) // End of buffer? { df_Read_BufferIdx=4; //(4 bytes for FileNumber, FilePage) PageToBuffer(df_Read_BufferNum,df_Read_PageAdr); // Write memory page to Buffer df_Read_PageAdr++; if (df_Read_PageAdr>DF_LAST_PAGE) // If we reach the end of the memory, start from the begining { df_Read_PageAdr = 0; df_Read_END = true; } // We are starting a new page - read FileNumber and FilePage df_FileNumber = BufferRead(df_Read_BufferNum,0); // High byte df_FileNumber = (df_FileNumber<<8) | BufferRead(df_Read_BufferNum,1); // Low byte df_FilePage = BufferRead(df_Read_BufferNum,2); // High byte df_FilePage = (df_FilePage<<8) | BufferRead(df_Read_BufferNum,3); // Low byte } return result; } int16_t DataFlash_APM1::ReadInt() { uint16_t result; result = ReadByte(); // High byte result = (result<<8) | ReadByte(); // Low byte return (int16_t)result; } int32_t DataFlash_APM1::ReadLong() { uint32_t result; result = ReadByte(); // First byte result = (result<<8) | ReadByte(); result = (result<<8) | ReadByte(); result = (result<<8) | ReadByte(); // Last byte return (int32_t)result; } void DataFlash_APM1::SetFileNumber(uint16_t FileNumber) { df_FileNumber = FileNumber; df_FilePage = 1; } uint16_t DataFlash_APM1::GetFileNumber() { return df_FileNumber; } uint16_t DataFlash_APM1::GetFilePage() { return df_FilePage; } // make one instance for the user to use DataFlash_APM1::DataFlash_APM1() {}