Add SPI FRAM driver
git-svn-id: https://rt-thread.googlecode.com/svn/trunk@1796 bbd45198-f89e-11dd-88c7-29a3b14d5316
This commit is contained in:
parent
1dd92a391d
commit
336a94a15e
|
@ -0,0 +1,296 @@
|
|||
#include "FM25Lx.h"
|
||||
#include "rtthread.h"
|
||||
#include "stm32f2xx_rcc.h"
|
||||
#include <stm32f2xx.h>
|
||||
|
||||
#define FLASH_TRACE(...)
|
||||
//#define FLASH_TRACE rt_kprintf
|
||||
|
||||
#define CS_LOW() GPIO_ResetBits(FM25_SPI_NSS_GPIO, FM25_SPI_NSS_PIN)
|
||||
#define CS_HIGH() GPIO_SetBits(FM25_SPI_NSS_GPIO, FM25_SPI_NSS_PIN)
|
||||
#define spi_config() rt_hw_spi2_baud_rate(SPI_BaudRatePrescaler_4);/* 72M/4=18M */
|
||||
|
||||
#define fram_lock() rt_sem_take(fram_lock, RT_WAITING_FOREVER);
|
||||
#define fram_unlock() rt_sem_release(fram_lock);
|
||||
|
||||
static uint32_t spi_timeout_cnt = 0;
|
||||
|
||||
rt_sem_t fram_lock;
|
||||
|
||||
void rt_hw_spi2_baud_rate(uint16_t SPI_BaudRatePrescaler)
|
||||
{
|
||||
SPI2->CR1 &= ~SPI_BaudRatePrescaler_256;
|
||||
SPI2->CR1 |= SPI_BaudRatePrescaler;
|
||||
}
|
||||
|
||||
/* FM25L256 using SPI2 */
|
||||
void fm25_spi_cfg()
|
||||
{
|
||||
GPIO_InitTypeDef GPIO_InitStructure;
|
||||
SPI_InitTypeDef SPI_InitStructure;
|
||||
|
||||
/* Enable SPI Periph clock */
|
||||
RCC_AHB1PeriphClockCmd(FM25_SPI_NSS_GPIO_CLK | FM25_SPI_GPIO_CLK, ENABLE);
|
||||
RCC_APB1PeriphClockCmd(FM25_SPI_CLK, ENABLE); //enable SPI clock
|
||||
|
||||
//Setup GPIO
|
||||
GPIO_InitStructure.GPIO_Pin = FM25_SPI_SCK | FM25_SPI_MISO | FM25_SPI_MOSI;
|
||||
|
||||
/*Connect Pin to AF*/
|
||||
GPIO_PinAFConfig(FM25_SPI_GPIO, GPIO_PinSource3, GPIO_AF_SPI3);
|
||||
GPIO_PinAFConfig(FM25_SPI_GPIO, GPIO_PinSource4, GPIO_AF_SPI3);
|
||||
GPIO_PinAFConfig(FM25_SPI_GPIO, GPIO_PinSource5, GPIO_AF_SPI3);
|
||||
|
||||
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
|
||||
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
|
||||
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
|
||||
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
|
||||
GPIO_Init(FM25_SPI_GPIO, &GPIO_InitStructure);
|
||||
|
||||
/* CS pin: PB12 */
|
||||
GPIO_InitStructure.GPIO_Pin = FM25_SPI_NSS_PIN;
|
||||
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
|
||||
GPIO_Init(FM25_SPI_NSS_GPIO, &GPIO_InitStructure);
|
||||
CS_HIGH();
|
||||
|
||||
SPI_Cmd(FM25_SPI, DISABLE);
|
||||
/*------------------------ SPI configuration ------------------------*/
|
||||
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//SPI_Direction_1Line_Tx;
|
||||
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
|
||||
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
|
||||
SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;
|
||||
SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;
|
||||
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
|
||||
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4;/* 72M/64=1.125M */
|
||||
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
|
||||
SPI_InitStructure.SPI_CRCPolynomial = 7;
|
||||
|
||||
//SPI_I2S_DeInit(FM25_SPI);
|
||||
SPI_Init(FM25_SPI, &SPI_InitStructure);
|
||||
|
||||
/* Enable SPI_MASTER */
|
||||
SPI_Cmd(FM25_SPI, ENABLE);
|
||||
//SPI_CalculateCRC(FM25_SPI, DISABLE);
|
||||
|
||||
fram_lock = rt_sem_create("framlock", 1, RT_IPC_FLAG_FIFO);
|
||||
}
|
||||
static uint8_t spi_readwrite(uint8_t data)
|
||||
{
|
||||
int32_t timeout = 0xFFFFF;
|
||||
//rt_kprintf("State 0x%X\n", SPI_I2S_GetFlagStatus(FM25_SPI, SPI_I2S_FLAG_TXE));
|
||||
//Wait until the transmit buffer is empty
|
||||
while (SPI_I2S_GetFlagStatus(FM25_SPI, SPI_I2S_FLAG_TXE) == RESET && --timeout >0);
|
||||
|
||||
if( timeout <= 0 ){ spi_timeout_cnt++; return 0;}
|
||||
// Send the byte
|
||||
SPI_I2S_SendData(FM25_SPI, data);
|
||||
|
||||
timeout = 0xFFFFF;
|
||||
//Wait until a data is received
|
||||
while (SPI_I2S_GetFlagStatus(FM25_SPI, SPI_I2S_FLAG_RXNE) == RESET && --timeout >0);
|
||||
if( timeout <= 0 ){ spi_timeout_cnt++; return 0;}
|
||||
// Get the received data
|
||||
data = SPI_I2S_ReceiveData(FM25_SPI);
|
||||
|
||||
// Return the shifted data
|
||||
return data;
|
||||
}
|
||||
static uint8_t fm25_read_status(void)
|
||||
{
|
||||
uint8_t tmp;
|
||||
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_RDSR );
|
||||
tmp=spi_readwrite(0xFF);
|
||||
CS_HIGH();
|
||||
return tmp;
|
||||
}
|
||||
|
||||
rt_size_t fm25_read(rt_device_t dev, rt_off_t offset, void * buf, rt_size_t size)
|
||||
{
|
||||
uint32_t index;
|
||||
|
||||
uint8_t *buffer = (uint8_t*) buf;
|
||||
|
||||
fram_lock();
|
||||
//spi_config();
|
||||
//rt_kprintf("READ: %d, size=%d\n", offset, size);
|
||||
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_READ);
|
||||
spi_readwrite( (offset >> 8)&0xFF );
|
||||
spi_readwrite( offset & 0xFF );
|
||||
for(index=0; index<size; index++)
|
||||
{
|
||||
*buffer++ = spi_readwrite(0xFF);
|
||||
|
||||
if( spi_timeout_cnt > 0 )
|
||||
{
|
||||
fram_unlock();
|
||||
spi_timeout_cnt = 0;
|
||||
rt_kprintf("Read time out\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
offset++;
|
||||
}
|
||||
CS_HIGH();
|
||||
|
||||
fram_unlock();
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
rt_size_t fm25_write(rt_device_t dev, rt_off_t offset, const void * buf, rt_size_t size)
|
||||
{
|
||||
uint32_t index = size;
|
||||
|
||||
uint8_t *buffer = (uint8_t*) buf;
|
||||
fram_lock();
|
||||
//spi_config();
|
||||
//rt_kprintf("WRITE: %d, size=%d\n", offset, size);
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_WREN );
|
||||
CS_HIGH();
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_WRITE);
|
||||
spi_readwrite( (offset >> 8)&0xFF );
|
||||
spi_readwrite( offset & 0xFF );
|
||||
while( index > 0 )
|
||||
{
|
||||
spi_readwrite( *buffer++ );
|
||||
|
||||
if( spi_timeout_cnt > 0 )
|
||||
{
|
||||
fram_unlock();
|
||||
rt_kprintf("Write time out\n");
|
||||
spi_timeout_cnt = 0;
|
||||
return -1;
|
||||
}
|
||||
index--;
|
||||
offset++;
|
||||
}
|
||||
CS_HIGH();
|
||||
//rt_thread_delay(100);
|
||||
|
||||
fram_unlock();
|
||||
|
||||
return size;
|
||||
}
|
||||
static rt_err_t fm25_init(rt_device_t dev)
|
||||
{
|
||||
return RT_EOK;
|
||||
}
|
||||
static rt_err_t fm25_open(rt_device_t dev, rt_uint16_t oflag)
|
||||
{
|
||||
char i;
|
||||
SPI_Cmd(FM25_SPI, ENABLE);
|
||||
|
||||
if( oflag != RT_DEVICE_FLAG_RDONLY )
|
||||
{
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_WRSR );
|
||||
spi_readwrite( FM25_WPEN );
|
||||
CS_HIGH();
|
||||
//rt_kprintf("RDSR=0x%X\n", fm25_read_status());
|
||||
|
||||
}
|
||||
return RT_EOK;
|
||||
}
|
||||
static rt_err_t fm25_close(rt_device_t dev)
|
||||
{
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_WRDI );
|
||||
CS_HIGH();
|
||||
SPI_Cmd(FM25_SPI, DISABLE);
|
||||
|
||||
return RT_EOK;
|
||||
}
|
||||
static rt_err_t fm25_control(rt_device_t dev, rt_uint8_t cmd, void *args)
|
||||
{
|
||||
RT_ASSERT(dev != RT_NULL);
|
||||
|
||||
if (cmd == RT_DEVICE_CTRL_BLK_GETGEOME)
|
||||
{
|
||||
struct rt_device_blk_geometry *geometry;
|
||||
|
||||
geometry = (struct rt_device_blk_geometry *)args;
|
||||
if (geometry == RT_NULL) return -RT_ERROR;
|
||||
|
||||
geometry->bytes_per_sector = 1;
|
||||
geometry->block_size = 1;
|
||||
geometry->sector_count = 8192;
|
||||
|
||||
}
|
||||
|
||||
return RT_EOK;
|
||||
}
|
||||
|
||||
static struct rt_device spi_flash_device;
|
||||
void fm25_hw_init()
|
||||
{
|
||||
int i = 0xFFFFF;
|
||||
fm25_spi_cfg();
|
||||
|
||||
while(i--);
|
||||
//spi_config();
|
||||
CS_LOW();
|
||||
spi_readwrite( FM25_WRDI );
|
||||
CS_HIGH();
|
||||
|
||||
spi_flash_device.type = RT_Device_Class_Block;
|
||||
spi_flash_device.init = fm25_init;
|
||||
spi_flash_device.open = fm25_open;
|
||||
spi_flash_device.close = fm25_close;
|
||||
spi_flash_device.read = fm25_read;
|
||||
spi_flash_device.write = fm25_write;
|
||||
spi_flash_device.control = fm25_control;
|
||||
/* no private */
|
||||
spi_flash_device.user_data = RT_NULL;
|
||||
|
||||
rt_device_register(&spi_flash_device, "fram0",
|
||||
RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_STANDALONE);
|
||||
|
||||
}
|
||||
|
||||
int fram_test(int x)
|
||||
{
|
||||
//rt_kprintf("SR=0x%X\nCR1=0x%X\nCR2=0x%X\n", FM25_SPI->SR, FM25_SPI->CR1,FM25_SPI->CR2);
|
||||
rt_device_t device = RT_NULL;
|
||||
char buf[256];
|
||||
char read[256];
|
||||
int i, j;
|
||||
|
||||
for(i =0; i< 256; i++ )
|
||||
{
|
||||
buf[i] = i;
|
||||
read[i] = 0;
|
||||
}
|
||||
// step 1:find device
|
||||
device = rt_device_find("fram0");
|
||||
if( device == RT_NULL)
|
||||
{
|
||||
rt_kprintf("device %s: not found!\r\n");
|
||||
return RT_ERROR;
|
||||
}
|
||||
device->open(device,RT_DEVICE_FLAG_RDWR);
|
||||
|
||||
for( j = 0; j < FM25_MAXSIZE; j+= 256 )
|
||||
//j = 256*x;
|
||||
{
|
||||
//rt_kprintf("RDSR=0x%X\n", fm25_read_status());
|
||||
device->write(device,j, buf,256);
|
||||
device->read(device,j, read,256);
|
||||
for(i =0; i< 256; i++ )
|
||||
{
|
||||
if( buf[i] != read[i] )
|
||||
rt_kprintf("error at %d: %d!=%d\n", i, buf[i], read[i]);
|
||||
}
|
||||
}
|
||||
device->close(device);
|
||||
rt_kprintf("Finsh test\n");
|
||||
}
|
||||
#ifdef RT_USING_FINSH
|
||||
#include <finsh.h>
|
||||
FINSH_FUNCTION_EXPORT(fram_test, test system);
|
||||
#endif
|
|
@ -0,0 +1,43 @@
|
|||
#ifndef FM25LX_H
|
||||
#define FM25LX_H
|
||||
|
||||
#define FM25_WREN 0x06
|
||||
#define FM25_WRDI 0x04
|
||||
#define FM25_RDSR 0x05
|
||||
#define FM25_WRSR 0x01
|
||||
#define FM25_READ 0x03
|
||||
#define FM25_WRITE 0x02
|
||||
#define FM25_WEL 0x02
|
||||
#define FM25_WPEN 0x80
|
||||
|
||||
#define FM25CL64B
|
||||
//#define FM25LC256
|
||||
|
||||
#ifdef FM25CL64B
|
||||
#define FM25_MAXSIZE 8192
|
||||
#elif define(FM25LC256)
|
||||
#define FM25_MAXSIZE 32768
|
||||
#endif
|
||||
|
||||
#define FM25_SPI SPI3
|
||||
#define FM25_SPI_GPIO GPIOB
|
||||
#define FM25_SPI_MOSI GPIO_Pin_5
|
||||
#define FM25_SPI_MISO GPIO_Pin_4
|
||||
#define FM25_SPI_SCK GPIO_Pin_3
|
||||
#define FM25_SPI_NSS_GPIO GPIOD
|
||||
#define FM25_SPI_NSS_PIN GPIO_Pin_10
|
||||
#define FM25_SPI_CLK RCC_APB1Periph_SPI3
|
||||
#define FM25_SPI_GPIO_CLK RCC_AHB1Periph_GPIOB
|
||||
#define FM25_SPI_NSS_GPIO_CLK RCC_AHB1Periph_GPIOD
|
||||
|
||||
#define FM25_SPI_DMA_CLK RCC_AHB1Periph_DMA1
|
||||
#define FM25_SPI_DMA_Channel DMA_Channel_0
|
||||
#define FM25_SPI_RX_DMA_Stream DMA1_Stream0
|
||||
#define FM25_SPI_RX_DMA_IRQ DMA1_Stream0_IRQn
|
||||
#define FM25_SPI_RX_DMA_FLAG DMA_IT_TCIF0
|
||||
#define FM25_SPI_TX_DMA_Stream DMA1_Stream5
|
||||
#define FM25_SPI_TX_DMA_IRQ DMA1_Stream5_IRQn
|
||||
#define FM25_SPI_TX_DMA_FLAG DMA_IT_TCIF5
|
||||
#define FM25_SPI_DR_Base 0x4003C00C
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue