1. 背景

客户的项目是无线控制灯具,目前采用2.4G芯片,一端是2.4G遥控器,一端是2.4G灯具。现在客户的需求是在不增加成本的条件下增加手机APP控制。因为BLE芯片一般会比纯2.4G芯片价格高,所以客户不想用BLE芯片替代掉2.4G芯片,毕竟省下的钱都进了客户自己的腰包。

2. 项目评估

BLE和2.4G都工作在2.4GHz频段,所以让他们互相通信在物理层上理论是可行的。在物理层之上只要手机BLE发送的数据包能被2.4G芯片解析就可以达到目的,所以本项目的技术点就转换到手机BLE模拟2.4G的数据包发送数据。

3. 技术实现

频移键控(FSK)接收机的一个特点:其接收连续相同比特的能力很差,当接收机收到一连串的“0000000000”时,会认为发射机的频率向左移了,进而导致频率失锁,以致导致数据接收失败。所以为了避免数据传输中出现一连串的全0或者全1,BLE设备会对数据做白化处理。

白化这个词,可能在深度学习领域比较常遇到,挺起来就是高大上的名词,然而其实白化是一个比PCA稍微高级一点的算法而已,所以如果熟悉PCA,那么其实会发现这是一个非常简单的算法。白化的目的是去除输入数据的冗余信息。假设训练数据是图像,由于图像中相邻像素之间具有很强的相关性,所以用于训练时输入是冗余的;白化的目的就是降低输入的冗余性。
输入数据集X,经过白化处理后,新的数据X'满足两个性质:
(1)特征之间相关性较低;
(2)所有特征具有相同的方差。
    其实我们之前学的PCA算法中,可能PCA给我们的印象是一般用于降维操作。然而其实PCA如果不降维,而是仅仅使用PCA求出特征向量,然后把数据X映射到新的特征空间,这样的一个映射过程,其实就是满足了我们白化的第一个性质:除去特征之间的相关性。因此白化算法的实现过程,第一步操作就是PCA,求出新特征空间中X的新坐标,然后再对新的坐标进行方差归一化操作。

然而手机端的白化处理是无法直接通过软件接口关掉的,所以在发送时就需要做反白化处理。另外,2.4G芯片接收机的特点是:从空中抓取2.4GHz信道的数据包,逐BIT对比,硬件过滤掉不是自己想要的数据包(不是只对比数据包头)。

手机端APP的实现原理如下:

文档中包头部分(Header)是指必须在手机端APP里添加的头,并非标准BLE的包头。
Packet Format:
                                                                |---------------------------------------------------------------------------|
|---------------------------------------------------------------|----------------------------------------|   |---------------------------|  |
|                                                               |                                        |   |                           |  |
| |---------------|   |-----------------|    |---------------|  | |--------------|   |-----------------| |   |  |---------------------|  |  |
| | Pdu_head(2B)  | + | Mac_address(6B) | +  | Menu_head(4B) | +| | Preamble(3B) | + | Dev_address(4B) | | + |  | Payload(Max is 18B) |  |  |
| |---------------|   |-----------------|    |---------------|  | |--------------|   |-----------------| |   |  |---------------------|  |  |
|                                                               |                                        |   |                           |  |
|----------------------------------------Header Format----------|----------------------------------------|   |------.4G Control---------|  |
                                                                |                                                                           |
                                                                |-----------------------.4G Packet Format----------------------------------|

Pdu_head:   标准BLE数据包的PDU head(BLE协议规定),此值可通过Android API设置。
Mac_address: BLE设备的MAC 地址(BLE协议规定),此字段可通过Android API设置。
Menu_head:   Menufacture data head(BLE协议规定), 此字段占4个字节。目前设置为1E FF F0 FF,此值是通过Android API设置的。
Preamble:    引导码,此字段占3个字节。此字段必须设置为2.4G接收芯片端规定的值,.4G接收芯片端规定此值为0x710F55,Android端发送值见下。
Dev_address:设备地址,此字段占4个字节。此字段也必须和2.4G接收端确认。
Payload:     有效数据,此字段最大为18个字节。用于2.4G应用控制数据包,此字段用于应用控制协议,若应用控制协议大于18Byte需另做说明。
//以下代码可以直接编译、运行,然后直接查看最后结果
#include <stdio.h>
/**数据准备:
定义一个37字节的数组(标准BLE广播数据包最大的payload为37字节),BLE数据包LSB发送在前。
根据Header Format结构,可得到data[11] - data[13]为Preamble字段,data[14] - data[17]为Dev_address字段。
而且Preamble和Dev_address必须进行比特的反转。
Example:
2.4G接收端Preamble为:    0x71  0x0F  0x55
                            |    |      |
那么,对应Android端为:   0x8E  0xF0   0xAA

2.4G接收端Dev_address为:    0xC0   0xC1   0xC2   0xC3
                               |      |      |      |
那么,对应Android端为:      0x03   0x83   0x43   0xC3
*/
] = {,,,,,,,,,,,,};
] = {};
;

void whitening_init(int channel_index)
{
    ;
    whitening_reg[] = ;

    ; i < ; i++)
    {
        whitening_reg[i] = (channel_index >> ( - i)) & 0x01;
    }
}

int whitening_output(void)
{
    ] ^ whitening_reg[];

    whitening_reg[] = whitening_reg[];
    whitening_reg[] = whitening_reg[];
    whitening_reg[] = whitening_reg[];
    whitening_reg[] = whitening_reg[];
    whitening_reg[] = whitening_reg[];
    whitening_reg[] = whitening_reg[];
    whitening_reg[] = temp;

    ];
}

int whitening_decode(int *data, int length)
{
    ;

    ; data_index < length; data_index++)
    {
        int data_input = data[data_index];
        ;
        ;

        ; bit_index < ; bit_index++)
        {
            data_bit = (data_input >> (bit_index)) & 0x01;

            data_bit ^= whitening_output();

            data_output += (data_bit << (bit_index));
        }

        data_re = data_output;
        //此处可以得到最后转换完成的数据
         && data_index < )
        {
            printf("Result == %x\n", data_re);
        }
    }

   return data_re;
}

int main(int argc, char * argv[])
{
   ;
   //37代表传输在2402频点, 38代表传输在2426频点, 39代表传输在2480频点, 参数只能为这3个值其中一个,需和2.4G接收端协调一致。
   whitening_init();
   //此处可以对data数组index=20以后赋值,赋值即为2.4G控制协议字段
   //whitening_decode(*,*)函数第二个参数代表要转换的数组个数
   result =  whitening_decode(data, );

    getchar();
    ;
}

该实现方式缺点:

1. 无法实现跳频,如果BLE的37/38/39某个频点严重拥堵的话可能会造成接受成功率低

2. 以上主要适用于Android手机版本BLE4.0/4.1/4.2,BLE5.0新增加了37个信道用于广播。

BLE和2.4G实现通信的更多相关文章

  1. ESP32 BLE蓝牙 微信小程序通信发送大于20字符数据

    由于微信小程序只支持BLE每次发送数据不大于20个字节,ESP32则有经典蓝牙.低功耗蓝牙两种模式. 要解决发送数据大于20个字节的问题,最简单实用的方式就是分包发送.如下图所示: 1.什么起始字符和 ...

  2. 关于Ble通信库BluetoothKit的使用 以及可能出现的问题分析

    首先,这个库是用于BLE(低功耗蓝牙)通信的,地址:https://github.com/dingjikerbo/BluetoothKit 当然,也可以选择根据andorid提供的底层接口自己完成这部 ...

  3. [nRF51822] 13、浅谈nRF51822和NRF24LE1/NRF24LU1/NRF24L01经典2.4G模块无线通信配置与流程

    前言:  nRF51可以支持基于2.4G的互相通信.与NRF24LE1的通信.与NRF24LU1的通信.与NRF24L01的通信. 一.nRF51822基于2.4G和nRF51822通信 其中nRF5 ...

  4. 【转】TI蓝牙BLE 协议栈代码学习

    BLE就是低功率蓝牙.要着重了解两种设备: dual-mode双模设备:简单说就是向下兼容. single-mode单模设备:仅仅支持BLE.   关于开发主要讲的是单模设备,它可以只靠纽扣电池即可持 ...

  5. 【树莓派】树莓派使用4G模块上网

    想了解一下树莓派通过4G网络模块通信如何实现,看到这篇文章(http://www.lxway.com/95811506.htm),准备接下来有机会实践一下,先留存学习: 一.4G Luci配置 1. ...

  6. nrf2401 - 最廉价的2.4G无线通信方案

    所有的使用Arduino 的朋友大多都会知道大名鼎鼎的XBee 这个土豪级的ZigBee 的通信模块.我们是做产品开发的,对于XBee这个产品可谓是又爱又恨,不得不承认他确实是一个好货,从做工到功能都 ...

  7. 蓝牙协议分析(5)_BLE广播通信相关的技术分析

    1. 前言 大家都知道,相比传统蓝牙,蓝牙低功耗(BLE)最大的突破就是加大了对广播通信(Advertising)的支持和利用.关于广播通信,通过“玩转BLE(1)_Eddystone beacon” ...

  8. 4G通信模块在ARM平台下的应用

    4G模块是连接物与物的重要载体,是终端设备接入物联网的核心部件之一.随着4G的普及,许多新兴市场对4G通信模块的需求都在日益扩大,那么在ARM平台的嵌入式设备上如何快速的应用4G模块呢? 4G通信模块 ...

  9. 4G 通信模块在ARM 平台下的应用

    收藏 评论(0) 分享到 微博 QQ 微信 LinkedIn 4G模块是连接物与物的重要载体,是终端设备接入物联网的核心部件之一,随着4G的普及,许多新兴市场对4G通信模块的需求都在日益扩大,那么在A ...

随机推荐

  1. Docker run 命令

    docker run -d -p 8084:80 --name weather --restart always --link fme-postgis 192.168.1.220:5000/weath ...

  2. 反射 day1

    1.Object objval = rs.getObject(fieldName);//如果数据库值为空时的返回值java中的null //这个如果是oracle数值型的数据的会返回的可能类型有很多, ...

  3. 获取百度地图POI数据三(模拟关键词搜索)

    上一篇博文中讲到如何获取用于搜索的关键词,并且已经准备好了一百五十万的关键词   这其中有门牌号码,餐馆酒店名称,公司名称,道路名称等.有了这些数据,我们就可以通过代码,模拟我们在百度地图的搜索框中搜 ...

  4. java基础知识—运算符和基本选择结构

    1.保存真假,使用boolean变量 boolean有两个值:true 真 false 假 2.从控制台接受输入信息,通过创建扫描器 Sacnner input=new Sacnner(System. ...

  5. 编写一个求和函数sum,使输入sum(2)(3)或输入sum(2,3),输出结果都为5

    昨天的笔试题,做的一塌糊涂,题目考的都很基础而且很细,手写代码对我来说是硬伤啊.其中有一道是这个,然而看到题目的时候,根本没有想到arguments:然后现在就恶补一下. arguments:用在函数 ...

  6. MyBatis最原始的实现curd的操作

    关于jdbc的缺点: 1.数据库链接创建释放频繁造成系统资源浪费从而影响系统性能.如果使用数据库连接池可以解决此问题. 2.sql语句在代码中硬编码,不利于维护,sql变动需要改变java代码 3.使 ...

  7. The eventual following stack trace is caused by an error thrown for debugging purposes as well as to attempt to terminate the thread which caused the illegal access, and has no functional impact.

    好久没有冒泡了,最近在新环境上搭建应用时,启动报错: INFO: Illegal access: this web application instance has been stopped alre ...

  8. CSS内嵌样式实现打字效果

    <style> *{margin:0;padding:0;} @keyframes typing { from { width: 0; } } @keyframes blink-caret ...

  9. C++入门程序作业2

    程序在Dev-C++5.5.3版本运行 结构体的使用 给结构体赋值,打印出结构体中学生姓名,分数,平均分 #include <iostream>#include <cassert&g ...

  10. SHELL输出带颜色字体

    输出特效格式控制:\033[0m  关闭所有属性  \033[1m   设置高亮度  \03[4m   下划线  \033[5m   闪烁  \033[7m   反显  \033[8m   消隐  \ ...