iOS 蓝牙固件升级

升级介绍

蓝牙固件升级是使用手机给固件进行版本升级,以达到修复bug或者添加新功能的作用。升级的大概流程是:首先,当蓝牙固件需要升级时,由嵌入式开发人员提供新的固件,由服务器管理人员将固件放到服务器上,此时,用户打开手机APP的时候会检测到服务器有更新,请求最新的蓝牙固件,确认更新后,手机会从服务器下载固件。下载完毕后,APP会读取固件内容,并根据升级协议将内容传到蓝牙固件里,完成升级。

DFU = Device Firmware Update (设备固件更新)

OTA = Over The Air (空中升级)

升级流程

各个蓝牙设备不尽相同,以下是我测试设备的升级流程:升级固件为.bin后缀的文件,文件名会有一定的格式。

我司的硬件工程师一般发送给我们的升级软件是hex文件,然后升级的时候需要的是bin文件这就需要我们把hex文件转换为bin文件。转换bin文件有两种方式:

1.通过软件的方式,软件的方式我是通过软件J-Flash来转换的。具体操作流程如下:

(1)打开J-Flash选择Create a new project。

(2)把hex文件拖入J-Flash 。

(3)找到hex文件对应的结束的最后一位的位置。

(4)选择Save data file as 保存类型选bin类型,然后点击保存之后弹出Enter address range框 start address 保持不变,End address 输入你想要转换文件的结束地址,然后点击OK hex转bin文件转换成功。

2.通过到代码的方式。转换的代码如下:

//

//  LBHexToBin.m

//

//  Created by lingbing on 2020/9/18.

//

#import

NS_ASSUME_NONNULL_BEGIN

@interface LBHexToBin : NSObject

+ (NSData*)convert:(NSData*)hex;

@end

NS_ASSUME_NONNULL_END

//

//  LBHexToBin.m

//

//  Created by lingbing on 2020/9/18.

//

#import "LBHexToBin.h"

@implementation LBHexToBin

+ (constByte)ascii2char:(constByte*)ascii

{

    if(*ascii >='A')

        return*ascii -0x37;


    if(*ascii >='0')

        return*ascii -'0';

    return-1;

}

+ (constByte)readByte:(constByte*)pointer

{

    Bytefirst = [LBHexToBinascii2char:pointer];

    Bytesecond = [LBHexToBinascii2char:pointer +1];


    return(first <<4) | second;

}

+ (constUInt16)readAddress:(constByte*)pointer

{

    Bytemsb = [LBHexToBinreadByte:pointer];

    Bytelsb = [LBHexToBinreadByte:pointer +2];


    return(msb <<8) | lsb;

}

+ (NSUInteger)calculateBinLength:(NSData*)hex

{

    if(hex ==nil|| hex.length==0)

    {

        return0;

    }


    NSUIntegerbinLength =0;

    constNSUIntegerhexLength = hex.length;

    constByte* pointer = (constByte*)hex.bytes;

    UInt32lastBaseAddress =0;


    do

    {

        constBytesemicollon = *pointer++;


        // Validate - each line of the file must have a semicollon as a firs char

        if(semicollon !=':')

        {

            return0;

        }


        constUInt8reclen = [LBHexToBinreadByte:pointer]; pointer +=2;

        constUInt16offset = [LBHexToBinreadAddress:pointer]; pointer +=4;

        constUInt8rectype = [LBHexToBinreadByte:pointer]; pointer +=2;


        switch(rectype) {

            case0x04: {

                // Only consistent hex files are supported. If there is a jump to non-following ULBA address skip the rest of the file

                constUInt32newULBA = [LBHexToBinreadAddress:pointer];

                if(binLength >0&& newULBA != (lastBaseAddress >>16) +1)

                    returnbinLength;

                lastBaseAddress = newULBA <<16;

                break;

            }

            case0x02: {

                // The same with Extended Segment Address. The calculated ULBA must not be greater than the last one + 1

                constUInt32newSBA = [LBHexToBinreadAddress:pointer] <<4;

                if(binLength >0&& (newSBA >>16) != (lastBaseAddress >>16) +1)

                    returnbinLength;

                lastBaseAddress = newSBA;

                break;

            }

            case0x00:

                // If record type is Data Record (rectype = 0), add it's length (only it the address is >= 0x1000, MBR is skipped)

                if(lastBaseAddress + offset >=0x1000)

                    binLength += reclen;

            default:

                break;

        }


        pointer += (reclen <<1);  // Skip the data when calculating length

        pointer +=2;  // Skip the checksum

        // Skip new line

        if(*pointer =='\r') pointer++;

        if(*pointer =='\n') pointer++;

    }while(pointer != hex.bytes+ hexLength);


    returnbinLength;

}

+ (NSData*)convert:(NSData*)hex

{

    constNSUIntegerbinLength = [LBHexToBincalculateBinLength:hex];

    constNSUIntegerhexLength = hex.length;

    constByte* pointer = (constByte*)hex.bytes;

    NSUIntegerbytesCopied =0;

    UInt32lastBaseAddress =0;


    Byte* bytes =malloc(sizeof(Byte) * binLength);

    Byte* output = bytes;


    do

    {

        constBytesemicollon = *pointer++;


        // Validate - each line of the file must have a semicollon as a firs char

        if(semicollon !=':')

        {

            free(bytes);

            returnnil;

        }


        constUInt8reclen = [LBHexToBinreadByte:pointer]; pointer +=2;

        constUInt16offset = [LBHexToBinreadAddress:pointer]; pointer +=4;

        constUInt8rectype = [LBHexToBinreadByte:pointer]; pointer +=2;


        switch(rectype) {

            case0x04: {

                constUInt32newULBA = [LBHexToBinreadAddress:pointer]; pointer +=4;

                if(bytesCopied >0&& newULBA != (lastBaseAddress >>16) +1)

                    return[NSDatadataWithBytesNoCopy:byteslength:bytesCopied];

                lastBaseAddress = newULBA <<16;

                break;

            }

            case0x02: {

                constUInt32newSBA = [LBHexToBinreadAddress:pointer] <<4; pointer +=4;

                if(bytesCopied >0&& (newSBA >>16) != (lastBaseAddress >>16) +1)

                    return[NSDatadataWithBytesNoCopy:byteslength:bytesCopied];

                lastBaseAddress = newSBA;

                break;

            }

            case0x00:

                // If record type is Data Record (rectype = 0), copy data to output buffer

                // Skip data below 0x1000 address (MBR)

                if(lastBaseAddress + offset >=0x1000)

                {

                    for(inti =0; i < reclen; i++)

                    {

                        *output++ = [LBHexToBinreadByte:pointer]; pointer +=2;

                        bytesCopied++;

                    }

                }

                else

                {

                    pointer += (reclen <<1);  // Skip the data

                }

                break;

            default:

                pointer += (reclen <<1);  // Skip the irrelevant data

                break;

        }


        pointer +=2;  // Skip the checksum

        // Skip new line

        if(*pointer =='\r') pointer++;

        if(*pointer =='\n') pointer++;

    }while(pointer != hex.bytes+ hexLength);


    return[NSDatadataWithBytesNoCopy:byteslength:bytesCopied];

}

@end

升级过程:

1.扫描并连接要升级的蓝牙设备

2.发送给外设升级前的验证指令,验证是否是合法升级

3.外设验证返回成功之后,发送授权升级指令,指令返回成功之后开始固件升级

3.判断随机数无误,准备发送打包好的数据

4.真正发送打包好的数据(每次发送10包,一包20个字节),这里会重复N多次,看你的原数据包有多大;每次接到我发的包后,外设都会给我会OK否,我收到OK后才会发一下个数据包

5.告诉外设我数据发送完毕,并发送一段指令(包括本次空中升级数据包的大小,还有加密参数什么的)

6.外设给我回OK无误后,才算真正升级完成

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容