低功耗蓝牙4.0BLE编程-nrf51822开发(2)

2024-05-28 12:48

本文主要是介绍低功耗蓝牙4.0BLE编程-nrf51822开发(2),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

首先看的示例是心率计一个示例程序:<KEIL path> \ARM\Device\Nordic\nrf51822\Board\pca10001\s110\ble_app_hrs\arm。

    打开工程前需要下载蓝牙协议栈S110 nRF51822 SoftDevice(s110_nrf51822_6.0.0_softdevice.hex)到板子中,这个手册上有说明。

    首先看的是main.c中的main函数:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /***************************************************************************** 
  2. * Main Function 
  3. *****************************************************************************/  
  4.   
  5. /**@brief Function for the application main entry. 
  6.  */  
  7. int main(void)  
  8. {  
  9.     uint32_t err_code;  
  10.   
  11.     timers_init();  
  12.     gpiote_init();  
  13.     buttons_init();  
  14.     ble_stack_init();  
  15.     bond_manager_init();  
  16.   
  17.     // Initialize Bluetooth Stack parameters  
  18.     gap_params_init();  
  19.     advertising_init();  
  20.     services_init();  
  21.     conn_params_init();  
  22.     sec_params_init();  
  23.   
  24.     // Start advertising  
  25.     advertising_start();  
  26.   
  27.     // Enter main loop  
  28.     for (;;)  
  29.     {  
  30.         // Switch to a low power state until an event is available for the application  
  31.         err_code = sd_app_evt_wait();  
  32.         APP_ERROR_CHECK(err_code);  
  33.     }  
  34. }  
它很简洁:初始化->start->loop。

(1)Timer初始化

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Function for the Timer initialization. 
  2.  * 
  3. * @details Initializes the timer module. This creates and starts application timers. 
  4. */  
  5. static void timers_init(void)  
  6. {  
  7.     uint32_t err_code;  
  8.   
  9.     // Initialize timer module  
  10.     APP_TIMER_INIT(APP_TIMER_PRESCALER, APP_TIMER_MAX_TIMERS, APP_TIMER_OP_QUEUE_SIZE, false);  
  11.   
  12.     // Create timers  
  13.     err_code = app_timer_create(&m_battery_timer_id,  
  14.                                 APP_TIMER_MODE_REPEATED,  
  15.                                 battery_level_meas_timeout_handler);  
  16.     APP_ERROR_CHECK(err_code);  
  17.   
  18.     err_code = app_timer_create(&m_heart_rate_timer_id,  
  19.                                 APP_TIMER_MODE_REPEATED,  
  20.                                 heart_rate_meas_timeout_handler);  
  21.     APP_ERROR_CHECK(err_code);  
  22. }  
使用app_timer_create创建了两个时钟,处理函数分别是battery_level_meas_timeout_handler和heart_rate_meas_timeout_handler。

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Function for handling the Battery measurement timer timeout. 
  2.  * 
  3.  * @details This function will be called each time the battery level measurement timer expires. 
  4.  *          This function will start the ADC. 
  5.  * 
  6.  * @param[in]   p_context   Pointer used for passing some arbitrary information (context) from the 
  7.  *                          app_start_timer() call to the timeout handler. 
  8.  */  
  9. static void battery_level_meas_timeout_handler(void * p_context)  
  10. {  
  11.     UNUSED_PARAMETER(p_context);  
  12.     battery_start();  
  13. }  
  14.   
  15.   
  16. /**@brief Function for handling the Heart rate measurement timer timeout. 
  17.  * 
  18.  * @details This function will be called each time the heart rate measurement timer expires. 
  19.  *          It will exclude RR Interval data from every third measurement. 
  20.  * 
  21.  * @param[in]   p_context   Pointer used for passing some arbitrary information (context) from the 
  22.  *                          app_start_timer() call to the timeout handler. 
  23.  */  
  24. static void heart_rate_meas_timeout_handler(void * p_context)  
  25. {  
  26.     uint32_t err_code;  
  27.   
  28.     UNUSED_PARAMETER(p_context);  
  29.   
  30.     err_code = ble_hrs_heart_rate_measurement_send(&m_hrs, m_cur_heart_rate);  
  31.   
  32.     if (  
  33.         (err_code != NRF_SUCCESS)  
  34.         &&  
  35.         (err_code != NRF_ERROR_INVALID_STATE)  
  36.         &&  
  37.         (err_code != BLE_ERROR_NO_TX_BUFFERS)  
  38.         &&  
  39.         (err_code != BLE_ERROR_GATTS_SYS_ATTR_MISSING)  
  40.     )  
  41.     {  
  42.         APP_ERROR_HANDLER(err_code);  
  43.     }  
  44. }  
时钟创建后并不会自动运行,当调用application_timers_start后时钟开始运行:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Function for starting the application timers. 
  2.  */  
  3. static void application_timers_start(void)  
  4. {  
  5.     uint32_t err_code;  
  6.   
  7.     // Start application timers  
  8.     err_code = app_timer_start(m_battery_timer_id, BATTERY_LEVEL_MEAS_INTERVAL, NULL);  
  9.     APP_ERROR_CHECK(err_code);  
  10.   
  11.     err_code = app_timer_start(m_heart_rate_timer_id, HEART_RATE_MEAS_INTERVAL, NULL);  
  12.     APP_ERROR_CHECK(err_code);  
  13. }  
services_init()初始化程序中的三个服务:ble_dis.c, ble_bas.c, ble_hrs.c。

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Function for initializing the services that will be used by the application. 
  2.  * 
  3.  * @details Initialize the Heart Rate, Battery and Device Information services. 
  4.  */  
  5. static void services_init(void)  
  6. {  
  7.     uint32_t       err_code;  
  8.     ble_hrs_init_t hrs_init;  
  9.     ble_bas_init_t bas_init;  
  10.     ble_dis_init_t dis_init;  
  11.     uint8_t        body_sensor_location;  
  12.   
  13.     // Initialize Heart Rate Service  
  14.     body_sensor_location = BLE_HRS_BODY_SENSOR_LOCATION_FINGER;  
  15.   
  16.     memset(&hrs_init, 0, sizeof(hrs_init));  
  17.       
  18.     hrs_init.is_sensor_contact_supported = false;  
  19.     hrs_init.p_body_sensor_location      = &body_sensor_location;  
  20.   
  21.     // Here the sec level for the Heart Rate Service can be changed/increased.  
  22.     BLE_GAP_CONN_SEC_MODE_SET_OPEN(&hrs_init.hrs_hrm_attr_md.cccd_write_perm);  
  23.     BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&hrs_init.hrs_hrm_attr_md.read_perm);  
  24.     BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&hrs_init.hrs_hrm_attr_md.write_perm);  
  25.   
  26.     BLE_GAP_CONN_SEC_MODE_SET_OPEN(&hrs_init.hrs_bsl_attr_md.read_perm);  
  27.     BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&hrs_init.hrs_bsl_attr_md.write_perm);  
  28.   
  29.     err_code = ble_hrs_init(&m_hrs, &hrs_init);  
  30.     APP_ERROR_CHECK(err_code);  
  31.   
  32.     // Initialize Battery Service  
  33.     memset(&bas_init, 0, sizeof(bas_init));  
  34.   
  35.     // Here the sec level for the Battery Service can be changed/increased.  
  36.     BLE_GAP_CONN_SEC_MODE_SET_OPEN(&bas_init.battery_level_char_attr_md.cccd_write_perm);  
  37.     BLE_GAP_CONN_SEC_MODE_SET_OPEN(&bas_init.battery_level_char_attr_md.read_perm);  
  38.     BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&bas_init.battery_level_char_attr_md.write_perm);  
  39.   
  40.     BLE_GAP_CONN_SEC_MODE_SET_OPEN(&bas_init.battery_level_report_read_perm);  
  41.   
  42.     bas_init.evt_handler          = NULL;  
  43.     bas_init.support_notification = true;  
  44.     bas_init.p_report_ref         = NULL;  
  45.     bas_init.initial_batt_level   = 100;  
  46.   
  47.     err_code = ble_bas_init(&bas, &bas_init);  
  48.     APP_ERROR_CHECK(err_code);  
  49.   
  50.     // Initialize Device Information Service  
  51.     memset(&dis_init, 0, sizeof(dis_init));  
  52.   
  53.     ble_srv_ascii_to_utf8(&dis_init.manufact_name_str, MANUFACTURER_NAME);  
  54.   
  55.     BLE_GAP_CONN_SEC_MODE_SET_OPEN(&dis_init.dis_attr_md.read_perm);  
  56.     BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&dis_init.dis_attr_md.write_perm);  
  57.   
  58.     err_code = ble_dis_init(&dis_init);  
  59.     APP_ERROR_CHECK(err_code);  
  60. }  
static ble_hrs_t的结构定义:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Heart Rate Service structure. This contains various status information for the service. */  
  2. typedef struct ble_hrs_s  
  3. {  
  4.     ble_hrs_evt_handler_t        evt_handler;                                          /**< Event handler to be called for handling events in the Heart Rate Service. */  
  5.     bool                         is_expended_energy_supported;                         /**< TRUE if Expended Energy measurement is supported. */  
  6.     bool                         is_sensor_contact_supported;                          /**< TRUE if sensor contact detection is supported. */  
  7.     uint16_t                     service_handle;                                       /**< Handle of Heart Rate Service (as provided by the BLE stack). */  
  8.     ble_gatts_char_handles_t     hrm_handles;                                          /**< Handles related to the Heart Rate Measurement characteristic. */  
  9.     ble_gatts_char_handles_t     bsl_handles;                                          /**< Handles related to the Body Sensor Location characteristic. */  
  10.     ble_gatts_char_handles_t     hrcp_handles;                                         /**< Handles related to the Heart Rate Control Point characteristic. */  
  11.     uint16_t                     conn_handle;                                          /**< Handle of the current connection (as provided by the BLE stack, is BLE_CONN_HANDLE_INVALID if not in a connection). */  
  12.     bool                         is_sensor_contact_detected;                           /**< TRUE if sensor contact has been detected. */  
  13.     uint16_t                     rr_interval[BLE_HRS_MAX_BUFFERED_RR_INTERVALS];       /**< Set of RR Interval measurements since the last Heart Rate Measurement transmission. */  
  14.     uint16_t                     rr_interval_count;                                    /**< Number of RR Interval measurements since the last Heart Rate Measurement transmission. */  
  15. } ble_h
ble_hrs.h/ble_hrs.c是心率计程序服务的代码。


buttons_init(void)初始化两个按钮:HR_INC_BUTTON_PIN_NO和HR_DEC_BUTTON_PIN_NO,分别模拟心率计的加减。

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Function for initializing the button module. 
  2.  */  
  3. static void buttons_init(void)  
  4. {  
  5.     // Configure HR_INC_BUTTON_PIN_NO and HR_DEC_BUTTON_PIN_NO as wake up buttons and also configure  
  6.     // for 'pull up' because the eval board does not have external pull up resistors connected to  
  7.     // the buttons.  
  8.     static app_button_cfg_t buttons[] =  
  9.     {  
  10.         {HR_INC_BUTTON_PIN_NO, false, BUTTON_PULL, button_event_handler},  
  11.         {HR_DEC_BUTTON_PIN_NO, false, BUTTON_PULL, button_event_handler}  // Note: This pin is also BONDMNGR_DELETE_BUTTON_PIN_NO  
  12.     };  
  13.       
  14.     APP_BUTTON_INIT(buttons, sizeof(buttons) / sizeof(buttons[0]), BUTTON_DETECTION_DELAY, false);  
  15. }  
当按下按钮时,处理程序是button_event_handler(),它处理心率计的加减模拟:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. /**@brief Function for handling button events. 
  2.  * 
  3.  * @param[in]   pin_no   The pin number of the button pressed. 
  4.  */  
  5. static void button_event_handler(uint8_t pin_no, uint8_t button_action)  
  6. {  
  7.     if (button_action == APP_BUTTON_PUSH)  
  8.     {  
  9.         switch (pin_no)  
  10.         {  
  11.             case HR_INC_BUTTON_PIN_NO:  
  12.                 // Increase Heart Rate measurement  
  13.                 m_cur_heart_rate += HEART_RATE_CHANGE;  
  14.                 if (m_cur_heart_rate > MAX_HEART_RATE)  
  15.                 {  
  16.                     m_cur_heart_rate = MIN_HEART_RATE; // Loop back  
  17.                 }  
  18.                 break;  
  19.                   
  20.             case HR_DEC_BUTTON_PIN_NO:  
  21.                 // Decrease Heart Rate measurement  
  22.                 m_cur_heart_rate -= HEART_RATE_CHANGE;  
  23.                 if (m_cur_heart_rate < MIN_HEART_RATE)  
  24.                 {  
  25.                     m_cur_heart_rate = MAX_HEART_RATE; // Loop back  
  26.                 }  
  27.                 break;  
  28.                   
  29.             default:  
  30.                 APP_ERROR_HANDLER(pin_no);  
  31.                 break;  
  32.         }  
  33.     }      
  34. }

这篇关于低功耗蓝牙4.0BLE编程-nrf51822开发(2)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1010617

相关文章

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

使用Python开发一个带EPUB转换功能的Markdown编辑器

《使用Python开发一个带EPUB转换功能的Markdown编辑器》Markdown因其简单易用和强大的格式支持,成为了写作者、开发者及内容创作者的首选格式,本文将通过Python开发一个Markd... 目录应用概览代码结构与核心组件1. 初始化与布局 (__init__)2. 工具栏 (setup_t

Spring Shell 命令行实现交互式Shell应用开发

《SpringShell命令行实现交互式Shell应用开发》本文主要介绍了SpringShell命令行实现交互式Shell应用开发,能够帮助开发者快速构建功能丰富的命令行应用程序,具有一定的参考价... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定义S

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.

Spring Security基于数据库的ABAC属性权限模型实战开发教程

《SpringSecurity基于数据库的ABAC属性权限模型实战开发教程》:本文主要介绍SpringSecurity基于数据库的ABAC属性权限模型实战开发教程,本文给大家介绍的非常详细,对大... 目录1. 前言2. 权限决策依据RBACABAC综合对比3. 数据库表结构说明4. 实战开始5. MyBA

使用Python开发一个简单的本地图片服务器

《使用Python开发一个简单的本地图片服务器》本文介绍了如何结合wxPython构建的图形用户界面GUI和Python内建的Web服务器功能,在本地网络中搭建一个私人的,即开即用的网页相册,文中的示... 目录项目目标核心技术栈代码深度解析完整代码工作流程主要功能与优势潜在改进与思考运行结果总结你是否曾经

揭秘Python Socket网络编程的7种硬核用法

《揭秘PythonSocket网络编程的7种硬核用法》Socket不仅能做聊天室,还能干一大堆硬核操作,这篇文章就带大家看看Python网络编程的7种超实用玩法,感兴趣的小伙伴可以跟随小编一起... 目录1.端口扫描器:探测开放端口2.简易 HTTP 服务器:10 秒搭个网页3.局域网游戏:多人联机对战4.

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优