前言
广义的物联网概念最早于 1999 年由麻省理工学院提出其定义是"通过射频识别(RFID)、全球定位系统等信息传感设备,按约定的协议,把任何物品通过物联网域名相连接,进行信息交换和通信,以实现智能化识别、定位、跟踪、监控和管理的一种网络概念"。
物联网(IoT, Internet of Things)的本质是将各种嵌入式单片机相连,而这一概念将无线连接和各类智能传感器相结合并搭配低功耗的微控制器实现设备成本更低、方式更简单的联网。
而近些年,基于 MEMS(微机电系统)技术的传感器、MCU(微控制器)、 LPWAN(低功耗广域物联网)、云计算以及云储存等技术的快速发展,使物联网这一概念重新成为热门话题,而这一领域,也成为全球互联网巨头未来的重要战略布局。是未来最具有想象力的市场。
也正是基于物联网的前景和对高新技术的渴望,我投身到物联网的学习和开发当中,写下了物联网分支—— LoRa 的开发与应用这篇教程。在开始学习之前,希望读者有以下准备:
Win10 及其以上系统的电脑、MacOS 虚拟机(仅可编译文档中的代码);
一定的 C 语言编程基础、电路知识基础、如果有一定嵌入式基础更好;
一颗热爱学习的心。
本教程的初衷是希望读者能够最大程度从零基础入门学习 LoRa 的开发,但是从实际的学习中来看,我低估了学习LoRa的门槛。学科都是有相关性的,所以在开发 LoRa 的过程中不可避免会涉及到嵌入式底层、电路元器件的基本使用以及电脑的一些复杂操作等内容。但是我想我会尽最大可能用最浅显易懂的语言帮你们理解其中的原理,以便日后在你们走上开发这条道路能有所裨益。
以上的几点准备是本教程所要求的最低限制,我建议开发人员还是需要使用 Windows 电脑,MacOS 系统我也尝试过运行程序,但是配置虚拟机的阻碍也还是挺大的,并且最终是没有办法通过串口将所编写的程序实现烧录运行的操作,但对于理解和编程的过程问题不大。我自己使用的编程学习环境是基于最新的 Windows11 来完成本教程的编写,也希望这篇教程不会"老得这么快"。当然了,如果有条件的同学,我希望你们能够自己准备两块配有 LoRa 模块的开发板,一块作为主机,一块作为从机,这样你们就可以烧录程序到开发板上,观察真实的效果。开发板上,我希望能具备以下传感器模块:直流电机(风扇)、MPU6050 三轴传感器、DHT11 温湿度传感器。这样你就能完美实现第七部分的 LoRa 智慧牧场项目集成开发的项目了。
做好准备,我们就正式开始教程的内容。
LoRa 简介
下一代生态网络
过去 20 年的互联网是"人联网",未来 20 年的互联网是"物联网"。互联网的下半场是将整个物理世界数字化,零售、物流、制造、道路、汽车、森林、河流、厂房......甚至一个垃圾桶都会被抽象到数字世界,连到互联网上,实现"物""物"交流,"人""物"交互。
物联网是互联网的外延,互联网是通过电脑、移动终端等设备将人联网,所形成的一种全新的人与人的连接方式。而物联网是通过传感器、通信模组和智能芯片使物体联网。
人类每一次连接方式的改变,都会催生出一批巨无霸型的企业,在数十亿连接量的语音网时代,产生了 AT&T 这样的百亿美元规模的公司;在实现百亿连接的 PC /移动互联网时代,产生了如 IBM、微软、Google、BAT 这样的干亿美金级别企业;而物联网实现的千亿连接量一定会诞生出下一个巨无霸企业。
这印证了人类每一次连接方式的改变,都会诞生出一批伟大的公司,而伟大的公司会反过来更深刻地改变我们的生活,推动社会的进步。

物联网为什么现在才开始爆发?
物联网概念于 1999 年被提出,但 2016 年却被称为物联网的元年,在这 17 年中,技术的进步是物联网从概念逐渐走向成熟的核心驱动力。
物联网的核心驱动力是通过多个维度的技术成熟而实现。
过去 70% 的传感器及设备无法连接;
2G/3G/4G、WIFI、ZigBee 只解决了视频、图像、语音的等方面应用;
2G/3G/4G 功耗大、成本高、覆盖并不完善;
WIFI、 ZigBee、蓝牙距场短,很多应用场景无法满足;
那些需要电池供电、广域覆盖、又需要长时闻低功耗待机的设备无法适应。

LPWAN 技术
在 LPWAN 技术出现以前,物联网网络难以平衡广覆盖和低功耗/低成本这对矛盾统一体。以目前物联网最常见的 GPRS (2G) 为例,技术存在功耗较高,每个扇区的连接量较低以及网络覆盖面积有限等缺陷,从而无法满足物联网在不同使用场景下的需求。
为了改进 GPRS 无法满足物联网连接需求的现状,业界提出了如 LoRa、SigFox 等工作于未授权频段的低功耗广域物联网(LPWAN)通信协议。与此同时,作为行业标准制定机构 3GPP 也提出了基于 LTE 发展而来的 eMTC 协议、基于 GSM 发展而来的 EC-GSM 协议和 NB-IoT协议。
相较于 3G/4G 蜂窝网络,LPWAN 具有成本低、能源消耗少、覆盖广等优势。
LPWAN:低功耗广域网络
优势:低带宽、低功耗、远距离、大量连接
LPWA 应用最广泛的三项技术:LoRa(美)、NB-IoT(华为参与)、sigfox(欧)

LoRa 无线技术
LoRa 作为低功耗广域网(LPWAN)中的一种无线技术,相对于其他无线技术(如 Sigfox、NB-IoT 等),LoRa 产业链较为成熟、商业化应用较早。基于 Sub-GHz 的频段使其更易以较低功耗远距离通信,可以使用电池供电或者其他能量收集的方式供电。
优势:
164 dB 链路预算、距离 >15 km、快速;
灵活的基础设施,易组网且投资成本较少;
LoRa节点模块仅用于通讯,电池寿命长达 10 年;
免牌照的频段、网关/路由器建设和运营、节点/终端成本低。

LoRa 开发环境搭建及驱动移植
STM32CubeMX 安装及使用
STM32CubeMX 简介
市面上常见的 LoRa 模块都是搭配 STM32 芯片的单片机来使用的,所以使用 ST 公司的软件和产品是入门学习 LoRa 开发的必要一环。

STM32 单片机是 ST(意法半导体)公司使用 ARM 公司的 Cortex-M3 为核心生产的 32 位系列的单片机。
STM32 字面含义:
- ST ——意法半导体(一个公司名),即 SOC 厂商。
一个芯片比如 STM32 里面有内核(ARM),而内核 ARM 由 ARM 公司( IP 厂商)生产,外设由 ST 公司( SOC 厂商)生产,再此基础上添加各种外设,比如 GPIO, IIC 等。
- M —— Microelectronics 的缩写即微控制器。
注意: 微处理器在微控制器的基础上有 MMU(内存管理单元),一般微控制器不跑系统,编写的程序为裸机例程。
- 32 —— 32 bit,表示是一个 32 位微控制器。
单片机位数指 CPU 处理的数据的宽度,参与运算的寄存器的数据长度。32 位单片机的数据总线宽度为 32 位,通常可直接处理 8 位或 16 位或 32 位数据。
使用到 LoRa 的时候不可避免会用到 STM32 上面的串口、IIC、RTC 时钟等功能,所以我们会需要对这些外设进行初始化、配置等操作。传统的 STM32 配置和开发可以单独开一门课来讲了,难度比较大,所以我们需要借助 ST 公司的 STM32CubeMX 软件来简化配置的门槛。
STM32CubeMX 是意法半导体推出的图形化配置工具,通过傻瓜化的操作便能实现相关配置,最终能够生成 C 语言代码,支持多种工具链,比如 MDK、IAR For ARM、TrueStudio 等。尤其值得一提的是,TrueStudio 已经被 ST 收购,提供完全免费的版本,并且,通过插件式安装,可以将 STM32CubeMX 集成在一个 IDE,使用十分方便。

直白点来说,我们可以直接通过图形化的方法,用 STM32CubeMX 这款软件完成一些外设的配置,它能够通过我们的选择,自动为我们生成配置外设所需的代码,大大提高了我们的效率。
STM32CubeMX 的优势:
直观的选择 STM32 微控制器;
微控制器图形化配置:
自动处理引脚冲突
动态设置确定的时钟树
可以动态确定参数设置的外围和中间件模式和初始化
功耗预测
- C代码工程生成器覆盖了STM32微控制器初始化编译软件,如IAR、KEIL、GCC可以独立使用,作为Eclipse插件使用。
HAL 库与 STD 库
库的概念大家应该不陌生,通常时候我们用到的一些传感器读取数据的函数,都是工程师们在库里面完成的底层代码编写。同样的,ST 的工程师们也为他们的 STM32 外设编写了不少外设库给我们调用,目前 ST 的库大致可以分为两大类:STD 库和 HAL 库。
STD 库\——标准外设库:寄存器操作,将一些基本的寄存器操作封装成函数;
HAL 库\——硬件抽象库:将这些象成了一个抽象层,从使用的角度来看,是与硬件无关的。
STD 库是早期 ST 推出的标准库,广泛应用在高校教育当中,库中对常见的一些寄存器功能进行了封装,成为了一个个独立的文件包。但是由于后来ST公司主推HAL库,STD 库已经缺少维护,并且文件需要自己去布局安排放置位置、使用门槛高,渐渐被HAL库所替代。

HAL 库优势:
HAL 库是 ST 未来主推的库,从 2015 开始 ST 新出的芯片已经没有 STD 库;
HAL 库的处理机制比 STD 库好很多,HAL 库支持 STM32 全线产品;
HAL 库跨芯片的可移植性非常好。
STM32CubeMX 安装
第一步:STM32CubeMX 依赖 Java 环境,JRE 官方下载地址:[https://www.oracle.com/java/technologies/downloads/#java8]{.underline}。以本文开发环境为例,选择 Windows x64,特别注意,需要选择 Accept License Agreement 才可以下载。
(百度网盘下载链接:[https://pan.baidu.com/s/1bYMQAemDrOwA688S70o8LA]{.underline} 提取码: 1p7o)



第二步:在 ST 官网选择对应版本的 STM32CubeMX 进行下载,下载时会要求你填邮箱信息,之后下载的链接会通过邮箱发送给你。在安装 SetupSTM32CubeMX.exe 时,最好不要有中文路径。
(百度网盘下载链接:[https://pan.baidu.com/s/1r0RK5mBP3gJjjIHpMWB9Cw]{.underline} 提取码: a9fm)









第三步:STM32CubeMX 固件包导入
固件包就是要选择我们 LoRa 开发板对应的 STM32 芯片类型,只有加载了正确的固件包,软件才能显示正确的芯片供我们配置。
首先需要检查固件包的路径,是否是中文路径,如果是中文的路径,修改路径。路径同样不能包含中文,具体修改固件包路径操作如下图所示:


导入固件包有两种方式:
- 方法一:在有网络的情况下,直接在线下载
我们的教程使用的是 STM32F051K8 这款芯片,所以下载的固件包是 STM32F0。



- 方法二:导入固件包文件


STM32CubeMX 创建工程
万事俱备之后,我们就开始创建我们的配置工程,步骤可以参考下面的图片。



选择好芯片之后,就可以看到芯片的图片显示在屏幕的中心位置。这个时候我们只需要通过简单地点击 GPIO 引脚,就可以将它配置成我们所需要的模式。
像在这里,我们就可以设置PA8引脚为输出模式:

完成工程的创建工作之后,我们还需要对工程进行一些常规配置:


完成配置之后,我们点击右上角的 "GENERATE CODE" 即可生成代码文件。

IAR 安装及使用
IAR 介绍
上一节内容带领大家尝试使用了一下 STM32CubeMX 来生成一个配置 PA8 引脚为输出模式的项目,但是这个项目我们应该使用什么软件来打开方便我们之后的开发步骤呢?在本篇教程中,我们选择的是 IAR 这款软件。
IAR Systems 是全球领先的嵌入式系统开发工具和服务的供应商。公司成立于 1983 年,提供的产品和服务涉及到嵌入式系统的设计、开发和测试的每一个阶段,包括:带有 C/C++ 编译器和调试器的集成开发环境(IDE)、实时操作系统和中间件、开发套件、硬件仿真器以及状态机建模工具。

IAR 与 Keil
IAR 即 IAR Embedded Workbench,Keil 即 Keil MDK-ARM,这两款都是 ARM 开发工具。那么,IAR 与 Keil 两款开发工具区别在于哪里呢?
Keil 可以自动配置启动代码,集成 Flash 烧写模块,而且支持性能分析功能;
而 IAR 是一套用于编译和调试嵌入式系统应用程序开发工具,支持汇编、C 和 C++ 语言,为伙伴们提供了完整的集成开发环境,而且还包括管理器、编辑器等。
Keil 默认只创建工程,工作区是不会直接创建的。如果想多个工程聚合,则首先需要创建一个工作区,然后再添加相应的工程,相对比较繁琐。
IAR 默认是创建工程和工作区,如果想多个工程并存,直接添加即可。
Keil 编译时,只有 level 的选择;IAR 有 debug 和 Release 的快速选择。
默认状态,Keil 的工具栏功能比较多,有点繁杂;IAM 的比较简洁。但相对,也比较单薄。Keil 的程序文件,最后必须要有一个新的空行,否则会有编译警告。
IAR 安装
第一步:从官网下载IAR软件([https://www.iar.com/products/architectures/arm/iar-embedded-workbench-for-arm/iar-embedded-workbench-for-arm---free-trial-version/]{.underline})
(百度网盘下载链接:[https://pan.baidu.com/s/1QKWNsdyseW59BIegrmDNlA]{.underline} 提取码: 0no0 )

第二步:参考以下图片步骤完成软件的安装。







附:破解方法参考([https://www.jb51.net/softs/767867.html]{.underline})
(破解工具百度网盘下载链接:[https://pan.baidu.com/s/1Sy-4_eGmwDOjgcSOA0Ra8g]{.underline} 提取码: 81e1 )

IAR 使用
我们先来记忆一些 IAR 的常用快捷键:
CTRL + B 括号匹配选择括号内的多行代码;
CTRL + T 自动缩进,格式化选中代码;
CTRL + K 快注释,屏蔽选中的代码;
CTRL + SHFT + K 取消快注释,取消屏蔽选中的代码。
这里我们使用在前面 STM32CubeMX 生成好的项目举例说明 IAR 软件的使用,打开 STM32CubeMX 生成好的工作区,首先启动 IAR 软件。

生成的工程文件位置在 EWARM 文件夹下,后缀为 .eww 的文件。


打开工作区之后我们可以看一下工作区的目录以及文件。

STM32CubeMX 生成的工作区文件夹分类相当清晰。目录树下分成了两个文件夹 Application 和 Drivers,将所有的库文件分成了应用和外设两个部分。对于 Application 文件夹的内容,一方面是启动芯片的配置库(在 EWARP 文件夹中),另一些就是用户的配置文件(在 Core 文件夹中)。用户配置的文件在图片当中就有 gpio.c(引脚配置)、main.c(主函数)、stm32f0xx_it.c(中断配置)等。
STM32F0xx_HAL_Driver 文件夹里面就是基于 HAL 库的外设配置代码,可以通过名称清晰地看到有 IIC、DMA、RCC 等的配置,当然了,由于我们刚才并未在 STM32CubeMX 中开启这些功能,所以部分内容可能是空白的。
在习惯上,我们打开工作区之后的第一步应该右击项目的名称,选择 "Make" 对工程进行编译,检查是否有错误。

只有编译没有问题,我们才能继续之后的开发步骤。

M0 工程建立
到目前为止,我们已经完成了 STM32CubeMX 和 IAR 软件的一些了解学习,接下来我们就结合 LoRa 开发板来创建一个 M0 工程。M0 工程是所有硬件开发的第一步,是根据我们所需的硬件内容和需求做出来的第一个工程文件。以后的配置工作甚至是其他 LoRa 板型都可以在 M0 工程的基础之上完成,是名副其实的"万能模板"。
在这部分的内容当中,希望大家能够梳理清楚整个LoRa驱动配置的步骤和过程,明白其中的原理;同时学习到 STM32CubeMX 和 IAR 软件使用方法。
IO 端口配置
从前面的学习中不难发现 STM32CubeMX 这个软件就是为了 STM32 芯片各个功能引脚配置和各种外设的开启功能配置的,所以在使用软件之前,我们得完成 STM32 的引脚配置,绘制出原理图。一般来讲这个工作是由硬件工程师来完成的,所以具体过程我们可以不用深究,拿到原理图我们就可以根据原理图来配置引脚的模式和功能了。
这里是我的 LoRa 开发板提供的芯片原理图。

然后根据上面的原理图,建立 IO 功能映射表如下。
| 序号 | IO编号 | IO信号网络 | 功能描述 |
|---|---|---|---|
| 0 | VSS | GND | 电源负 |
| 1 | VDD | 3.3V | 电源正 |
| 2 | PF0-OSC-IN | OSC_IN | 外部8M晶振 |
| 3 | PF1-OSC-OUT | OSC_OUT | |
| 4 | NRST | NRST | 外部复位 |
| 5 | VDDA | 3.3V | 电源正 |
| 6 | PA0 | BAT_ADC | 模拟量输入-电池电压 |
| 7 | PA1 | ADC_KEY | 模拟量输入-按键信息 |
| 8 | PA2 | DIO3 | 数字量输入-LoRa数字IO3 |
| 9 | PA3 | DIO2 | 数字量输入-LoRa数字IO2 |
| 10 | PA4 | A1 | 数字量输入-传感器扩展接口1 |
| 11 | PA5 | ID_1 | NC |
| 12 | PA6 | A2 | 模拟量输入-传感器扩展接口2 |
| 13 | PA7 | NSS_LoRa | LoRa模块片选接口 |
| 14 | PB0 | LED4 | 数字输出-无线通信网络指示灯 |
| 15 | PB1 | LED3 | 数字输出-无线通信发送指示灯 |
| 16 | PB2 | LED2 | 数字输出-无线通信接收指示灯 |
| 17 | VDD | 3.3V | 电源正 |
| 18 | PA8 | D3&KEY | 数字量输入3-按键状态 |
| 19 | PA9 | U1 USART1_TX | 串行通信接口 |
| 20 | PA10 | U1 USART1_RX | |
| 21 | PA11 | DIO0 | 数字量输入-LoRa数字IO0 |
| 22 | PA12 | DIO1 | 数字量输入-LoRa数字IO1 |
| 23 | PA13 | SWDIO | SWD调试烧写接口 |
| 24 | PA14 | SWCLK | |
| 25 | PA15 | NSS_LCD | LCD片选接口 |
| 26 | PB3 | SCLK | SPI时钟接口 |
| 27 | PB4 | MISO | SPI主机输入从机输出接口 |
| 28 | PB5 | MOSI | SPI主机输出从机输入接口 |
| 29 | PB6 | LCD | LCD背光控制接口 |
根据上面的映射表,我们通过 STM32CubeMX 配置每个 IO 的工作模式。配置的时候要留意晶振引脚 PIN2、PIN3 还有 SPI、USART、ADC、和系统时钟,它们需要单独配置。其他引脚根据实际的输入输出配置为常见的 GPIO_Output 或者 GPIO_Input 即可。

时钟配置
时钟系统是 CPU 的脉搏,就像人的心跳一样。所以时钟系统的重要性就不言而喻了。STM32有多个时钟来源的选择,采用一个系统时钟不是很简单吗?为什么 STM32 要有多个时钟源呢? 因为首先 STM32 本身非常复杂,外设非常的多,但是并不是所有外设都需要系统时钟这么高的频率,比如看门狗以及 RTC 只需要几十 k 的时钟即可。同一个电路,时钟越快功耗越大,同时抗电磁干扰能力也会越弱,所以对于较为复杂的 MCU 一般都是采取多时钟源的方法来解决这些问题。
对于 STM32F0 型号的芯片,我们固定选择时钟源,配置倍频,使能锁相环配置系统主时钟为48M。

外设配置
- USART1 配置
USART 是一个全双工通用同步/异步串行收发模块,该接口是一个高度灵活的串行通信设备。配置开启 USART 是为了方便我们进行 LoRa 的调试和通信。我们在学习使用 Arduino 的时候,因为 Arduino IDE 没有逐步调试功能,所以我们常用串口输出来检查某些代码编写的功能是否如我们所期望的一样,在这里也是同理。
配置 USART 的基本原则:
→选择异步通信
→无硬件流控
→设置波特率 115200
→设置 DMA 接收
异步通信是一种很常用的通信方式。相对于同步通信,异步通信在发送字符时,所发送的字符之间的时隙可以是任意的,当然,接收端必须时刻做好接收的准备。发送端可以在任意时刻开始发送字符,因此必须在每一个字符的开始和结束的地方加上标志,即加上开始位和停止位,以便使接收端能够正确地将每一个字符接收下来。内部处理器在完成了相应的操作后,通过一个回调的机制,以便通知发送端发送的字符已经得到了回复。
DMA,全称为:Direct Memory Access,即直接存储器访问。DMA 传输方式无需 CPU 直接控制传输,也没有中断处理方式那样保留现场和恢复现场的过程,通过硬件为 RAM 与 I/O 设备开辟一条直接传送数据的通路,能使 CPU 的效率大为提高。



- SPI 配置
配置 SPI 是为了方便我们使用 LoRa 和 LCD 等模块。SPI 是串行外设接口(Serial Peripheral Interface)的缩写,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为 PCB 的布局上节省空间,提供方便。在 LoRa 的开发当中,我们会用到各种传感器模块和 LCD 显示屏,这些可能都需要使用 SPI 进行通信,所以我们也配置开启 SPI 功能。
SPI 的配置原则:
→选择全双工主机模式
→硬件片选不使能
→设置波特率为 1.5MBit/s
→时钟极性为低电平驱动
→时钟相位为第一个边沿

我们继续尚未完成的 STM32CubeMX 工程,完成配置之后点击生成代码。

然后使用IAR打开工程,点击"make"编译,检查是否报错。

printf() 函数重定向
我们在学习 C 语言的第一课,就学会了使用 printf("hello world!") 来打印括号中的内容,printf() 函数的使用我们已经非常熟悉,没有想过它是怎么实现的甚至是怎么来的。
其实 printf() 函数是一个经过封装之后的函数,也就意味着它的底层还有一些汇编语言编写的代码段来帮助我们完成打印输出这一简单的功能。其实在C语言当中要想实现 printf() 函数的功能也并不是直接就能拿来用的,还需要在代码的最前面加上这么一句 #include <stdio.h>,这其实就是包含 printf() 函数的库。
所以在我们的 LoRa 代码开发中,如果想要使用 printf() 函数,并不能直接使用 printf() ,否则程序会报错,我们需要做一步操作叫"重定向"。
将重定向的代码复制到 main.c 最后面即可。
int fputc(int ch, FILE *f){
while((USART1->ISR&0X40) == 0);
USART1->TDR =(uint8_t)ch;
return ch;
}
```c


可以在初始化语句之后我们添加一段printf语句,结合开发板硬件,检查通信是否正常。


<img src="/images/lora-doc/image62.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
到这里就已经完成了 M0 工程的生成,之后的项目开发都会基于此 M0 工程的内容,为了保证 M0 工程的完整性和正确性,大家可以使用 M0 工程的副本来完成接下来的开发步骤。
## LoRa 驱动源码分析
### 驱动源码文件说明
(LoRa 官方驱动固件库百度网盘链接: [\[https://pan.baidu.com/s/1Yf3wsaZsa8BWqXTGf-9L3g\]{.underline}](https://pan.baidu.com/s/1Yf3wsaZsa8BWqXTGf-9L3g) 提取码: 92ed)
拿到官方提供的 LoRa 驱动固件库,我们先来仔细研究一下驱动固件库文件夹的目录和作用,以便日后我们对驱动代码的修改有一个初步的了解。

文件夹当中有四个子文件夹,分别是 doc、lst、obj、src。**doc 文件夹**存放的是关于此 LoRa 驱动库的官方说明;**lst 和 obj 文件夹**是程序编译之后自动生成的一些编译文件;**src 文件夹**则存放了 LoRa 驱动的所有源码。

综上,我们的重点是要研究 src 文件夹的组成。src 文件夹当中又有两个子文件夹,分别是 **platform** 和 **radio**,还有一个 main.c 文件。
<strong>platform文件夹:</strong>存放的是 LoRa 硬件平台的驱动源码,由于其他平台的开发我们用不到,这里只需要关心子文件夹 **sx12xxEiger** 即可。

**radio文件夹**:操作 LoRa 无线驱动的源码。这里只需要关心 SX1276 这款芯片即可( SX1276 芯片在美国使用,中国用 SX1278 芯片较多)。

**main.c**:是主程序文件,这里的主程序是 LoRa 官方提供的示例代码,我们可以拿来参考学习,但是我们开发很少会直接拿来使用。
简单了解了一下 LoRa 驱动固件的目录树之后,我们再来看看下面的这个表格,看看驱动移植当中我们需要处理的是哪些文件,先有一个初步的印象。

### 硬件抽象层分析
从表面看,终端驱动就是 MCU 通过读写 SX1278 的寄存器,实现射频收发功能。然而,一个优秀的驱动设计,至少满足以下设计目标。最具挑战的是,有些目标是相互抵触的。
提供机制:区分策略和机制,驱动仅提供机制,由用户进程实现策略;
接口简单:接口越简单,驱动越好使用,另外,更好实现"高内聚、低耦合";
提高效率:最大化硬件设备性能,是驱动的重要使命;
节能内存:内存复用和指针传递等方法可以节省 MCU 宝贵的内存;
易于移植:能在不同的 MCU 之间低成本移植,该驱动就越优异;
稳定可靠:驱动是硬件和系统软件的黏合层,不能有任何差错。

**搭建的LoRa终端的总体思路与步骤:**
1\. 搭建的LoRa终端的系统需求和目标
2\. 设计LoRa终端的目标系统
3\. 设计LoRa终端的主机开发环境
4\. 设计LoRa终端的软件架构
5\. 构建LoRa终端的实际硬件开发平台
6\. 构建LoRa终端的实际软件开发平台
7\. 迭代式实现LoRa终端的软件功能
**硬件接口设计**
MCU 与 LoRa 硬件接口如图所示。MCU 通过 SPI 总线与 LoRa 进行通信,包括设置参数和读写 FIFO;当 LoRa 有异步事件发生时,它通过 6 根连接线 DIO0 ~ DIO5 中断 MCU;MCU 为判断接收和发送数据包是否超时需要设置 TIMER ,该资源 LoRa 不需要,仅被 MCU 所用。

**硬件接口函数**
DIO 除初始化、函数化,还需要 DIO0 ~ DIO5 对应中断服务函数;SPI 需要初始化和输入、输出函数。


上述的函数都在 Hal.c 文件里面有他们的定义,不需要我们来重新写一遍这些功能,我们所需要做的,就是根据我们所选的 STM32 进行匹配,这也是接下来驱动移植所需要完成的主要任务。

也就是说,在接下来的硬件驱动移植的时候我们就只需要关心 SPI 和 IO 的配置即可。
## LoRa驱动移植
由上面的描述可知,LoRa 官方已经给我们提供了所需要外设的函数定义和全部驱动固件,但是我们不能要求这套固件拿到手之后对所有的 MCU 都适用,LoRa 提供的模板我们还是需要根据选择的芯片型号进行适配之后才能够进行使用的,这就是驱动移植当中我们需要完成的任务——将 LoRa 驱动固件与 M0 工程融合,使 LoRa 能够在所选择的芯片上编译运行。
### 驱动文件移植
首先第一步,我们要将所需的文件从官方提供的固件库中复制出来,放在我们的 M0 文件夹中进行开发,我们尽量不要对官方固件进行修改。所需的文件以及放置位置可以参考下图。

实操如下:在 M0 副本里面新建一个文件夹,命名为 "sx1278"。

然后我们找到官方固件库,将 platform 和 radio 文件夹全部复制到 sx1278 文件夹当中。

### IAR 工程文件添加
所需的驱动文件已经复制到我们的项目文件当中了,但是并不代表项目中也加载了我们所需的代码。所以我们还需要在 IAR 软件当中也添加驱动文件,添加之后的目录树如下图所示。
<img src="/images/lora-doc/image75.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
实操如下:打开 EWARM 文件夹下面的"Project.eww"文件。

然后我们右键点击项目名称,添加项目组名称为"platform"、"radio"。
<img src="/images/lora-doc/image77.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
分别为这两个项目组添加我们需要的源码文件:
<img src="/images/lora-doc/image78.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />


值得注意的是,这里添加的所有文件都需要从新拷贝过来的 platform 和 radio 文件夹里面去找,不要去官方固件库里面找,因为我们的原则是不破坏源代码,只对复制代码进行修改。
添加完成之后如下图所示:

### 驱动源码修改
LoRa 官方固件库不支持 Hal 库;SPI 和 IO 口跟自己开发的接口不一样,不修改是无法正常运行的。
**第一步:修改硬件平台**
为了确保程序能够成功烧录匹配 SX1278 芯片,我们需要将 SX1278 硬件平台添加到 IAR 软件中。关于硬件平台的定义在 platform.h 文件里。
打开 platform.h 文件


复制下图所示第 40 行的部分代码。

然后右击工程名称,选择 "**Options**",在图片指定位置添加宏定义 "PLATFORM=SX12xxEiger" 即完成硬件平台的添加。


**第二步:修改包含路径**
要想实现编译的时候软件能够自动检索 "#include" 包含的头文件和源文件,我们必须要在 IAR 设置当中将这些文件路径添加上,不然会导致软件找不到包含的文件而报错。

**第三步:修改 sx12xxEiger.h 和 fifo.h**
在开头我们已经说明,官方固件不支持 Hal 库,对于 IO 引脚也不适配,所以接下来就是比较繁琐和复杂的匹配过程。
我们打开 sx12xxEiger.h 文件,先修改让它支持我们的 STM32。

对代码的第 29 行开始进行修改,添加一个分支,内容为我们所选的芯片前缀型号。

修改后:
```c
#elif defined( STM32F1XX )
#include "stm32f10x.h"
#else
#include "stm32f0xx.h"
```c

除了这个文件有不支持的情况可能发生以外,可能还会有其他文件也存在同样的问题,那么我们现在可以编译文件查看错误,根据错误我们选择对应的文件进行相同的修改。

找到这类错误然后,双击这个错误即可打开需要修改的代码文件。我们复制修改代码进行粘贴即可。

修改后再次编译,确保编译错误中不再有 "cannot open source file 'stm32f10x.h'" 这个错误发生。
**第四步:修改 sx12xxEiger.c**
编译之后可以看见 sx12xxEiger.c 文件也有报错的情况发生,这里的错误因为缺少一些文件,而报错缺少的这些文件在我们的开发中并不需要,所以直接把代码 27 ~ 42 行注释(选中然后ctrl+K)即可。

47 ~ 59 行同样也注释。

在这里,所有关于初始化(Init)的函数也都需要注释,因为我们在使用 STM32CubeMX 生成项目的时候就已经使用 Hal 库完成初始化,不再需要 LoRa 官方的初始化步骤。
**\[(但要注意不要把函数名和大括号注释了,避免报错函数名字得留着!)\]{.mark}**

**第五步:修改 led.h 和 led.c**
打开led.h文件进行修改。

根据上面的"IO 功能映射表",确定 LED 的引脚号。可以看到 LED2 ~ LED4 对应的是 PB2 ~ PB0。

然后我们不需要去关心 STM32F4 等其他的平台的源码,我们直接找到"#else"的部分(表示我们自己使用的平台 STM32F0 )在 68 行附近:

将上图的引脚号改成我们自己指示灯的引脚标号(需要改成大写的原因是 Hal 库对于引脚的定义就是大写的)。
同时,在 Hal 库里面是不支持对时钟的定义的,所以我们还需要将引脚后面对时钟的定义全部注释。

第一个灯 LED1_PIN 因为我们用不到,但是删掉可能会导致报错,所以这里我们随便定义一个引脚。别忘了注释时钟的定义。

然后继续将剩余的一些与 Hal 库冲突的地方全部进行修改。

然后我们切换到 led.c 文件,把 Hal 库不支持的时钟和初始化的相关内容进行修改。


最后把灯开关的操作改成使用 Hal 库里面的函数来完成。

单击鼠标右键对 led.c 文件进行编译,检查是否无错。


补充一点:关于 Hal 库函数的介绍和使用可以参考 Hal 库的用户手册,下载地址在官网,选择你的芯片型号进行下载 [\[https://www.st.com/en/embedded-software/stm32cubef0.html#documentation\]{.underline}](https://www.st.com/en/embedded-software/stm32cubef0.html#documentation))

**第六步:修改 sx1276-Hal.c(工作量最大)**
打开 sx1276-Hal.c,首先注释 35 ~ 44 行。

片选部分我们在 else 当中修改为引脚 PA7 。(根据 IO 功能映射表)


修改 DIO 部分的引脚标号。


同时把其他芯片的内容注释。



将所有的 GPIO_WriteBit() 改成 Hal() 库的写法: HAL_GPIO_WritePin()。

将所有的 Bit_RESET() 和 Bit_SET() 全部改成 Hal 库的写法: GPIO_PIN_RESET() 和 GPIO_PIN_SET()。

将所有的 GPIO_ReadInputDataBit() 改为 Hal 库的写法: HAL_GPIO_ReadPin()。

由于 DIO3、DIO4、DIO5 我们不使用,所以在它们配置函数的内部直接设置为 return 1,避免报错。

我们不需要通过硬件去进行发送和接收,所以最后的函数内容也注释。

最后对 sx1276-Hal.c 文件进行编译,确保编译无错。

**第七步:修改 sx1276-Hal.h**
打开 sx1276-Hal.h 文件,把 39 行的 TickCounter 改成 Hal 库的写法:HAL_GetTick()。

**第八步:修改 spi.c**
打开 spi.c 文件,在用户配置代码的地方配置 SPI 的使用。
```c
uint8_t SpiInOut( uint8_t outData ){
uint8_t pData = 0;
if(HAL_SPI_TransmitReceive( &hspi1,&outData,&pData,1,0xffff) != HAL_OK)
return ERROR;
else
return pData;
}
```text

**注意:**
HAL_SPI_TransmitReceive() 函数的第一个参数要参见 spi.c 开头 STM32CubeMX 默认分配的结构体指针;

第二个参数是发送的数据指针,用我们定义的局部变量 outData 即可;
第三个参数是接收的数据指针,我们在函数内部定义的 pData 即作为接收数据指针;
第四、第五个参数是数据的长度和超时时间。
整个函数的设计逻辑就是,使用 HAL_SPI_TransmitReceive() 函数来查看返回值是否为定义好的 HAL_OK,如果是则代表 SPI 开启正常,则返回接收的数据,否则返回错误信息。
其他更多关于 HAL_SPI_TransmitReceive() 函数的使用可以参考 Hal 库的用户手册。
完成以上所有的修改之后,对整个项目进行编译,检查是否无错,到此,整个 LoRa 驱动移植步骤就顺利完成。

### LoRa 模块上电自检
为了检查上面驱动移植的代码没有问题,能够成功驱动 LoRa 运行,我们需要给开发板上电,并且设计一个程序能够通过 SPI 总线来读取芯片版本,判断读取值是否为 0x12,并打印出模块版本号。

首先我们来查找看一下通过 SPI 总线来读取芯片版本号这个功能是否有实现:

搜索 "**RegVersion**":

果然是能找到的,我们双击找到的位置,直接复制"SX1276Read( REG_LR_VERSION, &SX1276LR->RegVersion );",然后打开"main.c"文件,粘贴在初始化之后。

当然了,别忘了添加我们所需要的头文件。

新建一个变量 RegVersion,并且用它替换掉原来 SX1276Read() 函数的第二个参数——版本号。

然后用 C 语言编写判断语句,判断芯片版本是否为 0x12。

编译检查无错后,连接LoRa开发板,上电检查运行结果:

通过串口助手可以看到,最终打印出来的结果是"LoRa read Ok!",表示这部分驱动移植和芯片版本读取的代码已经顺利实现其应有的功能效果。
# LoRa 人机界面开发
使用串口助手查看 LoRa 通信的内容始终不是长久之计,因为我们不可能将 LoRa 一直连着电脑来看数据信息。因此,合理利用 LoRa 开发板上面的显示屏进行数据的显示会让整个产品的用户体验变得更好。
## TFT 液晶屏技术原理
TFT 即薄膜场效应晶体管,它可以"主动地"对屏幕上的各个独立的像素进行控制,这样可以大大提高反应时间。一般 TFT 的反应时间比较快,约 80 毫秒,而且可视角度大,一般可达到 130 度左右,主要运用在高端产品。

从而可以做到高速度、高亮度、高对比度显示屏幕信息。 TFT 属于有源矩阵液晶显示器,在技术上采用了"主动式矩阵"的方式来驱动,方法是利用薄膜技术所作成的电晶体电极,利用扫描的方法"主动拉"控制任意一个显示点的开与关,光源照射时先通过下偏光板向上透出,借助液晶分子传导光线,通过遮光和透光来达到显示的目的。


颜色深度:
Color 介绍 :
① R,G,B 三基色组合形成各种颜色。
② 能显示的颜色数由 RGB 的数字信号的位数来决定。(8bit 数字信号刚好能显示 16.7M 种颜色)

以 3bit 为例数字信号为例:

For 3 bit : 23(R) \* 23(G) \* 23(B) = 256 colors
For 6 bit : 26(R) \* 26(G) \* 26(B) = 262144 colors(242K)
For 8 bit: 28(R) \* 28(G) \* 28(B) = 16777216 colors(16.7M)
For 10 bit: 210(R) \* 210(G) \* 210(B) = 1073741824 colors(1 billion)
1.44寸TFT模块:

TFT 驱动控制器:

1.44 寸模块电路详解:

## TFT 液晶屏驱动开发
### TFT 液晶屏硬件接口驱动开发
SPI、GPIO 初始化 STM32CubeMX 已经帮助我们完成代码的设计了,最后我们需要手动完成数据模式的初始化。根据硬件设计,LoRa 与 LCD 共用 SPI 总线,且 LCD_MISO 用于命令/数据模式切换控制。同时需要修改 GPIO 初始化源码,让片选接口拉高。(片选脚拉低可以进行写指令,片选脚拉高使能 LoRa 和 LCD)

驱动源码移植概览:
<img src="/images/lora-doc/image147.png" alt="图片" data-width="70%" style="width: 70%" data-align="center" />
**实践:上电后液晶屏显示黄色屏幕**
我们继续使用 LoRa 移植完成的项目工程文件,先按照上图将我们需要的代码文件放在对应的文件夹当中:



(我们不难总结出来 Src 文件夹里面其实存放的都是 .c 源文件,Inc 里面存放的则是 .h 头文件,而在我们的 IAR 项目工程当中,导入的必须都是 .c 源文件)
然后打开 IAR 工程,把我们刚刚加的 lcd.c 文件添加进工程当中。

编译完成之后就可以看到 lcd.c 文件会包含所有的头文件(.h文件)。

编译之后可以看到有一处警告,双击进入警告的代码行,发现是HAL库没有定义的问题,我们对这个警告进行修正:

把数字"1"改成 "GPIO_PIN_SET" 即可,重新编译则无错误无警告。

然后我们需要将 gpio.c 文件当中初始化片选引脚设置为拉高,使能 LCD 和 LoRa,即把 GPIO_PIN_RESET 改为 GPIO_PIN_SET。

接下来我们来实现显示屏的清屏操作,在 main.c 文件当中先添加 lcd 的头文件,然后调用 lcd 的初始化函数以及清屏的函数,把屏幕设置为黄色。


然后我们就可以看到运行的结果如下:
<img src="/images/lora-doc/image158.png" alt="图片" data-width="61%" style="width: 61%" data-align="center" />
### TFT 液晶屏取模方式
TFT显示屏取模方式有四种,分别是:列扫描、行扫描、列行扫描、行列扫描(后面两种会在到达屏幕中间部分的时候切换扫描方式)

我们不可能手写完成所有图片或者文字的显示代码编写,所以我们会用到能够生成取模代码的软件。用到的取模软件有:文字取模和图片取模。


软件生成的内容以 logo.h 保存在 LoRa 人机界面开发的 Inc 文件夹当中,在之后的人机界面开发的实操中,我们会介绍 logo.h 的使用方法。

### TFT 液晶屏显示字符串
要想让 TFT 显示屏显示字体,除了上述取模的方法以外(取模方法常用于显示占满整个屏幕的字体或者中文),我们还可以直接使用库函数 Gui_DrawFont_GBK16() 来完成英文字体的输出。这个库函数的工作原理如下图所示。
<img src="/images/lora-doc/image163.png" alt="图片" data-width="57%" style="width: 57%" data-align="center" />

Gui_DrawFont_GBK16(uint16_t x, uint16_t y, uint16_t fc, uint16_t bc, uint8_t *s) 的五个参数分别是:显示的行地址、显示的列地址、字体颜色、背景色、要写的字符串(英文)。
## TFT 人机界面开发
### 开机界面设计
对于人机界面开发,我们的设计是:TFT 显示屏一开始显示公司的 logo 图标,然后显示出 LoRa 接收和发送的相关数据信息。
显示的 logo:
<img src="/images/lora-doc/image165.jpeg" alt="图片" data-width="53%" style="width: 53%" data-align="center" />
实现图像显示的最基本思路是模仿清屏函数 Lcd_Clear() 的代码。因为我们输入的数据设定为行扫描,并且显示设定的大小为 128*128,所以 for 循环当中循环的次数可以确定,同时低字节优先,高字节靠后。
```c
/*************************************************
函数名:showimage
功能:显示一副图片
入口参数:图片缓存
返回值:无
*************************************************/
//因为一副128*128的图片需要32768个8位数组成(数组元素有32768个),很显然数组太大,不能放到内存中,只能放在静态存储区,所以参数使用 const 修饰
void showimage(const unsigned char *p)
{
unsigned int i;
uint16_t HData,LData;
Lcd_SetRegion(0,0,X_MAX_PIXEL-1,Y_MAX_PIXEL-1);
Lcd_WriteIndex(0x2C);
for(i = 0;i < 128*128;i++)//因为我们已经设置显示屏显示范围为128*128,所以写满一行会自动换行,不需单独写代码
{
LData = *(p+i*2);
HData = *(p+i*2+1);
LCD_WriteData_16Bit(HData<<8|LData);
}
}
```text

这个函数的声明在 lcd.h 文件已经准备好了。

然后打开我们之前生成的 logo.h,进行宏处理:

并且在 main.c 里面添加头文件

之后即可调用 showimage() 函数来显示我们的 logo 图像。

**小结**:从上述编写 showimage() 函数的步骤我们可以总结得到程序开发的一般过程。首先确定函数的功能,**在源程序文件(.c)完成函数功能代码的编写,然后在头文件(.h)声明函数**。最后在需要**使用到该函数的位置包含头文件才能正常使用**。
### 菜单界面设计
显示收发数据的内容其实更多是排版的问题,显示英文内容我们直接使用函数 Gui_DrawFont_GBK16() 即可。
<img src="/images/lora-doc/image171.png" alt="图片" data-width="56%" style="width: 56%" data-align="center" />
```c
showimage(gImage_logo);
HAL_Delay(500);
Lcd_Clear(YELLOW);
Gui_DrawFont_GBK16(0,0,RED,GREEN," LoRa Topology ");
Gui_DrawFont_GBK16(0,16,RED,GREEN," Master ");
Gui_DrawFont_GBK16(0,32,BLACK,YELLOW,"SSID:");
Gui_DrawFont_GBK16(64,32,BLACK,YELLOW,"30");
Gui_DrawFont_GBK16(0,48,BLACK,YELLOW,"RX:");
Gui_DrawFont_GBK16(64,48,BLACK,YELLOW,"255");
Gui_DrawFont_GBK16(0,64,BLACK,YELLOW,"TX");
Gui_DrawFont_GBK16(64,64,BLACK,YELLOW,"255");
```text

# LoRa PingPang 系统设计
PingPang 是什么?其实大白话来讲就是收发数据的过程。在 LoRa 中,主机发送的数据叫Ping,从机发送的数据叫 Pang(也有地方用 Pong)。
LoRa 通信的整个过程就是 Master 主动发送 PING 数据,接收 PANG 数据,Slave 接收 PING 数据,回应 PANG 数据的过程,这个功能的设计就是 LoRa PingPang 系统的设计。
## 深入了解 LoRa 技术原理
### LoRa 扩频通信原理
无线电波指在自由空间传播的射频(RF)频段的电磁波,其基本原理是导体中电流强度的改变会产生无线电波。利用这一现象,通过调制可将信息加载于无线电波中。当电波通过空间传播到达接收方时,电波引起的电磁场变化又会在导体中产生电流。再通过解调将信息从电流变化中提取出来,即可实现信息传递。
模拟无线通信信号大概经历以下三大过程:**输入→检波→放大**。

数字无线通讯中,调制指将输入信息变换为适于信道传输的形式。信号源信息通常包含直流分量和频率较低的频率分量,称为基带信号。基带信号一般不能直接用于传输,需变换为一个远高于基带频率的信号,即已调信号。
调制过程改变了高频载波即信息载体信号的幅度、相位或频率,使其随基带信号幅度变化而变化。解调过程则将基带信号从载波中提取出来。
常用调制方式有:
- 模拟调制(幅度调制、角度调制)
- 数字调制(ASK、FSK、PSK)
- 脉冲调制(指用脉冲序列作为载波,最常用的是脉码调制)

无线通信传播方式有地波传播(低于 2MHz)、天波传播(2MHz~30MHz)和直线传播(30MHz 以上)。

无线通信传播路径有反射、散射和衍射三种。这些都会造成信号在移动环境中的衰落。

同时生活中还有许许多多无线通信噪声存在。

综上,也就导致了信号可能出现的不稳定的情况。
还有传输过程中的损耗:
```c
衰减和衰减失真
自由空间损耗
噪声
大气吸收
多径
```text
扩频(Spread Spectrum,SS)是一种重要的通信技术。发送方输入的数据首先进入信道编码器,生成模拟信号,该模拟信号围绕某个中心频率具有相对较窄的带宽。然后使用扩频码或扩展序列进一步调制,通常扩频码由伪噪声或伪随机数产生器产生。调制后传输信号的带宽显著增加,即扩展了频谱。 接收方使用同一扩频码进行解扩。解扩后的信号通过信号解码器,最终还原为数据。
扩频通信技术的作用:从各种类型的噪声和多径失真中获得**免疫性**。
扩频通信算法:C 表示信号质量

扩频通信原理:用户数据和扩频数据**异或**得到发送数据,增加了信号带宽,提高了信号质量。

### LoRa 关键技术参数
- **信号带宽(BW)**:
增加 BW,可以提高有效数据速率以缩短传输时间,但是以牺牲部分接受灵敏度为代价。对于 LoRa 芯片 SX127x,LoRa 带宽为双边带宽(全信道带宽),而 FSK 调制方式的 BW 是指单边带宽。
| 带宽(kHz) | 扩频因子 | 编码率 | 标称比特率(bps) |
| --- | --- | --- | --- |
| 7.8 | 12 | 4/5 | 18 |
| 10.4 | 12 | 4/5 | 24 |
| 15.6 | 12 | 4/5 | 37 |
| 20.8 | 12 | 4/5 | 49 |
| 31.2 | 12 | 4/5 | 73 |
| 41.7 | 12 | 4/5 | 98 |
| 62.5 | 12 | 4/5 | 146 |
| 125 | 12 | 4/5 | 293 |
| 250 | 12 | 4/5 | 586 |
| 500 | 12 | 4/5 | 1172 |
- **扩频因子(SF)**:
原本使用 1 位来表示的信号变成多位来表示这个信号,提高信号的通讯质量。
LoRa 采用多个信息码片来代表有效负载信息的每个位,扩频信息的发送速度称为符号速率(Rs),而码片速率与标称的 Rs 比值即为扩频因子(SF,SpreadingFactor),表示了每个信息位发送的符号数量。(信噪比越小,信号质量越好)
| 扩频因子(RegModulationCfg) | 扩频因子(码片/符号) | LoRa 解调器信噪比(SNR) |
| --- | --- | --- |
| 6 | 64 | -5 dB |
| 7 | 128 | -7.5 dB |
| 8 | 256 | -10 dB |
| 9 | 512 | -12.5 dB |
| 10 | 1024 | -15 dB |
| 11 | 2048 | -17.5 dB |
| 12 | 4096 | -20 dB |
- **编码率(CR)**:
提高信号质量的冗余,提高数据的可靠性。
编码率(或信息率)是数据流中有用部分(非冗余)的比例。也就是说,如果编码率是 k/n,则对每 k 位有用信息,编码器总共产生 n 位的数据,其中 n-k 是多余的。
LoRa 采用循环纠错编码进行前向错误检测与纠错。使用该方式会产生传输开销。
| 编码率(RegTxCfg1) | 循环编码率 | 开销比率 |
| --- | --- | --- |
| 1 | 4/5 | 1.25 |
| 2 | 4/6 | 1.5 |
| 3 | 4/7 | 1.75 |
| 4 | 4/8 | 2 |
---
LoRa 符号速率 Rs 计算:

LoRa 数据速率 DR 计算:

LoRaWAN 主要使用了 125kHz 信号带宽设置,但其他专用协议可以利用其他的信号带宽(BW)设置。改变 BW、SF 和 CR 也就改变了链路预算和传输时间,需要在电池寿命和距离上做个权衡。
配置 LoRa 收发数据需要设置的参数如下,分别是两个函数 SX1276LoRaInit() 和 LoRaSettings() 以及函数的若干事件。

### LoRa 数据收发任务
LoRa 数据发送序列:
数据发送流程:(1)LoRa 模块标准模式 →(2)发送模式 →(3)将数据写入发送队列 →(4)发送数据 →(5)等待发送完成(判断数据是否发送完成,如果发送完成则再次进入标准模式,如果还有数据需要发送,则进入第三步)

LoRa 数据接收序列:
数据接收流程分成两部分,分别是连续接收模式和单次接收模式。接收是通过等待接收中断来判断执行的。
数据接收流程:(1)等待接收中断 → (2)是否超时 → (3)读取数据 → (4)检查模式是否更换 → (5)等待下一次接收中断

Radio 事件任务:(官方固件提供的事件处理的任务,这些事件都是需要配置的)

我们只需要设置上面的这些工作状态,就可以实现 LoRa 数据的收发。这些事件任务都可以在 sx1276-LoRa.c 和 sx1276-LoRa.h 代码当中找到。
## LoRa PingPang 系统设计原理
### PingPang 系统设计需求
将 LoRa 终端定义成两种角色:Master(主机)和 Slave(从机),这也就决定了我们需要两块 LoRa 开发板才能完成收发数据的任务。
LoRa PingPang 系统设计需求:
- Master 主动发送 PING 数据,接收 PANG 数据;
- Slave 如果接收到 PING 数据,回应 PANG 数据。
- 终端在 LCD 屏幕上显示终端类型及收发数据包个数。
### PingPang 系统通信机制

发送数据的流程跟上面讲的差不多,我们可以看到,主机发送 Ping 是使用事件 SetTxPacket、StartRx 来完成的,从机发送 Pang 是使用事件 StartRx、SetTxPacket 来完成。是否停止收发数据我们可以看到是通过使用事件 TIMEOUT 或者 DONE 来判断的。
### PingPang 系统业务流程
PingPang 系统的第一步,对所有使用到的外设部分先进行初始化。并且添加宏定义判断,判断烧录程序的当前开发板是主机还是从机,对应执行不同的事件任务(主机发送数据,从机等待接收数据)。
<img src="/images/lora-doc/image187.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
PingPang 业务流程——Master 主机:
主机确定数据发送,则 LCD 屏幕显示发送的数据内容,发送数据完成后指示灯闪烁,并把主机从发送模式设置为接收模式,等待从机返回确认数据。
当主机处在接收模式时,等待接收中断,收到数据并需判断接收的数据是否为从机返回的 PANG 数据,如果是则接收指示灯闪烁,并且返回 PING 数据,最后更新 LCD 显示收到的数据内容。
<img src="/images/lora-doc/image188.png" alt="图片" data-width="70%" style="width: 70%" data-align="center" />
PingPang 业务流程——Slave:
<img src="/images/lora-doc/image189.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
LoRa 参数设置一览:
<img src="/images/lora-doc/image190.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
数据包结构:

## LoRa PingPang 系统功能开发
### IAR 工程配置
从上面的过程我们知道,我们需要对不同的两片开发板运行不同的收发过程,一个作为主机,一个作为从机。当主机处于发送状态时,从机需要在等待接收模式下;当主机处于等待接收状态时,从机需要在发送模式下。所以第一步,我们对不同的开发板先进行主从机设置。


新建两个配置,一个是主机"Master",另一个是从机"Slave"。



配置这段的意义是以后主机和从机的工程都是基于我们原来写好的 "learn" 的基础代码。
然后还需要添加主机和从机的宏定义,工作区先选择主机的配置:



然后再切换到 Slave 的工作区里面,宏定义"SLAVE"。

### 搭建框架
完成准备工作之后,我们分析一下需要在主程序中增加哪些函数功能来完成我们收发数据的任务。每个开发板至少应该有两个事件任务,则主从机应该一共配置四个函数。这两个事件函数一个负责 LCD 内容的显示,一个是收发数据的处理。
<img src="/images/lora-doc/image201.png" alt="图片" data-width="65%" style="width: 65%" data-align="center" />
我们把主机显示任务函数定义为 MLCD_Show(),从机显示任务定义为 SLCD_Show();主机无线任务定义为 Master_Task(),从机无线任务定义为 Slave_Task().
然后我们根据上面的功能函数框架,在 main.c 当中完成框架的搭建如下。
```c
/* main.c */
/* USER CODE BEGIN 0 */
//主机显示任务
void MLCD_Show(void){
}
//从机显示任务
void SLCD_Show(void){
}
//主机无线任务
void Master_Task(void){
}
//从机无线任务
void Slave_Task(void){
}
/* USER CODE END 0 */
```text
建立数据结构:声明全部变量、进行赋值初始化。
<img src="/images/lora-doc/image202.png" alt="图片" data-width="67%" style="width: 67%" data-align="center" />
接收的数据、发送的数据、接收的数据量、发送的数据量等都需要使用变量来存储,所以我们在程序的开始之初根据我们所需,先完成这些变量的赋值和初始化操作。
```c
/* main.c */
/* Private variables ——————————————----*/
#define BUFFERSIZE 4
uint8_t PingMsg[] = "PING";
uint8_t PingMsg[] = "PANG";
uint16_t BufferSize = BUFFERSIZE;
uint8_t Buffer[BUFFERSIZE];
#ifdef MASTER
uint8_t EnbleMaster = true;
#else
uint8_t EnbleMaster = false;
#endif
uint32_t Master_TxNumber = 0;
uint32_t Master_RxNumber = 0;
uint32_t Slave_TxNumber = 0;
uint32_t Slave_RxNumber = 0;
tRadioDriver *Radio = NULL; //在最前面要包含库 radio.h
```text
### 编码
(1)功能函数编码
我们要实现主机和从机的显示任务,就要使用到人机交互界面设计当中已经完成的一些用于显示的代码,将他们重新规整到 MLCD_Show() 和 SLCD_Show() 函数中。同时在这两个函数中,由于我们用到了 LCD 显示和 SPI 传输数据,所以也别忘了对这两者进行初始化。大致代码如下:
```c
//主机显示任务
void MLCD_Show(void){
uint8_t str[20] = {0};
LCD_GPIO_Init(); //LCD初始化
sprintf((char*)str,"%d",Master_RxNumber); //将接收的内容显示出来
Gui_DrawFont_GBK16(64,48,BLACK,YELLOW,str);
memset((char*)str,0,strlen((const char*)str)); //申请新的内存初始化
sprintf((char*)str,"%d",Master_TxNumber); //将发送的东西显示出来
Gui_DrawFont_GBK16(64,64,BLACK,YELLOW,str);
//SPI初始化
HAL_SPI_DeInit(&hspi1);
MX_SPI1_Init();
}
//从机显示任务
void SLCD_Show(void){
uint8_t str[20] = {0};
LCD_GPIO_Init(); //LCD初始化
sprintf((char*)str,"%d",Slave_RxNumber); //将接收的内容显示出来
Gui_DrawFont_GBK16(64,48,BLACK,YELLOW,str);
memset((char*)str,0,strlen((const char*)str)); //申请新的内存初始化
sprintf((char*)str,"%d",Slave_TxNumber); //将发送的东西显示出来
Gui_DrawFont_GBK16(64,64,BLACK,YELLOW,str);
//SPI初始化
HAL_SPI_DeInit(&hspi1);
MX_SPI1_Init();
}
```text
完成显示任务之后,我们开始编写主从机的无线任务。无线任务的编写思路就是我们上面主从机业务流程图的整个过程。在编写这部分代码的时候,我们不需要盲目自己去写,要学会去找官方已经提供好的参数和定义,灵活去配置。
举个例子,像在这里我们想要找到 LoRa 的数据处理的内容结构体,我们可以右键这个表示 LoRa 数据处理的变量"Radio",找到关于它的定义在第 80 行,双击跳转。

然后找到它原来宏定义的名字,再跳转到它的结构体。
<img src="/images/lora-doc/image204.png" alt="图片" data-width="70%" style="width: 70%" data-align="center" />
然后我们就能看到Radio的所有结构体了:

这些结构体表示的意思和内容在之前分析的图片当中已经有详细的注释了,大家要是不太记得可以翻回去再去了解一下。我们开发需要做的,就是根据需要取出这些结构体来使用就能完成系统的开发了。
完整代码如下:
```c
//主机无线任务
void Master_Task(void){
switch(Radio->Process()){ //无线任务处理进程判断
case RF_RX_DONE: //接收完成
Radio->GetRxPacket(Buffer,&BufferSize); //获取接收数据
if(strncmp((const char*)Buffer,(const char*)PangMsg,strlen((const char*)PangMsg)) == 0){ //判断是否为Pang数据
LedToggle(LED_RX); //接收指示灯闪烁
Master_RxNumber++; //统计接收数据量
Radio->SetTxPacket(PingMsg, strlen((const char*)PingMsg)); //发送Ping数据
}
break;
case RF_TX_DONE: //发送完成
LedToggle(LED_TX); //发送指示灯闪烁
Master_TxNumber++; //统计发送数据量
Radio->StartRx(); //设置为接收状态
break;
default:
break;
}
}
//从机无线任务
void Slave_Task(void){
switch(Radio->Process()){ //无线任务处理进程判断
case RF_RX_DONE: //接收完成
Radio->GetRxPacket(Buffer,&BufferSize); //获取接收数据
if(strncmp((const char*)Buffer,(const char*)PingMsg,strlen((const char*)PingMsg)) == 0){ //判断是否为Pang数据
LedToggle(LED_RX); //接收指示灯闪烁
Slave_RxNumber++; //统计接收数据量
Radio->SetTxPacket(PingMsg, strlen((const char*)PangMsg)); //发送Ping数据
}
break;
case RF_TX_DONE: //发送完成
LedToggle(LED_TX); //发送指示灯闪烁
Slave_TxNumber++; //统计发送数据量
Radio->StartRx(); //设置为接收状态
break;
default:
break;
}
}
```text
(2)main 函数编码
功能函数已经编写完成,接下来我们就来编写 main 主函数,按照上面的业务流程图,将整个过程复现在主函数当中,实现 LoRa 的数据收发。
```c
/* main.c */
#ifdef MASTER
Radio->SetTxPacket(PingMsg, strlen((const char*)PangMsg)); //发送Ping数据
#else
Radio->StartRx(); //开启接收
#endif
while (1){
if(EnbleMaster == true){
MLCD_Show(); //主机显示
Master_Task(); //主机无线任务
}
else{
SLCD_Show(); //从机显示
Slave_Task(); //从机无线任务
}
}
```text
(3)LoRa驱动源码
完成主函数功能的编写之后,我们再对 LoRa 驱动部分进行参数的设置和内容的修改。
首先我们打开 sx1276-LoRa.c 文件,注释掉其中所有的 DIO2 部分,因为我们不使用。


然后我们就要在 sx1276-LoRa.c 文件当中进行 LoRa 参数的设置(参数设置在前面的 **PingPang 系统业务流程**也有图片参考):

编写项目之后会出现程序报错,这是因为有一个代码文件还没有添加进来。



添加这两个文件之后再编译,就没有报错了。

那么到此,整个代码编写部分的任务就已经完成,LoRa PingPang 系统设计的整体思路也大致如此。
## LoRa PingPang 系统功能调试
为了验证我们程序编写的准确性,我们接下来进行功能调试环节。功能调试我们需要通信,所以需要准备好两块 LoRa 的开发板,一个作主机一个作从机,分别烧入不同的开发板当中。

接下来我们要通过串口打印出来的信息来验证我们的代码是否像期望的一样正常运行。
在宏定义里面加 printf() 函数打印主机、从机的确认信息。
```c
#ifdef MASTER
Radio->SetTxPacket(PingMsg, strlen((const char*)PangMsg)); //发送Ping数据
printf("I am Master!");
#else
Radio->StartRx(); //开启接收
printf("I am Salve!");
#endif
```text
然后在主机和从机的任务函数当中添加 printf() 函数,打印出接收的数据信息是什么。
```c
//主机无线任务
void Master_Task(void){
switch(Radio->Process()){ //无线任务处理进程判断
case RF_RX_DONE: //接收完成
Radio->GetRxPacket(Buffer,&BufferSize); //获取接收数据
printf("Master_Task:RX______%s\\n",Buffer);
if(strncmp((const char*)Buffer,(const char*)PangMsg,strlen((const char*)PangMsg)) == 0){ //判断是否为Pang数据
LedToggle(LED_RX); //接收指示灯闪烁
Master_RxNumber++; //统计接收数据量
Radio->SetTxPacket(PingMsg, strlen((const char*)PingMsg)); //发送Ping数据
}
break;
case RF_TX_DONE: //发送完成
LedToggle(LED_TX); //发送指示灯闪烁
Master_TxNumber++; //统计发送数据量
Radio->StartRx(); //设置为接收状态
break;
default:
break;
}
}
//从机无线任务
void Slave_Task(void){
switch(Radio->Process()){ //无线任务处理进程判断
case RF_RX_DONE: //接收完成
Radio->GetRxPacket(Buffer,&BufferSize); //获取接收数据
printf("Slave_Task:RX______%s\\n",Buffer);
if(strncmp((const char*)Buffer,(const char*)PingMsg,strlen((const char*)PingMsg)) == 0){ //判断是否为Pang数据
LedToggle(LED_RX); //接收指示灯闪烁
Slave_RxNumber++; //统计接收数据量
Radio->SetTxPacket(PingMsg, strlen((const char*)PangMsg)); //发送Ping数据
}
break;
case RF_TX_DONE: //发送完成
LedToggle(LED_TX); //发送指示灯闪烁
Slave_TxNumber++; //统计发送数据量
Radio->StartRx(); //设置为接收状态
break;
default:
break;
}
}
```text
然后就可以编译烧写程序检验代码的正确性。


可以看到,通过两个不同的串口助手连接两个不同的 LoRa 开发板,主机和从机之间已经实现了数据的收发任务,正常运行。
# LoRa 串口透传开发
## LoRa 串口透传系统设计
为什么要进行串口透传开发?因为很多人并不具备 LoRa 模块的开发能力,很多供应商会选择替用户设计好直观的串口透传系统,解决用户使用 LoRa 门槛高的问题。
### 串口透传系统设计需求

1、将 LoRa 终端定义成两种角色:Master(主机)和 Slave(从机);
2、一个模块发送任意字节长度(小于 128 Byte)数据,另一模块都可以接收到;
3、**PC 机上通过串口调试助手实现接收和发送**;
4、终端在 LCD 屏幕上显示终端类型及收发数据包个数;
LoRa 透传系统设计与 PingPang 系统设计最大的不同之处是后者没有完成自定义数据的发送和接收,我们之前 **PingPang 系统设计的数据接收和发送是固定数据**。
### 串口透传系统通信机制
透传机制如下图所示,主机通过串口助手发送数据 SetTxPacket() 到从机,从机 StartRx() 处于等待接收的状态,当从机接收到主机的数据之后,使用 printf() 将数据打印出来并返回主机,主机也通过 printf() 将数据打印出来。

### 串口透传业务流程
| 初始化 | 主程序 | LCD任务 |
| --- | --- | --- |
| | | |
| 串口接收任务 | 无线任务 | |
| | | |
## LoRa串口透传系统功能开发
### 串口功能开发
<img src="/images/lora-doc/image223.png" alt="图片" data-width="66%" style="width: 66%" data-align="center" />
对于串口的功能配置,我们在 STM32CubeMX 生成的代码 uart.c 中有 uart1 的接收中断初始化、设置中断优先级和打开全局中断,所以这一步不需要人为干预,我们只需要将启动串口,让串口进入空闲中断的代码添加在 main.c 中即可。
```c
/* main.c */
//启动串口1,使能串口空闲中断 。设置为空闲中断模式可以在整包之后才进行数据的收发,然后产生中断,避免重复单个数据的收发(减小开销)
__HAL_UART_ENABLE_IT(&huart1,UART_IT_IDLE);//注意:使能空闲中断后,不管有没有接收数据,会先触发一次中断,发送一次数据
HAL_UART_Receive_DMA(&huart1,a_Usart1_RxBuffer,RXLENGHT); //打开DMA接收,串口空闲中断模式配合DMA进行数据的接收
```text

然后我们打开 stm32f0xx_it.c 来配置串口中断。
```c
/* stm32f0xx_it.c */
void USART1_IRQHandler(void){
uint32_t temp;
//判断是否产生空闲中断
if(__HAL_UART_GET_FLAG(&huart1,UART_FLAG_IDLE) != RESET ){
//清除中断标志
__HAL_UART_CLEAR_IDLEFLAG(&huart1);
temp = huart1.Instance->ISR; //读取寄存器的值
temp = huart1.Instance->RDR; //读取寄存器的值
HAL_UART_DMAStop(&huart1); //停用DMA 关键点1:以上两行代码用于清除DMA的接收中断(只需要读取一次ISR和RDR寄存器的值)
temp = hdma_usart1_rx.Instance->CNDTR;
UsartType1.Usart_rx_len = RXLENGHT - temp; /*CNDTR为DMA通道接收数据的计数器(注意是一个递减计数器,所以需要将DMA的缓存区的总长度减去该计数器的值才是DMA通道接收数据的长度)*/
HAL_UART_RxCpltCallback(&huart1); //调用回调函数,在回调函数中具体实现数据的接收 关键点2
}
HAL_UART_IRQHandler(&huart1);
```text
在 usart.c 当中重写回调函数。回调函数是指使用者自己定义一个函数,实现这个函数的程序内容,然后把这个函数(入口地址)作为参数传入别人(或系统)的函数中,由别人(或系统)的函数在运行时来调用的函数。函数是你实现的,但由别人(或系统)的函数在运行时通过参数传递的方式调用,这就是所谓的回调函数。简单来说,就是由别人的函数运行期间来回调你实现的函数。
```c
/* usart.c */
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *UartHandle){
uint8_t Old_len = 0;
if(UartHandle->Instance == USART1){ //判断是否为USART1中断
Old_len = strlen(UsartType1.usartDMA_rxBuf);//读取发送缓冲区UsartType1.usartDMA_rxBuf的长度(如果为0则表示没有数据)
if(0 != Old_len){ /*如果发送缓冲区中还有数据,则将DMA中接收缓冲区a_Usart1_RxBuffer的数据拷贝到发送缓冲区的末尾
(不覆盖发送缓冲区前面的数据)*/
memcpy(&UsartType1.usartDMA_rxBuf[Old_len],a_Usart1_RxBuffer,UsartType1.Usart_rx_len);
Old_len=0;
}
else{ /*如果发送缓冲区UsartType1.usartDMA_rxBuf中没有数据,则直接将DMA中接收缓冲区a_Usart1_RxBuffer的数据
拷贝到发送缓冲区的起始位置*/
memcpy(UsartType1.usartDMA_rxBuf,a_Usart1_RxBuffer,UsartType1.Usart_rx_len);
// UsartxSendData_DMA(&huart1,a_Usart2_RxBuffer,UsartType2.Usart_rx_len);
memset(a_Usart1_RxBuffer,0,UsartType1.Usart_rx_len);//清空DMA中接收缓冲区a_Usart1_RxBuffer的数据
HAL_UART_Receive_DMA(&huart1,a_Usart1_RxBuffer,RXLENGHT);//继续打开DMA接收
UsartType1.receive_flag =1; //接收标识位,置1表示发送缓冲区中有新的数据,提醒相关函数将数据发送出去
}
}
```text
在 main.c 里面编写函数功能——将发送缓冲区里面的数据发送出去。
```c
/* main.c */
//串口数据获取//串口数据获取
void UartDmaGet(void){ //串口接收任务处理函数
if(UsartType1.receive_flag == 1){ //如果发送缓冲区中有新的数据,则将其发送给sx1278
//串口接收到的数据原封发给SX1278,sx1278通过无线将数据发送出去
Radio->SetTxPacket(UsartType1.usartDMA_rxBuf, UsartType1.Usart_rx_len);
memset(UsartType1.usartDMA_rxBuf,0,UsartType1.Usart_rx_len);
UsartType1.receive_flag = 0; //接收数据标志清零
}
}
```text
数据结构:
<img src="/images/lora-doc/image225.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
```c
/* uart.h */
#define RXLENGHT 128
#define RECEIVELEN 2048
#define USART_DMA_SENDING 1//发送未完成
#define USART_DMA_SENDOVER 0//发送完成
typedef struct
{
uint8_t receive_flag ; //空闲接收标记
uint8_t dmaSend_flag ; //发送完成标记
uint16_t Usart_rx_len; //接收长度
uint8_t usartDMA_rxBuf[RECEIVELEN]; //无线发送缓存
}USART_RECEIVETYPE;
extern USART_RECEIVETYPE UsartType1;
extern USART_RECEIVETYPE UsartType2;
//局部变量
//extern uint8_t g_Usart1_RxBuffer[RXLENGHT];
//extern uint8_t g_Usart2_RxBuffer[RXLENGHT];
//全局变量
extern uint8_t a_Usart1_RxBuffer[RXLENGHT]; //DMA接收缓存
//extern uint8_t a_Usart2_RxBuffer[RXLENGHT];
/* USER CODE END Private defines */
```text
### LCD 功能开发
在串口透传系统设计需求当中,我们明确了终端在 LCD 屏幕上显示终端类型及收发数据包个数,所以设计 LCD 的事件任务如下图所示。
<img src="/images/lora-doc/image226.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
<img src="/images/lora-doc/image227.png" alt="图片" data-width="80%" style="width: 80%" data-align="center" />
LED 闪烁代码:
```c
void LedBlink( tLed led )
{
LedPort[led]->ODR ^= LedPin[led];
HAL_Delay(50);
LedPort[led]->ODR ^= LedPin[led];
}
```text
LCD 显示代码:
这部分代码的理解可以结合 PingPang 系统设计中的系统功能调试以及串口透传系统通信机制的流程框图来加以理解。这里主要是把 PingPang 系统功能调试的 printf() 改成了使用串口将数据打印出来,思路大体是一致的。
```c
//读取sx127x射频射频数据
void Sx127xDataGet(void){
switch(Radio->Process( )){ //无线任务处理进程判断
case RF_RX_TIMEOUT: //接收超时
printf("RF_RX_TIMEOUT\\n");
break;
case RF_RX_DONE: //接收完成
Radio->GetRxPacket( Buffer, ( uint16_t* )&BufferSize ); //获取接收数据
if(EnbleMaster == true)
printf("master Rx__%s,__,%d,%d\\n",Buffer,strlen((char*)Buffer),BufferSize);
else
printf("slave Rx__%s,__,%d,%d\\n",Buffer,strlen((char*)Buffer),BufferSize);
if( BufferSize > 0 ){ //&& (BufferSize == strlen((char*)Buffer)))
//接收数据闪烁
LedBlink( LED_RX );
//计算接收数据的个数
RxDataPacketNum();
//清空sx127x接收缓冲区
memset(Buffer,0,BufferSize );
}
break;
case RF_TX_DONE:
//发送闪烁
LedBlink( LED_TX );
//计算发送数据的个数
TxDataPacketNum();
Radio->StartRx();//打开接收模式
break;
case RF_TX_TIMEOUT:
printf("RF_TX_TIMEOUT\\n");
break;
default:
break;
}
}
```text
### 无线收发功能开发
<img src="/images/lora-doc/image227.png" alt="图片" data-width="82%" style="width: 82%" data-align="center" />
无线收发任务跟前面的 PingPang 系统设计大致相同,这里不再进行赘述。
```c
/* main.c */
void RxDataPacketNum(void){ //接收数据包计数
if(EnbleMaster == true)
Master_RxNumber++;
else
Slave_RxNumber++;
}
void TxDataPacketNum(void){ //发送数据包计数
if(EnbleMaster == true)
Master_TxNumber++;
else
Slave_TxNumber++;
}
//主机显示任务
void MLCD_Show(void){
uint8_t str[20] = {0};
LCD_GPIO_Init(); //LCD初始化
sprintf((char*)str,"%d",Master_RxNumber); //将接收的内容显示出来
Gui_DrawFont_GBK16(64,48,BLACK,YELLOW,str);
memset((char*)str,0,strlen((const char*)str)); //申请新的内存初始化
sprintf((char*)str,"%d",Master_TxNumber); //将发送的东西显示出来
Gui_DrawFont_GBK16(64,64,BLACK,YELLOW,str);
//SPI初始化
HAL_SPI_DeInit(&hspi1);
MX_SPI1_Init();
}
```text

# LoRa 自组网络设计
## 深入了解 LoRaWAN
### LoRaWAN 是什么?
按照 LoRa 联盟官方白皮书《what is LoRaWAN》的介绍,LoRaWAN 是为 LoRa 远距离通信网络设计的一套通讯协议和系统架构。

可以看到一个 LoRaWAN 网络架构中包含了终端、基站、NS(网络服务器)、应用服务器这四个部分。基站和终端之间采用星型网络拓扑,由于 LoRa 的长距离特性,它们之间得以使用单跳传输。在终端部分官方列了 6 个典型应用,有个细节,你会发现终端节点可以同时发给多个基站。基站则对 NS 和终端之间的 LoRaWAN 协议数据做转发处理,将 LoRaWAN 数据分别承载在了 LoRa 射频传输和 TCP/IP 上。
### LoRaWAN 通信机制

首先是传感器端,传感器收集到的数据会存储在 LoRaWAN 的从机,LoRaWAN 底层部分应当有用 HAL 库配置好的 SPI 接口来传输这些数据,数据传输给服务器物理层。然后通过 LoRa 的 FSK 无线传输技术将数据传输到网关的物理层进行数据处理,通过 Ethernet、3G 或者 WIFI 加密传输到网络终端上面。但在用户看来,传感器端是直接跟网络服务器通信的。
### LoRaWAN 与其他组网协议对比

### LoRaWAN 通信协议
LoRaWAN 在协议和网络架构的设计上,充分考虑了节点功耗,网络容量,QoS,安全性和网络应用多样性等几个因素。

协议中有规定 Class A/B/C 三类终端设备,这三类设备基本覆盖了物联网所有的应用场景。

这是 Class A 上下行的时序图,目前接收窗口 RX1 一般是在上行后 1 秒开始,接收窗口 RX2 是在上行后 2 秒开始。(功耗低,但是实时性较差)

Class B 的时隙则复杂一些,它有一个同步时隙 beacon,还有一个固定周期的接收窗口 ping 时隙。(实时性高)

Class C 和 A 基本是相同的,只是在 Class A 休眠的期间,它都打开了接收窗口 RX2。(保证了实时性,但是功耗比 Class A 高)

### LoRaWAN 服务器框架
下面 LoRaWAN 的架构,它是基于 Internet 建设物联网,Gateway 是 IP 设备(运行 IP 协议栈),而 End Node 运行的是 LoRaMac-node(没有运行 IP 协议栈)。
LoRaWAN Server 共有 4 种角色,包括:NS(Network Server,网络服务器)、AS(Application Server,应用服务器)、NC(Network Controller,网络控制服务器)和 CS(Customer Server,客户服务器)。

### LoRaWAN 服务器通信接口

### LoRaWAN 服务器通信协议
它们之间的通信协议规律如下:
NS 和 Gateways 通过 JSON / GWMP / UDP / IP;
Command console 和 4 种服务器通过 JSON / UDP / IP;
4 种服务器之间通过 JSON / TCP / IP。
UDP 通信的优势:实时性
TCP 通信的优势:可靠性

## LoRa 自组网架构设计
### MAC 协议重要性
MAC 协议全称是 medium access control(介质访问控制)。介质访问控制的内容就是,采取一定的措施,使得两对节点之间的通信不会发生互相干扰的情况。它主要用于解决信号冲突的问题、尽可能地节省电能、保证通信的健壮和稳定性。
在 LoRa 自组网络设计中,会有多个节点同时收发数据,所以我们需要 MAC 协议帮助我们确定什么时间接收什么节点的数据,使得多个终端设备共享信道资源,提高信道利用率。

### MAC 协议的种类
(1)信道划分的 MAC 协议
- 时分(TDMA)、频分(FDMA)、码分(CDMA)划分
(2)随机访问 MAC 协议
- ALOHA,S-ALOHA,CSMA,CSMA/CD
- CSMA/CD 应用于以太网
- CSMA/CA 应用于 802.11 无线局域网
(3)轮讯访问 MAC 协议
- 主节点轮询
- 工业Modbus通信协议

### 常见的几种协议的优缺点分析
(1)时分复用

**优点:**
节省电能、最大化使用带宽;
**缺点:**
所有节点需要精确的时钟源,并且需要周期性校时、向网络中添加和删除节点都要有时隙分配和回收算法。
(2)频分复用

**优点:**
增加通信容量、提高通信可靠性;
**缺点:**
物理通道增加,成本增加。
(3)轮询访问

**优点:**
协议简单,易开发;
**缺点:**
通讯效率低、网络规模小。
### 基于时分复用 LoRa 自组网设计
由于能力和条件的限制,我们这里的 LoRa 自组网络就采用比较简单但是效率较高的时分复用 MAC 协议来完成。根据上述时分复用的原理,我们制定如下图所示的入网机制。

时分复用的基本原理是这样的,不同的节点随机等待一段时间进行数据的发送,假如当前的节点 i 等待时间到,且检测无其他节点正在与主机进行数据交换,则发送数据给主机,主机进行数据的接收。此时如果恰好节点 j 的等到时间也到了,则需继续等待一段随机时间,因为节点 i 正在传输数据,等到下一个时间片到达,且无其他节点发送数据,则节点 j 才可与主机通信。

### LoRa 自组网协调器设计

### LoRa自组网节点设计

## LoRa 自组网集中器程序开发
### 通讯协议
以下是 LoRa 规定的节点数据帧格式,贯穿于整个自组网的开发过程。
| 从机->入网请求 | | | |
| --- | --- | --- | --- |
| 名称 | 字节数 | 描述 | 举例 |
| 帧头 | 1 | 0x3C | 0x3C |
| 长度 | 1 | 最长 126 字节;长度范围内不检测包头;计算整个帧长度 | 0x0c |
| 网络类型 | 1 | 字符 代表入网请求 | J |
| 网络标识符 | 2 | PANID,用于网络区分,只有 PANID 一样才可以组网通信 | 0x0102 |
| 设备地址 | 2 | 设备唯一地址标识 | 0x1235 |
| CRC8 校验 | 1 | 数据包校验,整个数据包,除校验位 | 0x08 |
| 主机->入网应答 | | | |
| --- | --- | --- | --- |
| 名称 | 字节数 | 描述 | 举例 |
| 帧头 | 1 | 0x3C | 0x3C |
| 长度 | 1 | 最长 126 字节;长度范围内不检测包头;计算整个帧长度 | 0x0c |
| 网络类型 | 1 | 字符 代表入网成功 | A |
| 网络标识符 | 2 | PANID,用于网络区分,只有 PANID 一样才可以组网通信 | 0x0102 |
| 设备地址 | 2 | 设备唯一地址标识 | 0x1235 |
| 设备序号 | 1 | 设备是第几个入网 | 1 |
| CRC8 校验 | 1 | 数据包校验,整个数据包,除校验位 | 0x08 |
| 主机->时间同步 | | | |
| --- | --- | --- | --- |
| 名称 | 字节数 | 描述 | 举例 |
| 帧头 | 1 | 0x3C | 0x3C |
| 长度 | 1 | 最长126字节;长度范围内不检测包头;计算整个帧长度 | 0x0c |
| 网络类型 | 1 | 字符 代表入网请求 | T |
| 网络标识符 | 2 | PANID,用于网络区分,只有 PANID 一样才可以组网通信 | 0x0102 |
| 时 | 1 | RTC- 时 1-24 | 0x1235 |
| 分 | 1 | RTC- 分 1-60 | 12 |
| 秒 | 1 | RTC- 秒 1-60 | 12 |
| 亚秒 | 2 | RTC- 亚秒 1-1000 | 1000 |
| CRC8 校验 | 1 | 数据包校验,整个数据包,除校验位 | 0x08 |
| 网络数据包 | | | | |
| --- | --- | --- | --- | --- |
| 名称 | 字节数 | 描述 | 举例 | |
| 帧头 | 1 | 字符 代表网络数据包 | N | |
| 网络标识符 | 2 | 用于网络区分,只有 PANID 一样才可以组网通信 | 0x0102 | |
| 数据包 | 包头 | 1 | 0x21 | 0x21 |
| 包长 | 1 | 数据包长度,代表数据域数据长度 | 1 | |
| 数据类型 | 1 | 0x00:数据、0x01:命令 | 0x00 | |
| 设备地址 | 2 | 设备标识符 | 0x0001 | |
| 传感器类型 | 1 | 0x01:温湿度、0x02:三轴、0x03:水表、0x04:地磁、0x05:灌溉 | 0x01 | |
| 数据 | 4 | 每种传感器数值用两个字节标识,比如温湿度占四个字节 | 0x01298113 | |
| CRC8 校验 | 1 | 数据包校验,整个数据包,除校验位 | 0x08 | |
### 工程修改
业务流程参见下图:

根据协调器业务流程,需要在之前工程里添加两个外设的配置:RTC 和定时器。RTC 时钟是为了计算节点数据发送的时间而服务的。添加外设需要修改 STM32CubeMX 工程。

双击打开 .ioc 文件进入 STM32CubeMX,在左侧的引脚中选择 "Timers" 下的 "RTC",并且打开 "Activate Clock Source"、"Activate Calendar"、"Internal Alarm A"。

然后对 RTC 时钟进行一些简单的配置。


之后我们在 TIM2 中配置内部时钟。



修改完成后点击"生成"即可:

### 搭建框架
在 User 文件夹下面新建一个文件夹,然后将网络处理、协议解析的代码"dataprocess.c"、"netprocess.c"、"protocol.c"放在文件夹当中,方便多功能代码的开发,让结构更加清晰。

框架的基本内容有:
RTC 任务——RTC 初始化、Alarm 中断任务;
```c
/* rtc.c */
//闹钟事件回调函数
void HAL_RTC_AlarmAEventCallback(RTC_HandleTypeDef *hrtc){
}
//时分秒转换
void GetTimeHMS(uint32_t timeData,uint8_t *hours,uint8_t *minute,uint8_t *seconds,uint32_t *subSeconds) {
}
```text
定时器任务——定时器初始化、定时器中断任务;
```c
/* tim.c */
//定时器2溢出中断回调函数
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim){
}
```text
通信协议——CRC8 检验函数、协议数据结构;
```c
/* protocol.c */
/******************************************************************************
* Name: CRC-8 x8+x2+x+1
* Poly: 0x07
* Init: 0x00
* Refin: False
* Refout: False
* Xorout: 0x00
* Note:
*****************************************************************************/
uint8_t crc8(uint8_t *data, uint8_t length){
uint8_t i;
uint8_t crc = 0; // Initial value
while(length--){
crc ^= *data++; // crc ^= *data; data++;
for ( i = 0; i < 8; i++ ){
if ( crc & 0x80 )
crc = (crc << 1) ^ 0x07;
else
crc <<= 1;
}
}
return crc;
}
//CRC8校验
uint8_t DataCrcVerify(uint8_t * buff, uint8_t len){
uint8_t Crc8Data = 0;
//验证数据是否正确
Crc8Data = crc8(buff, len - 1);
if (Crc8Data == buff[len - 1]){
// PRINTF1("CRC8 Success!\\n");
return 1;
}
else{
// PRINTF1("CRC8 Failed!\\n");
return 0;
}
}
```text
数据处理任务——串口任务(串口接收)、无线任务(无线接收、主机协议解析、网络数据包解析、入网请求解析)
```c
/* dataprocess.c */
extern uint16_t BufferSize;
extern uint8_t Buffer[BUFFER_SIZE];
#if defined(MASTER)
extern uint8_t EnableMaster;
#elif defined(SLAVE)
extern uint8_t EnableMaster;
#endif
extern tRadioDriver *Radio;
extern uint32_t Master_RxNumber;
extern uint32_t Master_TxNumber;
extern uint32_t Slave_RxNumber;
extern uint32_t Slave_TxNumber;
extern volatile uint8_t SendDataOkFlag;
uint8_t startUpDateHours = 0;
uint8_t startUpDateMinute = 0;
uint8_t startUpDateSeconds = 0;
uint16_t startUpDateSubSeconds = 0;
//Master存储入网的设备信息
SlaveInfo slaveNetInfo_t[NodeNumber];
//Salve入网信息包
SlaveJionNet jionPacke_t;
//Salve保存自己的地址
SlaveInfo slaveNativeInfo_t;
//节点数据
SlaveDataNet DataPacke_t;
//串口数据获取
void UartDmaGet(void){
}
//接收数据包计数
void RxDataPacketNum(void){
}
//发送数据包计数
void TxDataPacketNum(void){
}
//读取sx127x射频射频数据
uint8_t Sx127xDataGet(void){
}
//**************从**************//
//**************机**************//
//从机入网数据发送
void SendJionNetPacke(void){
}
//从机协议解析
uint8_t SlaveProtocolAnalysis(uint8_t *buff,uint8_t len){
}
//上传节点传感器数据
void SendSensorDataUP(void){
}
//**************主**************//
//**************机**************//
//主机协议解析
uint8_t MasterProtocolAnalysis(uint8_t *buff,uint8_t len){
}
//入网协议解析
uint8_t JionNetProtocolAnalysis(uint8_t *buff,uint8_t len){
}
//网络数据包解析
void NetDataProtocolAnalysis(uint8_t *buff,uint8_t len){
}
```text
网络处理任务——等待入网完成、主机发送时钟同步数据包
```c
/* netprocess.c */
//所有节点的更新周期(在Time内上传所有数据) 单位Ms
volatile uint32_t DataUpTimePeriod = 1000 * 60 * 1; //1分钟
volatile static uint32_t currentTime = 0;
//当前加入设个的个数
volatile uint16_t currentDeviceNumber = 0;
//保存当前加入节点
volatile uint16_t oldNodeNumber = 0;
//节点时间片
volatile uint32_t DataUpTime = 0;
//节点入网状态
volatile DeviceJionFlag JionNodeTimeOutFlag = No_Node_Jion_Flag;
uint8_t startUpTimeHours = 0;
uint8_t startUpTimeMinute = 0;
uint8_t startUpTimeSeconds = 0;
uint32_t startUpTimeSubSeconds = 0;
uint8_t DataUpTimeHours = 0;
uint8_t DataUpTimeMinute = 0;
uint8_t DataUpTimeSeconds = 0;
uint32_t DataUpTimeSubSeconds = 0;
//时钟同步
SlaveRtcSync rtcSync_t;
//初始化网络状态
volatile DeviceJionStatus NetStatus = NO_JION;
extern tRadioDriver *Radio;
//**************从**************//
//**************机**************//
//生成随机数
uint16_t RandomNumber(void){
}
//从机加入网络
uint8_t SlaveJionNetFuction(void){
}
//节点获取时间片
void SlaveGetSendTime(void){
}
//**************主**************//
//**************机**************//
//等待入网完成
DeviceJionFlag WaitJionNetFinish(uint8_t timout){
}
//主机发送同步时钟
void MasterSendClockData(void){
}
```text
### 源码分析

```c
/* main.c */
int main(void){
...
#if SLAVE
//获取随机入网时间
DelayTime = RandomNumber();
printf("JionTime = %d\\n",DelayTime);
HAL_Delay(DelayTime);
//等待入网成功
while (SlaveJionNetFuction());
//获取节点发送时间片
SlaveGetSendTime();
#else
//主机直接初始化RTC
MX_RTC_Init();
HAL_TIM_Base_Start_IT(&htim2);
#endif
while (1){
Sx127xDataGet();
#if SLAVE
if(sendUpDataFlag == 1){
SendSensorDataUP();
sendUpDataFlag = 0;
}
#else
UartDmaGet();
//等待节点入网
if (JionDeviceStatu != Node_Jion_Finish_Flag){
printf("main 等待加入网络\\n");
JionDeviceStatu = WaitJionNetFinish(10);
}
/* 有新节点加入 */
if (currentDeviceNumber != oldNodeNumber){
printf("main 新节点加入网络\\n");
HAL_TIM_Base_Start_IT(&htim2);
JionDeviceStatu = New_Node_Jion_Flag;
SendClockFlag = 0; //发送分时时间片
}
/* 有旧节点加入 */
for (int i = 0; i < currentDeviceNumber;i++){
/* 查询是否有旧节点重新加入*/
if (slaveNetInfo_t[i].deviceNetStatus == AGAIN_JION_NET){
printf("main 旧节点加入网络\\n");
slaveNetInfo_t[i].deviceNetStatus = JIONDONE;
JionDeviceStatu = New_Node_Jion_Flag;
SendClockFlag = 0; //发送分时时间片
HAL_TIM_Base_Start_IT(&htim2);
}
}
/* 给从机分发时间片 */
if ((JionDeviceStatu == Node_Jion_Finish_Flag)&&(SendClockFlag == 0)
&&(currentDeviceNumber != 0)){
if (SendDataOkFlag == 1) {
SendDataOkFlag = 0;
printf("main 发送时钟同步\\n");
//告诉所有节点开始上传数据
MasterSendClockData();
SendClockFlag = 1;
while(!SendDataOkFlag){ //等待发送完成
Sx127xDataGet();
}
SendDataOkFlag = 1;
}
}
#endif
...
}
}
```text
LoRa 启动完毕并自检 OK 之后,从机自动生成入网的时间,等待入网成功之后,获取节点发送时间片。主机处于一直等待节点入网的状态,当有节点入网时,判断是新节点入网还是旧节点入网,然后分别发送分时时间片。节点时间到,则开始上传数据,主机等待数据发送完成。
## LoRa 自组网节点程序开发
### 硬件准备
- LoRa 设备 ×3
- ST-Link ×1
- USB mini 线 ×3
选择一块 LoRa 主板烧写一次 Master 主机程序。
烧写两次 Slave 程序(模拟多个节点)——需要配置从机设备地址(选择两个不同的地址即可),分别烧录。

### 实验现象
从机(节点)入网请求,主机入网应答,从机 1 分钟定时上传数据。

主机:

从机:


# LoRa 智慧牧场项目集成开发
在前面的课程内容当中,我们已经学习完了一整套 LoRa 开发的过程,并且在前面一节实现了基于时分复用 MAC 协议的自组网络设计,可以说已经具备了 LoRa 集成开发的基础。为了更加巩固我们所学习的成果,我们就拿应用比较多的智慧牧场项目来真正将我们所学习的内容应用在实操上。
开始设计智慧牧场项目之前,我们回顾一下之前学习的 LoRa 自组网络设计的步骤。传输数据确定 → LoRa PingPang 系统 → LoRa 透传系统 → LoRa 自组网。我们可以模仿这个步骤,先规划一下智慧牧场项目我们大概需要以下几步:数据采集 → 数据发送 → 数据接收处理 → 闭环控制。
## 体征数据采集
### 需求分析及传感器原理
需求分析——如何获取奶牛计步信息?

采集传感器——六轴运动处理传感器 MPU6050
三轴加速度测量原理:加速度测量计反应的加速向量与当前的受力方向是相反,单位为 g。

三轴陀螺仪测量:陀螺仪,是用来测量角速度的,单位为度每秒(deg/s)。

MPU6050 是使用 IIC 进行数据传输的,其初始化过程如下:

读取加速度初始值过程如下:

读取加速度数据过程如下:

### 功能配置及源码分析
首先我们需要打开 .ioc 文件,对 STM32 芯片的 IIC 通信接口进行配置。查询原理图可知,芯片的 PB8(SCL)和 PB7(SDA)是 IIC 的数据引脚。

然后生成代码即可。
在项目工程的 User 文件夹里面新建一个 Sensor 文件夹,将 MPU6050 传感器的配置代码放在里面。

```c
/* mpu6050 */
#include "mpu6050.h"
#include "string.h"
#include "stdio.h"
#include "i2c.h"
int16_t Accx,Accy,Accz;
// 初始化MPU6050
void InitMpu6050(void){
uint8_t WriteCmd = 0;
//解除休眠状态
WriteCmd = 0x00;
HAL_I2C_Mem_Write(&hi2c1, ADDRESS_Write, PWR_MGMT_1, I2C_MEMADD_SIZE_8BIT, &WriteCmd, 1, 0x10);
//时钟速率0x06(1Khz)陀螺仪采样率0x07(125Hz)
WriteCmd = 0x07;
HAL_I2C_Mem_Write(&hi2c1, ADDRESS_Write, SMPLRT_DIV, I2C_MEMADD_SIZE_8BIT, &WriteCmd, 1, 0x10);
WriteCmd = 0x06;
HAL_I2C_Mem_Write(&hi2c1, ADDRESS_Write, CONFIG, I2C_MEMADD_SIZE_8BIT, &WriteCmd, 1, 0x10);
//不自检,2000deg/s
WriteCmd = 0x18;
HAL_I2C_Mem_Write(&hi2c1, ADDRESS_Write, GYRO_CONFIG, I2C_MEMADD_SIZE_8BIT, &WriteCmd, 1, 0x10);
//(不自检,2G,5Hz)
WriteCmd = 0x01;
HAL_I2C_Mem_Write(&hi2c1, ADDRESS_Write, ACCEL_CONFIG, I2C_MEMADD_SIZE_8BIT, &WriteCmd, 1, 0x10);
HAL_Delay(10);
mpu6050_verify(&Accx, &Accy, &Accz); //读取第一次的值
}
// MPU6050校验
void mpu6050_verify(int16_t *x, int16_t *y, int16_t *z){
uint8_t ReadBuffer[10] = {0};
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_XOUT_L, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[0],1, 0x10);
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_XOUT_H, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[1],1, 0x10);
*x = (ReadBuffer[1]<<8)+ReadBuffer[0];
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_YOUT_L, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[0],1, 0x10);
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_YOUT_H, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[1],1, 0x10);
*y = (ReadBuffer[1]<<8)+ReadBuffer[0];
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_ZOUT_L, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[0],1, 0x10);
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_ZOUT_H, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[1],1, 0x10);
*z = (ReadBuffer[1]<<8)+ReadBuffer[0];
}
// MPU6060获取三轴数据
void mpu6050_ReadData(float *Mx, float *My, float *Mz){
int16_t x,y,z;
uint8_t ReadBuffer[10] = {0};
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_XOUT_L, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[0],1, 0x10);
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_XOUT_H, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[1],1, 0x10);
x = (ReadBuffer[1]<<8)+ReadBuffer[0];
x -= Accx;
*Mx = ((float)x)/16384;
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_YOUT_L, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[0],1, 0x10);
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_YOUT_H, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[1],1, 0x10);
y = (ReadBuffer[1]<<8)+ReadBuffer[0];
y -= Accy;
*My = ((float)y)/16384;
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_ZOUT_L, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[0],1, 0x10);
HAL_I2C_Mem_Read(&hi2c1, ADDRESS_Read, ACCEL_ZOUT_H, I2C_MEMADD_SIZE_8BIT,&ReadBuffer[1],1, 0x10);
z = (ReadBuffer[1]<<8)+ReadBuffer[0];
z -= Accz;
*Mz = ((float)z)/16384;
}
```text
更多有关 MPU6050 的配置可以查阅数据手册来理解。
然后我们需要在 main.c 当中对数据读取进行配置:定义全局变量 Mx、My、Mz,初始化 IIC,初始化 MPU6050,然后每两秒钟读取一次加速度值即可。
```c
float Mx,My,Mz;
...
int main(void){
MX_I2C1_Init();
InitMpu6050();
...
while(1){
...
mpu6050_ReadData(&Mx,&My,&Mz);
printf("Mx = %.3f\\n",Mx);
printf("My = %.3f\\n",My);
printf("Mz = %.3f\\n",Mz);
...
}
...
}
```text
实验现象:
将传感器安装在奶牛腿上,其体征数据会每两秒返回一次,主机可获得接收并通过串口助手打印出来。

## 饲养环境采集
### 需求分析及传感器原理
需求分析——如何获取牛舍温湿度信息?

采集传感器——DHT11 数字式空气温湿度传感器
DHT11 数据传输原理:一次完整的数据传输为 40 bit,高位先出
8bit 湿度整数数据+ 8bit 湿度小数数据;
8bit 温度整数数据+ 8bit 温度小数数据;
8bit 校验和。
由于 DHT11 是单总线协议传输数据,所以是通过高低电平持续的时间长短来进行数据的传输。

启动采集的过程:

获取数据的过程:

数字 0 信号的高低电平状态:

数字 1 信号的高低电平状态:

### 功能配置及源码分析

由于我们这里 DHT11 接的是 PB8 引脚,所以需要对 PB8 的两种模式(输入、输出)进行初始化操作。
```c
/* gpio.c */
void D2_IN_GPIO_Init(void){
GPIO_InitTypeDef GPIO_InitStruct;
GPIO_InitStruct.Pin = GPIO_PIN_8;
GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
}
void D2_OUT_GPIO_Init(void){
GPIO_InitTypeDef GPIO_InitStruct;
GPIO_InitStruct.Pin = GPIO_PIN_8;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
}
DHT11 配置代码如下:
/* dht11.c */
#include "stdint.h"
#include "tim.h"
#include "gpio.h"
#include "dht11.h"
#include "delay.h"
//温湿度定义
uint8_t ucharT_data_H=0,ucharT_data_L=0,ucharRH_data_H=0,ucharRH_data_L=0,ucharcheckdata=0;
void DHT11_TEST(void){ //温湿传感启动
uint8_t ucharT_data_H_temp,ucharT_data_L_temp,ucharRH_data_H_humidity,ucharRH_data_L_humidity,ucharcheckdata_temp;
uint8_t ucharFLAG = 0,uchartemp=0;
uint8_t ucharcomdata;
uint8_t i;
D2_OUT_GPIO_Init();
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_8,GPIO_PIN_RESET);
HAL_Delay_ms(18);
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_8,GPIO_PIN_SET);
D2_IN_GPIO_Init();
HAL_Delay_10us(4);
if(!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8)){
ucharFLAG=2;
while((!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
ucharFLAG=2;
while(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8)&&ucharFLAG++);
for(i=0;i<8;i++){
ucharFLAG=2;
while((!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
HAL_Delay_10us(3);
uchartemp=0;
if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))uchartemp=1;
ucharFLAG=2;
while(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8)&&ucharFLAG++);//超时判断
if(ucharFLAG==1)break;
ucharcomdata<<=1;
ucharcomdata|=uchartemp;
}
ucharRH_data_H_humidity = ucharcomdata;
for(i=0;i<8;i++){
ucharFLAG=2;
while((!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
HAL_Delay_10us(3);
uchartemp=0;
if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))uchartemp=1;
ucharFLAG=2;
while(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8)&&ucharFLAG++);
if(ucharFLAG==1)break;
ucharcomdata<<=1;
ucharcomdata|=uchartemp;
}
ucharRH_data_L_humidity = ucharcomdata;
for(i=0;i<8;i++){
ucharFLAG=2;
while((!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
HAL_Delay_10us(3);
uchartemp=0;
if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))uchartemp=1;
ucharFLAG=2;
while((HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
if(ucharFLAG==1)break;
ucharcomdata<<=1;
ucharcomdata|=uchartemp;
}
ucharT_data_H_temp = ucharcomdata;
for(i=0;i<8;i++){
ucharFLAG=2;
while((!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
HAL_Delay_10us(3);
uchartemp=0;
if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))uchartemp=1;
ucharFLAG=2;
while((HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
if(ucharFLAG==1)break;
ucharcomdata<<=1;
ucharcomdata|=uchartemp;
}
ucharT_data_L_temp = ucharcomdata;
for(i=0;i<8;i++){
ucharFLAG=2;
while((!HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
HAL_Delay_10us(3);
uchartemp=0;
if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))uchartemp=1;
ucharFLAG=2;
while((HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_8))&&ucharFLAG++);
if(ucharFLAG==1)break;
ucharcomdata<<=1;
ucharcomdata|=uchartemp;
}
ucharcheckdata_temp = ucharcomdata;
// humiture=1;
uchartemp=(ucharT_data_H_temp+ucharT_data_L_temp+ucharRH_data_H_humidity+ucharRH_data_L_humidity);
if(uchartemp==ucharcheckdata_temp){
ucharT_data_H = ucharT_data_H_temp;
ucharT_data_L = ucharT_data_L_temp;
ucharRH_data_H = ucharRH_data_H_humidity;
ucharRH_data_L = ucharRH_data_L_humidity;
ucharcheckdata = ucharcheckdata_temp;
}
}
else{ //没用成功读取,返回0
ucharT_data_H = 0;
ucharT_data_L = 0;
ucharRH_data_H = 0;
ucharRH_data_L = 0;
}
HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq()/1000); //恢复1ms嘀嗒定时器
}
```text
main 函数配置读取温湿度数据代码如下:
```c
...
while(1){
#if defined(MPU6050)
mpu6050_ReadData(&Mx,&My,&Mz);
printf("Mx = %.3f\\n",Mx);
printf("My = %.3f\\n",My);
printf("Mz = %.3f\\n",Mz);
#elif defined(DHT11)
DHT11_TEST();
printf("TEMP = %d\\n",ucharT_data_H);
printf("HUM = %d\\n",ucharRH_data_H);
#endif
SLCD_Show();
...
}
...
```text
实验现象:
将温湿度传感器放置在牛舍环境下,每隔两秒钟主机会获取牧场温湿度值并且通过串口助手打印出来。

## 饲养环境控制
### 需求分析及传感器原理
需求分析——如何驱动风扇启动关闭?

风扇转动是通过直流电机来实现的,直流电机的原理图如下所示,T2 和 T3 为 P-Mos 管,T1 为 N-Mos 管。当 D1 输出高电平时,T2 阻断,风扇不转;当 D1 输出低电平时,T2 通路,风扇开始转动。

### 功能配置及源码分析
配置 PB7 为输出模式,连接风扇直流电机。
```c
/* gpio.c */
void D1_OUT_GPIO_Init(void)
{
HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, GPIO_PIN_SET);
GPIO_InitTypeDef GPIO_InitStruct;
GPIO_InitStruct.Pin = GPIO_PIN_7;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
}
```text
配置直流电机代码如下:
```c
#include "gpio.h"
#include "fan.h"
#include <stdbool.h>
static uint8_t FanStaus = false;
// 开启风扇
void FanOn(void){
HAL_GPIO_WritePin( FAN_GPIO_PORT, FAN_PIN, FAN_ON );
FanStaus = true;
}
// 关闭风扇
void FanOff(void){
HAL_GPIO_WritePin( FAN_GPIO_PORT, FAN_PIN, FAN_OFF );
FanStaus = false;
}
// 读取风扇状态
uint8_t FanReadStaus( void ){
return FanStaus;
}
在 main.c 里面配置每隔五秒开启一次风扇。
/* main.c */
...
while(1){
...
#if defined(MPU6050)
mpu6050_ReadData(&Mx,&My,&Mz);
printf("Mx = %d\\n",Mx);
printf("My = %d\\n",My);
printf("Mz = %d\\n",Mz);
#elif defined(DHT11)
DHT11_TEST();
printf("TEMP = %d\\n",ucharT_data_H);
printf("HUM = %d\\n",ucharRH_data_H);
#elif defined(FAN)
FanOn();
FanStaus = FanReadStaus();
printf("Fan is On\\n");
SLCD_Show();
HAL_Delay(5000);
FanOff();
FanStaus = FanReadStaus();
printf("Fan is Off\\n");
#endif
...
}
```text
实验现象:
每隔五秒风扇状态改变一次。(这里只是介绍了风扇驱动的基础功能,对于如何根据已有的温湿度数据甚至更多牧场环境数据来控制风扇开闭,大家可以根据实际需要完成功能的开发)

## 项目集成开发
### 定时采集上传数据
第一步:修改 RTC 文件
在体征数据采集的步骤当中,我们使用 STM32cubeMX 开启了 IIC,这使得我们配置好的 RTC 时钟初始化被更改,所以在这里我们需要改回来。
```c
/* rtc.c */
#include "rtc.h"
#include "netprocess.h"
#include "string.h"
#include "dataprocess.h"
#define CLOCKHOURS 5
volatile uint8_t SendClockFlag = 0;
volatile uint8_t sendUpDataFlag = 0;
RTC_AlarmTypeDef gAlarm;
RTC_HandleTypeDef hrtc;
void MX_RTC_Init(void){
RTC_TimeTypeDef sTime;
RTC_DateTypeDef sDate;
RTC_AlarmTypeDef sAlarm;
hrtc.Instance = RTC;
hrtc.Init.HourFormat = RTC_HOURFORMAT_24;
hrtc.Init.AsynchPrediv = 125-1;
hrtc.Init.SynchPrediv = 2000-1;
hrtc.Init.OutPut = RTC_OUTPUT_DISABLE;
hrtc.Init.OutPutPolarity = RTC_OUTPUT_POLARITY_HIGH;
hrtc.Init.OutPutType = RTC_OUTPUT_TYPE_OPENDRAIN;
if (HAL_RTC_Init(&hrtc) != HAL_OK){
_Error_Handler(__FILE__, __LINE__);
}
sTime.Hours = startUpDateHours;
sTime.Minutes = startUpDateMinute;
sTime.Seconds = startUpDateSeconds;
sTime.DayLightSaving = RTC_DAYLIGHTSAVING_NONE;
sTime.StoreOperation = RTC_STOREOPERATION_RESET;
if (HAL_RTC_SetTime(&hrtc, &sTime, RTC_FORMAT_BCD) != HAL_OK){
_Error_Handler(__FILE__, __LINE__);
}
sDate.WeekDay = RTC_WEEKDAY_MONDAY;
sDate.Month = RTC_MONTH_APRIL;
sDate.Date = 0x1;
sDate.Year = 0x18;
if (HAL_RTC_SetDate(&hrtc, &sDate, RTC_FORMAT_BCD) != HAL_OK){
_Error_Handler(__FILE__, __LINE__);
}
if(HAL_RTCEx_BKUPRead(&hrtc, RTC_BKP_DR0) != 0x32F2){
HAL_RTCEx_BKUPWrite(&hrtc,RTC_BKP_DR0,0x32F2);
}
sAlarm.AlarmTime.Hours = DataUpTimeHours;
sAlarm.AlarmTime.Minutes = DataUpTimeMinute;
sAlarm.AlarmTime.Seconds = DataUpTimeSeconds;
sAlarm.AlarmTime.SubSeconds = DataUpTimeSubSeconds;
sAlarm.AlarmTime.DayLightSaving = RTC_DAYLIGHTSAVING_NONE;
sAlarm.AlarmTime.StoreOperation = RTC_STOREOPERATION_RESET;
sAlarm.AlarmMask = RTC_ALARMMASK_DATEWEEKDAY;
sAlarm.AlarmSubSecondMask = RTC_ALARMSUBSECONDMASK_ALL;
sAlarm.AlarmDateWeekDaySel = RTC_ALARMDATEWEEKDAYSEL_DATE;
sAlarm.AlarmDateWeekDay = 0x1;
sAlarm.Alarm = RTC_ALARM_A;
memcpy(&gAlarm, &sAlarm, sizeof(sAlarm));
if (HAL_RTC_SetAlarm_IT(&hrtc, &sAlarm, RTC_FORMAT_BIN) != HAL_OK){
_Error_Handler(__FILE__, __LINE__);
}
}
void HAL_RTC_MspInit(RTC_HandleTypeDef* rtcHandle){
if(rtcHandle->Instance==RTC){
__HAL_RCC_RTC_ENABLE();
HAL_NVIC_SetPriority(RTC_IRQn, 0, 0);
HAL_NVIC_EnableIRQ(RTC_IRQn);
}
}
void HAL_RTC_MspDeInit(RTC_HandleTypeDef* rtcHandle){
if(rtcHandle->Instance==RTC){
__HAL_RCC_RTC_DISABLE();
HAL_NVIC_DisableIRQ(RTC_IRQn);
}
}
//闹钟事件回调函数
void HAL_RTC_AlarmAEventCallback(RTC_HandleTypeDef *hrtc){
RTC_TimeTypeDef masterTime;
RTC_TimeTypeDef SlaveTime;
RTC_DateTypeDef masterDate;
#if MASTER
//置位同步时钟标志
SendClockFlag = 0;
//获取下次闹钟时间
HAL_RTC_GetTime(hrtc, &masterTime, RTC_FORMAT_BIN);
HAL_RTC_GetDate(hrtc, &masterDate, RTC_FORMAT_BIN);
gAlarm.AlarmTime.Hours = masterTime.Hours + CLOCKHOURS;
gAlarm.AlarmTime.Minutes = masterTime.Minutes;
gAlarm.AlarmTime.Seconds = masterTime.Seconds;
gAlarm.AlarmTime.SubSeconds = masterTime.SubSeconds;
#else
sendUpDataFlag = 1;
HAL_RTC_GetTime(hrtc, &SlaveTime, RTC_FORMAT_BIN);
HAL_RTC_GetDate(hrtc, &masterDate, RTC_FORMAT_BIN);
gAlarm.AlarmTime.Hours = SlaveTime.Hours + DataUpTimeHours;
gAlarm.AlarmTime.Minutes = SlaveTime.Minutes + DataUpTimeMinute;
gAlarm.AlarmTime.Seconds = SlaveTime.Seconds + DataUpTimeSeconds;
gAlarm.AlarmTime.SubSeconds = SlaveTime.SubSeconds + DataUpTimeSubSeconds;
#endif
if (gAlarm.AlarmTime.Seconds > 59){
gAlarm.AlarmTime.Seconds -= 60;
gAlarm.AlarmTime.Minutes += 1;
}
if ( gAlarm.AlarmTime.Minutes >59){
gAlarm.AlarmTime.Minutes -= 60;
gAlarm.AlarmTime.Hours += 1;
}
if (gAlarm.AlarmTime.Hours > 23){
gAlarm.AlarmTime.Hours -= 24;
}
printf("RTC\\n");
//使能闹钟中断
if (HAL_RTC_SetAlarm_IT(hrtc, &gAlarm, RTC_FORMAT_BIN) != HAL_OK){
_Error_Handler(__FILE__, __LINE__);
}
}
//时分秒转换
void GetTimeHMS(uint32_t timeData,uint8_t *hours,uint8_t *minute,uint8_t *seconds,uint32_t *subSeconds) {
/* 获得亚秒 */
*subSeconds = timeData % 1000;
/* 获得秒钟*/
timeData = timeData / 1000;
*seconds = timeData % 60;
/* 获得分钟*/
timeData = timeData / 60;
*minute = timeData % 60;
/* 获得小时 */
*hours = timeData / 60;
}
```text
第二步:修改传感器定时上传函数。
在之前完成的代码当中,我们发送的数据是固定的数据内容,这个函数是 dataprocess.c 文件当中的 SendSensorDataUP() 函数,现在我们需要将它修改为发送传感器的数据包。
这就涉及到网络数据包的解包过程了。
| 网络数据包 | | | | |
| --- | --- | --- | --- | --- |
| 名称 | 字节数 | 描述 | 举例 | |
| 帧头 | 1 | 字符 代表网络数据包 | N | |
| 网络标识符 | 2 | 用于网络区分,只有 PANID 一样才可以组网通信 | 0x0102 | |
| 数据包 | 包头 | 1 | 0x21 | 0x21 |
| 包长 | 1 | 数据包长度,代表数据域数据长度 | 1 | |
| 数据类型 | 1 | 0x00:数据、0x01:命令 | 0x00 | |
| 设备地址 | 2 | 设备标识符 | 0x0001 | |
| 传感器类型 | 1 | 0x01:温湿度、0x02:三轴、0x03:水表、0x04:地磁、0x05:灌溉 | 0x01 | |
| 数据 | 4 | 每种传感器数值用两个字节标识,比如温湿度占四个字节 | 0x01298113 | |
| CRC8 校验 | 1 | 数据包校验,整个数据包,除校验位 | 0x08 | |
```c
/* dataprocess.c */
// 上传节点传感器数据
void SendSensorDataUP(void){
printf("SendSensorDataUP\\n");
#if defined(MPU6050)
mpu6050_ReadData(&Mx,&My,&Mz);
printf("Mx = %.3f\\n",Mx);
printf("My = %3f\\n",My);
printf("Mz = %3f\\n",Mz);
DataPacke_t.netmsgHead = 'N';
DataPacke_t.netPanid[0] = HI_UINT16(PAN_ID);
DataPacke_t.netPanid[1] = LO_UINT16(PAN_ID);
DataPacke_t.msgHead = 0x21;
DataPacke_t.dataLength = 0x08;
DataPacke_t.dataType = 0;
DataPacke_t.deviceAddr[0] = HI_UINT16(ADDR);
DataPacke_t.deviceAddr[1] = LO_UINT16(ADDR);
DataPacke_t.sensorType = 0x2;
DataPacke_t.buff[0] = (int8_t)(Mx*10);
DataPacke_t.buff[1] = (int8_t)(My*10);
DataPacke_t.buff[2] = (int8_t)(Mz*10);
//校验码
DataPacke_t.crcCheck = crc8((uint8_t *)&DataPacke_t,DataPacke_t.dataLength + 4);
//发送数据包
Radio->SetTxPacket((uint8_t *)&DataPacke_t, DataPacke_t.dataLength + 5);
#elif defined(DHT11)
DHT11_TEST();
printf("TEMP = %d\\n",ucharT_data_H);
printf("HUM = %d\\n",ucharRH_data_H);
DataPacke_t.netmsgHead = 'N';
DataPacke_t.netPanid[0] = HI_UINT16(PAN_ID);
DataPacke_t.netPanid[1] = LO_UINT16(PAN_ID);
DataPacke_t.msgHead = 0x21;
DataPacke_t.dataLength = 0x07;
DataPacke_t.dataType = 0;
DataPacke_t.deviceAddr[0] = HI_UINT16(ADDR);
DataPacke_t.deviceAddr[1] = LO_UINT16(ADDR);
DataPacke_t.sensorType = 0x1;
DataPacke_t.buff[0] = ucharT_data_H;
DataPacke_t.buff[1] = ucharRH_data_H;
//校验码
DataPacke_t.crcCheck = crc8((uint8_t *)&DataPacke_t,DataPacke_t.dataLength + 4);
//发送数据包
Radio->SetTxPacket((uint8_t *)&DataPacke_t, DataPacke_t.dataLength + 5);
#elif defined(FAN)
FanStaus = FanReadStaus();
DataPacke_t.netmsgHead = 'N';
DataPacke_t.netPanid[0] = HI_UINT16(PAN_ID);
DataPacke_t.netPanid[1] = LO_UINT16(PAN_ID);
DataPacke_t.msgHead = 0x21;
DataPacke_t.dataLength = 0x06;
DataPacke_t.dataType = 0;
DataPacke_t.deviceAddr[0] = HI_UINT16(ADDR);
DataPacke_t.deviceAddr[1] = LO_UINT16(ADDR);
DataPacke_t.sensorType = 0x3;
DataPacke_t.buff[0] = FanStaus;
//校验码
DataPacke_t.crcCheck = crc8((uint8_t *)&DataPacke_t,DataPacke_t.dataLength + 4);
//发送数据包
Radio->SetTxPacket((uint8_t *)&DataPacke_t, DataPacke_t.dataLength + 5);
#endif
}
```text
其基本的原理也并不复杂,就是根据上面的表格,按照数据包的每一帧进行判断,判断是什么数据对应实现不同的功能即可。例如返回的数据第九个数据帧是 0x02,根据表格,我们知道这个传感器是温湿度传感器的数据。
### 实时接收控制指令
第三步:修改从机协议解析函数。
在原来编写好的从机协议解析函数 SlaveProtocolAnalysis() 当中,我们实现了将收到的数据打印出来,同时也实现了对网络数据包前面几帧的解包操作。现在我们需要实现发送指定命令,就开启和关闭风扇的功能,那么这里同样涉及解包的代码操作。
```c
/* dataprocess.c */
//从机协议解析
uint8_t SlaveProtocolAnalysis(uint8_t *buff,uint8_t len){
uint8_t Crc8Data;
printf("SlaveProtocolAnalysis\\n");
for (int i = 0; i < len; i++){
printf("0x%x ",buff[i]);
}
printf("\\n");
if (buff[0] == NETDATA){
if (buff[1] == HI_UINT16(PAN_ID) && buff[2] == LO_UINT16(PAN_ID)){
Crc8Data = crc8(&buff[0], len - 1);
if (Crc8Data != buff[len - 1]){
memset(buff, 0, len);
return 0;
}
if (buff[3] == 0x21){
printf("Slave_NETDATA\\n");
if(buff[5] == 0x1){
if (buff[6] == HI_UINT16(ADDR) && buff[7] == LO_UINT16(ADDR)){
if(buff[8] == 0x3){
#if defined(FAN)
if(buff[9] == true){
FanOn();
}
else{
FanOff();
}
#endif
}
}
}
}
return 0;
}
}
else if((buff[0] == 0x3C) && (buff[2] == 'A')){
if (DataCrcVerify(buff, len) == 0){
return 0;
}
if (buff[3] == HI_UINT16(PAN_ID) && buff[4] == LO_UINT16(PAN_ID)){
if (buff[5] == jionPacke_t.deviceAddr[0] && buff[6] == jionPacke_t.deviceAddr[1]){
slaveNativeInfo_t.deviceId = buff[7];
printf("Slave_ACK\\n");
return 0xFF;
}
}
}
else if((buff[0] == 0x3C) && (buff[2] == 'T')){
if (DataCrcVerify(buff, len) == 0){
return 0;
}
if (buff[3] == HI_UINT16(PAN_ID) && buff[4] == LO_UINT16(PAN_ID)){
uint32_t alarmTime = 0;
startUpTimeHours = buff[5];
startUpTimeMinute = buff[6];
startUpTimeSeconds = buff[7];
startUpTimeSubSeconds = buff[8] <<8 | buff[9];
printf("Slave_CLOCK\\n");
printf("H:%d,M:%d,S:%d,SUB:%d\\n", startUpTimeHours, startUpTimeMinute, startUpTimeSeconds, startUpTimeSubSeconds);
alarmTime = ((DataUpTimeHours * 60 + DataUpTimeMinute) * 60
+ DataUpTimeSeconds) * 1000 + (DataUpTimeSubSeconds / 2) + DataUpTime;
GetTimeHMS(alarmTime, &DataUpTimeHours, &DataUpTimeMinute, &DataUpTimeSeconds, &DataUpTimeSubSeconds);
printf("DataUpTime->H:%d,M:%d,S:%d,SUB:%d\\n", DataUpTimeHours, DataUpTimeMinute, DataUpTimeSeconds, DataUpTimeSubSeconds);
//使能RTC
MX_RTC_Init();
return 0xFF;
}
}
return 1;
}2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
1579
1580
1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
1612
1613
1614
1615
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
1681
1682
1683
1684
1685
1686
1687
1688
1689
1690
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713
1714
1715
1716
1717
1718
1719
1720
1721
1722
1723
1724
1725
1726
1727
1728
1729
1730
1731
1732
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
1786
1787
1788
1789
1790
1791
1792
1793
1794
1795
1796
1797
1798
1799
1800
1801
1802
1803
1804
1805
1806
1807
1808
1809
1810
1811
1812
1813
1814
1815
1816
1817
1818
1819
1820
1821
1822
1823
1824
1825
1826
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
1837
1838
1839
1840
1841
1842
1843
1844
1845
1846
1847
1848
1849
1850
1851
1852
1853
1854
1855
1856
1857
1858
1859
1860
1861
1862
1863
1864
1865
1866
1867
1868
1869
1870
1871
1872
1873
1874
1875
1876
1877
1878
1879
1880
1881
1882
1883
1884
1885
1886
1887
1888
1889
1890
1891
1892
1893
1894
1895
1896
1897
1898
1899
1900
1901
1902
1903
1904
1905
1906
1907
1908
1909
1910
1911
1912
1913
1914
1915
1916
1917
1918
1919
1920
1921
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931
1932
1933
1934
1935
1936
1937
1938
1939
1940
1941
1942
1943
1944
1945
1946
1947
1948
1949
1950
1951
1952
1953
1954
1955
1956
1957
1958
1959
1960
1961
1962
1963
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
2027
2028
2029
2030
2031
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
2042
2043
2044
2045
2046
2047
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
2061
2062
2063
2064
2065
2066
2067
2068
2069
2070
2071
2072
2073
2074
2075
2076
2077
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
2091
2092
2093
2094
2095
2096
2097
2098
2099
2100
2101
2102
2103
2104
2105
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
2119
2120
2121
2122
2123
2124
2125
2126
2127
2128
2129
2130
2131
2132
2133
2134
2135
2136
2137
2138
2139
2140
2141
2142
2143
2144
2145
2146
2147
2148
2149
2150
2151
2152
2153
2154
2155
2156
2157
2158
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
2172
2173
2174
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
2190
2191
2192
2193
2194
2195
2196
2197
2198
2199
2200
2201
2202
2203
2204
2205
2206
2207
2208
2209
2210
2211
2212
2213
2214
2215
2216
2217
2218
2219
2220
2221
2222
2223
2224
2225
2226
2227
2228
2229
2230
2231
2232
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
2244
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
2255
2256
2257
2258
2259
2260
2261
2262
2263
2264
2265
2266
2267
2268
2269
2270
2271
2272
2273
2274
2275
2276
2277
2278
2279
2280
2281
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
2292
2293
2294
2295
2296
2297
2298
2299
2300
2301
2302
2303
2304
2305
2306
2307
2308
2309
2310
2311
2312
2313
2314
2315
2316
2317
2318
2319
2320
2321
2322
2323
2324
2325
2326
2327
2328
2329
2330
2331
2332
2333
2334
2335
2336
2337
2338
2339
2340
2341
2342
2343
2344
2345
2346
2347
2348
2349
2350
2351
2352
2353
2354
2355
2356
2357
2358
2359
2360
2361
2362
2363
2364
2365
2366
2367
2368
2369
2370
2371
2372
2373
2374
2375
2376
2377
2378
2379
2380
2381
2382
2383
2384
2385
2386
2387
2388
2389
2390
2391
2392
2393
2394
2395
2396
2397
2398
2399
2400
2401
2402
2403
2404
2405
2406
2407
2408
2409
2410
2411
2412
2413
2414
2415
2416
2417
2418
2419
2420
2421
2422
2423
2424
2425
2426
2427
2428
2429
2430
2431
2432
2433
2434
2435
2436
2437
2438
2439
2440
2441
2442
2443
2444
2445
2446
2447
2448
2449
2450
2451
2452
2453
2454
2455
2456
2457
2458
2459
2460
2461
2462
2463
2464
2465
2466
2467
2468
2469
2470
2471
2472
2473
2474
2475
2476
2477
2478
2479
2480
2481
2482
2483
2484
2485
2486
2487
2488
2489
2490
2491
2492
2493
2494
2495
2496
2497
2498
2499
2500
2501
2502
2503
2504
2505
2506
2507
2508
2509
2510
2511
2512
2513
2514
2515
2516
2517
2518
2519
2520
2521
2522
2523
2524
2525
2526
2527
2528
2529
2530
2531
2532
2533
2534
2535
2536
2537
2538
2539
2540
2541
2542
2543
2544
2545
2546
2547
2548
2549
2550
2551
2552
2553
2554
2555
2556
2557
2558
2559
2560
2561
2562
2563
2564
2565
2566
2567
2568
2569
2570
2571
2572
2573
2574
2575
2576
2577
2578
2579
2580
2581
2582
2583
2584
2585
2586
2587
2588
2589
2590
2591
2592
2593
2594
2595
2596
2597
2598
2599
2600
2601
2602
2603
2604
2605
2606
2607
2608
2609
2610
2611
2612
2613
2614
2615
2616
2617
2618
2619
2620
实现效果:发送命令"4E 12 03 21 06 01 12 02 03 01 75",风扇运行;
发送命令"4E 12 03 21 06 01 12 02 03 00 72",风扇停止。


如何确定最后的校验位数字?由于使用的是 CRC8 校验,所以我们可以使用软件来生成 CRC 校验码。

LoRa 物联网系统设计
如何规划中小型 LoRa 物联网系统
在上一节的课程当中,我们动手完成了智慧牧场项目的集成开发,相信大家对整个 LoRa 项目开发过程有了初步的了解,那么这一节内容,我们就来好好梳理一下一个中小型 LoRa 物联网系统的设计都需要做哪些准备、需要考虑哪些问题。
为什么我们思考的是规划中小型的 LoRa 物联网系统?因为基于时分复用的私有协议只能做中小型的 LoRa 物联网系统,大型的需要基于 LoRaWAN 协议。
通讯距离——最远的节点与网关之间的通信距离
在发射功率、通信速率和天线相同的条件下,通信距离依赖于地形和环境的因素,所以在设计规划 LoRa 物联网系统的时候需要确定好通信所处的环境情况。

高空气球通信距离可达 40KM

铁塔通信距离可达 20KM

空旷地通信距离 5KM
那么我们如何增加通信距离呢?
1、增加发射功率。就 SX1278 芯片来说,它最大支持 30dBm 的全功率运行。
2、降低通信速率。
3、更换高增益天线。较好的增益天线可以增加 3dBm。
4、增加网关、集中器。
节点功耗
LoRa 节点一般用于户外,使用电池供电。针对电池供电的终端设备,我们需要通过它的工作模式、各个模式下的工作电流以及各个模式下的工作时长来合理计算 LoRa 节点的功耗情况,这样才能保证 LoRa 节点具备一定的使用寿命。
例:
LoRaWAN 终端,大约 10 分钟发送一次数据帧,约 1000ms;按协议,发送完毕后,1 秒内能唤醒,侦听时长为 160ms,接收时长为 1000ms;其他时间都处于 Sleep 休眠。以 10 分钟(600 秒)为单位。
| 工作模式 | 工作电流(mA) | 发送周期(min) | 工作时长(s) | 功耗 | 平均功耗(mA) | 电池容量(mA/H) | 使用天数 | 使用年数 |
|---|---|---|---|---|---|---|---|---|
| 休眠 | 0.0016 | 10 | 600-1.6-1 = 597.4 | 0.0016×597.4 = 0.95584 | (0.95584+20.8+88)/600 = 0.18292 | 2000 | 2000/0.18292/24 = 455.557 | 455.557/365 = 1.2481 |
| 接收 | 13 | - | 1.6 | 13×1.6 = 20.8 | - | - | - | - |
| 发送 | 88 | - | 1 | 88×1 = 88 | - | - | - | - |
LoRa 官方也提供给我们计算 LoRa 节点功耗的软件,方便我们在设计之初对 LoRa 的一些参数的配置:

在扩频因子设置为 12,负载长度为 8 字节,中心频率设置为 470000000(SX1278 芯片的固定频率),发射功率调整为 20dBm。计算后可以得到 8 个字节发送数据的时间为 190.46ms,发送电流是 125mA,接收电流是 13mA,睡眠电流为 100nA。

在通信距离可靠的情况下,我们把发射功率改为 18,可以发现信号强度降低了 1dBm,发送电流也降低了 20mA,说明功耗降低的同时,信号强度也会削弱。

这时候我们把扩频因子调整为 6。可以看到发送数据的时间大大缩短,同时信号的强度也较大程度地被削弱。也正是因为扩频因子越小,信号增益越差,而发送的数据量越少。

网络规模
网络规模与通信速率有直接关系。
| 按照 LoRaWan class A 计算 | |||
|---|---|---|---|
| 网络发送周期(min) | 单个节点发送时长(s) | 单个节点接收时长(s) | 节点数量 |
| 1 | 0.1 | 1 | 1min / (0.1s+1s) = 54.54545455 |
假设单个网关每天最多可以接收 a 个数据包,每个节点的应用发包频率是每小时 b 个数据包的话,那么,单个网关最多可以容纳的节点的数目的理论值的计算式如下:

我们可以看到,设计中小型的 LoRa 物联网系统,我们主要关注三个点:通讯距离、节点功耗以及网络规模。接下来我们结合上面的理论知识,再来分析一些生活中常见的 LoRa 物联网系统设计的案例。
LoRa 智能抄表系统开发
智能抄表解决方案
智能抄表系统架构图如下图所示。基本流程是:LoRa 水表收集数据,上传到 LoRa 网关,然后 LoRa 网关将数据信息上传到 LoRa 云,最后到用户的应用层面。

水表工作原理
水流从表壳进水口切向冲击叶轮使之旋转,然后通过齿轮减速机构连续记录叶轮的转数,从而记录流经水表的累积流量。
旋翼式多流束水表的工作原理与单流束水表基本相同,它是通过叶轮盒的分配作用,将多束水流从叶轮盒的进水口切向冲击叶轮,使水流对叶轮的轴向冲击力得到平衡,减少了叶轮支承部分的磨损,并从结构上减少水表安装、结垢对水表误差的影响,总体性能明显高于单流束水表。

常见的 LoRa 水表会在传统的水表指针上安装一个贴片,这可以是干簧管或者是霍尔传感器。将水流信号转换成电信号的原理就是通过指针上面的磁力让传感器接收到高低电平的信号。

信号输出——每 10 升(或 100 升)发一次信号
瞬时流量——单位时间内流经封闭管道有效截面的流体量,单位:m³/h
累计流量——从水表开始计量的一直累计的值,单位:m³
智能抄表功能开发
硬件设计:LoRa 水表的两端输出接开发板的 GPIO 引脚,引脚每次检测到由高电平到低电平跳跃,说明就有 10 升(或 100 升)水通过。可以通过模拟按键功能来实现。

GPIO初始化
→配置引脚为输入模式
→配置中断源为上升沿(或下降沿)触发
→编写中断回调函数
瞬时流量计量
→计算两次上升沿(或下降沿)中断时长(s)
→转换为瞬时流量(m³/h)
累计流量
→全局记录中断出发次数,转换为m³
LoRa 智能停车系统开发
智能停车解决方案
智能停车系统架构图如下图所示。基本流程是:LoRa 地磁传感器收集数据,上传到 LoRa 网关,然后 LoRa 网关将数据信息上传到 LoRa 云,最后到用户的手机 APP 界面。

地磁传感器工作原理
地磁传感器可用于检测车辆的存在和车型识别。数据采集系统在交通监控系统中起着非常重要的作用,地磁传感器是数据采集系统的关键部分,传感器的性能对数据采集系统的准确性起决定作用。

各向异性磁阻传感器由薄膜合金(透磁合金)制成,利用载流磁性材料在外部磁场存在时电阻特性将会改变的基本原理进行磁场变化的测量。当传感器接通以后,假设没有任何外部磁场,薄膜合金会有一个平行于电流方向的内部磁化矢量。

常用的地磁传感器为飞思卡尔的 MAG3110,MAG3110 是一款小型,低功耗数字 3D 磁性传感器,具有宽动态范围,可在具有高外部磁场的 PCB 中运行。具有如下特性:
测量局部磁场的成分,地磁场的总和以及电路板上的成分所产生的磁场;
可以与 3 轴加速度计结合使用,从而可以获得与方向无关的精确指南针航向信息;
具有标准的 I²C 串行接口,能够测量高达 10 高斯的局部磁场,输出数据速率高达 80 Hz。

智能停车功能开发
通过检测引脚是否为高电平判断车辆是否入驻。
GPIO初始化
- 配置 D3&KEY 为输入模式
车辆状态读取
- 读取 GPIO 状态,判断车辆入驻信息
LoRa 智能灌溉系统开发
智能灌溉解决方案
智能灌溉系统架构图如下图所示。

电磁阀工作原理
电磁阀从原理上分为三大类:直动式、分步直动式、先导式。而从阀瓣结构和材料上的不同与原理上的区别又分为六个分支小类:直动膜片结构、分步直动膜片结构、先导膜片结构、直动活塞结构、分步直动活塞结构、先导活塞结构。
直动式电磁阀
原理:通电时,电磁阀线圈产生电磁力把关闭件从阀座上提起,阀门打开; 断电时,电磁力消失,弹簧力把关闭件压在阀座上,阀门关闭。特点:在真空、 负压、零压时能正常工作,但通径一般不超过 25mm。

分步直动式电磁阀
原理:它是一种直动和先导式相结合的原理,当入口与出口没有压差时,通电后,电磁力直接把先导小阀和主阀关闭件依次向上提起,阀门打开。当入口与出口达到启动压差时,通电后,电磁力先打开先导小阀,主阀下压力上升,上腔压力下降,从而利用弹簧力或介质压力推动关闭件,向下移动,使阀门关闭。特点:在零压或真空、高压时亦能可靠动作,但功率较大,要求必须水平安装。

先导式电磁阀
原理:通电时,电磁力把先导孔打开,上腔室压力迅速下降,在关闭件周围形成上低下高的压差,流体压力推动关闭件移动,阀门打开;断电时,弹簧力把先导孔关闭,入口压力通过旁通孔迅速进入上腔室在关阀件周围形成下低上高的压差,流体压力推动关闭件向下移动,关闭阀门。特点:流体压力范围上限较高,可任意安装(需定制)但必须满足流体压差条件。

智能灌溉功能开发
D1 高电平,三极管导通,线圈通电,常开触点变常闭触点。

总结
开发物联网终端产品的关键要素:
传感器开发
无线模块开发
单片机开发
通信协议
人机交互