一. uboot启动流程
_main 函数中会调用 board_init_f 函数,本文继续简单分析一下 board_init_f 函数。
具体分析 board_init_f函数的第二部分:内存分配代码。
本文继上一篇文章的学习,地址如下:
uboot启动流程-涉及board_init_f 函数-CSDN博客
二. uboot内存分配
下面具体分析 board_init_f 函数的后半部分:内存分配代码。
board_init_f 函数里面有大量的条件编译代码,这里为了缩小篇幅,将条件编译部分删除掉了,去掉条件编译以后的board_init_f 函数 后半部分代码如下:
1 static init_fnc_t init_sequence_f[] = {
2 setup_mon_len,
......
32 /*
33 * Now that we have DRAM mapped and working, we can
34 * relocate the code and continue running from DRAM.
35 *
36 * Reserve memory at end of RAM for (top down in that order):
37 * - area that won't get touched by U-Boot and Linux (optional)
38 * - kernel log buffer
39 * - protected RAM
40 * - LCD framebuffer
41 * - monitor code
42 * - board info struct
43 */
44 setup_dest_addr,
45 reserve_round_4k,
46 reserve_mmu,
47 reserve_trace,
48 reserve_uboot,
49 reserve_malloc,
50 reserve_board,
51 setup_machine,
52 reserve_global_data,
53 reserve_fdt,
54 reserve_arch,
55 reserve_stacks,
56 setup_dram_config,
57 show_dram_config,
58 display_new_sp,
59 INIT_FUNC_WATCHDOG_RESET
60 reloc_fdt,
61 setup_reloc,
62 NULL,
63 };
第 44 行, setup_dest_addr 函数,设置目的地址,设 gd->ram_size , gd->ram_top , gd->relocaddr
这三个的值。 我可以修改 uboot 代码,直接将这些值通过串口打印出来,比如这里我们修改文件
common/board_f.c ,因为 setup_dest_addr 函数定义在文件 common/board_f.c 中。
在 setup_dest_addr 函数加入打印如下:
printf("gd->ram_size: %#x\n", gd->ram_size);printf("gd->ram_top: %#x\n", gd->ram_top);printf("gd->relocaddr: %#x\n", gd->relocaddr);
重新编译 uboot源码后,生成 u-boot.bin。将 u-boot.bin拷贝到 ubuntu的 tftp服务设置目录下。
通过 tftp服务将 u-boot.bin下载到开发板。开发板uboot命令模式下输入如下:
=> tftp 0x87800000 u-boot.bin
Using FEC1 device
TFTP from server 192.168.1.66; our IP address is 192.168.1.50
Filename 'u-boot.bin'.
Load address: 0x87800000
Loading: #################################2.7 MiB/s
done
Bytes transferred = 476700 (7461c hex)
运行下载到开发板的 DRAM的uboot。操作如下:
=> go 0x87800000
## Starting application at 0x87800000 ...U-Boot 2016.03 (Oct 02 2023 - 21:20:41 +0800)CPU: Freescale i.MX6ULL rev1.1 69 MHz (running at 396 MHz)
CPU: Industrial temperature grade (-40C to 105C) at 44C
Reset cause: unknown reset
Board: MX6ULL ALIENTEK NAND
I2C: ready
DRAM: gd->ram_size: 0x10000000
gd->ram_top: 0x90000000
gd->relocaddr: 0x90000000
256 MiB
......
可以看出, 这里三个参数:
gd->ram_size = 0X10000000 //ram 大小为 0X10000000= 256 MB
gd->ram_top = 0X90000000 //ram 最高地址为 0X80000000+0X10000000=0X90000000
gd->relocaddr = 0X90000000 //重定位后最高地址为 0X90000000
第 45 行 , reserve_round_4k 函数用于对 gd->relocaddr 做 4KB 对 齐 , 因 为gd->relocaddr=0XA0000000 ,已经是 4K 对齐了,所以调整后不变。
第 46 行, reserve_mmu 函数,留出 MMU 的 TLB 表的位置,分配 MMU 的 TLB 表内存以后,会对 gd->relocaddr 做 64K 字节对齐。完成以后, gd->arch.tlb_size 、 gd->arch.tlb_addr 和 gd->relocaddr如下所示:
DRAM: gd->arch.tlb_size: 0x4000 //MMU 的 TLB 表大小
gd->arch.tlb_addr: 0x8fff0000 //MMU 的 TLB 表起始地址,64KB 对齐以后
gd->relocaddr: 0x8fff0000 //relocaddr 地址
第 47 行,reserve_trace 函数,留出跟踪调试的内存,I.MX6ULL 没有用到!
第 48 行, reserve_uboot 函数, 留出重定位后的 uboot 所占用的内存区域, uboot 所占用大小由gd->mon_len 所指定,留出 uboot 的空间以后还要对 gd->relocaddr 做 4K 字节对齐,并且重新设 置 gd->start_addr_sp ,打印如下:
DRAM: gd->mon_len: 0xb7394
gd->start_addr_sp: 0x8ff38000
gd->relocaddr: 0x8ff38000
第 49 行, reserve_malloc 函数,留出 malloc 区域,调整 gd->start_addr_sp 位置, malloc 区域由宏
TOTAL_MALLOC_LEN 定义,打印如下:
DRAM: TOTAL_MALLOC_LEN: 0x1020000
gd->start_addr_sp: 0x8ef18000 //0X8FF38000-16MB-4MB=0X8EF18000