树莓派4BSPI读写flash
1.树莓派SPI介绍
4B的引脚如下图所示:
其中Pin19、21、23是SPI0,接口定义如下所示:
- 时钟(SPI CLK, SCLK)
- 主机输出、从机输入(MOSI)
- 主机输入、从机输出(MISO)
在使用 SPI 接口前,你需要使用 gpio 命令来加载 SPI 驱劢到内核中:
gpio load spi
如果您需要的缓冲区大于 4KB,需要在命令行迕行指定缓冲区的大小,单位是 KB:
gpio load spi 100
上述命令将会分配 100KB 的缓冲区.(您可能很少需要改变返项讴置,默认值对于绝大多数应用程序来说已经足够了). 为了使用 SPI 库,你也需要在你的程序中添加如下语句:
#include <wiringPiSPI.h>
程序在编译连接时,仍然需要添加-lwiringPi 选项
需要用到的函数如下所示:
int wiringPiSPISetup(int channel, int speed);//使用该函数可以初始化一个 SPI 通道,树莓派有两个 SPI 通道(0 和 1)。//speed 参数是一个整数值,其范围为 500000~32000000,代表 SPI 时钟速度,单位是 Hz。//返回值为-1,则失败。则需要检查一下电路连接和是否开启了树莓派的SPI。int wiringPiSPIDataRW(int channel, unsigned char* data, int len);
//该函数执行一个同时读写操作,通过选定的 SPI 总线。缓冲区中的数据,将会被 SPI总线的返回数据所覆盖。
void delay (unsigned int howLong)
//延时ms,最大传入32位无符号型整数,大约49天。
void delayMicroseconds (unsigned int howLong)
//延时微秒,最大传入32位无符号型整数,大约71分钟
2.FM25L16B芯片介绍
FM25L16是采用先进的铁电工艺制造的1024*16位的非易失性存储器(2048个字节)。铁电随机存储器(FRAM)具有非易失性,并且可以象RAM一样快速读写。FM25L16中的数据在掉电后可以保存45年。相对EEPROM或其他非易失性存储器,FM25L16具有结构更简单,系统可靠性更高等诸多优点。
与EEPROM系列不同的是,FM25L16以总线速度进行写操作,无须延时。数据发到FM25L16后直接写到具体的单元地址,下一个总线操作可以立即开始,无需数据轮询。此外,FM25L16的读/写次数几乎为无限次,比EEPROM高得多。同时,FM25L16的功耗也远比EEPROM低。
引脚定义如下所示:
- /CS: 片选,低电平为激活设备
- SCK: SPI输入时钟,频率最高支持20MHZ
- /HOLD: 输入保持,比如当我们在进行读写的时候,假如产生了一个中断,由于时序已经在进行了,这时可以给个低电平让芯片保持时序,等待中断处理完成后,再来置高,继续读写数据.
- /WP: 写保护,为低电平则不能写操作
- SI、SO: SPI输入输出数据引脚
指令如下所示:
2.1 状态寄存器(0x01)介绍
状态寄存器的每个bit位意义如下所示:
其中BP1和BP0是设置写保护区域的.如下图所示:
我们必须将BP1和BP0设置为0,才可以有写所有地址的权限.
2.2 写寄存器(0x06)介绍
所有对内存数组的写入都以WREN(0X06)操作码开头,下一个操作码是WRITE指令,这个操作码后面跟着一个双字节地址。地址的前5位将被忽略(最多存储2048字节),然后就可以一直写入数据.最后将CS置高则完成写操作.
3.最终代码
#include<stdio.h>#include
<cstring>#include
<wiringPi.h>#include
<wiringPiSPI.h>typedef unsigned
char u8;typedef unsigned
short u16;#define FM25CL16_WREN 0x06 // 写使能#define FM25CL16_WRITE 0x02 // 写寄存器
#define FM25CL16_READ 0x03 // 读寄存器
#define FM25CL64_WRSR 0x01 // 状态寄存器
#define FM25L16_CSPIN 21 // 片选引脚
void initSPI()
{
//初始化所用到的IO引脚
pinMode(FM25L16_CSPIN, OUTPUT);
digitalWrite(FM25L16_CSPIN, HIGH);
//初始化SPI通道0,并设置为速度
if(wiringPiSPISetup(0,5000000)==-1) {
printf("init spi failed!
");
}
}
// 片选
void fm25l16Cs(u8 select)
{
if (select == 0) {
digitalWrite(FM25L16_CSPIN, LOW);
} else {
digitalWrite(FM25L16_CSPIN, HIGH);
}
}
// 向FLASH写入一个字节数据
static u8 fm25l16WriteByte(u8 Temp)
{
return wiringPiSPIDataRW(0,&Temp,1);
}
// FLASH写使能
staticvoid fm25l16WriteEnable()
{
fm25l16Cs(0);
fm25l16WriteByte(FM25CL16_WREN);
fm25l16Cs(1);
}
// 写入一串数据
void fm25l16WriteBuff(u16 addr,u8* buff,u16 len)
{
u8 buffHead[3]; // 数据头
fm25l16WriteEnable();
fm25l16Cs(0);
buffHead[0] = FM25CL16_WRITE;
buffHead[1] = (addr&0xFF00)>>8;
buffHead[2] = (addr&0x00FF);
wiringPiSPIDataRW(0,buffHead,3);
wiringPiSPIDataRW(0,buff,len);
fm25l16Cs(1);
}
// 读出一串数据到buff中
void fm25l16ReadBuff(u16 addr,u8* buff,u16 len)
{
u8 buffHead[3]; // 数据头
fm25l16Cs(0);
buffHead[0] = FM25CL16_READ;
buffHead[1] = (addr&0xFF00)>>8;
buffHead[2] = (addr&0x00FF);
wiringPiSPIDataRW(0,buffHead,3);
wiringPiSPIDataRW(0,buff,len);
fm25l16Cs(1);
}
// 写状态
void fm25l16Status()
{
u8 buff[2] = {FM25CL64_WRSR, 0X00};
fm25l16WriteEnable();
fm25l16Cs(0);
wiringPiSPIDataRW(0,buff,2); // 取消写保护
fm25l16Cs(1);
}
int main()
{
u8 data[10];
u8 i=0;
u8 beginData = 0;
//初始化wiringPI的库函数
if(wiringPiSetup()<0) {
printf("init wiringPi error
");
}
initSPI(); //spi的初始化
fm25l16Status(); // 写状态寄存器
while(1) {
for(i=0;i<10;i++) {
data[i] = beginData++;
}
fm25l16WriteBuff(0,data,10);
printf("write ok
");
fm25l16ReadBuff(0,data,10);
printf("read: ");
for(i=0;i<10;i++) {
printf("%d ",data[i]);
}
printf("
");
delay(20);
}
return0;
}
运行效果如下所示:
以上是 树莓派4BSPI读写flash 的全部内容, 来源链接: utcz.com/z/519767.html