打印
[STM8]

STM8L的SPI1基本读写方法

[复制链接]
779|0
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
uiint|  楼主 | 2023-12-31 10:47 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
STM8L的硬件SPI1共有三个主要引脚,分别是MOSI,MISO,SCK。NSS引脚只有在SMT8L作为从设备时,才有用,用于判断是否被选择与SPI主设备通信。本文选取NRF24L01作为SPI从设备,STM8L作为主设备,进行SPI读写功能测试。

下图是淘宝上可以买到的最常见的NRF24L01模块引脚图,可以看到除了MOSI,MISO,SCK,三个和SPI1硬件有关的引脚外,还有CE,CSN,IRQ这三个引脚,这三个引脚用STM8L的普通IO驱动即可.

从下图可以看到,CSN为芯片的使能引脚,读写寄存器时CSN必须为低电平.CE配合的内部配置寄存器,决定NRF24L01状态.

本例中,STM8L作为SPI主设备,在STM8L用户手册关于SPI作为主设备的详细配置流程如下图.


本文只是简单的测试SPI的读写功能,通过标志位等待实现硬件SPI读写单个字节.没有使用中断和DMA功能.
注意:向NRF24L01写数据时,需要在寄存器地址加上0x20,用于指示下一字节发送的数据是写入这个寄存器的.
下图是实际调试结果截图,查看NRF24L01的数据手册,可以知道0x00,0x01,0x02这三个寄存器的复位值分别为0x08,0x3F,0x03.由于0x00这个寄存器中的数据被我改写过了,所以这里显示改写值0x55.
/*硬件连接*

//    STM8L          NRF24L01

//    PB4     -->      CSN

//    PD4     -->      CE

//    PB5     -->      SCK

//    PB6     -->      MOSI

//    PB7     <--      MISO

/****************************************************************************************

*开发环境:IAR for stm8 v1.40.1

*硬件平台:STM8L-DISCOVERY

*功能说明:通过硬件SPI等待的方法,实现对NRF24L01寄存器的读写,借助IAR软件的调试功能,查看变量的数值

*作    者:茗风

****************************************************************************************/

#include"iostm8l152c6.h"

#include"stdbool.h"

#include"stdint.h"

  

#define CSN_H           PB_ODR_ODR4=1

#define CSN_L                  PB_ODR_ODR4=0

#define CE_H            PD_ODR_ODR4=1

#define CE_L            PD_ODR_ODR4=0

uint16_t VDD_Value=0;

/******************************************************************************************************

*  名    称:void delay_10ms(uint8_t x_ms)

*  功    能:延时10ms

*  入口参数:无

*  出口参数:无

*  说    明:

*  范    例:无

******************************************************************************************************/

void delay_100ms(void)

{

  uint8_t i,j;

  for(i=0;i<255;i++)//2*255个指令周期

    for(j=0;j<255;j++);//2*255个指令周期

  //delay_10ms共消耗 x_ms*2*255+2*x_ms个指令周期  

  //255*2*255+2*255=130610us=130ms

  //此延时函数,延时时间为130ms

  //16M/8/2=1M 一个指令周期为1us

}


///******************************************************************** **********************************

//*  功  能   :SPI基本写一个字节函数

//*  入口参数   :address为寄存器地址

//            data为写入数据

//*  出口参数   :无

//*  说  明   :SPI文件提供基本的写函数,具体的器件写方式可能不同,建议使用基本函数进行封装

//*  范  例   :无

//******************************************************************************************************/

void SPI1_Write_REG(uint8_t address,uint8_t data)

{

    uint8_t tmp;

    address |=0x20;

    CSN_L;

    SPI1_DR=address;//写入需要操作的寄存器地址,

    while(!(SPI1_SR_RXNE));

    tmp=SPI1_DR;   //读取数据,仅仅是为了清除标志位

    while(!(SPI1_SR_TXE));//等待发送寄存器为空

    SPI1_DR=data;

    while(!(SPI1_SR_TXE));

    CSN_H;

}


///******************************************************************** **********************************

//*  功  能        :SPI基本读一个字节函数

//*  入口参数        :address为寄存器地址

//*  出口参数        :无

//*  说  明        :SPI文件提供基本的读函数,具体的器件读方式可能不同,建议使用基本函数进行封装

//*  范  例        :无

//******************************************************************************************************/

uint8_t SPI1_Read_REG(uint8_t address)

{

  volatile uint8_t value=0;

  CSN_L;

  value=SPI1_DR;//读一次,清除标志位

  while(!(SPI1_SR_TXE));

  SPI1_DR=address;//写入需要操作的寄存器地址,

  

  while(!(SPI1_SR_RXNE));

  value=SPI1_DR;

  

  while(!(SPI1_SR_TXE));

  SPI1_DR=0xFF;//写入一个无效值

  

  while(!(SPI1_SR_RXNE));//准备读数据

  value=SPI1_DR;

  CSN_H;

  return value;

}

/******************************************************************************************************

* 名 称: SPI_init()

* 功 能:初始化SPI

* 入口参数:无

* 出口参数:无

* 说 明: SP1传输速率设置为fmaster/2=8M,主模式,

* 范 例:无

******************************************************************************************************/

void SPI_Init(void)

{

  static uint8_t temp0=0,temp1=0,temp2=0,temp3=0;

//输出IO

  PD_DDR_DDR4 =1;//CE设置为输出

  PB_DDR_DDR4 =1;//CSN设置为输出

  PB_DDR_DDR5 =1;//SCK设置为输出

  PB_DDR_DDR6 =1;//SIMO设置为输出

  

  PD_CR1_C14 =1;//CE设置为推挽输出

  PB_CR1_C14 =1;//CSN设置为推挽输出

  PB_CR1_C15 =1;//SCK设置为推挽输出

  PB_CR1_C16 =1;//SIMO设置为推挽输出

  

  PB_CR2_C26 =1;//SIMO的IO输出速率为10MHz

  PB_CR2_C25 =1;//SCK的IO输出速率为10MHz

  PD_CR2_C24 =1;//CE的IO输出速率为10MHz

  PB_CR2_C24 =1;//CSN的IO输出速率为10MHz

  

//输入IO

  PD_DDR_DDR5 =0;//IRQ设置为输入

  PB_DDR_DDR7 =0;//SOMI设置为输入

  

  PD_CR1_C15 =1;//IRQ设置为带上拉电阻输入

  PB_CR1_C17 =1;//SOMI设置带上拉电阻输入

  

  PD_CR2_C25 =0;//关闭IRQ中断

  PB_CR2_C27 =0;//关闭中断



  

  CLK_PCKENR1_PCKEN14=1;//打开SPI1外设时钟

  

  SPI1_CR1_SPE=0;//关闭SPI设备

  //设置串行波特率

  SPI1_CR1_BR=0;//fmaster/2=1M

  

  //配置CPOL和CPHA,定义数据传输和串行时钟间的相位关系

  SPI1_CR1_CPHA=0;//数据采样从第一个时钟边沿开始

  SPI1_CR1_CPOL=0;//空闲状态时,SCK保持低电平  

  

   //定义帧格式

  SPI1_CR1_LSBFIRST=0;//先发送MSB

  

    //使能从设备管理//主模式需通过改变SSI位 来控制SPI_SEL

  SPI1_CR2_SSM=1;//禁止软件从设备

  SPI1_CR2_SSI=1;

  

  //主从设备模式选择

  SPI1_CR1_MSTR=1;//作为主设备

  

  SPI1_CR2_RXONLY=0;//全双工

  SPI1_CR2_BDM=0;//选择单向数据模式

  

  SPI1_CR1_SPE=1;//开启SPI设备

  

//NRF24L01上电复位后需要100ms才能进入到掉电模式

//配置NRF24L01之前必须要有100ms以上的延时

  delay_100ms();

  //至此NRF24L01进入到掉电模式,允许对NRF24L01寄存器进行读写操作

  CE_L;

  CSN_H;

  PB_DDR_DDR5=0;//SCK_L

  

/************以下,几个读写操作,是为了测试SPI读写功能*****************/  

  temp0=SPI1_Read_REG(0x00);//

  temp1=SPI1_Read_REG(0x01);//0x3F

  temp2=SPI1_Read_REG(0x02);//0x03



  SPI1_Write_REG(0x00,0x38);

  temp3=SPI1_Read_REG(0x00);

  asm("nop");

}

void main(void)

{

  SPI_Init();

//  asm("rim");               //enable interrupts

  while(1)

  {

    asm("wfi");

  }

}


使用特权

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

本版积分规则

25

主题

3938

帖子

1

粉丝