日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

Uboot启动全过程

發布時間:2024/9/21 编程问答 34 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Uboot启动全过程 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
? 1.1?????? U-Boot 工作過程

?

U-Boot啟動內核的過程可以分為兩個階段,兩個階段的功能如下:

?????? (1)第一階段的功能

?? 硬件設備初始化

?? 加載U-Boot第二階段代碼到RAM空間

?? 設置好棧

?? 跳轉到第二階段代碼入口

?????? (2)第二階段的功能

?? 初始化本階段使用的硬件設備

?? 檢測系統內存映射

?? 將內核從Flash讀取到RAM中

?? 為內核設置啟動參數

?? 調用內核


1.1.1???????????? U-Boot 啟動第一階段代碼分析

?????? 第一階段對應的文件是cpu/arm920t/start.S和board/samsung/mini2440/lowlevel_init.S。

?????? U-Boot啟動第一階段流程如下:

?

圖 2.1 U-Boot啟動第一階段流程

?

?????? 根據cpu/arm920t/u-boot.lds中指定的連接方式:

ENTRY(_start)

SECTIONS

{

?????? . = 0x00000000;

?

?????? . = ALIGN(4);

?????? .text :

?????? {

???????????????????? cpu/arm920t/start.o??? (.text)

???? ?????????? board/samsung/mini2440/lowlevel_init.o (.text)

??? ???????????? board/samsung/mini2440/nand_read.o (.text)

????????????? *(.text)

?????? }

?????? … …

}

?????? 第一個鏈接的是cpu/arm920t/start.o,因此u-boot.bin的入口代碼在cpu/arm920t/start.o中,其源代碼在cpu/arm920t/start.S中。下面我們來分析cpu/arm920t/start.S的執行。

1.????? 硬件設備初始化

(1)設置異常向量

?????? cpu/arm920t/start.S開頭有如下的代碼:

.globl _start

_start:??? b???? start_code???????????????????????? /* 復位 */

?????? ldr?? pc, _undefined_instruction????? /*?未定義指令向量 */

?????? ldr?? pc, _software_interrupt??????????? /* ?軟件中斷向量 */

?????? ldr?? pc, _prefetch_abort????????????????? /*? 預取指令異常向量 */

?????? ldr?? pc, _data_abort??????????????????????? /* ?數據操作異常向量 */

?????? ldr?? pc, _not_used?????????????????????????? /*? 未使用?? */

?????? ldr?? pc, _irq???????????????????????????????????? /*? irq中斷向量? */

?????? ldr?? pc, _fiq???????????????????????????????????? /*? fiq中斷向量? */

/*? 中斷向量表入口地址 */

_undefined_instruction:??? .word undefined_instruction

_software_interrupt:? .word software_interrupt

_prefetch_abort:? .word prefetch_abort

_data_abort:??????? .word data_abort

_not_used:????????? .word not_used

_irq:???????????????????? .word irq

_fiq:???????????????????? .word fiq

?

?????? .balignl 16,0xdeadbeef

?

?????? 以上代碼設置了ARM異常向量表,各個異常向量介紹如下:

表 2.1 ARM異常向量表

地址?

異常?

進入模式

描述

0x00000000?

復位

管理模式

復位電平有效時,產生復位異常,程序跳轉到復位處理程序處執行

0x00000004?

未定義指令

未定義模式

遇到不能處理的指令時,產生未定義指令異常

0x00000008

軟件中斷

管理模式

執行SWI指令產生,用于用戶模式下的程序調用特權操作指令

0x0000000c

預存指令

中止模式

處理器預取指令的地址不存在,或該地址不允許當前指令訪問,產生指令預取中止異常

0x00000010

數據操作

中止模式

處理器數據訪問指令的地址不存在,或該地址不允許當前指令訪問時,產生數據中止異常

0x00000014

未使用

未使用

未使用

0x00000018

IRQ

IRQ

外部中斷請求有效,且CPSR中的I位為0時,產生IRQ異常

0x0000001c

FIQ

FIQ

快速中斷請求引腳有效,且CPSR中的F位為0時,產生FIQ異常

?????? 在cpu/arm920t/start.S中還有這些異常對應的異常處理程序。當一個異常產生時,CPU根據異常號在異常向量表中找到對應的異常向量,然后執行異常向量處的跳轉指令,CPU就跳轉到對應的異常處理程序執行。

?????? 其中復位異常向量的指令“b start_code”決定了U-Boot啟動后將自動跳轉到標號“start_code”處執行。

(2)CPU進入SVC模式

start_code:

?????? /*

?????? ?* set the cpu to SVC32 mode

?????? ?*/

?????? mrs r0, cpsr

?????? bic? r0, r0, #0x1f??????? /*工作模式位清零 */

?????? orr?? r0, r0, #0xd3????????????? /*工作模式位設置為“10011”(管理模式),并將中斷禁止位和快中斷禁止位置1 */

?????? msr cpsr, r0

?????? 以上代碼將CPU的工作模式位設置為管理模式,并將中斷禁止位和快中斷禁止位置一,從而屏蔽了IRQ和FIQ中斷。

(3)設置控制寄存器地址

# if defined(CONFIG_S3C2400)

#? define pWTCON 0x15300000

#? define INTMSK? 0x14400008

#? define CLKDIVN????? 0x14800014

#else????? /* s3c2410與s3c2440下面4個寄存器地址相同 */

#? define pWTCON 0x53000000?????????????? /* WATCHDOG控制寄存器地址 */

#? define INTMSK? 0x4A000008???????????????????? /* INTMSK寄存器地址? */

#? define INTSUBMSK 0x4A00001C????? /* INTSUBMSK寄存器地址 */

#? define CLKDIVN????? 0x4C000014?????? ? ????????? /* CLKDIVN寄存器地址 */

# endif

?????? 對與s3c2440開發板,以上代碼完成了WATCHDOG,INTMSK,INTSUBMSK,CLKDIVN四個寄存器的地址的設置。各個寄存器地址參見參考文獻[4] 。

(4)關閉看門狗

?????? ldr?? r0, =pWTCON

?????? mov?????? r1, #0x0

?????? str?? r1, [r0]?? /* 看門狗控制器的最低位為0時,看門狗不輸出復位信號 */

?????? 以上代碼向看門狗控制寄存器寫入0,關閉看門狗。否則在U-Boot啟動過程中,CPU將不斷重啟。

(5)屏蔽中斷

?????? /*

?????? ?* mask all IRQs by setting all bits in the INTMR - default

?????? ?*/

?????? mov?????? r1, #0xffffffff???? /* 某位被置1則對應的中斷被屏蔽 */

?????? ldr?? r0, =INTMSK

?????? str?? r1, [r0]

?????? INTMSK是主中斷屏蔽寄存器,每一位對應SRCPND(中斷源引腳寄存器)中的一位,表明SRCPND相應位代表的中斷請求是否被CPU所處理。

???????? 根據參考文獻4,INTMSK寄存器是一個32位的寄存器,每位對應一個中斷,向其中寫入0xffffffff就將INTMSK寄存器全部位置一,從而屏蔽對應的中斷。

# if defined(CONFIG_S3C2440)

??? ????? ldr? r1, =0x7fff??????

??? ????? ldr? r0, =INTSUBMSK

??? ????? str? r1, [r0]

# endif

?????? INTSUBMSK每一位對應SUBSRCPND中的一位,表明SUBSRCPND相應位代表的中斷請求是否被CPU所處理。

?????? 根據參考文獻4,INTSUBMSK寄存器是一個32位的寄存器,但是只使用了低15位。向其中寫入0x7fff就是將INTSUBMSK寄存器全部有效位(低15位)置一,從而屏蔽對應的中斷。

(6)設置MPLLCON,UPLLCON, CLKDIVN

# if defined(CONFIG_S3C2440)?

#define MPLLCON?? 0x4C000004

#define UPLLCON?? 0x4C000008??

??? ????? ldr? r0, =CLKDIVN??

??? ????? mov? r1, #5

??? ????? str? r1, [r0]

?

??? ????? ldr? r0, =MPLLCON

??? ????? ldr? r1, =0x7F021?

??? ????? str? r1, [r0]

?

?? ?ldr? r0, =UPLLCON?

??? ????? ldr? r1, =0x38022

??? ????? str? r1, [r0]

# else

?????? /* FCLK:HCLK:PCLK = 1:2:4 */

?????? /* default FCLK is 120 MHz ! */

?????? ldr?? r0, =CLKDIVN

?????? mov?????? r1, #3

?????? str?? r1, [r0]

#endif

?????? CPU上電幾毫秒后,晶振輸出穩定,FCLK=Fin(晶振頻率),CPU開始執行指令。但實際上,FCLK可以高于Fin,為了提高系統時鐘,需要用軟件來啟用PLL。這就需要設置CLKDIVN,MPLLCON,UPLLCON這3個寄存器。

?????? CLKDIVN寄存器用于設置FCLK,HCLK,PCLK三者間的比例,可以根據表2.2來設置。

表 2.2 S3C2440 的CLKDIVN寄存器格式

CLKDIVN

說明

初始值

HDIVN

[2:1]

00 : HCLK = FCLK/1.

01 : HCLK = FCLK/2.

10 : HCLK = FCLK/4 (當 CAMDIVN[9] = 0 時)

HCLK= FCLK/8? (當 CAMDIVN[9] = 1 時)

11 : HCLK = FCLK/3 (當 CAMDIVN[8] = 0 時)

HCLK = FCLK/6 (當 CAMDIVN[8] = 1時)

00

PDIVN

[0]

0: PCLK = HCLK/1?? 1: PCLK = HCLK/2

0

?

?????? 設置CLKDIVN為5,就將HDIVN設置為二進制的10,由于CAMDIVN[9]沒有被改變過,取默認值0,因此HCLK = FCLK/4。PDIVN被設置為1,因此PCLK= HCLK/2。因此分頻比FCLK:HCLK:PCLK = 1:4:8 。

?????? MPLLCON寄存器用于設置FCLK與Fin的倍數。MPLLCON的位[19:12]稱為MDIV,位[9:4]稱為PDIV,位[1:0]稱為SDIV。

?????? 對于S3C2440,FCLK與Fin的關系如下面公式:

?????? MPLL(FCLK) = (2×m×Fin)/(p× < mso-application progid="Word.Document" >142s'>?)

?????? 其中: m=MDIC+8,p=PDIV+2,s=SDIV

?????? MPLLCON與UPLLCON的值可以根據參考文獻4中“PLL VALUE SELECTION TABLE”設置。該表部分摘錄如下:

表 2.3 推薦PLL值

輸入頻率

輸出頻率

MDIV

PDIV

SDIV

12.0000MHz

48.00 MHz

56(0x38)

2

2

12.0000MHz

405.00 MHz

127(0x7f)

2

1

?????? 當mini2440系統主頻設置為405MHZ,USB時鐘頻率設置為48MHZ時,系統可以穩定運行,因此設置MPLLCON與UPLLCON為:

?????? MPLLCON=(0x7f<<12) | (0x02<<4) | (0x01) = 0x7f021

?????? UPLLCON=(0x38<<12) | (0x02<<4) | (0x02) = 0x38022

(7)關閉MMU,cache

?????? 接著往下看:

#ifndef CONFIG_SKIP_LOWLEVEL_INIT

?????? bl??? cpu_init_crit

#endif

?????? cpu_init_crit這段代碼在U-Boot正常啟動時才需要執行,若將U-Boot從RAM中啟動則應該注釋掉這段代碼。

?????? 下面分析一下cpu_init_crit到底做了什么:

320? #ifndef CONFIG_SKIP_LOWLEVEL_INIT

321? cpu_init_crit:

322? ??? /*

323? ??? ?* 使數據cache與指令cache無效 */

324? ??? ?*/?

325? ??? mov?????? r0, #0

326? ??? mcr p15, 0, r0, c7, c7, 0??? /* 向c7寫入0將使ICache與DCache無效*/

327? ??? mcr p15, 0, r0, c8, c7, 0??? /* 向c8寫入0將使TLB失效 */

328?

329? ??? /*

330? ??? ?* disable MMU stuff and caches

331? ??? ?*/

332? ??? mrc p15, 0, r0, c1, c0, 0??? /*? 讀出控制寄存器到r0中? */

333? ??? bic? r0, r0, #0x00002300?? @ clear bits 13, 9:8 (--V- --RS)

334? ??? bic? r0, r0, #0x00000087?? @ clear bits 7, 2:0 (B--- -CAM)

335? ??? orr?? r0, r0, #0x00000002?? @ set bit 2 (A) Align

336? ??? orr?? r0, r0, #0x00001000?? @ set bit 12 (I) I-Cache

337? ??? mcr p15, 0, r0, c1, c0, 0??? /*? 保存r0到控制寄存器? */

338?

339? ??? /*

340? ??? ?* before relocating, we have to setup RAM timing

341? ??? ?* because memory timing is board-dependend, you will

342? ??? ?* find a lowlevel_init.S in your board directory.

343? ??? ?*/

344? ??? mov?????? ip, lr

345?

346? ??? bl??? lowlevel_init

347?

348? ??? mov?????? lr, ip

349? ??? mov?????? pc, lr

350? #endif /* CONFIG_SKIP_LOWLEVEL_INIT */

?????? 代碼中的c0,c1,c7,c8都是ARM920T的協處理器CP15的寄存器。其中c7是cache控制寄存器,c8是TLB控制寄存器。325~327行代碼將0寫入c7、c8,使Cache,TLB內容無效。

?????? 第332~337行代碼關閉了MMU。這是通過修改CP15的c1寄存器來實現的,先看CP15的c1寄存器的格式(僅列出代碼中用到的位):

表 2.3 CP15的c1寄存器格式(部分)

15

14

13

12

11

10

9

8

7

6

5

4

3

2

1

0

.

.

V

I

.

.

R

S

B

.

.

.

.

C

A

M

?????? 各個位的意義如下:

V : ?表示異常向量表所在的位置,0:異常向量在0x00000000;1:異常向量在 0xFFFF0000
I : ?0 :關閉ICaches;1 :開啟ICaches
R、S : 用來與頁表中的描述符一起確定內存的訪問權限
B : ?0 :CPU為小字節序;1 : CPU為大字節序
C : ?0:關閉DCaches;1:開啟DCaches
A : ?0:數據訪問時不進行地址對齊檢查;1:數據訪問時進行地址對齊檢查
M : ?0:關閉MMU;1:開啟MMU

?????? 332~337行代碼將c1的 M位置零,關閉了MMU。

(8)初始化RAM控制寄存器

?????? 其中的lowlevel_init就完成了內存初始化的工作,由于內存初始化是依賴于開發板的,因此lowlevel_init的代碼一般放在board下面相應的目錄中。對于mini2440,lowlevel_init在board/samsung/mini2440/lowlevel_init.S中定義如下:

45? #define BWSCON?? 0x48000000??????? /* 13個存儲控制器的開始地址 */

? … …

129? _TEXT_BASE:

130? ??? .word???? TEXT_BASE

131?

132? .globl lowlevel_init

133? lowlevel_init:

134? ??? /* memory control configuration */

135? ??? /* make r0 relative the current location so that it */

136? ??? /* reads SMRDATA out of FLASH rather than memory ! */

137? ??? ldr???? r0, =SMRDATA

138? ??? ldr?? r1, _TEXT_BASE

139? ??? sub? r0, r0, r1????????????? /* SMRDATA減 _TEXT_BASE就是13個寄存器的偏移地址 */

140? ??? ldr?? r1, =BWSCON?? /* Bus Width Status Controller */

141? ??? add???? r2, r0, #13*4

142? 0:

143? ??? ldr???? r3, [r0], #4??? /*將13個寄存器的值逐一賦值給對應的寄存器*/

144? ??? str???? r3, [r1], #4

145? ??? cmp???? r2, r0

146? ??? bne? ???0b

147?

148? ??? /* everything is fine now */

149? ??? mov?????? pc, lr

150?

151? ??? .ltorg

152? /* the literal pools origin */

153?

154? SMRDATA:??????????? /*? 下面是13個寄存器的值? */

155? .word? … …

156 ? .word? … …

?… …

?????? lowlevel_init初始化了13個寄存器來實現RAM時鐘的初始化。lowlevel_init函數對于U-Boot從NAND Flash或NOR Flash啟動的情況都是有效的。

?????? U-Boot.lds鏈接腳本有如下代碼:

?????? .text :

?????? {

???????????????????? cpu/arm920t/start.o??? (.text)

???? ?????????? board/samsung/mini2440/lowlevel_init.o (.text)

??? ???????????? board/samsung/mini2440/nand_read.o (.text)

????????????? … …

?????? }

?????? board/samsung/mini2440/lowlevel_init.o將被鏈接到cpu/arm920t/start.o后面,因此board/samsung/mini2440/lowlevel_init.o也在U-Boot的前4KB的代碼中。

?????? U-Boot在NAND Flash啟動時,lowlevel_init.o將自動被讀取到CPU內部4KB的內部RAM中。因此第137~146行的代碼將從CPU內部RAM中復制寄存器的值到相應的寄存器中。

?????? 對于U-Boot在NOR Flash啟動的情況,由于U-Boot連接時確定的地址是U-Boot在內存中的地址,而此時U-Boot還在NOR Flash中,因此還需要在NOR Flash中讀取數據到RAM中。

?????? 由于NOR Flash的開始地址是0,而U-Boot的加載到內存的起始地址是TEXT_BASE,SMRDATA標號在Flash的地址就是SMRDATA-TEXT_BASE。

?????? 綜上所述,lowlevel_init的作用就是將SMRDATA開始的13個值復制給開始地址[BWSCON]的13個寄存器,從而完成了存儲控制器的設置。

(9)復制U-Boot第二階段代碼到RAM

?????? cpu/arm920t/start.S原來的代碼是只支持從NOR Flash啟動的,經過修改現在U-Boot在NOR Flash和NAND Flash上都能啟動了,實現的思路是這樣的:

?

?????? bl??? bBootFrmNORFlash /*? 判斷U-Boot是在NAND Flash還是NOR Flash啟動? */

?????? cmp?????? r0, #0????????? /*? r0存放bBootFrmNORFlash函數返回值,若返回0表示NAND Flash啟動,否則表示在NOR Flash啟動? */

?????? beq nand_boot???????? /*? 跳轉到NAND Flash啟動代碼? */

?

/*? NOR Flash啟動的代碼? */

?????? b???? stack_setup???????? /* 跳過NAND Flash啟動的代碼 */

?

nand_boot:

/*? NAND Flash啟動的代碼? */

?

stack_setup:???????

?????? /* 其他代碼 */

?

?????? 其中bBootFrmNORFlash函數作用是判斷U-Boot是在NAND Flash啟動還是NOR Flash啟動,若在NOR Flash啟動則返回1,否則返回0。根據ATPCS規則,函數返回值會被存放在r0寄存器中,因此調用bBootFrmNORFlash函數后根據r0的值就可以判斷U-Boot在NAND Flash啟動還是NOR Flash啟動。bBootFrmNORFlash函數在board/samsung/mini2440/nand_read.c中定義如下:

int bBootFrmNORFlash(void)

{

??? volatile unsigned int *pdw = (volatile unsigned int *)0;

??? unsigned int dwVal;

??

??? dwVal = *pdw;?????? ??/* 先記錄下原來的數據 */

??? *pdw = 0x12345678;

??? if (*pdw != 0x12345678)?????? /* 寫入失敗,說明是在NOR Flash啟動 */

??? {

??????? return 1;?????

??? }

??? else?????? ??????????????????????????? /* 寫入成功,說明是在NAND Flash啟動 */

??? {

??????? *pdw = dwVal;??????? /* 恢復原來的數據 */

??????? return 0;

??? }

}

???? 無論是從NOR Flash還是從NAND Flash啟動,地址0處為U-Boot的第一條指令“ b??? start_code”。

?????? 對于從NAND Flash啟動的情況,其開始4KB的代碼會被自動復制到CPU內部4K內存中,因此可以通過直接賦值的方法來修改。

?????? 對于從NOR Flash啟動的情況,NOR Flash的開始地址即為0,必須通過一定的命令序列才能向NOR Flash中寫數據,所以可以根據這點差別來分辨是從NAND Flash還是NOR Flash啟動:向地址0寫入一個數據,然后讀出來,如果發現寫入失敗的就是NOR Flash,否則就是NAND Flash。

?????? 下面來分析NOR Flash啟動部分代碼:

208? ??? adr? r0, _start????????????? /* r0 <- current position of code?? */

209? ??? ldr?? r1, _TEXT_BASE??????????? /* test if we run from flash or RAM */

?

/* 判斷U-Boot是否是下載到RAM中運行,若是,則不用?再復制到RAM中了,這種情況通常在調試U-Boot時才發生 */

210? ??? cmp ???? r0, r1????? /*_start等于_TEXT_BASE說明是下載到RAM中運行 */

211? ??? beq stack_setup

212? /* 以下直到nand_boot標號前都是NOR Flash啟動的代碼 */

213? ??? ldr?? r2, _armboot_start

214? ??? ldr?? r3, _bss_start

215? ??? sub? r2, r3, r2????????????? /* r2 <- size of armboot??????????? */

216? ??? add r2, r0, r2????????????? /* r2 <- source end address???????? */

217? /* 搬運U-Boot自身到RAM中*/

218? copy_loop:

219? ??? ldmia???? r0!, {r3-r10} /* 從地址為[r0]的NOR Flash中讀入8個字的數據 */

220? ??? stmia????? r1!, {r3-r10} /* 將r3至r10寄存器的數據復制給地址為[r1]的內存 */

221? ??? cmp?????? r0, r2??????????????????? /* until source end addreee [r2]??? */

222? ??? ble? copy_loop

223? ??? b???? stack_setup???????? /* 跳過NAND Flash啟動的代碼 */

?????? 下面再來分析NAND Flash啟動部分代碼:

nand_boot:

??? mov r1, #NAND_CTL_BASE?

??? ldr r2, =( (7<<12)|(7<<8)|(7<<4)|(0<<0) )

??? str r2, [r1, #oNFCONF]?? /* 設置NFCONF寄存器 */

?

?????? /* 設置NFCONT,初始化ECC編/解碼器,禁止NAND Flash片選 */

??? ldr r2, =( (1<<4)|(0<<1)|(1<<0) )

??? str r2, [r1, #oNFCONT]?

?

??? ldr r2, =(0x6)?????????? /* 設置NFSTAT */

str r2, [r1, #oNFSTAT]

?

?????? /* 復位命令,第一次使用NAND Flash前復位 */

??? mov r2, #0xff???????????

??? strb r2, [r1, #oNFCMD]

??? mov r3, #0??????????????

?

??? /* 為調用C函數nand_read_ll準備堆棧 */

??? ldr sp, DW_STACK_START??

??? mov fp, #0??????????????

??? /* 下面先設置r0至r2,然后調用nand_read_ll函數將U-Boot讀入RAM */

??? ldr r0, =TEXT_BASE????? /* 目的地址:U-Boot在RAM的開始地址 */

??? mov r1, #0x0???? ????????? /* 源地址:U-Boot在NAND Flash中的開始地址 */

??? mov r2, #0x30000? ??????? /* 復制的大小,必須比u-boot.bin文件大,并且必須是NAND Flash塊大小的整數倍,這里設置為0x30000(192KB) */

??? bl? nand_read_ll?? ????????????? /* 跳轉到nand_read_ll函數,開始復制U-Boot到RAM */

tst? r0, #0x0???????????????????? /* 檢查返回值是否正確 */

beq stack_setup

bad_nand_read:

loop2: b loop2??? //infinite loop

?

.align 2

DW_STACK_START: .word STACK_BASE+STACK_SIZE-4

?????? 其中NAND_CTL_BASE,oNFCONF等在include/configs/mini2440.h中定義如下:

#define NAND_CTL_BASE? 0x4E000000? // NAND Flash控制寄存器基址

?

#define STACK_BASE? 0x33F00000???? //base address of stack

#define STACK_SIZE? 0x8000???????? //size of stack

?

#define oNFCONF? 0x00????? /* NFCONF相對于NAND_CTL_BASE偏移地址 */

#define oNFCONT? 0x04????? /* NFCONT相對于NAND_CTL_BASE偏移地址*/

#define oNFADDR? 0x0c???? /* NFADDR相對于NAND_CTL_BASE偏移地址*/

#define oNFDATA? 0x10????? /* NFDATA相對于NAND_CTL_BASE偏移地址*/

#define oNFCMD?? 0x08???? /* NFCMD相對于NAND_CTL_BASE偏移地址*/

#define oNFSTAT? 0x20??????? /* NFSTAT相對于NAND_CTL_BASE偏移地址*/

#define oNFECC?? 0x2c????????????? /* NFECC相對于NAND_CTL_BASE偏移地址*/

?????? NAND Flash各個控制寄存器的設置在S3C2440的數據手冊有詳細說明,這里就不介紹了。

?????? 代碼中nand_read_ll函數的作用是在NAND Flash中搬運U-Boot到RAM,該函數在board/samsung/mini2440/nand_read.c中定義。

?????? NAND Flash根據page大小可分為2種: 512B/page和2048B/page的。這兩種NAND Flash的讀操作是不同的。因此就需要U-Boot識別到NAND Flash的類型,然后采用相應的讀操作,也就是說nand_read_ll函數要能自動適應兩種NAND Flash。

?????? 參考S3C2440的數據手冊可以知道:根據NFCONF寄存器的Bit3(AdvFlash (Read only))和Bit2 (PageSize (Read only))可以判斷NAND Flash的類型。Bit2、Bit3與NAND Flash的block類型的關系如下表所示:

表 2.4 NFCONF的Bit3、Bit2與NAND Flash的關系


Bit2??? Bit3

0

1

0

256 B/page

512 B/page

1

1024 B/page

2048 B/page


?

?????? 由于的NAND Flash只有512B/page和2048 B/page這兩種,因此根據NFCONF寄存器的Bit3即可區分這兩種NAND Flash了。

?????? 完整代碼見board/samsung/mini2440/nand_read.c中的nand_read_ll函數,這里給出偽代碼:

int nand_read_ll(unsigned char *buf, unsigned long start_addr, int size)

{

//根據NFCONF寄存器的Bit3來區分2種NAND Flash

?????? if( NFCONF & 0x8 ) ?????? /* Bit是1,表示是2KB/page的NAND Flash */

?????? {

?????????????

????????????? 讀取2K block 的NAND Flash

?????????????

?

?????? }

?????? else????????????????????? /* Bit是0,表示是512B/page的NAND Flash */

?????? {

????????????? /

????????????? 讀取512B block 的NAND Flash

????????????? /

?

?????? }

??? return 0;

}

(10)設置堆棧

?????? /* ?設置堆棧 */

stack_setup:

?????? ldr?? r0, _TEXT_BASE??????????? /* upper 128 KiB: relocated uboot?? */

?????? sub? r0, r0, #CONFIG_SYS_MALLOC_LEN?? /* malloc area????????????? */

?????? sub? r0, r0, #CONFIG_SYS_GBL_DATA_SIZE /*? 跳過全局數據區?????????????? */

#ifdef CONFIG_USE_IRQ

?????? sub? r0, r0, #(CONFIG_STACKSIZE_IRQ+CONFIG_STACKSIZE_FIQ)

#endif

?????? sub? sp, r0, #12?????????? /* leave 3 words for abort-stack??? */

?????? 只要將sp指針指向一段沒有被使用的內存就完成棧的設置了。根據上面的代碼可以知道U-Boot內存使用情況了,如下圖所示:

?

?

圖2.2 U-Boot內存使用情況

?

(11)清除BSS段

clear_bss:

?????? ldr?? r0, _bss_start????????????? /* BSS段開始地址,在u-boot.lds中指定*/

?????? ldr?? r1, _bss_end?????????????? /* BSS段結束地址,在u-boot.lds中指定*/

?????? mov?????? r2, #0x00000000

clbss_l:str???? r2, [r0]????????? /* 將bss段清零*/

?????? add r0, r0, #4

?????? cmp ???? r0, r1

?????? ble? clbss_l

?????? 初始值為0,無初始值的全局變量,靜態變量將自動被放在BSS段。應該將這些變量的初始值賦為0,否則這些變量的初始值將是一個隨機的值,若有些程序直接使用這些沒有初始化的變量將引起未知的后果。

(12)跳轉到第二階段代碼入口

?????? ldr?? pc, _start_armboot

?

_start_armboot:?? .word? start_armboot

?????? 跳轉到第二階段代碼入口start_armboot處。


1.1.2???????????? U-Boot 啟動第二階段代碼分析

?????? start_armboot函數在lib_arm/board.c中定義,是U-Boot第二階段代碼的入口。U-Boot啟動第二階段流程如下:

?

圖 2.3 U-Boot第二階段執行流程

?????? 在分析start_armboot函數前先來看看一些重要的數據結構:

(1)gd_t結構體

?????? U-Boot使用了一個結構體gd_t來存儲全局數據區的數據,這個結構體在include/asm-arm/global_data.h中定義如下:

typedef? struct???? global_data {

?????? bd_t????????????? *bd;

?????? unsigned long????? flags;

?????? unsigned long????? baudrate;

?????? unsigned long????? have_console;????? /* serial_init() was called */

?????? unsigned long????? env_addr;???? /* Address? of Environment struct */

?????? unsigned long????? env_valid;??? /* Checksum of Environment valid? */

?????? unsigned long????? fb_base; /* base address of frame buffer */

?????? void????????????? **jt;????????????? /* jump table */

} gd_t;

?????? U-Boot使用了一個存儲在寄存器中的指針gd來記錄全局數據區的地址:

#define DECLARE_GLOBAL_DATA_PTR???? register volatile gd_t *gd asm ("r8")

?????? DECLARE_GLOBAL_DATA_PTR定義一個gd_t全局數據結構的指針,這個指針存放在指定的寄存器r8中。這個聲明也避免編譯器把r8分配給其它的變量。任何想要訪問全局數據區的代碼,只要代碼開頭加入“DECLARE_GLOBAL_DATA_PTR”一行代碼,然后就可以使用gd指針來訪問全局數據區了。

?????? 根據U-Boot內存使用圖中可以計算gd的值:

gd = TEXT_BASE -CONFIG_SYS_MALLOC_LEN - sizeof(gd_t)

(2)bd_t結構體

?????? bd_t在include/asm-arm.u/u-boot.h中定義如下:

typedef struct bd_info {

??? int??????????????? bi_baudrate; ????????????? /* 串口通訊波特率 */

??? unsigned long???? bi_ip_addr;?? ?????? /* IP 地址*/

??? struct environment_s ?????? *bi_env;????????????? /* 環境變量開始地址 */

??? ulong??? ??????? bi_arch_number;????? /* 開發板的機器碼 */

??? ulong??? ??????? bi_boot_params;?????? /* 內核參數的開始地址 */

??? struct???????????????????????? /* RAM配置信息 */

??? {

????????????? ulong start;

????????????? ulong size;

?? ?}bi_dram[CONFIG_NR_DRAM_BANKS];?

} bd_t;

?????? U-Boot啟動內核時要給內核傳遞參數,這時就要使用gd_t,bd_t結構體中的信息來設置標記列表。

(3)init_sequence數組

?????? U-Boot使用一個數組init_sequence來存儲對于大多數開發板都要執行的初始化函數的函數指針。init_sequence數組中有較多的編譯選項,去掉編譯選項后init_sequence數組如下所示:

typedef int (init_fnc_t) (void);

?

init_fnc_t *init_sequence[] = {

?????? board_init,??? ? ?? /*開發板相關的配置--board/samsung/mini2440/mini2440.c */

?????? timer_init,??????????? /* 時鐘初始化-- cpu/arm920t/s3c24x0/timer.c */

?????? env_init,? ????????? /*初始化環境變量--common/env_flash.c 或common/env_nand.c*/

?????? init_baudrate,????? /*初始化波特率-- lib_arm/board.c */

?????? serial_init,??????????? /* 串口初始化-- drivers/serial/serial_s3c24x0.c */

?????? console_init_f,??? /* 控制通訊臺初始化階段1-- common/console.c */

?????? display_banner,?? /*打印U-Boot版本、編譯的時間-- gedit lib_arm/board.c */

?????? dram_init,??????????? /*配置可用的RAM-- board/samsung/mini2440/mini2440.c */

?????? display_dram_config,????????????? /* 顯示RAM大小-- lib_arm/board.c */

?????? NULL,

};

?????? 其中的board_init函數在board/samsung/mini2440/mini2440.c中定義,該函數設置了MPLLCOM,UPLLCON,以及一些GPIO寄存器的值,還設置了U-Boot機器碼和內核啟動參數地址 :

/* MINI2440開發板的機器碼 */

gd->bd->bi_arch_number = MACH_TYPE_MINI2440;

?

/* 內核啟動參數地址 */

gd->bd->bi_boot_params?= 0x30000100;??

?????? 其中的dram_init函數在board/samsung/mini2440/mini2440.c中定義如下:

int dram_init (void)

{

????? /* 由于mini2440只有 */

????? gd->bd->bi_dram[0].start = PHYS_SDRAM_1;

????? gd->bd->bi_dram[0].size = PHYS_SDRAM_1_SIZE;

?

????? return 0;

}

mini2440使用2片32MB的SDRAM組成了64MB的內存,接在存儲控制器的BANK6,地址空間是0x30000000~0x34000000。

在include/configs/mini2440.h中PHYS_SDRAM_1和PHYS_SDRAM_1_SIZE 分別被定義為0x30000000和0x04000000(64M)。

?????? 分析完上述的數據結構,下面來分析start_armboot函數:

void start_armboot (void)

{

?????? init_fnc_t **init_fnc_ptr;

?????? char *s;

?????? … …

?????? /* 計算全局數據結構的地址gd */

?????? gd = (gd_t*)(_armboot_start - CONFIG_SYS_MALLOC_LEN - sizeof(gd_t));

?????? … …

?????? memset ((void*)gd, 0, sizeof (gd_t));

?????? gd->bd = (bd_t*)((char*)gd - sizeof(bd_t));

?????? memset (gd->bd, 0, sizeof (bd_t));

?????? gd->flags |= GD_FLG_RELOC;

?

?????? monitor_flash_len = _bss_start - _armboot_start;

?

/* 逐個調用init_sequence數組中的初始化函數? */

?????? for (init_fnc_ptr = init_sequence; *init_fnc_ptr; ++init_fnc_ptr) {

????????????? if ((*init_fnc_ptr)() != 0) {

???????????????????? hang ();

????????????? }

?????? }

?

/* armboot_start 在cpu/arm920t/start.S 中被初始化為u-boot.lds連接腳本中的_start */

?????? mem_malloc_init (_armboot_start - CONFIG_SYS_MALLOC_LEN,

???????????????????? CONFIG_SYS_MALLOC_LEN);

?

/* NOR Flash初始化 */

#ifndef CONFIG_SYS_NO_FLASH

?????? /* configure available FLASH banks */

?????? display_flash_config (flash_init ());

#endif /* CONFIG_SYS_NO_FLASH */

?

?????? … …

/* NAND Flash 初始化*/

#if defined(CONFIG_CMD_NAND)

?????? puts ("NAND:? ");

?????? nand_init();???????? /* go init the NAND */

#endif

?????? … …

?????? /*配置環境變量,重新定位 */

?????? env_relocate ();

?????? … …

?????? /* 從環境變量中獲取IP地址 */

?????? gd->bd->bi_ip_addr = getenv_IPaddr ("ipaddr");

?????? stdio_init (); /* get the devices list going. */

?????? jumptable_init ();

?????? … …

?????? console_init_r (); /* fully init console as a device */

?????? … …

?????? /* enable exceptions */

?????? enable_interrupts ();

?

#ifdef CONFIG_USB_DEVICE

?????? usb_init_slave();

#endif

?

?????? /* Initialize from environment */

?????? if ((s = getenv ("loadaddr")) != NULL) {

????????????? load_addr = simple_strtoul (s, NULL, 16);

?????? }

#if defined(CONFIG_CMD_NET)

?????? if ((s = getenv ("bootfile")) != NULL) {

????????????? copy_filename (BootFile, s, sizeof (BootFile));

?????? }

#endif

?????? … …

?????? /* 網卡初始化 */

#if defined(CONFIG_CMD_NET)

#if defined(CONFIG_NET_MULTI)

?????? puts ("Net:?? ");

#endif

?????? eth_initialize(gd->bd);

… …

#endif

?

?????? /* main_loop() can return to retry autoboot, if so just run it again. */

?????? for (;;) {

????????????? main_loop ();

?????? }

?????? /* NOTREACHED - no way out of command loop except booting */

}

?????? main_loop函數在common/main.c中定義。一般情況下,進入main_loop函數若干秒內沒有


1.1.3???????????? U-Boot 啟動Linux過程?

?????? U-Boot使用標記列表(tagged list)的方式向Linux傳遞參數。標記的數據結構式是tag,在U-Boot源代碼目錄include/asm-arm/setup.h中定義如下:

struct tag_header {

?????? u32 size;?????? /* 表示tag數據結構的聯合u實質存放的數據的大小*/

?????? u32 tag;??????? /* 表示標記的類型 */

};

?

struct tag {

?????? struct tag_header hdr;

?????? union {

????????????? struct tag_core?????????? core;

????????????? struct tag_mem32????? mem;

????????????? struct tag_videotext?? videotext;

????????????? struct tag_ramdisk???? ramdisk;

????????????? struct tag_initrd? initrd;

????????????? struct tag_serialnr?????? serialnr;

????????????? struct tag_revision????? revision;

????????????? struct tag_videolfb???? videolfb;

????????????? struct tag_cmdline???? cmdline;

?

????????????? /*

????????????? ?* Acorn specific

????????????? ?*/

????????????? struct tag_acorn? acorn;

????????????? /*

????????????? ?* DC21285 specific

????????????? ?*/

????????????? struct tag_memclk????? memclk;

?????? } u;

};

?????? U-Boot使用命令bootm來啟動已經加載到內存中的內核。而bootm命令實際上調用的是do_bootm函數。對于Linux內核,do_bootm函數會調用do_bootm_linux函數來設置標記列表和啟動內核。do_bootm_linux函數在lib_arm/bootm.c 中定義如下:

59?? int do_bootm_linux(int flag, int argc, char *argv[], bootm_headers_t *images)

60?? {

61?? ??? bd_t?????? *bd = gd->bd;

62?? ??? char?????? *s;

63?? ??? int?? machid = bd->bi_arch_number;

64? ???? void?????? (*theKernel)(int zero, int arch, uint params);

65??

66?? #ifdef CONFIG_CMDLINE_TAG

67?? ??? char *commandline = getenv ("bootargs");?? /* U-Boot環境變量bootargs */

68?? #endif

?????? … …

73?? ??? theKernel = (void (*)(int, int, uint))images->ep; /* 獲取內核入口地址 */

?????? … …

86?? #if defined (CONFIG_SETUP_MEMORY_TAGS) || \

87?????? defined (CONFIG_CMDLINE_TAG) || \

88?????? defined (CONFIG_INITRD_TAG) || \

89?????? defined (CONFIG_SERIAL_TAG) || \

90?????? defined (CONFIG_REVISION_TAG) || \

91?????? defined (CONFIG_LCD) || \

92?????? defined (CONFIG_VFD)

93?? ??? setup_start_tag (bd);???????????????????????????????????? /* 設置ATAG_CORE標志 */

?????? … …

100? #ifdef CONFIG_SETUP_MEMORY_TAGS

101? ??? setup_memory_tags (bd);????????????????????? ?????? /* 設置內存標記 */

102? #endif

103? #ifdef CONFIG_CMDLINE_TAG

104? ??? setup_commandline_tag (bd, commandline);????? /* 設置命令行標記 */

105? #endif

?????? … …

113? ??? setup_end_tag (bd);?????????????????????????????? /* 設置ATAG_NONE標志 */??????????

114? #endif

115?

116? ??? /* we assume that the kernel is in place */

117? ??? printf ("\nStarting kernel ...\n\n");

?????? … …

126? ??? cleanup_before_linux ();????????? /* 啟動內核前對CPU作最后的設置 */

127?

128? ??? theKernel (0, machid, bd->bi_boot_params);????? /* 調用內核 */

129? ??? /* does not return */

130?

131? ??? return 1;

132? }

?????? 其中的setup_start_tag,setup_memory_tags,setup_end_tag函數在lib_arm/bootm.c中定義如下:

?????? (1)setup_start_tag函數

static void setup_start_tag (bd_t *bd)

{

?????? params = (struct tag *) bd->bi_boot_params;? /* 內核的參數的開始地址 */

?

?????? params->hdr.tag = ATAG_CORE;

?????? params->hdr.size = tag_size (tag_core);

?

?????? params->u.core.flags = 0;

?????? params->u.core.pagesize = 0;

?????? params->u.core.rootdev = 0;

?

?????? params = tag_next (params);

}

?????? 標記列表必須以ATAG_CORE開始,setup_start_tag函數在內核的參數的開始地址設置了一個ATAG_CORE標記。

?????? (2)setup_memory_tags函數

static void setup_memory_tags (bd_t *bd)

{

?????? int i;

/*設置一個內存標記 */

?????? for (i = 0; i < CONFIG_NR_DRAM_BANKS; i++) {???

????????????? params->hdr.tag = ATAG_MEM;

????????????? params->hdr.size = tag_size (tag_mem32);

?

????????????? params->u.mem.start = bd->bi_dram[i].start;

????????????? params->u.mem.size = bd->bi_dram[i].size;

?

????????????? params = tag_next (params);

?????? }

}

?????? setup_memory_tags函數設置了一個ATAG_MEM標記,該標記包含內存起始地址,內存大小這兩個參數。

?????? (3)setup_end_tag函數

static void setup_end_tag (bd_t *bd)

{

?????? params->hdr.tag = ATAG_NONE;

?????? params->hdr.size = 0;

}

?????? 標記列表必須以標記ATAG_NONE結束,setup_end_tag函數設置了一個ATAG_NONE標記,表示標記列表的結束。

?????? U-Boot設置好標記列表后就要調用內核了。但調用內核前,CPU必須滿足下面的條件:

(1)??? CPU寄存器的設置

?? r0=0

?? r1=機器碼

?? r2=內核參數標記列表在RAM中的起始地址

(2)??? CPU工作模式

?? 禁止IRQ與FIQ中斷

?? CPU為SVC模式

(3)??? 使數據Cache與指令Cache失效

?????? do_bootm_linux中調用的cleanup_before_linux函數完成了禁止中斷和使Cache失效的功能。cleanup_before_linux函數在cpu/arm920t/cpu.中定義:

int cleanup_before_linux (void)

{

?????? /*

?????? ?* this function is called just before we call linux

?????? ?* it prepares the processor for linux

?????? ?*

?????? ?* we turn off caches etc ...

?????? ?*/

?

?????? disable_interrupts ();???????? /* 禁止FIQ/IRQ中斷 */

?

?????? /* turn off I/D-cache */

?????? icache_disable();?????????????? /* 使指令Cache失效 */

?????? dcache_disable();????????????? /* 使數據Cache失效 */

?????? /* flush I/D-cache */

?????? cache_flush();??????????????????? /* 刷新Cache */

?

?????? return 0;

}

?????? 由于U-Boot啟動以來就一直工作在SVC模式,因此CPU的工作模式就無需設置了。

do_bootm_linux中:

64?? ??? void?????? (*theKernel)(int zero, int arch, uint params);

… …

73?? ??? theKernel = (void (*)(int, int, uint))images->ep;

… …

128? ??? theKernel (0, machid, bd->bi_boot_params);

?????? 第73行代碼將內核的入口地址“images->ep”強制類型轉換為函數指針。根據ATPCS規則,函數的參數個數不超過4個時,使用r0~r3這4個寄存器來傳遞參數。因此第128行的函數調用則會將0放入r0,機器碼machid放入r1,內核參數地址bd->bi_boot_params放入r2,從而完成了寄存器的設置,最后轉到內核的入口地址。

?????? 到這里,U-Boot的工作就結束了,系統跳轉到Linux內核代碼執行。


1.1.4???????????? U-Boot 添加命令的方法及U-Boot命令執行過程

?????? 下面以添加menu命令(啟動菜單)為例講解U-Boot添加命令的方法。

(1)??? 建立common/cmd_menu.c

?????? 習慣上通用命令源代碼放在common目錄下,與開發板專有命令源代碼則放在board/<board_dir>目錄下,并且習慣以“cmd_<命令名>.c”為文件名。

(2)??? 定義“menu”命令

?????? 在cmd_menu.c中使用如下的代碼定義“menu”命令:

_BOOT_CMD(

?????? menu,??? 3,??? 0,??? do_menu,

?????? "menu - display a menu, to select the items to do something\n",

?????? " - display a menu, to select the items to do something"

);

?????? 其中U_BOOT_CMD命令格式如下:

U_BOOT_CMD(name,maxargs,rep,cmd,usage,help)

?????? 各個參數的意義如下:

name:命令名,非字符串,但在U_BOOT_CMD中用“#”符號轉化為字符串

maxargs:命令的最大參數個數

rep:是否自動重復(按Enter鍵是否會重復執行)

cmd:該命令對應的響應函數

usage:簡短的使用說明(字符串)

help:較詳細的使用說明(字符串)

?????? 在內存中保存命令的help字段會占用一定的內存,通過配置U-Boot可以選擇是否保存help字段。若在include/configs/mini2440.h中定義了CONFIG_SYS_LONGHELP宏,則在U-Boot中使用help命令查看某個命令的幫助信息時將顯示usage和help字段的內容,否則就只顯示usage字段的內容。

?????? U_BOOT_CMD宏在include/command.h中定義:

#define U_BOOT_CMD(name,maxargs,rep,cmd,usage,help) \

cmd_tbl_t __u_boot_cmd_##name Struct_Section = {#name, maxargs, rep, cmd, usage, help}

?????? “##”與“#”都是預編譯操作符,“##”有字符串連接的功能,“#”表示后面緊接著的是一個字符串。

?????? 其中的cmd_tbl_t在include/command.h中定義如下:

struct cmd_tbl_s {

?????? char????????????? *name;????????? /* 命令名 */

?????? int????????? maxargs;?????? /* 最大參數個數 */

?????? int????????? repeatable;??? /* 是否自動重復 */

?????? int????????? (*cmd)(struct cmd_tbl_s *, int, int, char *[]);? /* ?響應函數 */

?????? char????????????? *usage;???????? /* 簡短的幫助信息 */

#ifdef??? CONFIG_SYS_LONGHELP

?????? char????????????? *help;?????????? /* ?較詳細的幫助信息 */

#endif

#ifdef CONFIG_AUTO_COMPLETE

?????? /* 自動補全參數 */

?????? int????????? (*complete)(int argc, char *argv[], char last_char, int maxv, char *cmdv[]);

#endif

};

typedef struct cmd_tbl_s? cmd_tbl_t;

?????? 一個cmd_tbl_t結構體變量包含了調用一條命令的所需要的信息。

?????? 其中Struct_Section在include/command.h中定義如下:

#define Struct_Section? __attribute__ ((unused,section (".u_boot_cmd")))

?????? 凡是帶有__attribute__ ((unused,section (".u_boot_cmd"))屬性聲明的變量都將被存放在".u_boot_cmd"段中,并且即使該變量沒有在代碼中顯式的使用編譯器也不產生警告信息。

?????? 在U-Boot連接腳本u-boot.lds中定義了".u_boot_cmd"段:

?????? . = .;

?????? __u_boot_cmd_start = .;????????? /*將 __u_boot_cmd_start指定為當前地址 */

?????? .u_boot_cmd : { *(.u_boot_cmd) }

?????? __u_boot_cmd_end = .;?????????? /*? 將__u_boot_cmd_end指定為當前地址? */

?????? 這表明帶有“.u_boot_cmd”聲明的函數或變量將存儲在“u_boot_cmd”段。這樣只要將U-Boot所有命令對應的cmd_tbl_t變量加上“.u_boot_cmd”聲明,編譯器就會自動將其放在“u_boot_cmd”段,查找cmd_tbl_t變量時只要在__u_boot_cmd_start與__u_boot_cmd_end之間查找就可以了。

?????? 因此“menu”命令的定義經過宏展開后如下:

cmd_tbl_t __u_boot_cmd_menu __attribute__ ((unused,section (".u_boot_cmd"))) = {menu, 3, 0, do_menu, "menu - display a menu, to select the items to do something\n", " - display a menu, to select the items to do something"}

?????? 實質上就是用U_BOOT_CMD宏定義的信息構造了一個cmd_tbl_t類型的結構體。編譯器將該結構體放在“u_boot_cmd”段,執行命令時就可以在“u_boot_cmd”段查找到對應的cmd_tbl_t類型結構體。

(3)??? 實現命令的函數

?????? 在cmd_menu.c中添加“menu”命令的響應函數的實現。具體的實現代碼略:

int do_menu (cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])

{

?????? /* 實現代碼略 */

}

(4)??? 將common/cmd_menu.c編譯進u-boot.bin

?????? 在common/Makefile中加入如下代碼:

COBJS-$(CONFIG_BOOT_MENU) += cmd_menu.o

?????? 在include/configs/mini2440.h加入如代碼:

#define CONFIG_BOOT_MENU 1

?????? 重新編譯下載U-Boot就可以使用menu命令了

(5)menu命令執行的過程

?????? 在U-Boot中輸入“menu”命令執行時,U-Boot接收輸入的字符串“menu”,傳遞給run_command函數。run_command函數調用common/command.c中實現的find_cmd函數在__u_boot_cmd_start與__u_boot_cmd_end間查找命令,并返回menu命令的cmd_tbl_t結構。然后run_command函數使用返回的cmd_tbl_t結構中的函數指針調用menu命令的響應函數do_menu,從而完成了命令的執行。

?

轉自http://www.cnblogs.com/heaad/archive/2010/07/17/1779829.html

總結

以上是生活随笔為你收集整理的Uboot启动全过程的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

激情五月婷婷综合网 | 色视频在线免费 | 丰满少妇在线观看 | 99亚洲国产精品 | a级国产乱理论片在线观看 伊人宗合网 | 成人影片在线免费观看 | 久久视频国产精品免费视频在线 | 国产精彩在线视频 | www免费视频com━ | 99热精品免费观看 | 久久精品久久精品久久精品 | 亚洲成人av一区二区 | 久久久久久久久影视 | 日韩高清国产精品 | 黄色一级性片 | 日韩xxxx视频 | 日韩啪视频 | 中国一级片在线观看 | 国产视频一级 | 又污又黄网站 | 狠狠狠综合 | 国产精品乱码久久久 | 久久精品一区二区三区中文字幕 | 三级av黄色| 激情五月婷婷网 | 中文字幕色站 | 天天操天天射天天爱 | 97超碰人人模人人人爽人人爱 | 欧美激情视频一二三区 | 国产精品毛片一区 | 正在播放久久 | 欧美久久久久久久 | 天天操夜夜干 | 婷婷九月激情 | 最新av免费在线观看 | 性色在线视频 | 欧美精品中文在线免费观看 | 成人一区二区三区在线 | 国产私拍在线 | 99亚洲国产精品 | 国产日本在线播放 | 久久精品国产亚洲精品 | 日韩二区精品 | 久久99免费 | 中文字幕在线免费观看 | 在线观看av黄色 | 日本激情视频中文字幕 | 99久久精品免费视频 | 综合色站导航 | 中文字幕在线免费观看视频 | 天天射网 | 日韩美在线| 五月激情姐姐 | 99在线看| 天堂在线成人 | 狠狠88综合久久久久综合网 | 一区二区三区四区五区在线 | 激情视频网页 | 精品久久久久久久久久 | 国产高清视频在线播放一区 | 久久久久国产精品厨房 | 亚洲香蕉在线观看 | 91免费观看网站 | 欧美另类美少妇69xxxx | 涩av在线| 久草国产在线 | 国产精品2019 | 婷婷草| 久久成人国产精品一区二区 | 最近最新mv字幕免费观看 | 欧美激情第八页 | 久久夜色精品国产欧美乱极品 | 日韩精品一区二区三区在线视频 | 九九九热视频 | 久久免费视频3 | 日日夜夜天天操 | 99在线观看视频网站 | 欧美调教网站 | 成人激情开心网 | 99视屏 | 久久天堂影院 | 国产精品久久一区二区三区, | 国产美女精品 | av在线播放免费 | 欧美一区二区在线免费观看 | 三上悠亚一区二区在线观看 | 色妞色视频一区二区三区四区 | 亚洲成人资源 | 精品亚洲va在线va天堂资源站 | 狠狠色丁香婷婷 | 亚洲成免费 | 日韩有码在线观看视频 | 日本精品一区二区三区在线观看 | 色婷婷婷 | 日韩高清免费在线 | 精品久久1| 国产高清精 | 五月天亚洲综合小说网 | 午夜成人影视 | 在线视频你懂得 | 久久精品视频在线播放 | 久久精品国产亚洲精品 | 中文字幕在线观看资源 | 黄污视频网站大全 | 国内丰满少妇猛烈精品播放 | 在线观看免费av网站 | 日韩欧美高清在线观看 | 大胆欧美gogo免费视频一二区 | 永久免费的av电影 | 亚洲精品乱码久久久久久蜜桃欧美 | 在线免费观看黄网站 | 99国产精品久久久久久久久久 | 久久精品网| 国产中文字幕在线看 | 黄色aaa毛片 | 国产在线色视频 | 美女网站在线观看 | 国产精品成人av在线 | 色先锋av资源中文字幕 | 国产精品99免费看 | 国产中文字幕一区 | 99热高清 | 超碰97国产在线 | 久久情侣偷拍 | 开心激情五月婷婷 | 亚洲无毛专区 | 伊人网综合在线观看 | 色人久久 | 亚洲精品国产精品国自产观看浪潮 | 国产色拍拍拍拍在线精品 | 国产高清免费视频 | 一二三区高清 | 日韩高清国产精品 | 激情综合色播五月 | 亚洲专区中文字幕 | 国产a级片免费观看 | 欧美污污网站 | 亚洲三级精品 | 日韩成人xxxx | 国产伦理一区二区 | 天天操天天色天天射 | 亚洲精品在线视频观看 | 香蕉视频在线免费 | 精品99在线 | av电影在线观看 | 日韩精品一区二区三区免费视频观看 | 精品视频免费在线 | 国产日韩欧美在线免费观看 | 婷婷丁香六月 | 久久婷五月| 亚洲精品久久久蜜臀下载官网 | 免费看片亚洲 | 久草在线一免费新视频 | 久久99精品久久久久婷婷 | 成人影视免费 | 久久久一本精品99久久精品66 | 一区二区三区免费网站 | 日韩专区在线播放 | 午夜在线观看一区 | 亚洲伦理精品 | 水蜜桃亚洲一二三四在线 | 久草免费在线视频 | 天天操天天综合网 | 天天操天天爽天天干 | 欧美电影在线观看 | 日三级在线 | 麻豆视频在线免费观看 | 亚洲视频免费在线看 | 激情综合中文娱乐网 | av在线网站大全 | 久久看片 | 99视频这里有精品 | 免费黄色一区 | 久久少妇免费视频 | 九色精品免费永久在线 | 亚洲欧美久久 | 午夜影视av| 亚洲国产精品推荐 | 操操操日日日 | 狠狠色丁香婷婷综合基地 | 国产一区久久 | 久久久www | 欧美色888 | 成人毛片久久 | 国内精品久久久久久久久久久久 | 免费视频一区二区 | 亚洲精品国产电影 | 久久综合久久综合久久 | 久草观看视频 | 精品久久久久久综合日本 | 九九九九九九精品任你躁 | 欧美精品做受xxx性少妇 | 91精品国产自产91精品 | 一区二区国产精品 | 国产成人黄色网址 | 久久久黄视频 | 插婷婷| 18久久久 | 最新精品视频在线 | 欧美一级久久久久 | 午夜黄色一级片 | 国产精品青草综合久久久久99 | 色婷婷www | 国产一区在线观看免费 | 日韩午夜精品 | 午夜精品久久久久久久久久久久 | 夜夜澡人模人人添人人看 | 国产精品久久久久久久久久久久午夜 | 午夜手机看片 | 亚洲免费成人av电影 | 精品九九九九 | 91欧美日韩国产 | 国产精品视频 | 亚洲狠狠操 | 欧美综合在线观看 | 免费日韩一级片 | 国产精品视频999 | 日韩精品视频第一页 | 97免费在线观看视频 | 欧美日韩国产网站 | av天天澡天天爽天天av | 亚洲人成人99网站 | 久久少妇免费视频 | 99久久久国产精品免费99 | 亚洲在线资源 | 成人九九视频 | 亚洲综合五月天 | 在线看不卡av | 在线观看日韩专区 | 中文字幕在线看视频 | 国产99视频在线观看 | 看片网站黄色 | 欧美在线一级片 | 一级黄色片网站 | 日韩免费在线观看视频 | av中文国产 | 国产精品国产三级国产不产一地 | 日韩激情综合 | 中文字幕国产 | 久久图 | 欧美性精品 | 色婷婷国产精品一区在线观看 | 国产福利91精品一区 | 免费99精品国产自在在线 | 992tv人人网tv亚洲精品 | 黄色av一区二区 | 久久久久久久久久免费视频 | 天天射天天操天天 | 国产中文字幕视频在线观看 | 久久新 | 欧美日韩三级在线观看 | 91在线免费公开视频 | 久久激情视频 | 久久爱资源网 | 中文字幕2021 | 日韩精品中文字幕久久臀 | 成人一级黄色片 | 97超碰人人澡| 亚洲女人av | 天天色天天操综合网 | 91精品综合| 国产精品一区二区三区免费视频 | 久久久久久麻豆 | 久久久久国产成人免费精品免费 | 国产精品一区二区免费 | 久久黄色小说 | 中文字幕在线观看资源 | 中午字幕在线 | 美女网站在线免费观看 | 日本女人在线观看 | 久久夜色精品国产亚洲aⅴ 91chinesexxx | 999抗病毒口服液 | 久久毛片高清国产 | 欧美日韩不卡一区二区三区 | 国产精品福利一区 | 黄色激情网址 | 狠狠色丁香久久综合网 | 国产97视频在线 | 99看视频在线观看 | 99精品99| 黄p在线播放 | 黄色av网站在线观看免费 | 日韩精品不卡在线 | 亚洲免费永久精品国产 | 手机看片中文字幕 | 久久久久久久久久网 | 免费网站看v片在线a | 成人av影视| 久久毛片视频 | 超碰97免费观看 | 国产精品美女久久久久久2018 | 91九色porn在线资源 | 五月天久久 | 成年人视频在线免费播放 | 黄色www| 国产精品久久久久亚洲影视 | 91在线播放视频 | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 国语精品免费视频 | www.黄色| 91伊人影院| 亚洲精品在线视频播放 | 日韩字幕在线 | 亚洲精品97| 91视频91自拍 | 中文字幕麻豆 | 日韩首页| 欧美日韩在线精品 | 午夜精品一区二区三区视频免费看 | 五月婷亚洲| 黄色在线看网站 | 久久se视频 | 久久人人爽人人爽人人 | 97在线视频网站 | 亚洲码国产日韩欧美高潮在线播放 | 99久视频| 久久国产精品一区二区 | 91欧美在线 | 久久久精品 一区二区三区 国产99视频在线观看 | 免费美女久久99 | 免费看黄20分钟 | 欧美成人h版 | 中文综合在线 | 亚洲国产欧美一区二区三区丁香婷 | 国产区av在线 | 中文字幕一区二区在线播放 | 色婷婷成人网 | bbbb操bbbb | 人人澡人人添人人爽一区二区 | 97色涩 | 久久任你操| 亚洲国产精品一区二区尤物区 | 免费一级特黄毛大片 | 久久99影院 | 久久99电影 | 成年人视频免费在线 | 一区二区三区精品在线视频 | 又大又硬又黄又爽视频在线观看 | 看国产黄色大片 | 97成人资源站 | 国产精品 日韩 | 中文字幕在线视频第一页 | 中文字幕在线观 | 成年人在线免费视频观看 | 国产在线不卡 | 国产精品电影一区二区 | 国产无遮挡又黄又爽馒头漫画 | 色偷偷网站视频 | 久久久久久久看片 | 久久草av| 亚洲天堂网在线观看视频 | 国产一级免费播放 | 中文字幕观看在线 | 日韩三级在线观看 | 国产午夜三级一区二区三桃花影视 | 国产一级电影在线 | 色婷婷激情综合 | 亚洲资源在线观看 | 最新av观看 | 综合网伊人 | 人人爽人人做 | 九九精品久久 | 在线观看深夜视频 | 国产高清在线精品 | 久久久久国产一区二区 | 免费a v在线 | 色91在线视频| 日韩天堂网 | 狠狠干夜夜爱 | 最新av网址大全 | 国产精品一区二区av影院萌芽 | 欧美午夜一区二区福利视频 | 在线免费视频 你懂得 | 日韩在线观看第一页 | 婷婷av资源| 日韩免费精品 | 久久在草 | 91av视频在线观看免费 | 国产高清成人av | 国产激情小视频在线观看 | 天天干,天天操,天天射 | 夜夜嗨av色一区二区不卡 | 九九久久影院 | 日韩av片无码一区二区不卡电影 | 亚洲永久国产精品 | 日本视频高清 | 国产999视频在线观看 | 日韩一级黄色av | 日本少妇久久久 | 又爽又黄又无遮挡网站动态图 | 亚洲伦理精品 | 久久久久国产精品午夜一区 | 久久精品99国产精品日本 | 亚洲电影一级黄 | 日韩精品中文字幕有码 | 91视频久久久 | 美女视频免费精品 | 天天色宗合 | 韩国精品福利一区二区三区 | 国产成人久久av免费高清密臂 | 日韩av伦理片 | 外国av网 | 成人国产精品免费 | 人人看黄色 | www,黄视频| 狠狠88综合久久久久综合网 | 99九九热只有国产精品 | 欧美精品被 | 麻豆传媒视频在线 | 久久免费黄色 | 日日夜夜免费精品视频 | 精品视频99 | 国产99亚洲| 96视频免费在线观看 | 国产午夜精品福利视频 | 久久久久久国产精品 | 久久久久久久18 | 中文字幕一区二区三区乱码不卡 | 精品国产精品一区二区夜夜嗨 | 日韩欧在线 | 国产又粗又长的视频 | 天天操比| 黄色天堂在线观看 | 亚洲乱亚洲乱妇 | 中午字幕在线 | 日本黄色免费网站 | 99热九九这里只有精品10 | 色偷偷人人澡久久超碰69 | 在线亚州| 人人爽人人看 | 91看片淫黄大片一级在线观看 | 在线不卡中文字幕播放 | 久久精品精品电影网 | 欧美日韩免费看 | 久久99在线视频 | 亚洲精品视频在线免费播放 | 天天爽人人爽 | 精品一区二区三区久久久 | 日韩电影在线视频 | 天天操综 | 国内精品久久久久久久影视简单 | 91在线公开视频 | 黄色高清视频在线观看 | 狠狠狠狠狠狠狠 | 国产91全国探花系列在线播放 | 久久久久久久久久电影 | 亚洲一区精品人人爽人人躁 | 免费成人在线网站 | 在线天堂视频 | 99久久99视频只有精品 | 四川妇女搡bbbb搡bbbb搡 | 最近中文字幕 | av理论电影 | 992tv人人草 黄色国产区 | 午夜av在线免费 | 久久免费精彩视频 | 99精品亚洲 | 精品国产乱码久久久久久久 | 天天射天天爽 | 日日操网| 91精品视频在线 | 国产高清小视频 | 久久久久久国产一区二区三区 | 亚洲精品国偷自产在线99热 | 日韩av不卡在线播放 | 久久人网 | 国产亚洲精品久久久久久大师 | 国产精品精品久久久 | 亚洲国产三级在线 | 99re久久资源最新地址 | 国产综合视频在线观看 | 在线免费观看视频一区 | 91自拍视频在线观看 | 91香蕉视频黄色 | 不卡国产在线 | 国语精品免费视频 | 国产69精品久久久久久久久久 | www.eeuss影院av撸 | 91精品一区在线观看 | 国产日韩高清在线 | av黄网站 | 亚洲一区欧美激情 | 在线免费观看一区二区三区 | 婷婷在线不卡 | 国产成人黄色网址 | 黄色a级片在线观看 | 国产69精品久久久久99尤 | 精品视频免费久久久看 | 国产日韩欧美在线 | 九九九热精品免费视频观看网站 | 玖玖精品在线 | 日本中文乱码卡一卡二新区 | 中文在线中文a | 国产黄网站在线观看 | 亚洲色图激情文学 | 91热爆视频 | 色婷婷亚洲精品 | 久久天天躁狠狠躁亚洲综合公司 | 国产黄色在线看 | 欧美大片mv免费 | 精品视频久久久久久 | 欧美午夜精品久久久久久浪潮 | 激情丁香5月 | 国产精品理论在线观看 | 超碰在线亚洲 | 色a网| 国产精品久久久久久a | 国产视频色 | av电影av在线| 日韩av在线高清 | 天天操天天干天天操天天干 | 午夜精品福利一区二区三区蜜桃 | 国产精品国产亚洲精品看不卡15 | 色网站视频 | 狠狠综合网 | 成人欧美日韩国产 | 亚洲精品综合一区二区 | 91av电影网 | 91精品免费在线视频 | 久久亚洲视频 | 天天射天天干天天插 | 久久高清片 | 亚洲九九九在线观看 | 美女久久久久 | 久久亚洲二区 | 国产经典 欧美精品 | 在线视频婷婷 | 久草视频在线看 | 久久99久久精品国产 | 五月天久久激情 | 超碰在线官网 | 国产精品日韩在线播放 | 色久天| 国产一级黄色电影 | 日韩精品免费一区二区在线观看 | 人人澡人人添人人爽一区二区 | 午夜久久影视 | 一区二区视频免费在线观看 | 日日天天av | 黄av免费| 中文在线a√在线 | 中文字幕资源在线观看 | 国产一区福利 | 天天综合狠狠精品 | 精品你懂的 | 亚洲视频高清 | 亚洲一区二区高潮无套美女 | 色婷婷激情综合 | 亚洲午夜精品福利 | 九九九国产| 成人精品国产免费网站 | 国产精品乱码在线 | 亚洲精品一区二区三区高潮 | 国产精品久久久久aaaa九色 | 高清av在线免费观看 | 成人免费视频网站 | 国产日韩欧美视频在线观看 | 日日躁夜夜躁xxxxaaaa | av在线免费播放 | 国产亚洲精品久久久久久网站 | 日韩h在线观看 | 欧美一区二区三区免费看 | 亚洲激情五月 | 国产黄色av影视 | 最近更新好看的中文字幕 | 国产看片免费 | 永久免费在线 | 成人h视频在线播放 | 操操操日日 | 中国一级片免费看 | 探花视频免费观看高清视频 | 日韩动漫免费观看高清完整版在线观看 | 亚洲区另类春色综合小说校园片 | 午夜精品视频一区 | 日韩一区二区三免费高清在线观看 | 国产精品伦一区二区三区视频 | 亚洲一区二区视频 | 激情婷婷六月 | 五月婷香蕉久色在线看 | 精品国产视频一区 | 久久国产午夜精品理论片最新版本 | 99热9| 99在线视频免费观看 | 成人免费视频在线观看 | 激情欧美日韩一区二区 | 亚洲 欧美日韩 国产 中文 | 国产精品一区二区三区久久久 | 国产精品毛片完整版 | 亚洲乱亚洲乱亚洲 | 超碰人人91| 国产精品一区专区欧美日韩 | 亚洲精品国偷拍自产在线观看蜜桃 | 香蕉久草 | 亚洲免费资源 | 久草在线观看视频免费 | 国产精品婷婷午夜在线观看 | 狠狠色丁香婷婷综合最新地址 | 久久av福利| 亚洲国产精品视频在线观看 | 免费在线精品视频 | 涩涩资源网| 亚洲精品美女久久久久网站 | 亚洲精品久久久蜜桃 | 国产日韩精品一区二区 | 国产欧美精品一区二区三区四区 | 日日夜夜天天久久 | 91精品综合| 久久er99热精品一区二区 | 五月开心婷婷网 | www黄在线| 国产精品久久久久久久7电影 | 日韩久久精品一区二区三区 | 高清免费在线视频 | 三级性生活视频 | 久久久精品二区 | 日韩视频免费在线观看 | 三级毛片视频 | 精品国产黄色片 | 天天躁天天躁天天躁婷 | 制服丝袜欧美 | 婷色| 中文字幕在线播放视频 | 中文字幕免费在线 | 麻豆一区在线观看 | www黄色软件 | 久久精品视频网址 | 成人av亚洲| 91一区二区三区久久久久国产乱 | 友田真希x88av | 国产一级片播放 | 韩日精品在线 | 国产小视频在线免费观看视频 | 精品一区在线看 | 女人18片 | 天天搞天天干天天色 | 91精品综合在线观看 | 日韩中文久久 | 国产免费看| 国产高清网站 | 热热热热热色 | 亚洲 中文 欧美 日韩vr 在线 | 国产日韩欧美自拍 | 性色视频在线 | 亚洲片在线观看 | 2021av在线 | 免费亚洲视频 | 91九色蝌蚪国产 | 日韩av中文在线 | www.狠狠色.com| 免费a级大片 | 成年人在线播放视频 | 99高清视频有精品视频 | 天天操天天玩 | 日韩精品一区二区久久 | 国产精品精品国产色婷婷 | 狠狠色伊人亚洲综合网站野外 | 人人插人人草 | 成人蜜桃| 日韩黄色av网站 | 亚洲国产中文字幕 | 中午字幕在线 | 欧美久久久久久久久久久久 | 一区免费视频 | 欧洲精品久久久久毛片完整版 | 国产超碰在线 | 亚洲精品乱码久久久久久按摩 | 片网址| 久久伊人精品一区二区三区 | 99久久久久久国产精品 | 欧美日韩一区二区久久 | av三级在线免费观看 | 亚洲综合五月天 | 久久久久久高清 | 国产伦理剧| 天天躁天天操 | 九九九九九精品 | 亚洲 欧美变态 另类 综合 | 国产亚洲欧美精品久久久久久 | 国产一级二级在线 | 欧美一区二区三区在线 | 色窝资源 | 99久久精品费精品 | 日韩成人精品一区二区 | 久久久久久久久久免费 | 在线亚洲日本 | 国产流白浆高潮在线观看 | 日韩欧美在线不卡 | 国产原创在线观看 | 99日韩精品| 日本美女xx| 91传媒免费在线观看 | 99久久婷婷国产 | 久久一级片 | av中文天堂在线 | 在线看片一区 | 麻豆影音先锋 | 精品视频亚洲 | 天天综合成人网 | 久久永久视频 | 国产伦理久久精品久久久久_ | 日韩网站免费观看 | 91在线网址 | 精品99久久 | 久久久久亚洲精品中文字幕 | 婷婷六月天丁香 | 免费观看视频的网站 | 久久精品这里都是精品 | 国产精品久久久久久久妇 | 午夜电影中文字幕 | 成人黄色视 | 午夜手机看片 | 国产1区2 | 丝袜美腿亚洲综合 | 国产一区二区三区高清播放 | 婷婷日日 | 五月天丁香| 婷婷射五月 | 天天色天天 | 成人av在线直播 | 摸bbb搡bbb搡bbbb | 国产一级大片在线观看 | 在线黄色av | 超碰人人舔| 91视频 - 114av | 1区2区视频| 97国产精品久久 | 久草在线国产 | 国产不卡视频在线播放 | 国产精品永久 | 色综合天天综合在线视频 | 日韩精品三区四区 | 久草在线免费在线观看 | 午夜av剧场| 日韩精品第1页 | 黄色a视频 | 亚洲一级国产 | 91精品影视 | 国产黄在线观看 | 日韩一区二区三区在线观看 | 超碰人人舔 | 国产精品日韩在线观看 | 激情欧美在线观看 | 久久艹免费 | 国产精品久久久久久久久免费看 | 国产不卡一 | 精品国产欧美一区二区三区不卡 | 亚洲国产精品免费 | 奇米7777狠狠狠琪琪视频 | 久久精品xxx | 亚洲第一中文字幕 | 亚洲专区 国产精品 | 久久久久久久免费 | 激情综合电影网 | 国产精品亚洲综合久久 | 欧美日韩三级 | 国产综合精品一区二区三区 | 天天操天天插 | 久草视频99| 亚洲国产网站 | 久久久久成人免费 | 2023亚洲精品国偷拍自产在线 | 日日操天天操狠狠操 | 毛片888 | 黄色三级在线看 | 日日天天av | 涩涩伊人 | 国产一区视频导航 | www.五月婷婷.com | 丁香九月婷婷 | 日韩精品在线一区 | 国产精品字幕 | 亚洲有 在线 | 国产专区日韩专区 | 在线观看一区二区视频 | 欧美a免费| 成人福利在线 | a级片久久久 | 亚洲永久精品在线 | 一级黄色片在线播放 | av一级网站 | 日韩av一区二区在线影视 | 天天射综合网视频 | 日韩在线观看中文字幕 | 在线观看一 | 久久国产视屏 | 亚州中文av | 91精品福利在线 | 九九99视频 | 日韩毛片在线播放 | 国产精品久久在线观看 | 在线黄色免费av | 天天操夜夜干 | 又黄又爽又色无遮挡免费 | 日韩激情在线视频 | 免费日韩 | 国产精品久久久久av | 久草视频首页 | 黄色亚洲| 91中文在线观看 | 欧美一进一出抽搐大尺度视频 | 亚州精品天堂中文字幕 | 国产精品理论片在线播放 | 狠狠干综合 | 狠狠色丁香久久婷婷综合五月 | 最近字幕在线观看第一季 | 在线观看中文字幕2021 | 亚洲久久视频 | 黄色视屏av | 亚洲性xxxx | 免费a一级 | 国产精品18久久久久vr手机版特色 | a√天堂中文在线 | 91精品999 | 狠狠躁日日躁狂躁夜夜躁av | 精品久久久久久亚洲 | 玖玖在线播放 | 美女视频黄在线观看 | 国产97视频在线 | 九九99| 91毛片在线| 日韩中文字幕在线不卡 | 久久99久久99免费视频 | 黄色av电影网 | 国产精品一区二区免费视频 | 亚洲激精日韩激精欧美精品 | 亚洲成av人影院 | 黄色片网站大全 | 伊人婷婷| 一区二区在线影院 | 日日夜精品 | 六月婷婷网 | 婷婷色吧| 天天草夜夜 | 成人一级黄色片 | 91热精品 | 国产精品 日韩精品 | 天天操天天操天天操 | 久久久久久久久久久久久9999 | 最近最新中文字幕 | 国内外激情视频 | 久久久噜噜噜久久久 | 亚洲精品中文字幕在线 | 欧美日韩精品在线 | 天天干天天射天天操 | 正在播放亚洲精品 | 成人av免费播放 | 玖玖色在线观看 | 在线观看中文字幕一区 | 激情av一区二区 | 免费毛片一区二区三区久久久 | 激情欧美一区二区免费视频 | 国产欧美在线一区二区三区 | 91九色国产| 小草av在线播放 | 亚洲激精日韩激精欧美精品 | 韩国一区二区三区在线观看 | 免费a网址 | 99视频99| 国产成人av网站 | 亚洲欧美va| 国产高清不卡 | 国产精品久久久久毛片大屁完整版 | 婷婷久久婷婷 | 黄色av影院 | 免费福利视频网 | 天天综合中文 | 久久一区精品 | 国产亚洲精品久久久久久久久久 | 天天操天天透 | 日韩久久久久久久久久 | 亚洲日本欧美在线 | 欧美福利久久 | 丁香婷婷综合网 | 91大神精品视频在线观看 | 亚洲国产mv| 色综合天天综合 | 亚洲日韩精品欧美一区二区 | 黄色中文字幕 | 青青河边草免费视频 | 国产小视频在线免费观看视频 | 久久99网 | 99视频在线精品国自产拍免费观看 | 日韩久久久| 成年人网站免费在线观看 | 国产一区二区影院 | 成人免费看片98欧美 | 在线亚洲成人 | 国产精品成人久久久久久久 | 国产无套一区二区三区久久 | 成人h视频在线播放 | 日韩av电影手机在线观看 | 99高清视频有精品视频 | 中文字幕二区在线观看 | 97视频免费播放 | 丁香综合| 夜夜天天干| 久久久精品国产免费观看同学 | 亚洲精品视频网站在线观看 | av在线超碰 | 天天干天天干天天色 | 激情黄色av | 久久福利国产 | 天天天干天天射天天天操 | 国产精品久久久久毛片大屁完整版 | 天天色天天爱天天射综合 | 亚洲激情小视频 | 色综合久久综合中文综合网 | 日韩毛片久久久 | 91精品视频在线观看免费 | 久久综合在线 | 欧美一区影院 | 久草视频在线新免费 | 蜜臀av夜夜澡人人爽人人桃色 | 久草在线久草在线2 | 免费日韩一区二区三区 | 久久草在线视频国产 | 国产婷婷久久 | 久久av伊人 | 欧美污污网站 | 久综合网 | 国产91对白在线播 | 成人a毛片 | 天天干夜夜夜操天 | 午夜性色| 亚洲精品www久久久久久 | 亚洲综合日韩在线 | 成年在线观看 | 久久 亚洲视频 | 日韩av在线影视 | 日批在线看 | 四虎伊人 | 久久在线免费观看视频 | 91在线网址 | 国产高清专区 | 在线中文字幕网站 | 亚洲日b视频 | 婷婷六月网| 97成人在线免费视频 | 97网在线观看 | 香蕉久久久久久久 | 五月天激情婷婷 | 亚洲精品动漫久久久久 | 99久久精品国产亚洲 | 99视频一区二区 | 在线色吧 | 欧美一区日韩一区 | 毛片无卡免费无播放器 | 狠狠干狠狠艹 | 伊人影院在线观看 | 精品久久久久久久久久久久久久久久 | 中文字幕人成不卡一区 | 国产精品theporn | 久草免费看 | 国产精选在线观看 | 日韩电影一区二区三区 | 亚洲乱码精品久久久久 | 999久久久国产精品 高清av免费观看 | 国产精品久久久久久麻豆一区 | 91麻豆免费版 | 九九九电影免费看 | 亚洲日本va中文字幕 | 国产免费黄色 | 婷婷色网站 | 欧美日韩视频精品 | 国产精品美女久久久久久网站 | 亚洲黄色免费网站 | 久久九九久久精品 | 国产精品第7页 | 日韩激情av在线 | 欧美精品一区二区在线播放 | 久久国产精品免费观看 | 97精品国产91久久久久久久 | 国产精品一区二区三区免费视频 | 人人澡人人爱 | 免费看色的网站 | 亚洲精品国产免费 | 91精品视频在线观看免费 | 成人毛片在线观看 | 国产麻豆精品久久一二三 | 99热在线精品观看 | 8x成人在线| 欧美日本在线视频 | 在线视频日韩精品 | 天天色天天操天天爽 | 精品亚洲午夜久久久久91 | 亚洲欧美精品在线 | 少妇性色午夜淫片aaaze | 国产一二三区av | 88av网站| 欧美在线观看视频 | 一区二区三区日韩在线 | 99精品欧美一区二区 | 黄色成年片 | 免费看三片 | 国产成人精品久久亚洲高清不卡 | 国产精品孕妇 | 久草在线免费新视频 | 国产精品久久久久久99 |