建议看该教程前,先看一下  简单扫描器实现  教程

讲解基于sdk目录下central中的两个例子。

关于主机的程序框架其实和从机都是一样的,都是基于事件驱动的框架。 Main函数中完成初始化, 从机的话就启动广播,主机就启动扫描。 之后都是基于事件驱动的方式将ble相关事件返回给上层app,并且再通过ble_evt_dispatch函数将事件分发给各个处理函数。如下图所示:


蓝牙的使用无非就是 进行数据的通信,从机通常都是作为服务端(一次数据交互中数据的提供者),而主机通常都是作为客户端(一次数据交互中数据的使用者)。(并不是固定的,从机也可以作为客户端,主机也可以作为服务端。 从机和主机的概念是针对链路层来说的,而客户端和服务端是针对GATT层来说的。)

之前的教程中 从机例子中 都是作为服务端的,所以我们发送数据给手机时都是使用notify方式或者indication方式并通过sd_ble_gatts_hvx函数发送。

而这里主机例子ble_app_multilink_central,是作为客户端的,客户端不仅可以发数据给服务端,也可以从服务端读取数据。 所以 有 读写函数,如下图

先说明sdk中主从demo的现象,分别烧录其中的主从例子在两个开发板中,打开主机和从机便会自动连接,连接上后 按从机的button0,主机上就会根据连接时的顺序翻转对应灯的状态。

比如你有三个开发板,一个主机两个从机,当你全部打开时,假设A设备先连接上,B设备后连接上。那么你按A的button0按键时,主机的第一个灯就会状态反转,你按B的button0时,主机就会反转第二个灯的状态。

Sdk中的demo主机时可以连接8个从机的,不过因为板子的led有限你只能看到四个从机连接后 按键时 主机的灯会对应亮。

PS:demo中的例子是针对PCA10028板子的,如果你的板子的按键,LED灯不兼容这个板子的话对应你的板子改下就行了。

下面来分析sdk中的主从通信demo是如何建立连接并通信的。解决一下几个问题:

1:主机怎么判断哪个从机是我要连接的那个设备?

2:如何实现连接多个设备?

3:因为连接上后 从机上按键时会通过notify发送数据给主机,所以主机需要先 使能从机上的notify功能,具体就是写从机上的 CCCD(客户端特性配置描述       符) 描述符。该功能如何实现?

4:连接上后如何通信?

下面首先解决第一个问题:主机怎么判断哪个从机是我要连接的那个设备。

 

在从机工程ble_app_multilink_peripheral中,看以看到其设置了自己的设备名为Multilink。


在主机工程ble_app_multilink_central 当初始化完成后就会调用 scan_start() 开始监听BLE广播,每当监听到BLE广播时协议栈就会给上层一个 广播事件, 该事件由ble_evt_dispatch派发函数递交给on_ble_evt函数。

在on_ble_evt中会对提取广播数据中的设备名字,并且判断设备是不是Multilink,如果是就启动停止扫描,并且连接改设备,如下图所示

所以 主机判断广播设备是不是自己想要连接的就是通过 名字来判断的。

第二个问题:如何实现可以连接多个设备

第一个问题中找到了如何识别广播设备是不是自己想连的方式,当找到了名字为Multilink设备后就关闭扫描侦听而发起连接,也就是这个时候主机已经不能再监听别的ble设备的广播了,自然也就不能再和别的设备发起连接了(因为没收到广播不知道对方设备地址),

因为demo中主机可以连接8个从机,那么如果能够继续 启动扫描侦听,并且周围还有别的广播设备,并且其名字也是Multilink,那么收到该广播时,就能够在广播事件处理函数再次对这第二个设备发起连接。

所以主机例子中虽然在找到了一个设备名为Multilink的设备后会关闭扫描并且发起连接,但是当这个连接完成后在收到连接事件时 会做当前连接的设备是否已经大于最大可连接数,不过没有超过那么就会再次开启广播。 这样就可以再次链接其他设备名为Multilink的设备了。如下图所示。

PS:这里的 device_manager_event_handler函数并不是由派发函数ble_evt_dispatch调用的,而是派发函数ble_evt_dispatch先调用了dm_ble_evt_handler(p_ble_evt),而该函数内部会调用device_manager_event_handler

函数关系链如下:main中首先调用了device_manager_init()初始化函数,该初始化函数会将device_manager_event_handler函数设置为回调函数。

当连接完成后ble_evt_dispatch函数将 连接事件 派发给dm_ble_evt_handler 函数,改函数中判断如果当前收到的事件是 连接事件就内部 设定了一个DM_EVT_CONNECTION事件然后调用之前注册的device_manager_event_handler这个回调函数,来最终处理关于连接事件的后续工作,也就是上面提到的,判断是否要继续开启广播。

nrf51822-主从通信分析1的更多相关文章

  1. MINIX3 进程通信分析

    MINIX3 进程通信分析 6.1MINIX3 进程通信概要 MINIX3 的进程通信是 MINIX3 内核部分最重要的一个部件,我个人认为其实这 是内核中的“内核”,怎么来理解这个概念呢?其实 MI ...

  2. STM32F10x_硬件I2C主从通信(轮询发送,中断接收)

    Ⅰ.写在前面 关注我分享文章的朋友应该知道我在前面讲述过(软件.硬件)I2C主机控制从机EEPROM的例子.在I2C通信主机控制程序是比较常见的一种,可以说在实际项目中,很多应用都会使用到I2C通信. ...

  3. 有关RDS上只读实例延时分析-同适用于自建MySQL主从延时分析判断

    个人不是很喜欢在技术上跟人互喷,尤其是不在同一个岗位上的人.一方面本人的性格如此,另一方面,我自身的口水也确实是不行,人生经历了第一次的双11洗礼,在大促的环境下,总算知道了有些东西是否应该规避,这次 ...

  4. nrf51822-主从通信分析2

    解决第三个问题:如何使能从机上的特征值的 notify功能,使其能通过notify方式发送数据   使能从机的notify功能是通过写0x0001到从机的那个具有notify功能的特征值的CCCD描述 ...

  5. S7-300之间的PROFIBUS-DP主从通信

    一.PROFIBUS-DP简介 1.由来 2..总线连接器 二.系统结构示例 三.组态过程示例 三,DP网络组态 1.新建一个项目和两个300站点如下 2.组态从站DP网络 1)点击常规中的属性 2) ...

  6. nrf51822 RAM不足分析

    之前了解过STM32 的内存分配问题,对于蓝牙芯片51822的内存分配问题把项目中,遇到了.bss和.data部分超了的问题,这其实就是声明的变量和stask 及 heap的大小总和超出了单片机的RA ...

  7. 基于STM32Cube的IIC主从通信

    1.建立STM32Cube工程,我使用的芯片是STM32F429,为了更简单的建立单独的IIC通信任务,我使用了FREERTOS,另外选择了RCC(系统时钟初始化),I2C1,I2C2(为了验证代码的 ...

  8. mycat 主从切换分析过程

    67 68互为主从 66为67从 区分双主写的数据,设置不同的自增id 67: SET @@auto_increment_offset=2;SET @@auto_increment_increment ...

  9. Redis主从同步分析(转)

    一.Redis主从同步原理 1.1 Redis主从同步的过程 配置好slave服务器连接的master后,slave会建立和master的连接,然后发送sync命令.无论是第一次同步建立的连接还是连接 ...

随机推荐

  1. ls命令

    ls(list) 命令可以说是Linux下最常用的命令之一 #ls -l;列出文件的详细信息 #ll 以上两个命令一样,ll是ls -l的简写 #ls -al;列出目录下的所有文件,包括以 . 开头的 ...

  2. js:数据结构笔记7--哈希表

    哈希表(散列表):通过哈希函数将键值映射为一个字典; 哈希函数:依赖键值的数据类型来构建一个哈希函数: 一个基本的哈希表:(按字符串计算键值) function HashTable() { this. ...

  3. POJ1204 Word Puzzles(AC自动机)

    给一个L*C字符矩阵和W个字符串,问那些字符串出现在矩阵的位置,横竖斜八个向. 就是个多模式匹配的问题,直接AC自动机搞了,枚举字符矩阵八个方向的所有字符串构成主串,然后在W个模式串构造的AC自动机上 ...

  4. 【wikioi】1191 数轴染色(线段树+水题)

    http://wikioi.com/problem/1191/ 太水的线段树了,敲了10分钟就敲完了,但是听说还有一种并查集的做法?不明觉厉. #include <cstdio> #inc ...

  5. linux中shell变量$#,$@,$*,$?,$$,$!,$_,$0,$1,$2的含义解释

    变量说明:    $$    Shell本身的PID(ProcessID)    $!    Shell最后运行的后台Process的PID    $?    最后运行的命令的结束代码(返回值)    ...

  6. Servlet的生命周期,并说出Servlet和CGI的区别,Servlet与JSP的区别

    一.Servlet 生命周期 1.加载 2.实例化 3.初始化 4.处理请求 5.销毁 二.Servlet与cgi的区别: Servlet处于服务器进程中,它通过多线程方式运行其service方法,一 ...

  7. Linux 高级网络编程

    设置套接字函数: #include<sys/socket.h> int setsockopt(int sockfd, int level, int optname, const void* ...

  8. #undef

    #undef 是在后面取消以前定义的宏定义 该指令的形式为 #undef 标识符 其中,标识符是一个宏名称.如果标识符当前没有被定义成一个宏名称,那么就会忽略该指令. 一旦定义预处理器标识符,它将保持 ...

  9. WebUI-自用

    一.常用插件 1.弹出框架: http://layer.layui.com/ http://layer.layui.com/mobile/ 移动端 http://aui.github.io/artDi ...

  10. IOS 今天学到太多的知识了,赶快记录下来

    TabBarController 修改tabbar的背景颜色和选中时候的颜色: func application(application: UIApplication, didFinishLaunch ...