转自:http://blog.csdn.net/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.csdn.net/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数据擦除。
因此,实验成功。