查看: 1682|回复: 0
收起左侧

第四十七章:SPI-单工通信(1条时钟线和1条双向数据线)

[复制链接]

  离线 

  • TA的每日心情
    慵懒
    2021-7-23 17:16
  • 签到天数: 17 天

    [LV.4]

    发表于 2021-4-28 17:44:16 | 显示全部楼层 |阅读模式

    有人预言,RISC-V或将是继Intel和Arm之后的第三大主流处理器体系。欢迎访问全球首家只专注于RISC-V单片机行业应用的中文网站

    您需要 登录 才可以下载或查看,没有帐号?立即注册

    x
    本帖最后由 草帽王子 于 2021-9-10 17:36 编辑

    CH32V103应用教程——
    SPI-单工通信(1条时钟线和1条双向数据线),主机接收从机发送


    本章教程主要在SPI单工通信方式下进行1条时钟线和1条双向数据线配置,并进行主机接收从机发送。


    1、SPI简介及相关函数介绍

    关于SPI单工通信模式下1条时钟线和1条双向数据线配置介绍,在第46章已经进行介绍,在此不再赘述。
    关于CH32V103 SPI具体信息,可参考CH32V103应用手册。SPI标准库函数在第十五章节已介绍,在此不再赘述。


    2、硬件设计

    本章教程主要在SPI单工通信模式下选择1条时钟线和1条双向数据线进行主机接收从机发送实验,需用到两个开发板,且由于采用1条时钟线和1条双向数据线配置,因此主设备使用MOSI引脚,从设备使用MISO引脚进行通讯。此处使用外设为SPI1,主设备MOSI对应引脚为PA7引脚,从设备MISO对应引脚为PA6引脚,将主设备PA7引脚与从设备PA6引脚连接起来,此外还需将两个开发板SPI1对应的SCK引脚PA5连接起来。

    此外,由于两个开发板需要同时进行上电传输,因此将两个开发板的3.3V引脚和GND引脚进行连接。


    3、软件设计

    本章教程主要进行SPI单工通信模式1条时钟线和1条双向数据线配置下的主机接收从机发送实验,具体程序如下:
    spi.h文件
    1. #ifndef __SPI_H
    2. #define __SPI_H

    3. #include "ch32v10x_conf.h"

    4. /* SPI Mode Definition */
    5. #define HOST_MODE    0
    6. #define SLAVE_MODE   1

    7. /* SPI Communication Mode Selection */
    8. #define SPI_MODE   HOST_MODE
    9. //#define SPI_MODE   SLAVE_MODE

    10. #define  Size  18

    11. extern volatile u8 Txval;
    12. extern volatile u8 Rxval;
    13. extern u16 TxData[Size];
    14. extern u16 RxData[Size];

    15. void SPI_1Lines_HalfDuplex_Init(void);
    16. void SPI1_IRQHandler(void);

    17. #endif
    复制代码
    spi.h文件主要进行相关宏定义和函数声明;
    spi.c文件
    1. #include "spi.h"

    2. /* Global Variable */
    3. volatile u8 Txval=0, Rxval=0;

    4. u16 TxData[Size] = { 0x0101, 0x0202, 0x0303, 0x0404, 0x0505, 0x0606,
    5.                      0x1111, 0x1212, 0x1313, 0x1414, 0x1515, 0x1616,
    6.                      0x2121, 0x2222, 0x2323, 0x2424, 0x2525, 0x2626 };
    7. u16 RxData[Size];

    8. void SPI1_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));

    9. /*******************************************************************************
    10. * Function Name  : SPI_1Lines_HalfDuplex_Init
    11. * Description    : Configuring the SPI for half-duplex communication.
    12. * Input          : None
    13. * Return         : None
    14. *******************************************************************************/
    15. void SPI_1Lines_HalfDuplex_Init(void)
    16. {
    17.     GPIO_InitTypeDef GPIO_InitStructure;
    18.     SPI_InitTypeDef  SPI_InitStructure;
    19.     NVIC_InitTypeDef NVIC_InitStructure;

    20.     RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA | RCC_APB2Periph_SPI1, ENABLE );

    21. #if (SPI_MODE == HOST_MODE)
    22.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
    23.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   //推挽复用输出
    24.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    25.     GPIO_Init( GPIOA, &GPIO_InitStructure );

    26.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
    27.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;  //浮空输入
    28.     GPIO_Init( GPIOA, &GPIO_InitStructure );

    29. #elif (SPI_MODE == SLAVE_MODE)
    30.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
    31.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;  //浮空输入
    32.     GPIO_Init( GPIOA, &GPIO_InitStructure );

    33.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
    34.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   //推挽复用输出
    35.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    36.     GPIO_Init( GPIOA, &GPIO_InitStructure );
    37. #endif


    38. #if (SPI_MODE == HOST_MODE)
    39.     //配置为单工通信(1条时钟线和1条双向数据线),SPI_Direction_1Line_Rx值为0x8000,即配置SPI控制寄存器位15 BIDIMODE位 为1(单线双向模式)、位14 BIDIOE位为0(禁止输出,仅接收)
    40.     SPI_InitStructure.SPI_Direction = SPI_Direction_1Line_Rx;
    41.     SPI_InitStructure.SPI_Mode = SPI_Mode_Master;  //配置为主机模式

    42. #elif (SPI_MODE == SLAVE_MODE)
    43.     //配置为单工通信(1条时钟线和1条双向数据线),SPI_Direction_1Line_Tx值为0xC000,即配置SPI控制寄存器位15 BIDIMODE位 为1(单线双向模式)、位14 BIDIOE位为1(使能输出,仅发送)
    44.     SPI_InitStructure.SPI_Direction = SPI_Direction_1Line_Tx;
    45.     SPI_InitStructure.SPI_Mode = SPI_Mode_Slave;   //配置为从机模式

    46. #endif

    47.     SPI_InitStructure.SPI_DataSize = SPI_DataSize_16b;  //设置SPI通讯数据帧大小
    48.     SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;   //设置时钟极性
    49.     SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;  //设置时钟相位
    50.     SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;     //设置NSS引脚(即片选引脚)的使用模式
    51.     SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64;  //设置波特率分频因子
    52.     SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;  //设置数据传输高位在前
    53.     SPI_InitStructure.SPI_CRCPolynomial = 7;      //设置用于CRC计算的多项式
    54.     SPI_Init( SPI1, &SPI_InitStructure );

    55.     NVIC_InitStructure.NVIC_IRQChannel = SPI1_IRQn;
    56.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
    57.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    58.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    59.     NVIC_Init(&NVIC_InitStructure);


    60. #if (SPI_MODE == SLAVE_MODE)

    61.     SPI_I2S_ITConfig( SPI1, SPI_I2S_IT_TXE , ENABLE ); //使能开启SPI1接收中断

    62.     //从模式双向模式数据发送,软件必须保证在SPI主设备开始数据传输(接收数据)之前在发送寄存器中写入要发送的数据,即先开启从机SPI再开启主机SPI
    63.     SPI_Cmd( SPI1, ENABLE );

    64. #endif


    65. }

    66. /*******************************************************************************
    67. * Function Name  : SPI1_IRQHandler
    68. * Description    : This function handles SPI1 exception.
    69. * Input          : None
    70. * Return         : None
    71. *******************************************************************************/
    72. void SPI1_IRQHandler(void)
    73. {
    74. #if (SPI_MODE == SLAVE_MODE)

    75.     if( SPI_I2S_GetITStatus( SPI1, SPI_I2S_IT_TXE ) != RESET )            //发送缓冲区非空
    76.     {
    77.         SPI_I2S_SendData( SPI1, TxData[Txval++] );  //发送数据
    78.         if( Txval == 18 )
    79.         {
    80.             SPI_I2S_ITConfig( SPI1, SPI_I2S_IT_TXE , DISABLE ); //关闭发送中断
    81.         }
    82.     }

    83. #elif (SPI_MODE == HOST_MODE)

    84.     if( Rxval<18 )
    85.     {
    86.         if( SPI_I2S_GetFlagStatus( SPI1, SPI_I2S_FLAG_RXNE ) != RESET )   //接收缓冲区非空
    87.         {
    88.                 RxData[Rxval++] = SPI_I2S_ReceiveData( SPI1 ); //接收数据

    89.         }
    90.         //主模式下单向只接收模式,需等待倒数第二个RXNE=1且在关闭SPI之前等待一个SPI时钟周期,此处利用延时函数实现
    91.         if( Rxval == 16 )
    92.         {
    93.             Delay_Us(2);
    94.         }

    95.         //在进入停机模式或关闭该模块时钟之前等待最后一个RXNE=1。
    96.         if( Rxval == 17 )
    97.         {
    98.             SPI_Cmd( SPI1, DISABLE );  //关闭SPI1
    99.         }
    100.     }

    101. #endif
    102. }

    复制代码
    spi.c文件主要包括两个函数:SPI_1Lines_HalfDuplex_Init函数和SPI1_IRQHandler函数。SPI_1Lines_HalfDuplex_Init函数主要进行SPI1单工通信模式1条时钟线和1条双向数据线配置下的主机接收从机发送配置。首先,由于选择1条时钟线和1条双向数据线配置,在这种配置下,主机需要用到SCK引脚和MOSI引脚,从机需要用到SCK引脚和MISO引脚,此处,由于用到SPI1,对应SCK引脚为PA5,MOSI引脚为PA7,MISO引脚为PA6,因此,在SPI_1Lines_HalfDuplex_Init函数中首先对主机和从机对应GPIO引脚进行初始化配置,此处需要注意,由于是主机接收从机发送,需要将PA7设置为浮空输入模式,PA6设置为复用推挽输出模式。

    此外,由于主机作为接收,从机作为发送,此处在SPI_1Lines_HalfDuplex_Init函数中需要对SPI进行主机接收和从机发送初始化配置,此配置可根据CH32V103应用手册主模式和从模式配置步骤进行,主要对SPI通信的通信方向、主从模式、数据帧大小、时钟极性、时钟相位、NSS引脚使用方式、波特率等进行配置,可对照手册参考标准库函数ch32v10x_spi.c文件中SPI_Init函数进行配置。

    注意:此外,此处需要注意的是,从机在双向模式发送数据时,软件必须保证在SPI主设备开始数据接收之前在发送寄存器写入要发送的数据,即先开启从机SPI功能,再开启主机SPI功能。

    SPI1_IRQHandler函数为SPI1中断服务函数,主要进行主机模式下的数据接收和从机模式下的数据发送。

    关于主机接收,此处需要注意的是,如果在最后一个数据传输后关闭SPI模块,需进行以下操作进行处理,保证SPI不会开始一次新的传输:
    1、等待倒数第二个RXNE=1;
    2、在关闭SPI之前等待一个SPI时钟周期,此处可以利用延时函数进行等待;
    3、在进入停机模式或关闭该模块时钟之前等待最后一个RXNE=1。

    main.c文件
    1. /********************************** (C) COPYRIGHT *******************************
    2. * File Name          : main.c
    3. * Author             : WCH
    4. * Version            : V1.0.0
    5. * Date               : 2020/04/30
    6. * Description        : Main program body.
    7. *******************************************************************************/

    8. #include "debug.h"
    9. #include "spi.h"

    10. /*******************************************************************************
    11. * Function Name  : main
    12. * Description    : Main program.
    13. * Input          : None
    14. * Return         : None
    15. *******************************************************************************/
    16. int main(void)
    17. {
    18.     u8 i;

    19.     NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
    20.     Delay_Init();
    21.     USART_Printf_Init(115200);
    22.     printf("SystemClk:%d\r\n",SystemCoreClock);

    23.     SPI_1Lines_HalfDuplex_Init();

    24. #if (SPI_MODE == SLAVE_MODE)
    25.     printf("SLAVE Mode\r\n");
    26.     Delay_Ms(1000);

    27. #endif

    28. #if (SPI_MODE == HOST_MODE)
    29.     printf("HOST Mode\r\n");
    30.     Delay_Ms(2000);
    31.     SPI_Cmd( SPI1, ENABLE );
    32.     SPI_I2S_ITConfig( SPI1, SPI_I2S_IT_RXNE , ENABLE );

    33. #endif

    34.     while(1)
    35.     {
    36. #if (SPI_MODE == SLAVE_MODE)
    37.         while( Txval<18 );

    38.         /* 等待TXE=1,BSY=0,等待发送完成,关闭SPI1 */
    39.         while( SPI_I2S_GetFlagStatus( SPI1, SPI_I2S_FLAG_TXE ) != RESET )
    40.         {
    41.             if( SPI_I2S_GetFlagStatus( SPI1, SPI_I2S_FLAG_BSY ) == RESET )
    42.             {
    43.                 SPI_Cmd( SPI1, DISABLE );  //关闭SPI1
    44.                 printf("Tx End\r\n");
    45.                 while(1);
    46.             }
    47.         }

    48. #elif (SPI_MODE == HOST_MODE)

    49.         while( Rxval<18 );

    50.         for( i=0; i<18; i++ )
    51.         {
    52.             printf( "Rxdata:%04x\r\n", RxData[i] );
    53.         }

    54.         while(1);

    55. #endif
    56.     }
    57. }

    复制代码
    main.c文件主要进行函数初始化以及主机模式下的接收数据打印输出以及从机模式下的数据发送结束提示。


    4、下载验证

    将编译好的程序分别在主机模式和从机模式下下载到两个开发版,并将主机的PA7引脚与从机PA6引脚进行连接,开发板上电后,串口打印如下:

    主机打印:
    CH32V CH573单片机芯片-第四十七章:SPI-单工通信(1条时钟线和1条双向数据线)risc-v单片机中文社区(1)

    从机打印:
    CH32V CH573单片机芯片-第四十七章:SPI-单工通信(1条时钟线和1条双向数据线)risc-v单片机中文社区(2)

    SPI-单工通信(1条时钟线和1条双向
    CH32V CH573单片机芯片-第四十七章:SPI-单工通信(1条时钟线和1条双向数据线)risc-v单片机中文社区(3) 46、SPI-单工通信(1条时钟线和1条双向数据线),主机接收从机发送.rar.rar.rar (483.87 KB, 下载次数: 10)
    链接:https://pan.baidu.com/s/15xdrB3kshPWXxNWu5lYKaw
    提取码:f7qj
    复制这段内容后打开百度网盘手机App,操作更方便哦








    上一篇:第四十六章:SPI-单工通信(1条时钟线和1条双向数据线)
    下一篇:第四十八章:SPI-单工通信(1条时钟线和1条单向数据线)
    RISCV作者优文
    全球首家只专注于RISC-V单片机行业应用的中文网站
    回复

    使用道具 举报

    高级模式
    B Color Image Link Quote Code Smilies

    本版积分规则

    关闭

    RISC-V单片机中文网上一条 /2 下一条



    版权及免责声明|RISC-V单片机中文网 |网站地图

    GMT+8, 2025-1-11 01:56 , Processed in 0.295890 second(s), 48 queries .

    快速回复 返回顶部 返回列表