手把手帶你做LiteOS的樹莓派移植

華為雲開發者社群發表於2021-10-20
摘要:樹莓派是英國的慈善組織“Raspberry Pi 基金會”開發的一款基於arm的微型電腦主機板。本文介紹基於LiteOS的樹莓派移植過程。

本文分享自華為雲社群《2021 LiteOS樹莓派移植指南(一)》,作者: Lionlace 。

樹莓派是英國的慈善組織“Raspberry Pi 基金會”開發的一款基於arm的微型電腦主機板。本文介紹基於LiteOS的樹莓派移植過程。

硬體資訊

開發板:Raspberry Pi 2 Model B(樹莓派2B)

CPU:Broadcom BCM2836

主頻:900MHz

記憶體:1GB

GPU:VideoCore IV GPU

移植準備

硬體環境

本實驗使用了Raspberry Pi 2 Model B開發板、USB轉TTL模組、SDcard和讀卡器。

軟體環境

移植步驟

建立目錄結構

在targets目錄下新增Raspberry_Pi2B目錄,參考與cortex-A7架構差異較小的realview-pbx-a9的啟動流程進行移植。

 將realview-pbx-a9目錄下的reset_vector.S和main.c拷貝到Raspberry_Pi2B目錄下並將reset_vector.S重新命名為los_startup_gcc.S。
 將realview-pbx-a9目錄下的board.ld和liteos.ld中內容合併到Raspberry_Pi2B目錄下liteos.ld檔案中。
 拷貝realview-pbx-a9目錄下include、os_adapt資料夾到Raspberry_Pi2B目錄下,並刪除不需要的dma相關標頭檔案include/asm/dma.h。

關閉SMP和MMU

在los_startup_gcc.S檔案中增加關閉SMP和MMU的程式碼。

  • 關閉SMP功能
mrc p15, 0, r0, c1, c0, 1
bic r0, r0, #0x40
mcr p15, 0, r0, c1, c0, 1

image.png

上表是ACTLR(Auxiliary Control Register)暫存器bit6功能描述資訊,瞭解更多暫存器相關資訊可以參考Cortex-A7 MPCore Technical Reference Manual。

  • 關閉MMU的功能
mrc     p15, #0, r0, c1, c0, #0
bic     r0, r0, #1
mcr     p15, #0, r0, c1, c0, #0    @ clear mmu bit

image.png
上表是SCTLR (System Control Register)暫存器bit0功能描述資訊,瞭解更多暫存器相關資訊可以參考Cortex-A7MPCore Technical Reference Manual。

  • 刪除呼叫SMP相關函式

刪除los_startup_gcc.S中的enable_scu和secondary_cpu_start。

使能FPU/ENON

配置FPU/NEON:

/* enable fpu+neon */
LDR     r0, =(0xF << 20)
MCR     p15, 0, r0, c1, c0, 2
MOV     r3, #0x40000000
VMSR    FPEXC, r3

以上前兩行程式碼用於設定CP10和CP11的訪問許可權,後兩行用於設定暫存器FPEXC的EN位來使能FPU。

注:在arm的協處理器設計中,最多可以支援16個協處理器,通常被命名為cp0~cp15。
image.png

上表為暫存器CPACR bit20-23功能描述資訊,瞭解更多暫存器相關資訊可以參考Cortex-A7 MPCore Technical Reference Manual。

修改連結指令碼

樹莓派啟動時首先載入SD卡中的start.elf檔案,該程式會讀取SD卡中的config.txt檔案內容,該檔案記錄了一些配置資訊。如果沒有設定啟動地址和啟動檔案,則預設會載入kernel8.img檔案,該檔案是aarch64編譯的程式,啟動地址為0x80000。如果SD卡中無kernel8.img映象檔案,則會載入kernel7.img映象檔案,該檔案是32位編譯器編譯的程式,啟動地址為0x8000。樹莓派2B的cpu是32位架構,因此設定liteos.ld檔案中啟動地址為0x8000。

棧初始化

樹莓派2B啟動檔案los_startup_gcc.S中只設定了SVC模式的sp暫存器,新增cpuInit函式來初始化其他模式的sp指標。如下所示:

VOID cpuInit(VOID)
{
    __asm__ (
    "msr    cpsr_c, %1\n\t"
    "mov    sp,     %0\n\t"
    "msr    cpsr_c, %3\n\t"
    "mov    sp,     %2\n\t"
    "msr    cpsr_c, %5\n\t"
    "mov    sp,     %4\n\t"
    "msr    cpsr_c, %7\n\t"
    "mov    sp,     %6\n\t"
    "msr    cpsr_c, %8\n\t"
        :
        : "r" (__irq_stack_top),
          "I" (PSR_F_BIT | PSR_I_BIT | CPSR_IRQ_MODE),
          "r" (__abt_stack_top),
          "I" (PSR_F_BIT | PSR_I_BIT | CPSR_ABT_MODE),
          "r" (__undef_stack_top),
          "I" (PSR_F_BIT | PSR_I_BIT | CPSR_UNDEF_MODE),
          "r" (__fiq_stack_top),
          "I" (PSR_F_BIT | PSR_I_BIT | CPSR_FIQ_MODE),
          "I" (PSR_F_BIT | PSR_I_BIT | CPSR_SVC_MODE)
        : "r14");
}

配置動態記憶體地址

#define OS_SYS_MEM_ADDR        ((void *)(&__bss_end))
#define LOS_HEAP_ADDR_END      (void*)(0x0 + 4 * 1024 * 1024)
#define OS_SYS_MEM_SIZE        (UINT32)(((UINT32)LOS_HEAP_ADDR_END - (UINT32)OS_SYS_MEM_ADDR + (64 - 1)) & ~(64 - 1))

以上程式碼定義OS_SYS_MEM_ADDR為動態記憶體起始地址,LOS_HEAP_ADDR_END為動態記憶體結束地址,OS_SYS_MEM_SIZE為動態記憶體大小。

串列埠實現

樹莓派2B原理圖引出了mini_uart串列埠TXD0、RXD0,對應的引腳為GPIO14、GPIO15,如下圖所示:
image.png

建立usart.c和usart.h檔案,在usart.c中編寫串列埠初始化函式UartInit,並實現uart_debug.c檔案中uart_getc、uart_hwiCreate、uart_write介面,實現printf函式從串列埠輸出。

適配中斷

樹莓派2B的中斷屬於bcm特定的中斷控制器。在drivers/interrupt目錄下新增arm_control.c檔案,並在該檔案中實現HwiControllerOps結構體內的回撥函式。

STATIC const HwiControllerOps g_armControlOps = {
    .enableIrq      = HalIrqUnmask,
    .disableIrq     = HalIrqMask,
    .getCurIrqNum   = HalCurIrqGet,
    .getIrqVersion  = HalIrqVersion,
    .getHandleForm  = HalIrqGetHandleForm,
    .handleIrq      = IrqEntryArmControl,
    .clearIrq       = HalIrqClear,
    .triggerIrq     = HalIrqPending,
};

image.png

以上表格是interrupt暫存器偏移地址,讀者想了解詳細暫存器相關資訊請參考官方晶片手冊。

適配zhongduan

樹莓派2B通過Timer(arm side)來觸發systick中斷。具體操作細節請參考檔案:drivers\timer\rasp_systick.c。

/* systime=250000000 */
timer->preDivider = (OS_SYS_CLOCK / OS_SYS_US_PER_SECOND - 1);
timer->reload   = 0;    
timer->load     = 0;    
timer->IRQClear = 0;    
timer->control  = 0;    
timer->reload   = LOSCFG_BASE_CORE_TICK_PER_SECOND;    
timer->load     = LOSCFG_BASE_CORE_TICK_PER_SECOND;    
/* 23-bit counter, enable interrupt, enable timer */    timer->control = (1 << 1) | (1 << 5) | (1 << 7);    
UINT32 ret = LOS_HwiEnable(ARM_TIMER_INI);

以上程式碼配置定時器Timer為每1ms觸發一次systick中斷。
image.png

以上是Timer暫存器偏移地址,讀者想了解詳細暫存器相關資訊請參考官方晶片手冊。

配置編譯

在targets目錄下新增kconfig.raspberry檔案:

ConfigLOSCFG_PLATFORM
    config LOSCFG_PLATFORM
    string
    default "Raspberry_Pi2B"      if LOSCFG_PLATFORM_Raspberry_Pi2B
choice
    prompt "Board"
    depends on LOSCFG_FAMILY_RASPBERRY
    default LOSCFG_PLATFORM_Raspberry_Pi2B
    help
      Raspberry_Pi2B
config LOSCFG_PLATFORM_Raspberry_Pi2B
    bool "Raspberry_Pi2B"
    select LOSCFG_ARCH_CORTEX_A7
    select LOSCFG_USING_BOARD_LD
    select LOSCFG_PLATFORM_ARM_CONTROL
    select LOSCFG_Raspberry_Pi2B_SYSTICK
endchoice

修改Makefile檔案

分別修改以下路徑Makefile(詳情請參考gitee倉庫對應檔案):driver/timer/Makefiledriver/interrupt/Makefiletargets/Raspberry_Pi2B/Makefile

新增.img生成指令

在根目錄下Makefile中新增指令$(OBJCOPY) -O binary $(OUT)/$@.elf $(OUT)/kernel7.img,用來將生成的elf檔案轉換生成kernel7.img檔案。

製作啟動SDcard

  • 使用Raspberry Pi Imager工具製作Raspberry Pi系統。
    image.png

Raspberry Pi Imager 下載連結:https://www.raspberrypi.org/s...

  • 將編譯生成的kernel7.img檔案替換掉SDcard中kernel7.img檔案。
  • 將寫入映象檔案的SDcard插入樹莓派2B中並上電,樹莓派2B即可執行LiteOS系統。執行結果如下:

    ********Hello Huawei LiteOS********
    LiteOS Kernel Version : 5.1.0
    build data : Jul 13 2021 16:40:42
    **********************************
    OsAppInit
    cpu 0 entering scheduler
    app init!
    Hello, welcome to liteos demo!
    Huawei LiteOS #

至此,LiteOS系統成功啟動和執行。該移植工程已經在Gitee LiteOS社群上線,相關程式碼連結地址為:https://gitee.com/LiteOS/Lite...

參考文獻連結

[1] Raspberry Pihardware - Raspberry Pi Documentation:https://www.raspberrypi.org/d...

[2] 樹莓派官方晶片手冊:

https://datasheets.raspberryp...

[3] Cortex-A7 MPCore Technical Reference Manual:

https://developer.arm.com/doc...

點選關注,第一時間瞭解華為雲新鮮技術~

相關文章