转自:http://blog.youkuaiyun.com/feilusia/article/details/57374684
一、简介
本文以SimpleBLEPeripheral工程为例,介绍如何使用SNV。
二、实验平台
协议栈版本:ble_cc26xx_2_01_00_44423
编译软件:IAR Embedded Workbench for ARM Version 7.40
硬件平台:CC26xxDK开发板
仿真器:XDS100V3(香瓜)
四、
实验前提
1、在进行本文步骤前,请先
阅读
以下博文:
1)《
CC2541之SNV
》:
http://blog.youkuaiyun.com/feilusia/article/details/50190859
2)《SWRU393_CC2640_BLE_Software_Developer's_Guide》(第3.10.4章节):C:\ti\simplelink\ble_cc26xx_2_01_00_44423\Documents(协议栈路径)
2、在进行本文步骤前,请先
实现以下博文:
暂无
五、基础知识
1、CC2640的SNV是怎么样的?
答:
1)总大小
两页,共4K(协议栈已使用一部分)。
2)可用SNV的ID号
-
- #define BLE_NVID_CUST_START 0x80 //!< Start of the Customer's NV IDs
- #define BLE_NVID_CUST_END 0x8F //!< End of the Customer's NV IDs
0x80~0x8F都可以用(包含0x80、0x8F),每个ID号最多一次可写入252字节。
注:实际可以写几个ID取决于所剩余SNV空间,并不是说可以把所有ID号都写满252字节。
3)工程中可选的SNV大小
可在IAR的预编译处写入三种宏:
①OSAL_SNV=0:0个可写SNV页。工程中不使用SNV,由于绑定信息时需要SNV,该操作将导致无法使用绑定功能。
②OSAL_SNV=1:1个可写SNV页,共2K。
③OSAL_SNV=2(默认):2个可写SNV页,共4K。
2、多次“仿真、退出仿真、仿真”的操作,是否会把SNV擦除?
答:实测不会。
注:CC2541使用仿真时会擦除SNV。
3、如果我想写1K字节到SNV,怎么写?
答:可以每个ID写252字节,一共需要4个ID(1000/252=3.96)。
4、为什么有些工程使用不了SNV(比如组网的例程)?
答:因为组网例程所需flash较大,因此该工程的SNV部分被充分利用在了组网部分。
可以看工程的预编译中包含了“OSAL_SNV=0”。
六、实验步骤
1、编写并添加自定义的SNV驱动
1)写一个驱动GUA_SNV.c(存放在“……\ble_cc26xx_2_01_00_44423\Projects\ble\SimpleBLEPeripheral\CC26xx\Source\Application\GUA”路径下)
-
-
-
-
-
-
-
-
- #include "GUA_SNV.h"
- #include "osal_snv.h"
-
-
-
-
-
-
-
-
-
-
-
-
-
- GUA_U8 GUA_SNV_Control(GUA_U8 nGUA_SNV_ID, GUA_U8 nGUA_WriteReadFlag, GUA_U8 *npGUA_Data, GUA_U8 nGUA_Len)
- {
-
- if(nGUA_WriteReadFlag == GUA_SNV_READ)
- {
- return osal_snv_read(nGUA_SNV_ID, nGUA_Len, npGUA_Data);
- }
-
- else
- {
- return osal_snv_write(nGUA_SNV_ID, nGUA_Len, npGUA_Data);
- }
- }
2)写一个驱动头文件GUA_SNV.h(存放在“……\ble_cc26xx_2_01_00_44423\Projects\ble\SimpleBLEPeripheral\CC26xx\Source\Application\GUA”路径下)
-
-
-
-
-
-
-
-
- #ifndef _GUA_SNV_H_
- #define _GUA_SNV_H_
-
- /*********************宏定义************************/
- #ifndef GUA_U8
- typedef unsigned char GUA_U8;
- #endif
-
- #ifndef GUA_8
- typedef signed char GUA_8;
- #endif
-
- #ifndef GUA_U16
- typedef unsigned short GUA_U16;
- #endif
-
- #ifndef GUA_16
- typedef signed short GUA_16;
- #endif
-
- #ifndef GUA_U32
- typedef unsigned long GUA_U32;
- #endif
-
- #ifndef GUA_32
- typedef signed long GUA_32;
- #endif
-
- #ifndef GUA_U64
- typedef unsigned long long GUA_U64;
- #endif
-
- #ifndef GUA_64
- typedef signed long long GUA_64;
- #endif
-
-
- #define GUA_SNV_READ 0x00
- #define GUA_SNV_WRITE 0x01
-
-
- GUA_U8 GUA_SNV_Control(GUA_U8 nGUA_SNV_ID, GUA_U8 nGUA_WriteReadFlag, GUA_U8 *npGUA_Data, GUA_U8 nGUA_Len);
-
- #endif
3)工程中添加GUA_SNV.c

4)在IAR设置中添加驱动源文件路径
- $PROJ_DIR$/../../../Source/Application/GUA
2、应用层调用
1)添加头文件(simpleBLEperipheral.c中)
2)定义一个数据缓存区(simpleBLEPeripheral.c中)
-
- static uint8_t gbGUA_Data[256] = {0};
3)
定义一个SNV的测试ID(simpleBLEPeripheral.c中)
-
- #define GUA_SNV_TEST_ID 0x81
4)添加测试代码
(simpleBLEPeripheral.c的SimpleBLEPeripheral_Init函数末尾)
-
-
- uint8_t nGUA_Ret = 0;
-
-
- nGUA_Ret = GUA_SNV_Control(GUA_SNV_TEST_ID, GUA_SNV_READ, gbGUA_Data, 252);
- if(nGUA_Ret == NV_OPER_FAILED)
- {
-
- memset(gbGUA_Data, 0x38, sizeof(gbGUA_Data));
-
-
- nGUA_Ret = GUA_SNV_Control(GUA_SNV_TEST_ID, GUA_SNV_WRITE, gbGUA_Data, 252);
- }
-
-
- memset(gbGUA_Data, 0x55, sizeof(gbGUA_Data));
- nGUA_Ret = GUA_SNV_Control(GUA_SNV_TEST_ID, GUA_SNV_WRITE, gbGUA_Data, 252);
-
-
- memset(gbGUA_Data, 0x00, sizeof(gbGUA_Data));
- nGUA_Ret = GUA_SNV_Control(GUA_SNV_TEST_ID, GUA_SNV_READ, gbGUA_Data, 252);
-
-
七、注意事项
rebuild all,
设置断点在测试代码处,
仿真并全速运行。
测试可发现如下现象:
1、当该ID之前未写过数据时,读取的返回值为“NV_OPER_FAILED”。
2、一次最多只能对一个ID写252个字节,写多时虽然返回值仍然是“SUCCESS”,但实测读出来的数据是错的。
3、再次仿真时不会将上次写入的SNV数据擦除。
因此,实验成功。