打印
[RISC-V MCU 应用开发]

【RISC-V MCU CH32V103测评7】+ SPI操作外部flash测试

[复制链接]
357|0
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
板载一块spi的flash芯片W25Q16,16兆的大小。官方的例程其实已经把这个flash芯片驱动起来了,我这里做的主要就是再过一遍spi的配置,检查一下flash芯片的读写函数是否完整,读写是否正确。然后就是把用到的几个引脚增加了宏定义,主要是因为pA2这个引脚,既连接了Flash芯片的CS引脚,又连接了触摸按键TK2,其实在设计的时候应该能避免这种情况的,因此在这个测评中我们需要修改触摸按键,把触摸按键2变为触摸按键1。

在操作flash芯片之前,首先我们做的应该是把板上CS引脚的那个电阻焊接上,电阻在板子上的丝印是R11,位于flash芯片的上方。换个0欧姆的电阻或者直接短接都是可以的,我这里是直接短接的。R11的位置如下:


Flash芯片的操作函数如下,基本上都是官方改革的例子:
#include "w25q16.h"


/* Global Variable */
u8 SPI_FLASH_BUF[4096];

/*******************************************************************************
* Function Name  : SPI_Flash_Init
* Description    : Configuring the SPI for operation flash.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_Flash_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    SPI_InitTypeDef  SPI_InitStructure;

    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA|RCC_APB2Periph_SPI1, ENABLE );

    GPIO_InitStructure.GPIO_Pin = SPI_CS_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;      //推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    GPIO_SetBits(SPI_PORT, SPI_CS_PIN);                      //CS引脚PA2置高电平

    GPIO_InitStructure.GPIO_Pin = SPI_SCK_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;       //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init( SPI_PORT, &GPIO_InitStructure );

    GPIO_InitStructure.GPIO_Pin = SPI_MISO_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
    GPIO_Init( SPI_PORT, &GPIO_InitStructure );

    GPIO_InitStructure.GPIO_Pin = SPI_MOSI_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;       //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init( SPI_PORT, &GPIO_InitStructure );

    SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //设置SPI通讯方向为双线全双工方式
    SPI_InitStructure.SPI_Mode = SPI_Mode_Master;      //设置SPI为主机端模式
    SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;  //设置SPI通讯的数据帧大小为8位
    SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;        //设置SPI的时钟极性为高电平
    SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;       //设置SPI的时钟相位为在SCK的偶数边沿采集数据
    SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;          //设置NSS引脚(即片选引脚)的使用模式为软件模式
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4; //设置波特率分频因子为4分频
    SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //设置数据传输为高位数据在前
    SPI_InitStructure.SPI_CRCPolynomial = 7;           //SPI的CRC校验中多项式的值
    SPI_Init(SPI1, &SPI_InitStructure);                //初始化SPI

    SPI_Cmd(SPI1, ENABLE);                             //使能SPI1外设
}

/*******************************************************************************
* Function Name  : SPI1_ReadWriteByte
* Description    : SPI1 read or write one byte.
* Input          : TxData: write one byte data.
* Return         : Read one byte data.
*******************************************************************************/
u8 SPI1_ReadWriteByte(u8 TxData)
{
    u8 i=0;

    while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET)  //等待发送缓冲区为空,TXE事件
    {
        i++;
        if(i>200)return 0;
    }

    SPI_I2S_SendData(SPI1, TxData);    //写入数据寄存器,把要写入的数据写入发送缓冲区,即通过外设SPI1发送一个数据
    i=0;

    while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET) //等待接收缓冲区非空,RXNE事件
    {
        i++;
        if(i>200)return 0;
    }

    return SPI_I2S_ReceiveData(SPI1);  //读取数据寄存器,获取接收缓冲区数据,即返回SPI1最近接收到的数据
}

/*******************************************************************************
* Function Name  : SPI_Flash_ReadSR
* Description    : Read W25Qxx status register.
*       ——BIT7  6   5   4   3   2   1   0
*       ——SPR   RV  TB  BP2 BP1 BP0 WEL BUSY
* Input          : None
* Return         : byte: status register value.
*******************************************************************************/
u8 SPI_Flash_ReadSR(void)
{
    u8 byte=0;

    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);    //PA2=0,使能片选信号
    SPI1_ReadWriteByte(W25X_ReadStatusReg); //发送读状态寄存器命令
    byte=SPI1_ReadWriteByte(0Xff);          //读取一个字节
    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);    //取消片选信号

    return byte;
}

/*******************************************************************************
* Function Name  : SPI_FLASH_Write_SR
* Description    : Write W25Qxx status register.
* Input          : sr:status register value.
* Return         : None
*******************************************************************************/
void SPI_FLASH_Write_SR(u8 sr)
{
    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
    SPI1_ReadWriteByte(W25X_WriteStatusReg); //发送写状态寄存器命令
    SPI1_ReadWriteByte(sr);                  //写入一个字节
    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
}

/*******************************************************************************
* Function Name  : SPI_Flash_Wait_Busy
* Description    : Wait flash free.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_Flash_Wait_Busy(void)
{
    while((SPI_Flash_ReadSR()&0x01)==0x01); //等待FLASH空闲
}

/*******************************************************************************
* Function Name  : SPI_FLASH_Write_Enable
* Description    : Enable flash write.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_FLASH_Write_Enable(void)
{
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_WriteEnable); //发送写使能
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
}

/*******************************************************************************
* Function Name  : SPI_FLASH_Write_Disable
* Description    : Disable flash write.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_FLASH_Write_Disable(void)
{
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_WriteDisable); //发送写禁止指令
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
}

/*******************************************************************************
* Function Name  : SPI_Flash_ReadID
* Description    : Read flash ID.
* Input          : None
* Return         : Temp: FLASH ID.
*******************************************************************************/
u16 SPI_Flash_ReadID(void)
{
    u16 Temp = 0;

    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
    SPI1_ReadWriteByte(W25X_ManufactDeviceID); //发送读取ID命令
    SPI1_ReadWriteByte(0x00);
    SPI1_ReadWriteByte(0x00);
    SPI1_ReadWriteByte(0x00);
    Temp|=SPI1_ReadWriteByte(0xFF)<<8;
    Temp|=SPI1_ReadWriteByte(0xFF);
    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);

    return Temp;
}

/*******************************************************************************
* Function Name  : SPI_Flash_Erase_Sector
* Description    : Erase one sector(4Kbyte).
* Input          : Dst_Addr:  0 —— 2047
* Return         : None
*******************************************************************************/
void SPI_Flash_Erase_Sector(u32 Dst_Addr)
{
    Dst_Addr*=4096;
    SPI_FLASH_Write_Enable();
    SPI_Flash_Wait_Busy();
    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
    SPI1_ReadWriteByte(W25X_SectorErase);     //发送扇区擦除指令
    SPI1_ReadWriteByte((u8)((Dst_Addr)>>16)); //发送24bit地址
    SPI1_ReadWriteByte((u8)((Dst_Addr)>>8));
    SPI1_ReadWriteByte((u8)Dst_Addr);
    GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
    SPI_Flash_Wait_Busy(); //等待擦除完成
}

/*******************************************************************************
* Function Name  : SPI_Flash_Read
* Description    : Read data from flash.
* Input          : pBuffer:
*                  ReadAddr:Initial address(24bit).
*                  size: Data length.
* Return         : None
*******************************************************************************/
void SPI_Flash_Read(u8* pBuffer,u32 ReadAddr,u16 size)
{
  u16 i;

  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_ReadData);        //发送读取命令
  SPI1_ReadWriteByte((u8)((ReadAddr)>>16)); //发送24bit地址
  SPI1_ReadWriteByte((u8)((ReadAddr)>>8));
  SPI1_ReadWriteByte((u8)ReadAddr);

  for(i=0;i<size;i++)
  {
     pBuffer[i]=SPI1_ReadWriteByte(0XFF); //循环读数
  }

  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
}

/*******************************************************************************
* Function Name  : SPI_Flash_Write_Page
* Description    : Write data by one page.
* Input          : pBuffer:
*                  WriteAddr:Initial address(24bit).
*                  size:Data length.
* Return         : None
*******************************************************************************/
void SPI_Flash_Write_Page(u8* pBuffer,u32 WriteAddr,u16 size)
{
  u16 i;

  SPI_FLASH_Write_Enable();
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_PageProgram);      //发送写页命令
  SPI1_ReadWriteByte((u8)((WriteAddr)>>16)); //发送24bit地址
  SPI1_ReadWriteByte((u8)((WriteAddr)>>8));
  SPI1_ReadWriteByte((u8)WriteAddr);

  for(i=0;i<size;i++)
    {
        SPI1_ReadWriteByte(pBuffer[i]); //循环写数
    }

  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
  SPI_Flash_Wait_Busy(); //等待写入结束
}

/*******************************************************************************
* Function Name  : SPI_Flash_Write_NoCheck
* Description    : Write data to flash.(need Erase)
*                  All data in address rang is 0xFF.
* Input          : pBuffer:
*                  WriteAddr: Initial address(24bit).
*                  size: Data length.
* Return         : None
*******************************************************************************/
void SPI_Flash_Write_NoCheck(u8* pBuffer,u32 WriteAddr,u16 size)
{
    u16 pageremain;

    pageremain=256-WriteAddr%256; //单页剩余的字节数

    if(size<=pageremain)pageremain=size; //不大于256个字节

    while(1)
    {
        SPI_Flash_Write_Page(pBuffer,WriteAddr,pageremain);

        if(size==pageremain)
        {
            break; //写入结束了
        }
        else
        {
            pBuffer+=pageremain;
            WriteAddr+=pageremain;
            size-=pageremain; //减去已经写入了的字节数

            if(size>256)pageremain=256; //一次可以写入256个字节
            else pageremain=size; //不够256个字节了
        }
    }
}

/*******************************************************************************
* Function Name  : SPI_Flash_Write
* Description    : Write data to flash.(no need Erase)
* Input          : pBuffer:
*                  WriteAddr: Initial address(24bit).
*                  size: Data length.
* Return         : None
*******************************************************************************/
void SPI_Flash_Write(u8* pBuffer,u32 WriteAddr,u16 size)
{
    u32 secpos;
    u16 secoff;
    u16 secremain;
    u16 i;

    secpos=WriteAddr/4096; //扇区地址
    secoff=WriteAddr%4096; //在扇区内的便宜
    secremain=4096-secoff; //扇区剩余空间大小

    if(size<=secremain)secremain=size; //不大于4096个字节

    while(1)
    {
        SPI_Flash_Read(SPI_FLASH_BUF,secpos*4096,4096); //读出整个扇区的内容

        for(i=0;i<secremain;i++) //校验数据
        {
            if(SPI_FLASH_BUF[secoff+i]!=0XFF)break; //需要擦除
        }

        if(i<secremain) //需要擦除
        {
            SPI_Flash_Erase_Sector(secpos); //擦除这个扇区

            for(i=0;i<secremain;i++) //复制
            {
                SPI_FLASH_BUF[i+secoff]=pBuffer[i];
            }

            SPI_Flash_Write_NoCheck(SPI_FLASH_BUF,secpos*4096,4096); //写入整个扇区

        }
        else
        {
            SPI_Flash_Write_NoCheck(pBuffer,WriteAddr,secremain); //写已经擦除了的,直接写入扇区剩余空间
        }

        if(size==secremain)
        {
            break; //写入结束
        }
        else
        {
            secpos++; //扇区地址增1
            secoff=0; //偏移位置为0

          pBuffer+=secremain;   //指针偏移
          WriteAddr+=secremain; //写地址偏移
          size-=secremain;      //字节数递减

            if(size>4096)
            {
                secremain=4096; //下一个扇区还是写不完
            }
            else
            {
                secremain=size; //下一个扇区可以写完了
            }
        }
    }
}

/*******************************************************************************
* Function Name  : SPI_Flash_Erase_Chip
* Description    : Erase all FLASH pages.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_Flash_Erase_Chip(void)
{
  SPI_FLASH_Write_Enable();
  SPI_Flash_Wait_Busy();
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_ChipErase);  //发送片擦除指令
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
  SPI_Flash_Wait_Busy();
}

/*******************************************************************************
* Function Name  : SPI_Flash_PowerDown
* Description    : Enter power down mode.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_Flash_PowerDown(void)
{
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_PowerDown);  //发送进入断电模式之灵
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
  Delay_Us(3);
}

/*******************************************************************************
* Function Name  : SPI_Flash_WAKEUP
* Description    : Power down wake up.
* Input          : None
* Return         : None
*******************************************************************************/
void SPI_Flash_WAKEUP(void)
{
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 0);
  SPI1_ReadWriteByte(W25X_ReleasePowerDown); //发送断电唤醒指令
  GPIO_WriteBit(SPI_PORT, SPI_CS_PIN, 1);
  Delay_Us(3);
}


然后是更改触摸通道2为触摸通道1:
void Touch_Key_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE );

    GPIO_InitStructure.GPIO_Pin = AD1_CHANNEL1_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    ADC_Cmd(ADC1, ENABLE);
    TKEY_CH =0x00000001;     // TouchKey Channel
    TKEY_CR |= 0x51000000;   // Enable TouchKey
}

初始化触摸通道1的Io引脚后,只需要修改下面这条语句就好了:
TKEY_CH =0x00000001; // TouchKey Channel

工程运行后flash读写OK:


全部的工程打包如下:
CH32V103-SPI-FLASH.rar (553.67 KB)

使用特权

评论回复

相关帖子

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

本版积分规则

个人签名:qq群: 嵌入式系统arm初学者 224636155←← +→→点击-->小 i 精品课全集,21ic公开课~~←←→→点击-->小 i 精品课全集,给你全方位的技能策划~~←←

2710

主题

19166

帖子

103

粉丝