三,ESP8266 SPI(基于Lua脚本语言)

本文涉及的产品
数据传输服务 DTS,数据同步 small 3个月
推荐场景:
数据库上云
数据传输服务 DTS,数据迁移 small 3个月
推荐场景:
MySQL数据库上云
数据传输服务 DTS,数据同步 1个月
简介: 重点是说SPI通信协议,,,,不要害怕协议因为协议是人规定的,,刚好我也是人......规定的协议既然能成为规范让所有人所接受,那么必然有它的优势和优点,必然值得学习,,害怕协议的人是因为当初碰到了不懂的老师,他只会告诉你这很难.

重点是说SPI通信协议,,,,

不要害怕协议因为协议是人规定的,,刚好我也是人......规定的协议既然能成为规范让所有人所接受,那么必然有它的优势和优点,必然值得学习,,

害怕协议的人是因为当初碰到了不懂的老师,他只会告诉你这很难............其实是他不会........

CS      :   Chip Selection    片选引脚,多个设备时可以用这个引脚选择和哪个设备通信

MOSI  :   Master Out   Slave In   主机输出数据引脚,,,,从机接收数据引脚

MISO :   Master In  Slave Out     主机接收数据引脚....从机输出数据引脚

CLK  :    时钟

还有两个参数----假设有的单片机自带硬件SPI,一定会提供设置下面两个参数,,,当然有的支持SPI通信的芯片也可能涉及这两个参数

CPOL    :表示时钟信号(CLK引脚)在空闲时是高电平还是低电平

CPHA    :表示数据在时钟信号(CLK)的第几个沿开始数据传输

现在假如说

CPOL   = 0  ;//时钟信号(CLK引脚)在空闲时是低电平

CPHA   = 0;//数据在时钟信号(CLK)的第1个沿开始数据传输

 

通信的时候先传输高位

假如说  主机发给从机   10101010       从机发给主机    01010101

再假如 主机接收数据存到  MasterData 里面

            从机接收数据存到  SlaveData 里面

 

第一个上升沿  主机的10101010 最高位是 1 所以主机会让MOSI引脚输出高电平

                        从机的01010101最高位是 0  所以从机会让MISO引脚为低电平

 

第一个下降沿  主机接收MISO引脚的数据,因为是低电平所以 MasterData= 0000 0000;

                       从机接收MOSI引脚的数据,因为是高电平所以从机SlaveData = 0000 0001;

 

 

第二个上升沿  主机左移一位  0101010X 最高位是 0 所以主机会让MOSI引脚输出低电平

                        从机左移一位 1010101X 最高位是 1  所以从机会让MISO引脚为高电平

 

 

第二个下降沿  主机接收MISO引脚的数据,因为是高电平所以 MasterData= 0000 0001;

                       从机接收MOSI引脚的数据,因为是低电平所以从机SlaveData = 0000 0010;

 

 就这样8个上升和8个下降沿之后

                        MasterData  = 01010101

                              SlaveData    =  10101010

 

 

 

现在假如说

CPOL   = 0  ;//时钟信号(CLK引脚)在空闲时是低电平

CPHA   = 1;//数据在时钟信号(CLK)的第2个沿开始数据传输

 

第一个下降沿  主机的10101010 最高位是 1 所以主机会让MOSI引脚输出高电平

                        从机的01010101最高位是 0  所以从机会让MISO引脚为低电平

 

第一个上升沿  (注意是标号为1的那个,,,实际上是第二个上升沿)

                       主机接收MISO引脚的数据,因为是低电平所以 MasterData= 0000 0000;

                       从机接收MOSI引脚的数据,因为是高电平所以从机SlaveData = 0000 0001;

 

 

 

第二个下降沿  主机左移一位  0101010X 最高位是 0 所以主机会让MOSI引脚输出低电平

                        从机左移一位 1010101X 最高位是 1  所以从机会让MISO引脚为高电平

 

第二个上升沿  主机接收MISO引脚的数据,因为是高电平所以 MasterData= 0000 0001;

                       从机接收MOSI引脚的数据,因为是低电平所以从机SlaveData = 0000 0010;

 

 第八个下降沿  主机把最后一位0放在了 MOSI引脚输出低电平  

                               从机把最后一位1放在了 MISO引脚输出高电平  

 

其实现在我也有疑惑,,,这样就完了吗??????只把数据放在引脚上就行了吗????然后内部硬件就自动接收了吗????

看一下摩托罗拉的数据手册

难道最后一位会自动的接收?????搞不懂咧咧.....改天自己测试一下,,,让主机工作在

CPOL   = 0  ;//时钟信号(CLK引脚)在空闲时是低电平

CPHA   = 1;//数据在时钟信号(CLK)的第2个沿开始数据传输

然后让主机发一个字节的数据看看CLK怎样变化的..............然后再尝试手写从机接收....然后知道结果了再来修改这个地方....

其余的两种就不说了

其实说白了就是

CPOL控制在空闲状态下CLK是高电平还是低电平

CPHA控制数据是在第一个沿就开始传输还是在第二个沿

其实呢!!写程序最终还是要看芯片的资料,,,,,

现在看一下ESP8266的SPI

 

 

 

spi.setup(1, spi.MASTER, spi.CPOL_HIGH, spi.CPHA_HIGH, 8, 8)

用SPI1,主机模式,,空闲状态下Clk为高电平,,第二个沿开始数据传输,8位数据,8分频(10Mhz),默认半双工

发送数据呢就简单了

比如向从机发送0xaa,0x55,0x02,0x01

 

spi.send(1,0xaa,0x55,0x02,0x01)

接收数据呢还另有个函数
假设需要接收4个数据

 

ReadData = spi.recv(1,4,0xaa)

其实就是让CLK产生32个脉冲信号,每8个代表一个字节的数据,,后面的0xaa哈,是在产生脉冲信号的时候MOSI引脚发送的数据,,如果不写

默认发0xFF

 

对了接收到的数据都是以字符串的形式保存在变量里面,,可能会问我怎么自己解析数据呢

假如说判断是不是接收到0xaa 和0x55

aa = spi.recv(1,2,0xaa)

if   aa:byte(1)==0xaa  and  aa:byte(2)==0x55  then

    自己的执行函数

end

大家肯定会说你咋知道要这样写

因为有API文档

链接:http://pan.baidu.com/s/1i5lBi7N%20密码:2b5h

 链接:http://pan.baidu.com/s/1crUXUe%20密码:bs83

 还有一点,,我一开始看到过别人这样做过,,所以呢我就知道可以这样用....知识这东西一定要活学活用

http://blog.csdn.net/silno/article/details/72866623?locationNum=10&fps=1

 

 最后说一下和STM32进行SPI通信的一些注意,,,当然是把stm32配置成从机模式,,然后呢其余的设置一定要一样

列如我配置的

void Spi2SlaveInit(void)
{
  SPI_InitTypeDef   SPI_InitStructure;
  GPIO_InitTypeDef  GPIO_InitStructure;

    
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO,ENABLE);
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2,  ENABLE );//SPI2时钟使能       
         
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;//CS
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
    GPIO_Init(GPIOB,&GPIO_InitStructure);
    
    //GPIO_ResetBits(GPIOB,GPIO_Pin_12);  //PB13/14/15上拉
    
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;        
    GPIO_Init(GPIOB,&GPIO_InitStructure);
  
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15 ;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; 
    GPIO_Init(GPIOB,&GPIO_InitStructure);    
    
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14 ;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 
    GPIO_Init(GPIOB,&GPIO_InitStructure);    
  
    GPIO_ResetBits(GPIOB,GPIO_Pin_14);  //PB13/14/15上拉
  
    GPIO_ResetBits(GPIOB,GPIO_Pin_15);  //PB13/14/15上拉
    

    SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
    SPI_InitStructure.SPI_Mode = SPI_Mode_Slave;
    SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
    SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
    SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
//     SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
    SPI_InitStructure.SPI_NSS = SPI_NSS_Hard;  //如果这里使用硬件模式,从机低电平
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;
    SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
    SPI_InitStructure.SPI_CRCPolynomial = 7;
    SPI_Init(SPI2, &SPI_InitStructure); 
    
    SPI_I2S_ITConfig(SPI2,SPI_I2S_IT_RXNE,ENABLE);
    
    SPI_Cmd(SPI2 , ENABLE);
}

 

要注意

CS引脚在模块刚启动的时候一定是低电平............

再说一点,我的32程序用的中断接收的SPI的数据,然后在中断里面准备发送的数据,,大家这样想

主机的数据发过来一个字节数据之后才进的中断,,所以如果想在中断里面发送数据,,应该在进中断之前准备好数据的第一个字节

假设主机需要读四个字节就会进四次中断

我从机发给主机的数据是0xaa 0x55 0x01 0x02

char table[4] = {0xaa, 0x55, 0x01, 0x02}

在主机发送数据之前

我需要   SPI2->DR = table;

              Spi2SendBuff = table+1;

void SPI2_IRQHandler(void)  
{  
  if(SPI_I2S_GetITStatus(SPI2, SPI_I2S_IT_RXNE) == SET)  
  {    
        SPI2->DR = *Spi2SendBuff;
        Spi2ReadBuff[Spi2ReadCnt] = SPI2->DR;
        
        Spi2ReadCnt ++;    
        Spi2SendBuff ++;
  }  
} 

这样的话来第一个中断的时候我的第一个数据0xaa也发向了主机,然后又准备了第二个数据,,,,,,,四次中断就会把数据发给了主机,,注意指针溢出乱指了一下

这篇文章呢主要是会用ESP8266的SPI就行哈......


 

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
Sqoop 企业级大数据迁移方案实战
Sqoop是一个用于在Hadoop和关系数据库服务器之间传输数据的工具。它用于从关系数据库(如MySQL,Oracle)导入数据到Hadoop HDFS,并从Hadoop文件系统导出到关系数据库。 本课程主要讲解了Sqoop的设计思想及原理、部署安装及配置、详细具体的使用方法技巧与实操案例、企业级任务管理等。结合日常工作实践,培养解决实际问题的能力。本课程由黑马程序员提供。
目录
相关文章
|
6月前
|
数据挖掘 Linux 数据处理
探索Linux下的Lua命令:轻量级脚本语言在数据处理和分析中的应用
**探索Linux上的Lua:轻量级脚本语言用于数据处理。Lua通过命令行解释器执行,适用于游戏开发、数据分析及自动化。特点包括小巧、高效、可扩展和动态类型。使用`lua`或`luajit`,配合-e、-l、-i参数执行脚本或互动模式。示例:执行`hello.lua`脚本打印"Hello, Lua!"。最佳实践涉及版本兼容、性能优化、使用C API、测试和文档编写。**
|
Ubuntu 网络协议 Linux
【Lua基础入门】解密世界上最快的脚本语言
【Lua基础入门】解密世界上最快的脚本语言
2273 1
|
存储 NoSQL Java
Lua高性能脚本语言快速入门
Lua高性能脚本语言快速入门
268 0
|
XML 存储 Java
【Lua基础 第1章】初识Lua脚本语言、数据类型、全局变量、关键字的使用
初识Lua脚本语言、数据类型、全局变量、关键字的使用
165 0
【Lua基础 第1章】初识Lua脚本语言、数据类型、全局变量、关键字的使用
|
存储 JavaScript 编译器
Lua脚本语言——Lua脚本基础语法
Lua脚本语言——Lua脚本基础语法
695 0
Lua脚本语言——Lua脚本基础语法
(一)Lua脚本语言入门
今天开始自己的Lua语言学习,Lua脚本语言,是介于应用程序和开发其应用程序的底层编程语言之间,,它很方便调用其它语言,它只是在载入时对其进行编译,而不像我们写的单片机程序是预编译的,先编译好然后写入单片机,它只是在调用时编译,所以脚本语言的灵活性很高,,,,,对于自己为什么又开始学习脚本语言了,,因为需要了   .
1326 0
|
C语言 数据安全/隐私保护
(二)Lua脚本语言入门
上一篇文章忘了插入代码了,方便粘贴复制...... 函数 对于c语言就是  void aa()//c语言是用void { print("这是一个函数") } Lua就变成了 function aa()--定义一个函数 print("这是一个函数") end aa()--调用函数   function aa(str)--定义一个函数 print("传进来的是".
1097 0
|
11天前
|
监控 安全
公司用什么软件监控电脑:Lua 脚本在监控软件扩展功能的应用
在企业环境中,电脑监控软件对保障信息安全、提升效率至关重要。Lua 脚本在此类软件中用于扩展功能,如收集系统信息、监控软件使用时长及文件操作,向指定服务器发送数据,支持企业管理和运营。
29 6
|
6月前
|
消息中间件 NoSQL Java
Redis系列学习文章分享---第六篇(Redis实战篇--Redis分布式锁+实现思路+误删问题+原子性+lua脚本+Redisson功能介绍+可重入锁+WatchDog机制+multiLock)
Redis系列学习文章分享---第六篇(Redis实战篇--Redis分布式锁+实现思路+误删问题+原子性+lua脚本+Redisson功能介绍+可重入锁+WatchDog机制+multiLock)
246 0
|
2月前
|
缓存 分布式计算 NoSQL
大数据-43 Redis 功能扩展 Lua 脚本 对Redis扩展 eval redis.call redis.pcall
大数据-43 Redis 功能扩展 Lua 脚本 对Redis扩展 eval redis.call redis.pcall
37 2