自己用C語言寫NXP S32K116 serial bootloader

来源:https://www.cnblogs.com/geekygeek/archive/2020/02/25/hyperbootloader_s32k116.html
-Advertisement-
Play Games

HyperBoot_S32K116 is a UART bootloader developed for my S32K116 EVB hardware. so that the application can be reflashed by UART. ...


瞭解更多關於bootloader 的C語言實現,請加我QQ: 1273623966 (驗證信息請填 bootloader),歡迎咨詢或定製bootloader(線上升級程式)。

 

  到目前為止,“自己用C語言寫 xxx serial bootloader"已經有7篇博文了,7篇博文,7款不同的MCU。今天給大家介紹第8款MCU的串口bootloader, 也就是NXP S32K116 serial boot-loader。 NXP S32K116 是ARM Cortex-M0 內核的32-bit MCU。 有豐富的外設,卓越的性能以及成熟的工具鏈。Processor Expert更是非常不錯,我的這個bootloader 的底層驅動都是用Processor Expert自動生成的。我只需要寫中間層和bootloader應用層的代碼。看完S32Kxxx datasheet和S32Kxxx reference manual,才開始寫bootloader,整個實現過程除了"bootloader jumping to application"卡了一段時間,一切都非常順利。Processor Expert確實可以節省不少時間。但NXP S32Kxxx 開發環境也有不足的地方,比如IDE S32DS (S32 design studio) 非常慢,電腦配置不高的話容易卡死。還有就是S32DS 不支持simulator debug。 一定要有硬體板才可以debug.

  Bootloader 是獨立的一個程式,和Application分別存儲在ROM中不同的區間,不能有重疊。我的S32K116 bootloader 是放置在頭部,區間範圍為:0x00000000~0x00003FFF. Application的區間範圍為:0x00004000~0x0001FFFF. 為此分別對Bootloader 和Application的linker script做了一下改動。

Bootloader linker script 的改動如下:

MEMORY
{
  /* Flash */
  m_interrupts          (RX)  : ORIGIN = 0x00000000, LENGTH = 0x000000C0
  m_flash_config        (RX)  : ORIGIN = 0x00000400, LENGTH = 0x00000010
  m_text                (RX)  : ORIGIN = 0x00000410, LENGTH = 0x00003BF0

  /* SRAM_L */

  /* SRAM_U */
  m_data                (RW)  : ORIGIN = 0x20000000, LENGTH = 0x000020C0
  m_data_2              (RW)  : ORIGIN = 0x200020C0, LENGTH = 0x00001740
}

Application linker script 的改動如下:

MEMORY
{
  /* Flash */
  m_interrupts          (RX)  : ORIGIN = 0x00004000, LENGTH = 0x000000C0
  m_flash_config        (RX)  : ORIGIN = 0x00004400, LENGTH = 0x00000010
  m_text                (RX)  : ORIGIN = 0x00004410, LENGTH = 0x0001BBF0

  /* SRAM_L */
  m_custom              (RW)  : ORIGIN = 0x1FFFFC00, LENGTH = 0x00000400
  /* SRAM_U */
  m_data                (RW)  : ORIGIN = 0x20000000, LENGTH = 0x000020C0
  m_data_2              (RW)  : ORIGIN = 0x200020C0, LENGTH = 0x00001740
}

  Bootloader實現Application的更新需要上位機的協助,上位機一般是PC端的host 軟體工具。我的S32K116 bootloader 使用的上位機是HyperTerminal.  HyperTerminal 可以建立串口連接,傳輸Application 的Hex文件。並可以配置傳輸方式為每發送一行delay 50ms, 這樣可以預留時間讓bootloader處理數據完成燒寫。使用HyperTerminal可以省掉開發專門的上位機時間,但是由於傳送方式是純文本發送,沒有使用協議,沒有應答機制。某種意義上講是不可靠的。並且速度慢,一般不可以用於量產產品,只能用於學習或內部人員使用。

  Bootloader程式是MCU的程式,MCU一上電就進入Bootloader。Bootloader程式先完成CLOCK, PIN, UART 初始化,然後就運行一個Bootloader狀態機,狀態機如下所示:

switch (bootState)
        {
            case BOOT_HANDSHAKE:
                mbootStatus = M_Bootloader_Handshake();
                if (mbootStatus == BT_OK)
                {
                    bootState = BOOT_INIT;
                }
                else if (mbootStatus == BT_HS_TIMEOUT)
                {
                    mbootStatus = BT_DONE;
                    bootState = BOOT_JUMPTO_APP;
                }
                break;
            case BOOT_INIT:
                LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootInitMsg,strlen((char*)bootInitMsg));
                mbootStatus = M_Bootloader_Init();
                if (mbootStatus == BT_OK)
                {
                    LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootJobDoneMsg,strlen((char*)bootJobDoneMsg));
                    bootState = BOOT_ERASE;
                }
                break;
            case BOOT_ERASE:
                LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootEraseMsg,strlen((char*)bootEraseMsg));
                mbootStatus = M_Bootloader_Erase();
                if (mbootStatus == BT_OK)
                {
                    LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootJobDoneMsg,strlen((char*)bootJobDoneMsg));
                    bootState = BOOT_RECEIVE;
                    LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootPrgmMsg,strlen((char*)bootPrgmMsg));
                }
                break;
            case BOOT_RECEIVE:
                mbootStatus = M_Bootloader_Receive();
                if (mbootStatus == BT_OK)
                {
                    LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootFbHdrMsg,strlen((char*)bootFbHdrMsg));
                    LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)lineRcdBuf,LINE_RECORD_BUF_SIZE);
                    bootState = BOOT_PROGRAM;
                }
                break;
            case BOOT_PROGRAM:
                mbootStatus = M_Bootloader_Write();
                if (mbootStatus == BT_BUSY)
                {
                    bootState = BOOT_RECEIVE;
                }
                else if (mbootStatus == BT_OK)
                {
                    bootState = BOOT_PREJUMP;
                }
                break;
            case BOOT_PREJUMP:
                mbootStatus = Prejump_To_Application();
                if (mbootStatus == BT_OK)
                {
                    bootState = BOOT_JUMPTO_APP;
                    LPUART_DRV_SendDataPolling(INST_LPUART1,(uint8_t*)bootPrgmDoneMsg,strlen((char*)bootPrgmDoneMsg));
                }
                break;
            case BOOT_JUMPTO_APP:
                mbootStatus = BT_DONE;
                //LPUART_DRV_Deinit(INST_LPUART1);
                //Jump_To_Application(*((uint32_t*)APP_START_ADDRESS),*((uint32_t*)APP_JUMP_ADDRESS));
                break;
            default:
                break;
        }

總共7個狀態:BOOT_HANDSHAKE,BOOT_INIT,BOOT_ERASE,BOOT_RECEIVE,BOOT_PROGRAM,BOOT_PREJUMP,BOOT_JUMPTO_APP。
BOOT_HANDSHAKE:初始狀態,數秒6秒,收到更新請求就切換狀態為BOOT_INIT,超時就切換狀態為BOOT_JUMPTO_APP,出錯就重啟。                  BOOT_INIT: 初始化flash, 成功就切換狀態為BOOT_ERASE,出錯就重啟。                                                    BOOT_ERASE:擦除flash的Application區間,成功就切換狀態為BOOT_RECEIVE,出錯就重啟。                                      BOOT_RECEIVE:接收Hex數據,每成功接收一行數據,就切換狀態為BOOT_PROGRAM,出錯就重啟。                                                                                  BOOT_PROGRAM:解析數據,數據ready就完成燒寫,如果數據是最後一行數據,就切換狀態為BOOT_PREJUMP,否則切回BOOT_RECEIVE,出錯就重啟。          BOOT_PREJUMP: 檢查數據並處理未處理的數據,成功則切換狀態為BOOT_JUMPTO_APP,出錯就重啟。                                 BOOT_JUMPTO_APP: 設置Application中斷向量,設置Application的Stack首地址。跳轉到Application Reset 向量地址。                                    

Jump_To_Application這個函數功能很簡單,卻花了很多時間,Application已經燒寫完成,始終無法跳轉過去,包括NXP官網上bootloader常式的跳轉方法也不行,後來經過不斷試錯,總算成功了,最終實現如下:

void Jump_To_Application(uint32_t userSP)
{
    void (*entry)(void);
    uint32_t pc;
    if(userSP == 0xFFFFFFFF)
    {
        return;
    }
    else
    {
        /* Set up stack pointer */
        __asm("msr msp, r0");
        __asm("msr psp, r0");
        /* Relocate vector table */
        S32_SCB->VTOR = (uint32_t)APP_START_ADDRESS;
        /* Jump to application PC */
        pc = *((volatile uint32_t *)(APP_START_ADDRESS + 4));
        entry = (void (*)(void))pc;
        entry();
    }
}

Bootloader 的開發環境:

IDE: S32DS

Compiler: S32DS 自帶的gcc

Hardware:  S32K116 EVB

SDK:   S32DS/S32SDK_S32K116_EAR_1.8.7


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 預計是通過三篇來將清楚asp.net core 3.x中的授權:1、基本概念介紹;2、asp.net core 3.x中授權的預設流程;3、擴展。 在完全沒有概念的情況下無論是看官方文檔還是源碼都暈乎乎的,希望本文能幫到你。不過我也是看源碼結合官方文檔看的,可能有些地方理解不對,所以只作為參考 ...
  • 區別 OpenId: Authentication :認證 Oauth: Aurhorize :授權 輸入賬號密碼,QQ確認輸入了正確的賬號密碼可以登錄 認證 下麵需要勾選的覆選框(獲取昵稱、頭像、性別) 授權 OpenID 當你需要訪問A網站的時候,A網站要求你輸入你的OpenId,即可跳轉到你的 ...
  • gRPC的結構 在我們搭建gRPC通信系統之前,首先需要知道gRPC的結構組成。 首先,需要一個server(伺服器),它用來接收和處理請求,然後返迴響應。 既然有server,那麼肯定有client(客戶端),client的作用就是向server發送請求,具體就是生成一個請求,然後把它發送到ser ...
  • 1. 需求 上圖這種包含多選(CheckBox)和單選(RadioButton)的菜單十分常見,可是在WPF中只提供了多選的MenuItem。順便一提,要使MenuItem可以多選,只需要將MenuItem的 屬性設置為True: 不知出於何種考慮,WPF沒有為MenuItem提供單選的功能。為了在 ...
  • Winform控制項的雙緩衝。控制項的雙緩衝屬性是隱藏的,可以通過反射改變其屬性值。 lv.GetType().GetProperty("DoubleBuffered", BindingFlags.Instance | BindingFlags.NonPublic).SetValue(lv, true, ...
  • 微信公眾號dotnet跨平臺2020年初做的一個關於中國.NET開發者調查收到了開發者近 1400 條回覆。這份調查報告涵蓋了開發者工具鏈的所有部分,包括編程語言、應用架構、應用伺服器、運行時平臺、框架技術、框架配置、IDE、.NET/.NET Core 發行版部署模式、構建工具和Kubernete... ...
  • screen是一款由GNU計劃開發的用於命令行終端切換的自由軟體。用戶可以通過該軟體同時連接多個本地或遠程的命令行會話,併在其間自由切換。GNU Screen可以看作是視窗管理器的命令行界面版本。它提供了統一的管理多個會話的界面和相應的功能。 screen重要性 screen的重要性,主要是體現在它 ...
  • 通過前面十餘篇文章的介紹,相信已經初步入門Linux本地管理的基本方法了,後續的文章將介紹Linux中常用的服務部署以及如何為外部提供相應的服務。 系列文章第三篇“linux入門系列3 linux遠程登陸工具”初步介紹了幾款用於Linux遠程登錄管理的工具,本文再來詳細講解下SSH協議以及對應的服務 ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...