目录

Xilinx-ZYNQ-FSBL解读LoadBootImage

Xilinx ZYNQ FSBL解读:LoadBootImage()

篇首

最近突发奇想,Xilinx 的集成开发环境已经很好了,很多必要的代码都直接生成了,这给开发者带来了巨大便利的同时,也让人错过了很多代码的精彩,可能有很多人用了很多年了,都还无法清楚的理解其中过程。博主准备以FSBL为例,与大家深入探讨一番,从而加深对ZYNQ的加载过程的理解,以便大家作出更精彩的设计!


LoadBootImage() 解读

本文以Zynq7000 FSBL工程代码为基础,分析启动流程核心函数

L o a d B o o t I m a g e ( ) LoadBootImage()

L

o

a

d

B

oo

t

I

ma

g

e

(

) 的执行逻辑与关键技术细节。

一、函数调用框架

int LoadBootImage(void) {
    FsblHookBeforeBitstreamDdr(); // 钩子函数
    Status = XFsbl_LoadPartitions(...); // 核心加载
    FsblHookBeforeHandoff(); // 移交前预处理
    return Status;
}

二、 函数执行全流程分解

** 函数入口与预处理**
int LoadBootImage(void) {
    u32 Status = XFSBL_SUCCESS;
    XTime tStart, tEnd;  // 64位计时器(若启用性能分析)
  • 硬件依赖
    • 依赖 psu_init.c 完成的PS端基础初始化(时钟、MIO、SLCR锁等)
    • DDR物理层已通过 Xil_DDRInit() 完成训练( psu_ddr_phyinit.c
** F s b l H o o k B e f o r e B i t s t r e a m D d r ( ) FsblHookBeforeBitstreamDdr() F s b l Hoo k B e f ore B i t s t re am D d r ( ) 钩子函数**
#ifdef FSBL_PERF
    XTime_GetTime(&tStart);  // 记录TSC起始值(AXI Timer 0)
#endif
/* 用户自定义扩展点:可插入DDR重配置代码 */
  • 关键寄存器操作
    • DDRC控制 :通过 Xil_Out32(0xFD070000, 0x00040010) 设置 DDRC_ADDRMAP0 调整地址映射
    • OCM重映射 :关闭OCM缓存( SLCR.OCM_CFG 寄存器位3置1)
X F s b l L o a d P a r t i t i o n s ( ) XFsbl_LoadPartitions() XF s b l L ​ o a d P a r t i t i o n s ( ) 核心加载
阶段1:Boot Header解析
XFsblPs_BootHdr Header;
XFsbl_CheckBootHeader(ImageAddr, &Header); // 从QSPI/NAND读取头部
  • 头部结构体xfsbl_ps_boothdr.h ):

    typedef struct {
        u32 ImageID;          // 魔数0xAA995566
        u32 NumPartitions;    // 分区总数(含PL比特流+应用)
        u32 AuthType;         // 加密类型:0=None, 1=RSA-2048
        u32 Checksum;         // 头部的CRC32校验
        // ... 其他字段(分区表偏移、证书偏移等)
    } XFsblPs_BootHdr;
阶段2:安全认证(以RSA-2048为例)
XSecure_Sha3Init(&Sha3Instance);  // 初始化SHA-3引擎
XSecure_Sha3Update(&Sha3Instance, (u8*)ImageAddr, Header.HashLength);
XSecure_Sha3Final(&Sha3Instance, CalculatedHash);  // 计算哈希
XSecure_VerifySignature(CalculatedHash, StoredSignature); // RSA验签
  • 硬件加速
    • 使用PS内置的CSU模块(Crypto Subsystem)
    • RSA密钥存储在eFUSE或BBRAM中(通过 XSecure_GetEfuseKek() 读取)
阶段3:分区加载循环
for (u8 i=0; i<Header.NumPartitions; i++) {
    XFsblPs_PartitionHdr PartHdr;
    XFsbl_ReadPartitionHdr(ImageAddr + Offset, &PartHdr);
    
    if (PartHdr.Attr & PART_ATTR_PL) {  // PL比特流分区
        XFsbl_LoadPlBitstream(PartHdr.LoadAddr, PartHdr.Size);
    } else {  // PS应用程序分区
        XFsbl_LoadElf(PartHdr.LoadAddr, PartHdr.Size); // ELF解析
    }
}
  • 关键操作细节
    • PL加载 :通过DevCfg接口( XDcfg_CfgInitialize() )写入PCAP
    • ELF加载 :解析Program Headers,使用 Xil_Out32() 逐段写入DDR
    • 地址对齐 :通过 XLAT_FSBL_TABLE 处理非32位对齐访问(触发Data Abort时自动转换)
F s b l H o o k B e f o r e H a n d o f f ( ) FsblHookBeforeHandoff() F s b l Hoo k B e f oreH an d o ff ( ) 移交前处理
Xil_DCacheFlush();  // 数据缓存刷新(确保DDR数据一致性)
Xil_Out32(CRL_APB_BASE + 0x24, 0x01000F00);  // 配置时钟分频
  • 寄存器详解
    • CRL_APB (0xFF5E0024)
      设置 RPLL_CTRL 分频系数(CPU=1.3GHz, DDR=1066MHz)
    • SLCR_UNLOCK (0xF8000008)
      写入 0xDF0D 解锁保护寄存器

三、关键子函数解析

  1. **F s b l H o o k B e f o r e B i t s t r e a m D d r ( ) FsblHookBeforeBitstreamDdr()

    F

    s

    b

    l

    Hoo

    k

    B

    e

    f

    ore

    B

    i

    t

    s

    t

    re

    am

    D

    d

    r

    (

    )**

    • 作用:DDR初始化前的预处理钩子

    • 执行内容:

      #ifdef FSBL_PERF
      XTime_GetTime(&tStart); // 性能计数器启动
      #endif
  2. **X F s b l L o a d P a r t i t i o n s ( ) XFsbl_LoadPartitions()

    XF

    s

    b

    l

    L

    o

    a

    d

    P

    a

    r

    t

    i

    t

    i

    o

    n

    s

    (

    )**

    • 流程分解:
      • **X F s b l C h e c k B o o t H e a d e r ( ) XFsbl_CheckBootHeader()

        XF

        s

        b

        l

        C

        h

        ec

        k

        B

        oo

        t

        He

        a

        d

        er

        (

        )**

        验证BIN文件头结构(含

        s i z e o f ( X F s b l P s B o o t H d r ) sizeof(XFsblPs_BootHdr)

        s

        i

        zeo

        f

        (

        XF

        s

        b

        lP

        s

        B

        oo

        t

        H

        d

        r

        ) )

      • **X F s b l A u t h e n t i c a t i o n ( ) XFsbl_Authentication()

        XF

        s

        b

        l

        A

        u

        t

        h

        e

        n

        t

        i

        c

        a

        t

        i

        o

        n

        (

        )**

        执行RSA-2048签名验证(通过

        X S e c u r e S h a 3 I n i t ( ) XSecure_Sha3Init()

        XS

        ec

        u

        r

        e

        S

        ha

        3

        I

        ni

        t

        (

        ) 等加密API)

      • 分区加载循环

        遍历分区表加载:

        for(u8 PartNum=0; PartNum<Header.NumPartitions; PartNum++){
            XFsbl_LoadPartition(...); // 加载单个分区
            #ifdef FSBL_DEBUG
            xil_printf("Partition %d Loaded\r\n", PartNum);
            #endif
        }
  3. **F s b l H o o k B e f o r e H a n d o f f ( ) FsblHookBeforeHandoff()

    F

    s

    b

    l

    Hoo

    k

    B

    e

    f

    oreH

    an

    d

    o

    ff

    (

    )**

    • 执行DDR刷新操作(

      X i l D C a c h e F l u s h ( ) Xil_DCacheFlush()

      X

      i

      l

      D

      C

      a

      c

      h

      e

      Fl

      u

      s

      h

      (

      ) )

    • 配置时钟分频器(通过

      X i l O u t 32 ( ) Xil_Out32()

      X

      i

      l

      O

      u

      t

      32

      (

      ) 写CRL_APB寄存器)

四、核心宏定义

  • $FSBL_DEBUG

    控制调试输出(默认关闭)

  • KaTeX parse error: Double subscript at position 15: XPAR_PSU_DDR_0_̲S_AXI_BASEADDR

    DDR控制器基地址宏(值

    0 x 00100000 0x00100000

    0

    x

    00100000 )

  • **X L A T F S B L T A B L E XLAT_FSBL_TABLE

    X

    L

    A

    T

    F

    SB

    L

    T

    A

    B

    L

    E**

    地址转换表(处理非对齐访问)

五、执行流程图

初始化硬件 → 验证头部 ↓ ↓ DDR预处理 → 加载分区 ↘ ↓ 移交控制权 \begin{array}{ccc} \text{初始化硬件} & \rightarrow & \text{验证头部} \ \downarrow & & \downarrow \ \text{DDR预处理} & \rightarrow & \text{加载分区} \ & \searrow & \downarrow \ & & \text{移交控制权} \end{array}

初始化硬件

DDR

预处理

验证头部

加载分区

移交控制权

六、 关键数据流与硬件交互

数据加载路径

QSPI Flash → AXI Quad-SPI控制器 OCM缓存 → DMA DDR3 \text{QSPI Flash} \xrightarrow{\text{AXI Quad-SPI控制器}} \text{OCM缓存} \xrightarrow{\text{DMA}} \text{DDR3}

QSPI Flash

AXI Quad-SPI

控制器

OCM

缓存

DMA

DDR3

  • 性能优化
    • 启用DMA传输( XQspiPs_DmaTransfer()
    • 使用线性突发模式(QSPI配置为DDR模式,时钟速率83MHz)
安全认证流程

原始镜像 → SHA-3/384 哈希值 哈希值 → RSA-2048签名 验签结果 \begin{aligned} &\text{原始镜像} \xrightarrow{\text{SHA-3/384}} \text{哈希值} \ &\text{哈希值} \xrightarrow{\text{RSA-2048签名}} \text{验签结果} \end{aligned}

原始镜像

SHA-3/384

哈希值

哈希值

RSA-2048

签名

验签结果

  • 抗攻击设计
    • 哈希计算前会清空CSU的密钥缓存( XSecure_CsuAesKcvClear()
    • 签名失败触发安全锁定(通过 XSecure_SetTamperConfig()

七、调试与错误处理

调试宏启用
#define FSBL_DEBUG  // 启用调试输出
  • 典型输出

    XFsbl_Debug: Partition 0 Loaded at 0x00100000 (Size 1MB)
    XFsbl_Debug: PL Bitstream CRC Check Passed
** 错误码定义**
#define XFSBL_ERROR_BOOTHEADER   0x1000  // 头部校验失败
#define XFSBL_ERROR_AUTHFAIL     0x1001  // RSA验签错误
#define XFSBL_ERROR_PLLLOCK      0x1002  // 时钟锁相环失锁
  • 错误处理
    • 记录错误到PMU全局状态寄存器( XFsbl_WriteReg(PMU_GLOBAL_GLOB_GEN_STORAGE, errCode)
    • 触发系统复位( XFsbl_FallbackReset()

**八、 总结 **

L o a d B o o t I m a g e ( ) LoadBootImage()

L

o

a

d

B

oo

t

I

ma

g

e

(

) 作为Zynq7000启动链的核心,其执行涵盖硬件初始化、安全认证、多阶段加载三大模块。函数首先通过

F s b l H o o k B e f o r e B i t s t r e a m D d r ( ) FsblHookBeforeBitstreamDdr()

F

s

b

l

Hoo

k

B

e

f

ore

B

i

t

s

t

re

am

D

d

r

(

) 完成DDR时序微调与性能监控启动,随后

X F s b l L o a d P a r t i t i o n s ( ) XFsbl_LoadPartitions()

XF

s

b

l

L

o

a

d

P

a

r

t

i

t

i

o

n

s

(

) 深度解析Boot Header结构,利用CSU硬件模块实现RSA-2048/SHA-3安全认证,并依据分区属性(PL比特流或PS应用)选择PCAP配置或ELF加载机制。关键点包括:通过DevCfg接口的PL动态重配置、基于XLAT表的非对齐地址访问补偿、以及DMA加速的QSPI数据传输。移交控制权前,函数会强制刷新数据缓存(确保内存一致性)并通过CRL_APB寄存器组重配时钟域。调试方面,FSBL_DEBUG宏可实时输出分区加载状态,而错误处理机制将异常状态固化至PMU寄存器,为后续故障分析提供关键日志。该函数的设计充分体现了Zynq架构中PS-PL协同、硬件安全加速、以及多级启动链的技术特点。


:具体实现细节需参考对应版本的

f s b l _ h o o k s . c fsbl_hooks.c

f

s

b

l

_

h

oo

k

s

.

c 和

x f s b l _ p a r t i t i o n l o a d . c xfsbl_partition_load.c

x

f

s

b

l

_

p

a

r

t

i

t

i

o

n

l

o

a

d

.

c 源码文件。