Browse Source

Add SPI FRAM driver

git-svn-id: https://rt-thread.googlecode.com/svn/trunk@1796 bbd45198-f89e-11dd-88c7-29a3b14d5316
yungchi@cs.nctu.edu.tw 13 years ago
parent
commit
336a94a15e
2 changed files with 339 additions and 0 deletions
  1. 296 0
      bsp/stm32f20x/Drivers/FM25Lx.c
  2. 43 0
      bsp/stm32f20x/Drivers/FM25Lx.h

+ 296 - 0
bsp/stm32f20x/Drivers/FM25Lx.c

@@ -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

+ 43 - 0
bsp/stm32f20x/Drivers/FM25Lx.h

@@ -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