打印
[范例教程]

c语言实现SPI通信源程序

[复制链接]
1096|22
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
eefas|  楼主 | 2023-7-13 21:42 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
*************
***************/
void SPI_SendData(uint8_t data)
{
        uint8_t cnt;
        for(cnt=0;cnt<8;cnt++)
        {
                SPI_CLK_L;//拉低
          Delay1us(10);//这个延时时间任意,但要大于芯片数据手册上的(纳秒级的)
                if(data &0x80)
                {
                        SPI_MOSI_H;
                }
                else
                {
                        SPI_MOSI_L;
                }
                data <<= 1;//
                Delay1us(10);
                SPI_CLK_H;//拉高
                Delay1us(10);
        }
}

/*****************************************/
uint8_t SPI_ReadData(void)
{
        uint8_t i = 0;
        uint8_t value=0;
        SPI_CLK_H;      
        Delay1us(2);
        for(i=0;i<8;i++)
        {
                SPI_CLK_L;               
                value<<=1;
                if(SPI_MISO_READ())
                {
                        value |= 0x01;
                }
                Delay1us(2);
                SPI_CLK_H;
                Delay1us(2);
        }
      
        return value;
}

/*************
等待擦除完成
**************/
void SPI_WaitErase(void)
{
        uint8_t status = 0x01;
        SPI_CS_L;
        SPI_SendData(0x05); //读取状态寄存器命令
  do
        {
                status = SPI_ReadData();
        }
        while((status&0x01) == 1);
        SPI_CS_H;
}

/************
扇区擦除
4k
**************/
void SPI_EraseSector(uint32_t addr)
{      
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x06);
        SPI_CLK_L;
        SPI_CS_H;
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x20);
        SPI_SendData((uint8_t)(addr&0xFF0000)>>16);
        SPI_SendData((addr&0xFF00)>>8);
        SPI_SendData((addr&0xFF));
        SPI_CLK_L;
        SPI_CS_H;
        SPI_WaitErase();
}


/************
块擦除
32k
**************/
void SPI_EraseBlock(uint32_t addr)
{      
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x06);
        SPI_CLK_L;
        SPI_CS_H;
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x52);
        SPI_SendData((uint8_t)(addr&0xFF0000)>>16);
        SPI_SendData((addr&0xFF00)>>8);
        SPI_SendData((addr&0xFF));
        SPI_CLK_L;
        SPI_CS_H;
        SPI_WaitErase();
}


/************
块擦除
64k
**************/
void SPI_Erase64kBlock(uint32_t addr)
{      
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x06);
        SPI_CLK_L;
        SPI_CS_H;
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0xd8);
        SPI_SendData((uint8_t)(addr&0xFF0000)>>16);
        SPI_SendData((addr&0xFF00)>>8);
        SPI_SendData((addr&0xFF));
        SPI_CLK_L;
        SPI_CS_H;
        SPI_WaitErase();
}

/*****************
检测设备是否正确
设备ID为ID=0XEF4017
正常才继续后面的页写入和读取数据
******************/
uint32_t SPI_CheckDevice(void)
{
        uint8_t temp0,temp1,temp2;
        uint32_t Device_ID=0;
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x9F);
        temp0 = SPI_ReadData();
  temp1 = SPI_ReadData();
        temp2 = SPI_ReadData();
        SPI_CLK_L;
        SPI_CS_H;
        Device_ID = (temp0<<16) | (temp1<<8) | (temp2);
        return Device_ID;
}


/**************
读取数据
readlen: 读取长度
***************/
void SPI_BufferRead(uint8_t *pBuffer,uint32_t addr,uint32_t readlen)
{
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x03);//读数据命令
        SPI_SendData((uint8_t)(addr&0xFF0000)>>16);
        SPI_SendData((addr&0xFF00)>>8);
        SPI_SendData((addr&0xFF));
        while(readlen--)
        {
                *pBuffer = SPI_ReadData();
                 pBuffer++;
        }
        SPI_CLK_L;
        SPI_CS_H;
}

/************
页写入---写入字节数低于256可用页编程
**************/
void SPI_PageWrite(uint8_t *pBuffer,uint32_t addr,uint16_t writelen)
{      
        SPI_CLK_L;
        SPI_CS_L;
        SPI_SendData(0x06);
        SPI_CLK_L
        SPI_CS_H;
        SPI_CLK_L;//拉低
        SPI_CS_L;
        SPI_SendData(0x02);
        SPI_SendData((addr&0xFF0000)>>16);
        SPI_SendData((addr&0xFF00)>>8);
        SPI_SendData((addr&0xFF));
        if(writelen>Page_Size)
        {
//                printf("\r\n页写入最大为256字节\r\n");
                writelen = Page_Size ;
        }
        while(writelen--)
        {
                SPI_SendData(*pBuffer);
                pBuffer++;
        }
        SPI_CLK_L;//拉低
        SPI_CS_H;
        SPI_WaitErase();
}


/*************
写入任意长度数据
*pBuffer :待写入数据
addr                 :需写入flash的地址
numToWrite :待写入数据长度
***************/
void SPI_BufferWrite(uint8_t *pBuffer,uint32_t addr,uint32_t numToWrite)
{
        uint32_t count=0,numPage=0,numSingle=0,Addr=0;
        Addr = addr%Page_Size;//某个page的地址
        count = Page_Size - Addr;//某page剩余字节数
        numPage = numToWrite/Page_Size;//待写入的数据占用page数
        numSingle = numToWrite%Page_Size;//待写入数据不够一个page的字节数
        if(Addr == 0)//在某page 起始位置 1
        {
                if(numPage==0)//写入字节数小于一个page
                {
                        SPI_PageWrite(pBuffer,addr,numToWrite);
                }
                else//写入字节数大于等于1个page
                {
                        while(numPage--)
                        {
                                SPI_PageWrite(pBuffer,addr,Page_Size);
                                pBuffer+=Page_Size;
                                addr+=Page_Size;
                        }
                        SPI_PageWrite(pBuffer,addr,numSingle);
                }
        }
        else //在某page的2~256位置
        {
                if(numPage==0)//写入字节数小于一个page
                {
                        if(numSingle>count)//剩余字节数>该page剩余位置
                        {
                                SPI_PageWrite(pBuffer,addr,count);//向该page的剩余位置写入对应字节
                                pBuffer+=count;
                                addr+=count;
                                SPI_PageWrite(pBuffer,addr,(numSingle-count));//其余的写入下一个page
                        }
                        else
                        {
                                SPI_PageWrite(pBuffer,addr,numToWrite);
                        }
                }
                else//写入字节数大于等于一个page
                {
                        numToWrite -= count;
                        numPage = numToWrite/Page_Size;
                        numSingle        =        numToWrite%Page_Size;
                        SPI_PageWrite(pBuffer,addr,count);//写入该page还可写入的字节数
                        addr+=count;
                        pBuffer+=count;
                        while(numPage--)//page个数
                        {
                                SPI_PageWrite(pBuffer,addr,Page_Size);
                                pBuffer+=Page_Size;
                                addr+=Page_Size;
                        }
                        if(numSingle!=0)//剩余字节数
                        {
                                SPI_PageWrite(pBuffer,addr,numSingle);
                        }
                }
        }
        SPI_WaitErase();
}


/************
数据比较
len :数据长度
不等:0
相等:1
************/
uint8_t pBuffer_Cmp(uint8_t *buffer1,uint8_t *buffer2,uint32_t len)
{
        while(len--)
        {
                if(*buffer1!=*buffer2)
                {
                        return 0;
                }
                buffer1++;
                buffer2++;
        }
        return 1;
}

uint8_t Write_pBuffer[]={"软件模拟SPI成功!"};
uint8_t Read_pBuffer[3500]={0};


/***********
读写测试,验证是否正常
***********/
void SPI_Test(void)
{
        uint32_t Device_ID = 0;
        Device_ID = SPI_CheckDevice();
//        printf("ID=0x%x \r\n", Device_ID);
        if(Device_ID == 0xEF4016)
        {
                if(comparm.DebugComm & F_DebugStart)
                        DebgStr_FramProc("SPI FLASH设备正常!\r\n");
                SPI_EraseSector(0x001000);//sector1
                SPI_BufferWrite(Write_pBuffer,0x001000,(sizeof(Write_pBuffer)/sizeof(*(Write_pBuffer))-1));
                SPI_BufferRead(Read_pBuffer,0x001000,(sizeof(Write_pBuffer)/sizeof(*(Write_pBuffer))-1));
                if(pBuffer_Cmp(Write_pBuffer,Read_pBuffer,(sizeof(Write_pBuffer)/sizeof(*(Write_pBuffer))-1)) == 1)
                {      
                       
                        if(comparm.DebugComm & F_DebugStart)
                                DebgStr_FramProc("软件模拟SPI的读写数据一致!\r\n");
//                        DebgStr_FramProc("\r\n 读出的数据为:%s \r\n", Read_pBuffer);
                }
                else
                {
                        if(comparm.DebugComm & F_DebugStart)
                                DebgStr_FramProc("软件模拟SPI的读写数据不一致!\r\n");
                }
        }
        else
        {
                if(comparm.DebugComm & F_DebugStart)
                        DebgStr_FramProc("SPI FLASH设备异常!\r\n");
        }
}


使用特权

评论回复
沙发
zerorobert| | 2023-8-9 14:55 | 只看该作者
如果支持中断方式的SPI通信,可以使用中断来提高效率和响应性

使用特权

评论回复
板凳
mollylawrence| | 2023-8-9 17:18 | 只看该作者
可以使用示波器或逻辑分析仪来监视和分析SPI信号,以验证通信的正确性。

使用特权

评论回复
地板
juliestephen| | 2023-8-9 22:39 | 只看该作者
在进行SPI通信之前,需要设置SPI通信参数,包括SPI主机和从机的极性、数据位数、传输方式等。

使用特权

评论回复
5
robincotton| | 2023-8-14 11:21 | 只看该作者
在编写代码时,考虑到代码的可重用性和可维护性

使用特权

评论回复
6
kkzz| | 2023-8-14 12:24 | 只看该作者
在C语言中实现SPI通信源程序时,需要注意SPI控制器的初始化、通信参数的设置、数据的控制和错误处理等

使用特权

评论回复
7
fengm| | 2023-8-14 13:06 | 只看该作者
在程序开始时,需要对SPI控制器进行初始化,包括选择SPI模式、时钟极性、数据位数等参数。

使用特权

评论回复
8
benjaminka| | 2023-8-14 14:06 | 只看该作者
SPI通信是全双工的,意味着主设备和从设备可以同时发送和接收数据

使用特权

评论回复
9
sdCAD| | 2023-8-14 17:26 | 只看该作者
SPI通信对时序有严格要求。              

使用特权

评论回复
10
jtracy3| | 2023-8-14 19:39 | 只看该作者
SPI通信过程中,可能会出现错误,如数据传输错误、时钟同步错误等,需要对这些错误进行处理。

使用特权

评论回复
11
lzmm| | 2023-8-14 19:48 | 只看该作者
SPI通信有多种模式,包括时钟极性(CPOL)和时钟相位(CPHA)

使用特权

评论回复
12
alvpeg| | 2023-8-15 13:07 | 只看该作者
设置时钟频率、选择主从机角色以及配置其他相关参数

使用特权

评论回复
13
pixhw| | 2023-8-15 17:47 | 只看该作者
在SPI通信中,可能会发生错误,如传输错误或超时

使用特权

评论回复
14
maudlu| | 2023-8-15 18:31 | 只看该作者
使用适当的数据结构和函数来组织代码

使用特权

评论回复
15
yeates333| | 2023-8-15 19:09 | 只看该作者
选择适当的时钟速率以满足目标设备的要求,并确保主设备和从设备的时钟速率匹配。

使用特权

评论回复
16
jimmhu| | 2023-8-15 19:22 | 只看该作者
SPI通信的速率由时钟频率决定              

使用特权

评论回复
17
ingramward| | 2023-8-15 19:34 | 只看该作者
在编程之前,你需要确定SPI接口的引脚分配。通常,SPI接口有三个信号线:MOSI(主从机数据输出/输入线)、SCK(时钟线)和SS(片选线)

使用特权

评论回复
18
bartonalfred| | 2023-8-15 20:04 | 只看该作者
可以利用中断或DMA来处理SPI通信

使用特权

评论回复
19
saservice| | 2023-8-15 22:04 | 只看该作者
连接SPI设备,包括主设备和从设备之间的物理连接,以及正确的引脚配置。

使用特权

评论回复
20
mollylawrence| | 2023-8-18 22:37 | 只看该作者
需要确定合适的缓冲区大小,并确保在发送和接收之间正确管理缓冲区的数据。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

86

主题

2887

帖子

2

粉丝