前言
本文主要讲述了如何在 0.96 寸 OLED上显示汉字及采集显示温湿度数据
一、SPI协议是什么?
SPI 协议是由摩托罗拉公司提出的通讯协议(Serial Peripheral Interface),即串行外围设备接口,是一种高速全双工的通信总线。它被广泛地使用在 ADC、LCD 等设备与 MCU 间,要求通讯速率较高的场合。
SPI总线是一种4线总线,因其硬件功能很强,所以与SPI有关的软件就相当简单,使中央处理器(Central Processing Unit,CPU)有更多的时间处理其他事务。正是因为这种简单易用的特性,越来越多的芯片集成了这种通信协议,比如AT91RM9200。SPI是一种高速、高效率的串行接口技术。通常由一个主模块和一个或多个从模块组成,主模块选择一个从模块进行同步通信,从而完成数据的交换。SPI是一个环形结构,通信时需要至少4根线(事实上在单向传输时3根线也可以) [1] 。
SPI的通信原理很简单,它以主从方式工作,这种模式通常有一个主设备和一个或多个从设备,需要至少4根线,事实上3根也可以(单向传输时)。也是所有基于SPI的设备共有的,它们是MISO(主设备数据输入)、MOSI(主设备数据输出)、SCLK(时钟)、CS(片选)。
(1)MISO– Master Input Slave Output,主设备数据输入,从设备数据输出;
(2)MOSI– Master Output Slave Input,主设备数据输出,从设备数据输入;
(3)SCLK – Serial Clock,时钟信号,由主设备产生;
(4)CS – Chip Select,从设备使能信号,由主设备控制。
其中,CS是从芯片是否被主芯片选中的控制信号,也就是说只有片选信号为预先规定的使能信号时(高电位或低电位),主芯片对此从芯片的操作才有效。这就使在同一条总线上连接多个SPI设备成为可能。
接下来就负责通讯的3根线了。通讯是通过数据交换完成的,这里先要知道SPI是串行通讯协议,也就是说数据是一位一位的传输的。这就是SCLK时钟线存在的原因,由SCLK提供时钟脉冲,SDI,SDO则基于此脉冲完成数据传输。数据输出通过 SDO线,数据在时钟上升沿或下降沿时改变,在紧接着的下降沿或上升沿被读取。完成一位数据传输,输入也使用同样原理。因此,至少需要8次时钟信号的改变(上沿和下沿为一次),才能完成8位数据的传输。
时钟信号线SCLK只能由主设备控制,从设备不能控制。同样,在一个基于SPI的设备中,至少有一个主设备。这样的传输方式有一个优点,在数据位的传输过程中可以暂停,也就是时钟的周期可以为不等宽,因为时钟线由主设备控制,当没有时钟跳变时,从设备不采集或传送数据。SPI还是一个数据交换协议:因为SPI的数据输入和输出线独立,所以允许同时完成数据的输入和输出。芯片集成的SPI串行同步时钟极性和相位可以通过寄存器配置,IO模拟的SPI串行同步时钟需要根据从设备支持的时钟极性和相位来通讯。
最后,SPI接口的一个缺点:没有指定的流控制,没有应答机制确认是否接收到数据。
SPI的片选可以扩充选择16个外设,这时PCS输出=NPCS,说NPCS03接4-16译码器,这个译码器是需要外接4-16译码器,译码器的输入为NPCS03,输出用于16个外设的选择。
二、实验步骤
1.实验准备
野火 stm32 指南者开发板
ST-LINK V2 STM8/STM32仿真器编程器
0.96寸OLED显示屏模块0.91 1.3寸液晶屏供原理图12864屏 IIC/SPI
Keil5 MDK
野火串口调试助手
具体连接请参考博客:https://blog.csdn.net/ssj925319/article/details/111588662?spm=1001.2014.3001.5502
2.代码实现
首先下载源码:
链接:https://pan.baidu.com/s/1HS33ftk3Pb7nWJRhBTLqUw
提取码:57x8
解压缩后,在 1-Demo 下选择相应的项目,
这里我选择的是 Demo_STM32 下的 0.96inch_OLED_Demo_STM32F103ZET6_Hardware_4-wire_SPI ,
如图所示:
之后双击打开 PROJECT 下的工程 OLED.uvprojx 即可,如图所示:
接下来我们需要移植代码,首先移植温度数据代码中的bsp_i2c.h文件
代码如下:
#ifndef __BSP_I2C_H
#define __BSP_I2C_H
#include "sys.h"
#include "delay.h"
#include "usart.h"
#define SDA_IN() {
GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)8<<28;}
//CRL = 0000 1111 1111 1111 1111 1111 1111 1111
//8<<28 = 1000 1111 1111 1111 1111 1111 1111 1111
//CRL = 1000 1111 1111 1111 1111 1111 1111 1111 = 0x8fffffff 表示 SDA 输入
#define SDA_OUT() {
GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)3<<28;}
//CRL = 0x3fffffff 表示 SDA 输出
#define IIC_SCL PBout(6) //SCL
#define IIC_SDA PBout(7) //SDA
#define READ_SDA PBin(7) //SDA 数据读取 7 管脚
void IIC_Init(void);
void read_AHT20_once(void);
void reset_AHT20(void);
void init_AHT20(void);
void startMeasure_AHT20(void);
void read_AHT20(void);
uint8_t Receive_ACK(void);
void Send_ACK(void);
void SendNot_Ack(void);
void I2C_WriteByte(uint8_t input);
uint8_t I2C_ReadByte(void);
void set_AHT20sendOutData(void);
void I2C_Start(void);
void I2C_Stop(void);
#endif
之后我们需要移植bsp_i2c.c文件(此处最好更改名称为 AHT20_sys.h,不然会重名)
代码如下:
#include "bsp_i2c.h"
#include "delay.h"
#include "string.h"
uint8_t ack_status=0;
uint8_t readByte[6];
uint32_t H1=0; //Humility
uint32_t T1=0; //Temperature
uint8_t AHT20_OutData[4];
/****************
*初始化 I2C 函数
****************/
void IIC_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
//启用高速 APB (APB2) 外围时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE );
//GPIO 定义
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
//初始化 SCL(Pin6)高电平
IIC_SCL=1;
//初始化 SDA(Pin7)高电平
IIC_SDA=1;
}
/*********************
*AHT20 数据操作总函数
*********************/
void read_AHT20_once(void)
{
printf("读取数据中");
//延时 10 微妙
delay_ms(10);
//传输数据前进行启动传感器和软复位
reset_AHT20();
delay_ms(10);
//查看使能位
init_AHT20();
delay_ms(10);
//触发测量
startMeasure_AHT20(