Ymodem 協(xié)議固件升級(jí)(上)
Ymodem 協(xié)議作為 Xmodem 協(xié)議家族的增強(qiáng)版,是嵌入式設(shè)備通過串口(UART/RS232/RS485)實(shí)現(xiàn)固件升級(jí)的經(jīng)典方案,其核心定位是 “低成本、高可靠的串行數(shù)據(jù)傳輸協(xié)議”,尤其適用于無網(wǎng)絡(luò)模塊(如 ENC28J60)、資源極度受限(如 8 位 MCU、KB 級(jí) RAM)或工業(yè)場景中依賴串口通信的設(shè)備 —— 例如老舊工業(yè) PLC、STM32F103 嵌入式開發(fā)板、智能儀表等,這些設(shè)備往往無法支持 HTTP OTA 的網(wǎng)絡(luò)傳輸,卻可通過 Ymodem 利用現(xiàn)有串口鏈路完成固件更新。相較于 Xmodem 的單包 128 字節(jié)傳輸與簡單校驗(yàn),Ymodem 在保留 “幀結(jié)構(gòu)清晰、實(shí)現(xiàn)輕量化” 優(yōu)勢的基礎(chǔ)上,新增了文件信息預(yù)傳輸、1024 字節(jié)大數(shù)據(jù)包支持、雙向握手重傳機(jī)制及文件完整性校驗(yàn),既能提升大固件(如 512KB~2MB)的傳輸效率,又能通過多重校驗(yàn)避免固件傳輸錯(cuò)誤導(dǎo)致的設(shè)備 “變磚”,成為嵌入式領(lǐng)域串口升級(jí)的 “標(biāo)配協(xié)議”。
Ymodem 協(xié)議的幀結(jié)構(gòu)是其實(shí)現(xiàn)可靠傳輸?shù)幕A(chǔ),采用 “固定幀頭 + 可變數(shù)據(jù) + 強(qiáng)校驗(yàn)” 的設(shè)計(jì),確保每幀數(shù)據(jù)在串口噪聲干擾下仍能被正確解析。完整的 Ymodem 幀分為兩種核心類型:用于數(shù)據(jù)傳輸?shù)? “數(shù)據(jù)幀” 與用于控制交互的 “控制幀”。數(shù)據(jù)幀的結(jié)構(gòu)固定為 “起始符(1 字節(jié))+ 幀序號(hào)(1 字節(jié))+ 幀序號(hào)反碼(1 字節(jié))+ 數(shù)據(jù)段(128/1024 字節(jié))+ 校驗(yàn)字段(2 字節(jié) CRC16 或 4 字節(jié) CRC32)”:起始符(SOH=0x01)對(duì)應(yīng) 128 字節(jié)數(shù)據(jù)段,起始符(STX=0x02)對(duì)應(yīng) 1024 字節(jié)數(shù)據(jù)段,后者在大固件升級(jí)中可將傳輸幀數(shù)減少 75%,大幅提升效率;幀序號(hào)從 0x00 開始遞增,幀序號(hào)反碼為 0xFF 與幀序號(hào)的異或值(如幀序號(hào) 0x01 的反碼為 0xFE),用于接收端驗(yàn)證幀的連續(xù)性,避免幀丟失或亂序;校驗(yàn)字段默認(rèn)采用 CRC16-CCITT(多項(xiàng)式 0x1021),工業(yè)高干擾場景可升級(jí)為 CRC32,進(jìn)一步降低誤碼率??刂茙瑒t用于升級(jí)流程中的交互,包括 “文件信息幀(C 幀,0x43)”—— 用于傳輸固件文件名、大小等元數(shù)據(jù);“結(jié)束幀(EOT,0x04)”—— 標(biāo)識(shí)固件數(shù)據(jù)傳輸完成;“確認(rèn)幀(ACK,0x06)” 與 “否定幀(NACK,0x15)”—— 接收端對(duì)每幀數(shù)據(jù)的校驗(yàn)結(jié)果反饋,構(gòu)成雙向握手的基礎(chǔ)。
Ymodem 固件升級(jí)的完整流程需設(shè)備端 Bootloader 與上位機(jī)(如 PC 端串口助手、工業(yè)控制軟件)協(xié)同,遵循 “初始化 - 協(xié)商 - 數(shù)據(jù)傳輸 - 校驗(yàn) - 重啟” 的閉環(huán)邏輯,每個(gè)環(huán)節(jié)均設(shè)計(jì)容錯(cuò)機(jī)制以應(yīng)對(duì)串口傳輸風(fēng)險(xiǎn)。首先是 “升級(jí)初始化” 階段:設(shè)備上電后,Bootloader 先檢測升級(jí)觸發(fā)條件(如特定引腳電平拉低、串口收到 “進(jìn)入升級(jí)模式” 指令 0x11),若滿足則進(jìn)入 Ymodem 接收模式,初始化串口參數(shù)(波特率通常為 9600~115200bps,需與上位機(jī)一致,數(shù)據(jù)位 8、停止位 1、無校驗(yàn)),并向上位機(jī)發(fā)送 “等待文件信息” 信號(hào)(通常是連續(xù)的 NACK 幀,提示上位機(jī)開始傳輸);上位機(jī)收到信號(hào)后,準(zhǔn)備固件文件(需預(yù)先轉(zhuǎn)換為二進(jìn)制格式,避免文本編碼干擾),并構(gòu)建 Ymodem 文件信息幀(C 幀),包含固件名(如 “firmware_v1.2.bin”)、文件大?。ㄈ? “524288”,單位字節(jié))及 CRC 校驗(yàn)值,通過串口發(fā)送至設(shè)備端。
進(jìn)入 “文件信息協(xié)商” 階段后,設(shè)備端 Bootloader 接收 C 幀并解析固件元數(shù)據(jù),驗(yàn)證文件大小是否匹配 Flash 預(yù)留的應(yīng)用分區(qū)(如 STM32F103 的應(yīng)用分區(qū)大小為 512KB,需確保固件大小≤512KB),若匹配則發(fā)送 ACK 幀確認(rèn),上位機(jī)收到 ACK 后開始準(zhǔn)備數(shù)據(jù)分塊傳輸;若不匹配(如固件過大),則發(fā)送 NACK 幀并重啟協(xié)商流程,避免 Flash 寫入溢出。隨后進(jìn)入核心的 “數(shù)據(jù)分塊傳輸” 階段:上位機(jī)將固件按 1024 字節(jié)(或 128 字節(jié))分塊,為每塊構(gòu)建 Ymodem 數(shù)據(jù)幀(STX 幀),幀序號(hào)從 0x00 開始遞增,每發(fā)送一幀后等待設(shè)備端反饋;設(shè)備端 Bootloader 接收數(shù)據(jù)幀后,先驗(yàn)證幀序號(hào)連續(xù)性(如當(dāng)前應(yīng)接收 0x01 幀,若收到 0x03 幀則判定幀丟失),再計(jì)算數(shù)據(jù)段的 CRC 值與幀尾校驗(yàn)字段比對(duì),雙重驗(yàn)證通過后,將數(shù)據(jù)塊寫入 Flash 應(yīng)用分區(qū)(通過 STM32 的 HAL_FLASH_Program 接口實(shí)現(xiàn)),并發(fā)送 ACK 幀請(qǐng)求下一塊;若校驗(yàn)失敗或幀序號(hào)異常,設(shè)備端發(fā)送 NACK 幀,上位機(jī)收到后立即重傳該幀,重傳次數(shù)通常設(shè)為 3 次,若 3 次均失敗則終止升級(jí),避免無限重試導(dǎo)致的資源浪費(fèi)。





