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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

U-Boot启动过程完全分析

發布時間:2024/4/18 58 豆豆
生活随笔 收集整理的這篇文章主要介紹了 U-Boot启动过程完全分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

?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×)

?????? 其中: 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,從而完成了命令的執行。

?

?  作者:heaad

?  ?http://www.cnblogs.com/heaad/?

  郵箱:heaad@qq.com

?   本文摘選自作者所寫的一篇文章。轉載請注明,水平有限,歡迎拍磚。?

總結

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

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

亚洲免费永久精品国产 | 国产视频综合在线 | 久久99精品久久久久婷婷 | 五月激情五月激情 | 久久久国产精品一区二区中文 | 免费网站观看www在线观看 | 色av资源网 | 狠狠狠色丁香婷婷综合久久88 | 激情电影影院 | av中文字幕在线播放 | 免费在线激情视频 | 伊人日日干 | 黄网站免费大全入口 | 免费在线激情电影 | 亚洲成人国产精品 | a特级毛片| 91精品视频一区二区三区 | 天天舔夜夜操 | 亚洲欧美视频一区二区三区 | 日韩小视频 | 久久精品国产亚洲精品2020 | 黄色网中文字幕 | 国产亚洲在线观看 | 久久精品观看 | 久久香蕉影视 | ,午夜性刺激免费看视频 | 日韩精品久久久免费观看夜色 | 国产精品婷婷午夜在线观看 | 久久久久久美女 | 一级久久精品 | www国产亚洲精品久久麻豆 | 成人亚洲综合 | 视频一区亚洲 | 成片人卡1卡2卡3手机免费看 | 91porny九色91啦中文 | 九九在线精品视频 | 亚洲一区二区三区在线看 | 久草视频在线观 | 欧美成人精品在线 | 日韩欧在线 | 免费 在线 中文 日本 | 成人一区二区三区在线 | 亚洲精品乱码久久久久久高潮 | 久久久国产一区二区三区四区小说 | 午夜久久福利影院 | 观看免费av | 亚洲一区二区精品3399 | 久青草视频在线观看 | 久久综合之合合综合久久 | 日韩69av | 99色亚洲| 天天综合亚洲 | 国产九九九九九 | 久久午夜免费视频 | 中文字幕观看在线 | 精品免费99久久 | 香蕉视频最新网址 | 久久久黄视频 | 日韩久久一区二区 | 在线电影 你懂得 | 国产精品免费成人 | 午夜在线资源 | 免费三级黄色片 | 免费看三级黄色片 | 麻豆网站免费观看 | 亚洲精品 在线视频 | 成人国产亚洲 | 91精品成人久久 | 久久综合色影院 | 欧美日韩精品在线视频 | 国产主播大尺度精品福利免费 | 亚洲一区二区三区四区在线视频 | 国产尤物视频在线 | 久久久久久久国产精品视频 | 天天干夜夜 | 国产精品日韩在线观看 | 国产精品欧美在线 | 色噜噜狠狠狠狠色综合 | 91精品国产99久久久久 | 亚洲97在线 | 激情av网址 | 久久久国产精品电影 | 激情综合五月网 | 日韩网页 | 久久艹精品 | 国产99re| 国产玖玖精品视频 | 最新中文字幕在线播放 | 天天操狠狠操夜夜操 | 国产美女被啪进深处喷白浆视频 | 国产黄色电影 | 成人免费在线看片 | 国产精久久久久久妇女av | 亚洲精品日韩在线观看 | 99在线观看免费视频精品观看 | 日韩精品一区二区三区高清免费 | 婷婷丁香九月 | 精品久久久久久一区二区里番 | 伊人久在线 | 日韩久久久久久久久 | 亚洲狠狠丁香婷婷综合久久久 | 精品国产一二三四区 | 日本不卡一区二区三区在线观看 | 美女国产精品 | 狠狠操夜夜 | 欧美一区二区在线刺激视频 | 97视频免费观看2区 亚洲视屏 | 中文av一区二区 | 日韩视频中文 | 亚洲欧美日韩国产 | 久久久久国产精品一区二区 | 久久草在线免费 | 99亚洲国产精品 | 国产福利在线 | 一区在线观看 | 免费一级片视频 | 久久人人97超碰国产公开结果 | 成人免费共享视频 | 麻豆高清免费国产一区 | 日韩一区二区在线免费观看 | 亚洲精品久久久久999中文字幕 | 免费视频a | 国产精品美女999 | 精品毛片在线 | 波多野结衣小视频 | 狠狠躁18三区二区一区ai明星 | 免费电影播放 | 五月婷婷在线综合 | 麻花传媒mv免费观看 | 久久精品国产精品亚洲 | 国产精品福利无圣光在线一区 | 一区二区日韩av | 91av福利视频 | 99久久精品免费视频 | 国产中文字幕在线看 | www99久久| 黄色软件视频大全免费下载 | 三级av小说| 欧美精彩视频 | 亚洲无吗av | 久久成人国产精品一区二区 | 又紧又大又爽精品一区二区 | 91视频免费观看 | 日本大尺码专区mv | 免费在线日韩 | 国产免费又粗又猛又爽 | 日韩高清免费在线观看 | 日韩精品偷拍 | 亚洲国产午夜视频 | 国产精品九九九九九 | 在线中文字幕一区二区 | 玖玖玖在线 | 日本精品久久久久 | 国内精品久久久久久中文字幕 | 久久精品99国产精品酒店日本 | 一级片色播影院 | 成人黄色片免费 | 日韩在线观看精品 | 丁香花在线视频观看免费 | www.狠狠干| 丁香久久激情 | 精品免费观看视频 | 2023国产精品自产拍在线观看 | 91成人精品国产刺激国语对白 | 激情自拍av | 亚洲首页 | 天天翘av | 人人人爽| 92av视频| 狠狠的干狠狠的操 | 丁香午夜婷婷 | 青青射| 99久久精品国产毛片 | 日韩大片在线免费观看 | 狠狠色噜噜狠狠 | 国产人在线成免费视频 | 久久艹精品 | 国产视频999 | 伊人婷婷网 | 波多野结衣电影一区 | 国产午夜剧场 | 日韩欧美视频在线播放 | 久草在线免费在线观看 | 男女激情网址 | 亚洲精品国偷自产在线99热 | 婷婷六月丁香激情 | 国产成人av网站 | 亚洲va在线va天堂va偷拍 | 99在线观看免费视频精品观看 | 成人av电影免费在线观看 | 国产精品自产拍在线观看蜜 | 麻豆小视频在线观看 | 综合色狠狠 | 韩国视频一区二区三区 | 干干干操操操 | 日本久久免费电影 | av韩国在线 | 国产在线高清视频 | www.黄色小说.com| 日日干网| 成人全视频免费观看在线看 | 国产 欧美 日产久久 | 国内精品99| 毛片网在线观看 | 国产精品丝袜久久久久久久不卡 | 99久久99久久精品国产片 | 91精品亚洲影视在线观看 | 青青草久草在线 | 亚洲少妇xxxx| 夜夜视频资源 | 欧洲精品久久久久毛片完整版 | 精品福利在线 | 伊人久操 | 日韩av免费在线看 | 国产视频精品免费 | 久久国产精品区 | 久久在线精品 | 99视频精品全部免费 在线 | 香蕉视频网址 | 在线视频精品播放 | 日日爽夜夜爽 | 国产激情小视频在线观看 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 亚州精品天堂中文字幕 | 九九日韩| 综合网成人 | 中文字幕婷婷 | 亚洲 欧洲 国产 日本 综合 | 在线观看成人毛片 | 日韩在线观看三区 | 久久久综合精品 | 五月天精品视频 | 五月天亚洲激情 | 日韩激情片在线观看 | 亚洲在线综合 | 91福利在线观看 | 亚洲精品在线资源 | 在线观看免费日韩 | 成人av电影在线观看 | 中文字幕在线观看免费高清电影 | 成 人 黄 色 视频 免费观看 | 在线一二区 | 欧美福利视频一区 | 色婷久久 | 国精产品999国精产品视频 | 91av欧美 | 中文字幕在线一区二区三区 | 一区二区三区免费在线观看视频 | 18+视频网站链接 | 久久午夜免费观看 | 久久久久久久久久久久久影院 | www免费看片com| 九九久久免费视频 | 综合色影院 | 97超碰站 | 精品极品在线 | 午夜性福利| 欧美精品一二三 | 午夜视频久久久 | 午夜精品一区二区三区免费视频 | 国产91在线播放 | 国产精品久久一区二区无卡 | 中文字幕在线观看2018 | 天天拍天天色 | 成人黄色片在线播放 | 蜜臀av夜夜澡人人爽人人桃色 | 中文字幕网站视频在线 | 色婷婷狠狠五月综合天色拍 | 色停停五月天 | 夜夜爱av | 美女在线观看网站 | 免费在线观看视频a | 国产精品久久久久久超碰 | 夜色资源站wwwcom | 九九九在线观看视频 | 日韩精品一区二区免费视频 | 精品欧美乱码久久久久久 | 成人黄色片免费看 | 日本精品久久久久影院 | 又色又爽又激情的59视频 | 91亚洲精品国偷拍 | 色网站中文字幕 | 精品一二三区 | 91精品人成在线观看 | 国产一级黄| 草久久影院 | 91精品久久久久久久99蜜桃 | 日韩免费视频观看 | 色综合天天综合网国产成人网 | 揉bbb玩bbb少妇bbb| 久久免费视频5 | 在线成人性视频 | 国产精品久久99综合免费观看尤物 | 亚洲精品午夜久久久久久久 | 99久热在线精品视频观看 | 国产美女视频免费观看的网站 | www.夜夜| 91av视频| 久久久免费精品 | 麻花豆传媒mv在线观看 | 国产精品久久9 | 亚洲另类视频 | 在线视频免费观看 | 欧美黑人性爽 | 91桃色视频| 日韩精品一区二区免费视频 | 日本黄色大片免费看 | 成人a在线观看高清电影 | 天天插综合| 亚洲精品综合欧美二区变态 | 亚洲午夜精品福利 | 中文字幕在线观看资源 | 国产精品嫩草影院99网站 | 一区二区电影在线观看 | 久久色视频 | 免费看毛片在线 | 在线国产片 | 九九久久电影 | 欧美精品久久久久久久 | 日本精品一区二区在线观看 | 国产精品在线看 | 欧美激情精品久久久 | 中文字幕在线观看1 | 日本久久免费视频 | 亚洲日本中文字幕在线观看 | 91伊人久久大香线蕉蜜芽人口 | 久久久久久久久久影视 | 日韩精品一区不卡 | 国产精品中文字幕在线观看 | 日韩久久精品一区二区三区下载 | 国产高清免费在线播放 | 粉嫩一二三区 | av色网站| 亚洲精品乱码久久久一二三 | 国产一级做a | 精品视频国产 | 久久理论视频 | 亚洲永久在线 | 成人久久久久久久久久 | 97超碰在线免费观看 | 久久免费视频在线观看6 | 色综合狠狠干 | 欧美一级久久 | 日韩经典一区二区三区 | 蜜臀久久99精品久久久久久网站 | 成人毛片在线观看 | 水蜜桃亚洲一二三四在线 | 91秒拍国产福利一区 | 午夜少妇av | 久久爱影视i | 中文字幕在线一区二区三区 | 亚洲涩涩网 | 久久精品久久久精品美女 | 成人在线视频你懂的 | 国产一区久久 | 日韩在线资源 | 久久久久久久久亚洲精品 | 亚洲一二三区精品 | 在线播放亚洲 | 综合亚洲视频 | 在线播放 日韩专区 | 欧美大片mv免费 | 久久久国产一区二区三区 | 午夜精品久久久久久久99热影院 | 日韩在线首页 | 日韩av图片 | 欧美另类网站 | 国产一级片播放 | 开心综合网 | 欧美一二区视频 | 天天爱天天干天天爽 | 99视频在线播放 | 久视频在线播放 | 色香网 | 亚洲国产精品成人av | 色丁香婷婷 | 三级黄色大片在线观看 | 国产精品久久久久久久久费观看 | 一本一本久久a久久精品综合妖精 | 国产综合激情 | 久久精品视频日本 | 五月网婷婷 | 国产精品毛片久久久久久久 | 欧美日韩午夜 | 久久免费高清视频 | 国产美女精彩久久 | 93久久精品日日躁夜夜躁欧美 | 超碰官网 | 国产精品自在欧美一区 | 亚洲精品国产欧美在线观看 | 97超碰人人模人人人爽人人爱 | 亚洲视频综合 | 很污的网站 | 亚洲春色综合另类校园电影 | 国产一区二区三区久久久 | 亚洲精品视频免费在线观看 | 精品视频www | 狠狠色狠狠色终合网 | 午夜精品av | 三级av免费 | 999亚洲国产996395 | 久久免费视频在线 | 亚洲欧美日韩不卡 | 久久成人高清视频 | 人人插人人做 | 中文字幕中文字幕 | 麻豆视频在线免费 | 日韩精品一区在线观看 | 亚洲精品国产精品国 | 国产日韩精品在线 | 国产精品美女免费视频 | 成人在线视频免费观看 | 夜夜夜夜夜夜操 | 精品国产一区在线观看 | 一区二区视频在线免费观看 | 国产一级高清 | 国内视频在线观看 | 精品国产免费一区二区三区五区 | 欧美午夜精品久久久久久孕妇 | 中文字幕二区三区 | 日日夜夜天天操 | 亚洲精品999 | 日韩av中文字幕在线免费观看 | www成人av| 日日干狠狠操 | 中文字幕刺激在线 | 久久久免费视频播放 | www.夜夜操.com | 天天射天天爱天天干 | 色a网| 免费av一级电影 | 国产亚洲精品美女 | 久久久久久久久久网站 | 一区久久久 | 三级黄色大片在线观看 | 在线免费亚洲 | 国产精品99精品久久免费 | 国产福利一区二区三区在线观看 | 日韩精品一区二区三区第95 | 最近中文字幕大全中文字幕免费 | 欧美久久久久久久久久久 | 丁香六月激情婷婷 | 久久狠狠婷婷 | 婷婷综合国产 | 日韩免费av网址 | 久久99热这里只有精品国产 | 综合久久综合久久 | 九九热有精品 | 麻花豆传媒mv在线观看 | 又爽又黄又无遮挡网站动态图 | 午夜国产福利视频 | 91精品啪在线观看国产线免费 | 国产91aaa | 国产精品久久久久四虎 | 国产成人亚洲在线观看 | 91精品秘密在线观看 | 欧美在线视频第一页 | 亚洲黄色在线免费观看 | 日韩免费小视频 | 91九色国产蝌蚪 | 中文字幕在线观看视频一区 | 1区2区视频| 亚洲成人动漫在线观看 | 九草视频在线 | 精品一二三区 | 久久女同性恋中文字幕 | 九九色综合 | 人人爱人人添 | 久草在线手机视频 | 99久久精品免费看国产免费软件 | 91香蕉久久 | 日韩一级电影在线观看 | 久草视频在线免费 | 国产视频一区在线免费观看 | 在线日韩三级 | 911久久香蕉国产线看观看 | 美女又爽又黄 | 胖bbbb搡bbbb擦bbbb| 亚洲日本国产精品 | 911精品美国片911久久久 | av线上免费观看 | 97人人模人人爽人人少妇 | 亚洲欧美日本国产 | 国产精品资源网 | 午夜视频一区二区三区 | 国产91区 | 91精品夜夜| 91精品国自产在线观看 | 精品福利视频在线观看 | 久久,天天综合 | 日本三级国产 | 91精品办公室少妇高潮对白 | 成人在线一区二区三区 | 中文字幕你懂的 | 最新99热| 国产精品对白一区二区三区 | 欧美色综合天天久久综合精品 | 视频直播国产精品 | 超碰97.com| 国产精品欧美激情在线观看 | 婷婷中文字幕在线观看 | 深爱五月网| 成人国产精品 | 国产精品毛片网 | 婷婷中文在线 | 黄色大全在线观看 | 日韩精品电影在线播放 | 亚洲欧美国产精品 | 色综合五月天 | 日韩videos | 成人久久18免费网站 | 中文字幕在线视频国产 | 国产伦精品一区二区三区照片91 | 国产精品亚洲片夜色在线 | 激情婷婷在线观看 | 精品视频免费在线 | 97福利在线 | 久久人人爽人人爽人人片av免费 | 婷婷丁香av | 精品久久精品 | av再线观看| 亚洲 欧洲av | 国产大尺度视频 | 久久久久亚洲国产精品 | 五月婷社区| 97超级碰碰碰碰久久久久 | 欧美 日韩 国产 成人 在线 | 欧美日韩国产精品一区 | 成人黄色毛片视频 | 国产精品国产三级国产aⅴ无密码 | 在线观看成人av | 9久久精品 | 国产精品福利一区 | 99色婷婷 | 色天天天 | av一级一片 | 99久久精品午夜一区二区小说 | 日韩欧美综合 | av一区二区三区在线观看 | 欧美精品一区二区免费 | 日韩影片在线观看 | 国语精品久久 | 成人免费在线播放视频 | 免费看的黄色网 | 亚洲精品午夜久久久久久久久久久 | 国产精品美女免费视频 | 国内久久久久久 | 91av99 | 在线电影日韩 | 欧美性久久久久久 | 亚洲精品国产欧美在线观看 | 欧美一区,二区 | 午夜性福利 | 91精品一区二区三区蜜臀 | 国产剧情一区在线 | 精品在线观看视频 | 天堂av最新网址 | 日本巨乳在线 | 亚州欧美精品 | 免费观看全黄做爰大片国产 | 久久综合久久综合这里只有精品 | 国产夫妻自拍av | 日本精品视频一区二区 | 91视频免费看 | 成人av视屏 | 国产精品一区二区av影院萌芽 | 91尤物国产尤物福利在线播放 | 日本69hd| 亚洲精品自拍视频在线观看 | 日韩av电影国产 | 亚洲激情 在线 | 日日干天天射 | 久久成人一区二区 | 99人成在线观看视频 | 人人澡人人舔 | 不卡精品视频 | 欧美在线a视频 | 国产91在线播放 | 夜夜操网站 | 中文字幕999 | 免费男女羞羞的视频网站中文字幕 | 97香蕉久久超级碰碰高清版 | 精品视频中文字幕 | 久久精品一区二区三区中文字幕 | 国产九九九精品视频 | 一本—道久久a久久精品蜜桃 | 天天操天天操天天干 | 国产69精品久久app免费版 | 在线观看国产区 | 日韩在线免费播放 | 欧美一级免费高清 | 国产精品久久久久久吹潮天美传媒 | 国产精选在线观看 | 色婷婷亚洲婷婷 | 日日碰狠狠添天天爽超碰97久久 | 久久精品国产免费看久久精品 | 日韩av在线影视 | 九色免费视频 | 国内免费久久久久久久久久久 | 中文字幕在线影视资源 | 国产在线欧美日韩 | 婷婷激情av | 亚洲理论片在线观看 | 中文字幕最新精品 | 国产黄色在线网站 | 国产在线 一区二区三区 | 国产精品一区久久久久 | 超碰精品在线观看 | 91精品影视 | 亚洲国产久 | 亚洲伊人婷婷 | 成人久久久久久久久久 | 丁香婷婷色| 精品久久91 | 99资源网| 国产精品剧情在线亚洲 | 国产精品福利在线播放 | 91精品国产99久久久久久红楼 | 黄色中文字幕在线 | 91一区一区三区 | 久久免费看av | 亚洲一区二区三区在线看 | 国产美女免费看 | 欧美人操人 | 国产精品久久久久久久7电影 | 国产无遮挡猛进猛出免费软件 | 97超碰在线视 | 久久只有精品 | 欧美精品在线免费 | 一区二区三区中文字幕在线观看 | 97天天干| 国产一级片视频 | 日本一区二区免费在线观看 | 亚洲成人av片 | 国产精品久久一区二区无卡 | 日韩成人高清在线 | 亚洲1级片 | 国产福利精品一区二区 | 99视频一区二区 | 久久免费99精品久久久久久 | 日韩二区三区在线观看 | 久久精品99国产精品酒店日本 | 欧美成亚洲| 久久精品国产亚洲精品2020 | 91精品少妇偷拍99 | 成人免费在线看片 | 91天天操 | 中文字幕色在线视频 | 亚洲视频专区在线 | 亚洲精品国产麻豆 | 夜夜躁天天躁很躁波 | 五月激情婷婷丁香 | 91最新地址永久入口 | 成人久久18免费网站麻豆 | 96香蕉视频 | 色香蕉在线视频 | 国产一区二区在线看 | 国产亚洲日 | 日韩免费三区 | 国产精品福利午夜在线观看 | 国产日韩视频在线观看 | 91av在线播放视频 | 欧美日韩伦理在线 | 91网页版在线观看 | 婷婷亚洲最大 | 久久久久国产成人精品亚洲午夜 | 黄色一级免费 | 五月天久久狠狠 | 天天综合操 | 综合久久精品 | 日韩电影中文 | 欧美一级在线观看视频 | 日韩高清一区在线 | 视频在线亚洲 | 天天操天天操天天操天天操天天操天天操 | 免费在线视频一区二区 | 欧美另类一二三四区 | 婷婷新五月 | 成av人电影| 狠狠操电影网 | 亚洲国产精品人久久电影 | 97精品久久人人爽人人爽 | 亚洲国产中文字幕在线观看 | 天天精品视频 | 国产成人三级在线播放 | 欧美日韩国产在线精品 | 国产精品不卡一区 | 最近中文字幕在线中文高清版 | 亚洲在线色 | 免费视频久久久久 | 国产一级视频在线 | av高清影院 | 成人三级视频 | 成人黄色在线看 | 亚洲网站在线看 | 免费a级毛片在线看 | 超碰九九| 久久久久久久久久久免费视频 | 97成人在线观看 | 国产精品你懂的在线观看 | 亚洲无吗天堂 | 日韩激情免费视频 | 国产成人黄色片 | 欧美一级在线观看视频 | 久久手机免费视频 | 国产在线欧美 | 久久极品 | 国产馆在线播放 | 国内精品一区二区 | 日韩精品欧美专区 | 黄网站www| 免费福利影院 | 超碰97在线看 | 精品九九久久 | 精品国产一区二区三区久久影院 | 五月婷婷在线综合 | 日韩精品一区电影 | 欧美精品v国产精品v日韩精品 | 久久久2o19精品 | 国产色黄网站 | 日日夜夜草 | 欧美日韩精品在线播放 | 欧美另类高清 videos | 久久久久久久综合色一本 | 婷婷久久久久 | 色偷偷男人的天堂av | 91天堂素人约啪 | 欧美一级裸体视频 | 91人人澡人人爽人人精品 | 免费福利视频网站 | 手机在线中文字幕 | 精品久久久久免费极品大片 | 欧美精品在线观看一区 | 玖玖色在线观看 | 91在线精品视频 | av中文字幕在线看 | 中文字幕一区二区三区四区视频 | 黄色影院在线播放 | 成人久久久久久久久 | 手机av电影在线观看 | 国产高清视频在线播放 | 亚洲免费国产视频 | 欧美一区二区伦理片 | 欧美精品久久久久久久免费 | 日韩欧美一区二区三区黑寡妇 | 免费看片成年人 | 国产h片在线观看 | 日韩精品久久久 | 色 中文字幕| 97热久久免费频精品99 | 亚洲人成人天堂h久久 | 五月婷婷在线综合 | 最新av在线播放 | 国产探花视频在线播放 | 二区精品视频 | 久久艹综合 | 97国产一区 | 五月天综合婷婷 | 日批网站在线观看 | 日韩欧美国产免费播放 | 中文字幕婷婷 | 少妇搡bbbb搡bbb搡69 | 国模精品一区二区三区 | 青青色影院 | 黄色av影视| 久久成人国产精品免费软件 | 亚洲日b视频 | 久久精品理论 | 亚洲视频h | 久久色中文字幕 | 黄色一区二区在线观看 | 国产一线二线三线性视频 | 久草.com| 99精品视频免费看 | 久久久久欠精品国产毛片国产毛生 | 国产精品岛国久久久久久久久红粉 | 成人a毛片 | 182午夜在线观看 | 亚洲精品久久激情国产片 | 中文字幕人成不卡一区 | av噜噜噜在线播放 | 午夜av色 | 五月婷婷综合在线视频 | 午夜在线观看一区 | 韩国在线一区二区 | 亚洲成人网在线 | 久久亚洲福利 | 永久免费精品视频网站 | 最近更新中文字幕 | 不卡国产在线 | 欧美一二区在线 | 狠狠色2019综合网 | 蜜臀av在线一区二区三区 | 91传媒在线观看 | 国产精品欧美激情在线观看 | 国产又粗又硬又长又爽的视频 | 91超碰免费在线 | 欧美日韩xxxxx | 欧洲色综合 | 激情欧美国产 | 欧美日韩69| 久久蜜桃av | 国产亚洲一级高清 | www.888av| 91热精品| 麻豆视频国产精品 | 午夜久久网站 | 超碰人人乐 | 91精品日韩 | 色婷婷激情电影 | 亚洲欧洲美洲av | 久久综合九色综合欧美就去吻 | 国产不卡视频在线播放 | 久久免费av | 在线免费观看欧美日韩 | 蜜桃视频在线观看一区 | 亚洲成人黄色av | 97在线看 | 日韩视频免费 | 国产在线不卡 | 91香蕉视频好色先生 | 日本久久综合网 | 激情综合狠狠 | 一级黄色片在线免费观看 | 欧美性生活免费 | 日韩视频在线一区 | 免费看在线看www777 | 夜夜骑日日 | 国产精品99久久久精品免费观看 | 国产小视频你懂的在线 | 九九色在线观看 | 国产精品中文久久久久久久 | 亚洲黄网站 | 欧美亚洲国产一卡 | 中文字幕免费不卡视频 | 久久久国产精品一区二区中文 | 日韩精品在线观看av | 亚洲视频www | 日韩在线观看中文 | 一区二区三区日韩在线观看 | 亚洲免费永久精品国产 | www日韩欧美 | 在线观看视频福利 | 美女av免费看 | 亚州av一区 | 蜜臀久久99精品久久久久久网站 | 99 色| 欧美大片在线观看一区 | 狠狠色丁香婷婷综合久小说久 | 手机看片1042| 欧美 日韩精品 | 大胆欧美gogo免费视频一二区 | 黄色精品久久 | 久久国产精品99久久人人澡 | 日韩在线观看视频免费 | 免费黄色av电影 | 亚洲另类久久 | 亚洲欧美日韩国产一区二区三区 | 日韩偷拍精品 | 日韩中文字幕免费看 | 国产精品网址在线观看 | 在线观看色网站 | 免费三级骚 | 视频一区二区免费 | 99视频网站 | www.五月天婷婷 | 久久久麻豆 | 色综合 久久精品 | 视频在线观看99 | 国产高清成人av | 久久黄色a级片 | 欧美日韩不卡一区二区三区 | 成人黄色一级视频 | 亚洲精品女 | 久久五月婷婷丁香社区 | 中文字幕在线播放视频 | 蜜臀久久99精品久久久酒店新书 | 涩涩成人在线 | 国产中文字幕视频在线观看 | 国产精品一码二码三码在线 | 天天操天天干天天综合网 | 欧美色黄 | 狠狠色丁香婷婷 | 欧美精品久久久久久久免费 | 丁香激情综合国产 | 2020天天干天天操 | 欧美日性视频 | 中文字幕日韩无 | 国产综合福利在线 | 热久久99这里有精品 | 97综合视频 | 国产视频精选 | 天天射天天射 | 日韩经典一区二区三区 | 亚洲国产精品99久久久久久久久 | 久久中文精品视频 | 黄网站免费大全入口 | 91视频在线免费看 | 97久久精品午夜一区二区 | 美女国内精品自产拍在线播放 | 亚洲视频 一区 | 国产亚洲精品久久久久久久久久 | 91视频麻豆 | 成人av在线一区二区 | 亚洲国产成人精品在线 | 中文字幕欧美三区 | 欧美日韩一区二区在线 | avcom在线 | 在线观看久久 | 亚洲污视频 | 亚洲精品高清视频在线观看 | 成年人黄色在线观看 | 久久九九国产精品 | 麻豆视频免费在线观看 | 9999精品免费视频 | 九九国产视频 | wwwwww国产 | 91成人在线网站 | 欧美二区三区91 | 99久久99热这里只有精品 | 91视频在线免费观看 | 91九色蝌蚪视频在线 | 久久这里有精品 | 国产露脸91国语对白 | 激情自拍av | 欧美成人h版在线观看 | 91高清不卡 | 国产精品一区二区三区视频免费 | 狠狠色丁香婷婷综合最新地址 | 亚洲高清在线视频 | 久久专区 | 好看的国产精品视频 | 国产精品久久久久久久久久久久久久 | av免费在线看网站 | 亚洲高清精品在线 | 亚洲人在线 | 一区二区三区四区五区在线视频 | 深夜福利视频在线观看 | 欧美91精品| 久久久久久久久久久免费 | 日韩欧美精选 | 人人干网站 | 狠狠操91| 日本公乱妇视频 | 国模视频一区二区三区 | 91麻豆免费视频 | 国产98色在线 | 日韩 | 99色亚洲 | 日本视频久久久 | 天天搞夜夜骑 | 99精品免费久久久久久日本 | 天堂麻豆 | 欧美日韩国产成人 | 日本3级在线观看 | 亚洲国产日本 | 在线视频日韩一区 | 国产男女免费完整视频 | 91黄色免费网站 | 五月婷久久 | 不卡的av片 | 91传媒在线播放 | 操夜夜操 | 91精品视频在线免费观看 | 黄污视频大全 | 九九热免费观看 | 一区二区男女 | 99综合电影在线视频 | 97人人澡人人添人人爽超碰 | 日韩电影在线一区 | 91精品一区国产高清在线gif | 91最新地址永久入口 | 国产欧美在线一区二区三区 | 亚洲香蕉在线观看 | 久久人人爽爽人人爽人人片av | 麻豆传媒视频在线 | 99精品欧美一区二区蜜桃免费 | 国产中文在线观看 | 免费看的黄色的网站 | 西西www444 | 日韩av在线一区二区 | 97视频入口免费观看 | 一区二区电影网 | 亚洲免费在线观看视频 | 国产一区国产二区在线观看 | 手机看片福利 | www.天天操.com | 99热精品在线观看 | av高清影院| 亚洲一级理论片 | 久久曰视频| 天天视频色版 | 国产女人18毛片水真多18精品 | 欧美久久久影院 | 永久黄网站色视频免费观看w | 9992tv成人免费看片 | 久久久久久毛片精品免费不卡 | 日本精品一区二区三区在线播放视频 | 亚洲少妇影院 |