日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]Ok6410的硬件相比mini6410強(qiáng)大許多(同樣的價錢),但是ok6410的uboot制作用起來不方便,需要輸入很多命令才可以燒寫完一個系統(tǒng)。我還是比較懷念在2440上方便、靈活的燒寫方式。 下面我們就來修改出一個簡單的uboot

Ok6410的硬件相比mini6410強(qiáng)大許多(同樣的價錢),但是ok6410的uboot制作用起來不方便,需要輸入很多命令才可以燒寫完一個系統(tǒng)。我還是比較懷念在2440上方便、靈活的燒寫方式。
下面我們就來修改出一個簡單的uboot,實(shí)現(xiàn)快速更新系統(tǒng)。
一、首先簡單的說明uboot的啟動過程:
1)、從文件層面上看主要流程是在兩個文件中:cpu/xxxx/start.s,lib_arm/board.c。
Start.s
在flash中執(zhí)行的引導(dǎo)代碼,也就是bootloader中的stage1,負(fù)責(zé)初始化硬件環(huán)境,把u-boot從flash加載到RAM中去,然后跳到lib_arm/board.c中的start_armboot中去執(zhí)行。
1.1.6版本的start.s流程:
硬件環(huán)境初始化:
進(jìn)入svc模式-->關(guān)閉watch dog-->屏蔽所有IRQ掩碼-->設(shè)置時鐘頻率FCLK、HCLK、PCLK-->清I/D cache-->禁止MMU和CACHE-->配置memory control-->重定位:如果當(dāng)前代碼不在連接指定的地址上(對smdk2410是0x3f000000)則需要把u-boot從當(dāng)前位置拷貝到RAM指定位置中;-->建立堆棧,堆棧是進(jìn)入C函數(shù)前必須初始化的。-->清.bss區(qū)。-->跳到start_armboot函數(shù)中執(zhí)行。(lib_arm/board.c)
2)、lib_arm/board.c:
start_armboot是U-Boot執(zhí)行的第一個C語言函數(shù),完成系統(tǒng)初始化工作,進(jìn)入主循環(huán),處理用戶輸入的命令。這里只簡要列出了主要執(zhí)行的函數(shù)流程:
void start_armboot (void)
{
//全局?jǐn)?shù)據(jù)變量指針gd占用r8。
DECLARE_GLOBAL_DATA_PTR;
/* 給全局?jǐn)?shù)據(jù)變量gd安排空間*/
gd = (gd_t*)(_armboot_start - CFG_MALLOC_LEN - sizeof(gd_t));
memset ((void*)gd, 0, sizeof (gd_t));
/* 給板子數(shù)據(jù)變量gd->bd安排空間*/
gd->bd = (bd_t*)((char*)gd - sizeof(bd_t));
memset (gd->bd, 0, sizeof (bd_t));
monitor_flash_len = _bss_start - _armboot_start;//取u-boot的長度。
/* 順序執(zhí)行init_sequence數(shù)組中的初始化函數(shù) */
for (init_fnc_ptr = init_sequence; *init_fnc_ptr; ++init_fnc_ptr)
if ((*init_fnc_ptr)() != 0) { hang ();}
/*配置可用的Flash */
size = flash_init ();
 ……
/* 初始化堆空間 */
mem_malloc_init (_armboot_start - CFG_MALLOC_LEN);
/* 重新定位環(huán)境變量, */
env_relocate ();
/* 從環(huán)境變量中獲取IP地址 */
gd->bd->bi_ip_addr = getenv_IPaddr ("ipaddr");
/* 以太網(wǎng)接口MAC 地址 */
……
devices_init (); /* 設(shè)備初始化 */
jumptable_init ();//跳轉(zhuǎn)表初始化
console_init_r (); /* 完整地初始化控制臺設(shè)備 */
enable_interrupts (); /* 使能中斷處理 */
/* 通過環(huán)境變量初始化 */
if ((s = getenv ("loadaddr")) != NULL)
load_addr = simple_strtoul (s, NULL, 16);
/* main_loop()循環(huán)不斷執(zhí)行 */
for (;;)
main_loop (); /* 主循環(huán)函數(shù)處理執(zhí)行用戶命令 -- common/main.c */
}
其中,初始化函數(shù)序列init_sequence[]
init_sequence[]數(shù)組保存著基本的初始化函數(shù)指針。這些函數(shù)名稱和實(shí)現(xiàn)的程序文件在下列注釋中。
init_fnc_t *init_sequence[] = {
cpu_init, /* 基本的處理器相關(guān)配置 -- cpu/arm920t/cpu.c */
board_init, /* 基本的板級相關(guān)配置 -- board/smdk2410/smdk2410.c */
interrupt_init,/* 初始化例外處理 -- cpu/arm920t/s3c24x0/interrupt.c */
env_init, /* 初始化環(huán)境變量 -- common/env_flash.c */
init_baudrate, /* 初始化波特率設(shè)置 -- lib_arm/board.c */
serial_init, /* 串口通訊設(shè)置 -- cpu/arm920t/s3c24x0/serial.c */
console_init_f, /* 控制臺初始化階段1 -- common/console.c */
display_banner, /* 打印u-boot信息 -- lib_arm/board.c */
dram_init, /* 配置可用的RAM -- board/smdk2410/smdk2410.c */
display_dram_config,/* 顯示RAM的配置大小 -- lib_arm/board.c */
NULL,
};
整個u-boot的執(zhí)行就進(jìn)入等待用戶輸入命令,解析并執(zhí)行命令的死循環(huán)中。
二、修改main_loop()函數(shù)
我們期望此時能進(jìn)入一個菜單,通過輸入一些簡單的指令來更新uboot、kernel、fs等。
1)、分析/common/main.c的main_loop函數(shù):系統(tǒng)進(jìn)入到main_loop
后首先判斷在3秒內(nèi)是否有輸入。如果有輸入就進(jìn)入命令行模式,我們可以在此命令行下通過輸入指令來更新系統(tǒng)。如果沒有輸入則執(zhí)行bootm 指令。
首先,我自己試過幾次,如果用run_command來保持環(huán)境變量“setenv bootcmd nand read 0xc0008000 0x100000 0x300000;bootm 0xc0008000 ”,系統(tǒng)會直接重啟。比較郁悶,想了變通的方法就是在/include/configs/smdk6410.h文件里直接修改CONFIG_BOOTCOMMAND 為nand read c0008000 100000 300000;bootm c0008000。據(jù)我分析系統(tǒng)啟動后會從這個宏里讀取bootcmd參數(shù)。(如果有不對的,請高手指出) 那么我們后面就不用再設(shè)定這個參數(shù)了。


void main_loop (void)
{
…….
s = getenv ("bootcmd"); //獲取bootcmd指令
debug ("### main_loop: bootcmd="%s"n", s ? s : "");
if (bootdelay >= 0 && s && !abortboot (bootdelay)) {
// abortboot()主要是判斷bootdelay的時間內(nèi),是否有按鍵輸入。
……
run_command (s, 0); //如果沒有輸入就執(zhí)行bootcmd指令
}
//進(jìn)入一個命令行模式,循環(huán)接受用戶指令。
//我們就在這里加入一個我們自己的函數(shù)MainMenu()用來執(zhí)行菜單。
MainMenu();
/*
* Main Loop for Monitor Command Processing
*/
……
}
復(fù)制代碼
2)、在該文件的開頭申明一個函數(shù)MainMenu():void MainMenu()
3)、定義我們的MainMenu()
void main_menu_usage(void)
{
printf("rn##### ok6410 Bootloader #####rn");
printf("[u] Download u-bootrn");
printf("[k] Download Linux kernelrn");
//printf("[y] Download YAFFS imagern");
printf("[c] Download cramfs imagern");
//printf("[d] Download to SDRAM & Runrn");
printf("[b] Boot the systemrn");
printf("[f] Format the Nand Flashrn");
printf("[s] Set the boot parametersrn");
printf("[r] Reboot u-bootrn");
printf("[q] Quit from menurn");
printf("Enter your selection: ");
}
void MainMenu()
{
char c;
char cmd_buf[256];
char name_buf[20];
char val_buf[256];
while (1)
{
main_menu_usage(); //輸出菜單的函數(shù)
c = getc(); //獲取輸入的字符
printf("%cn", c);
switch (c)
{
case ''u'': //燒寫uboot
printf("nand erase nand and write uboot n");
strcpy(cmd_buf, "dnw c0008000 ; nand erase 0 100000 ; nand write 0xc0008000 0 100000");
printf("%s n",cmd_buf);
run_command(cmd_buf, 0);
break;
case ''k'': //燒寫kernel
//先設(shè)定環(huán)境變量
strcpy(cmd_buf,"setenv bootargs "root=/dev/mtdblock2 console=ttySAC0,115200"“);
run_command(cmd_buf,0);
run_command("saveenv",0);
strcpy(cmd_buf, "dnw c0008000; nand erase 100000 300000 ; nand write.e 0xc0008000 100000 300000");
printf("%s n",cmd_buf);
run_command(cmd_buf, 0);
break;
case ''c'': //燒寫cramfs文件系統(tǒng)
strcpy(cmd_buf, "dnw 0xc0008000; nand erase 400000 5000000 ; nand write.e 0xc0008000 400000 5000000");
printf("%s n",cmd_buf);
run_command(cmd_buf, 0);
strcpy(cmd_buf, "setenv bootargs "root=/dev/mtdblock2 rootfstype=cramfs console=ttySAC0,115200"");
printf("%s n",cmd_buf);
run_command(cmd_buf, 0);
strcpy(cmd_buf, "saveenv");
printf("%s n",cmd_buf);
run_command(cmd_buf, 0);
break;
case ''b'': //bootm 重啟
printf("Booting Linux ...n");
strcpy(cmd_buf, "nand read 0xc0008000 0x100000 0x300000;bootm 0xc0008000");
printf("%sn",cmd_buf);
run_command(cmd_buf, 0);
break;
case ''f'': //format flash
strcpy(cmd_buf, "nand scrub");
printf("%sn",cmd_buf);
run_command(cmd_buf, 0);
break;
case ''s'': //更改環(huán)境參數(shù)
param_menu_shell();這部分函數(shù)需要自己寫 :)
break;
case ''q'': //退出菜單
return;
break;
}
}
}
復(fù)制代碼
4)、加入?yún)?shù)修改的菜單函數(shù)
void param_menu_usage()
{
printf("rn##### Parameter Menu #####rn");
printf("[v] View the parametersrn");
printf("[s] Set parameter rn");
printf("[d] Delete parameter rn");
printf("[w] Write the parameters to flash memeory rn");
printf("[q] Quit rn");
printf("[l] load env 1 rn"); //設(shè)置參數(shù)1,跟新系統(tǒng)時用
printf("[m] load env 2 rn"); //設(shè)置參數(shù)2,更新完系統(tǒng)后恢復(fù)的參數(shù)
printf("Enter your selection: ");
}
void param_menu_shell(void)
{
char c;
char cmd_buf[256];
char name_buf[20];
char val_buf[256];
while (1)
{
param_menu_usage();
c = getc();
printf("%cn", c);
switch (c)
{
case ''v'':
strcpy(cmd_buf, "printenv ");
run_command(cmd_buf, 0);
break;

case ''s'':
sprintf(cmd_buf, "setenv ");
printf("Name: ");
readline(NULL);
strcat(cmd_buf, console_buffer);
run_command(cmd_buf, 0);
break;

case ''d'':
sprintf(cmd_buf, "setenv ");
printf("Name: ");
readline(NULL);
strcat(cmd_buf, console_buffer);
run_command(cmd_buf, 0);
break;
case ''w'':
sprintf(cmd_buf, "saveenv");
run_command(cmd_buf, 0);
break;

case ''l'':
sprintf(cmd_buf, "setenv bootargs "root=/dev/mtdblock2 console=ttySAC0,115200"");
printf("%sn",cmd_buf);
run_command(cmd_buf, 0);
//保存參數(shù)
run_command("saveenv", 0);
break;
case ''m'':
sprintf(cmd_buf, "setenv bootargs "root=/dev/mtdblock2 rootfstype=cramfs console=ttySAC0,115200"");
printf("%sn",cmd_buf);
run_command(cmd_buf, 0);
run_command("saveenv", 0);
break;
case ''q'':
return;
break;
}
}
}
復(fù)制代碼

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除( 郵箱:macysun@21ic.com )。
換一批
延伸閱讀

U-Boot提供命令行接口(CLI),允許開發(fā)者實(shí)時查看硬件信息、修改啟動參數(shù)、讀寫存儲設(shè)備等操作。例如,通過串口或網(wǎng)絡(luò)連接可實(shí)時調(diào)試內(nèi)存狀態(tài)或監(jiān)控系統(tǒng)運(yùn)行,部分工業(yè)設(shè)備仍依賴其硬實(shí)時響應(yīng)能力。

關(guān)鍵字: uboot 嵌入式

同大多數(shù)的Bootloader一樣,uboot的啟動過程也分為BL1、BL2兩個階段,分別對應(yīng)著SPL和Uboot。

關(guān)鍵字: Bootloader uboot SPL

為什么要有uboot?它能解決哪些問題?

關(guān)鍵字: uboot 計算機(jī) 嵌入式

針對傳統(tǒng)類人機(jī)器人在控制系統(tǒng)實(shí)時性和視覺識別方面的不足,以S3C6410作為主控芯片,設(shè)計了具有視覺識別功能的類人機(jī)器人控制系統(tǒng),通過改進(jìn)和簡化視頻識別算法取得了良好的目標(biāo)識別效果。實(shí)驗(yàn)表明,基

關(guān)鍵字: s3c6410 機(jī)器人 機(jī)器視覺

一、為什么要有uboot 1.1、計算機(jī)系統(tǒng)的主要部件 (1)計算機(jī)系統(tǒng)就是以CPU為核心來運(yùn)行的系統(tǒng)。典型的計算機(jī)系統(tǒng)有:PC機(jī)(臺式機(jī)+筆記本)、嵌入式設(shè)備(手機(jī)、平板電腦、游戲機(jī))、單片機(jī)(家用電器像電飯鍋、空調(diào))...

關(guān)鍵字: uboot

早在今年8月份的時候就將jpeg解碼弄好了,但是一直以來非常的不穩(wěn)定,如果修改了任意地方的代碼都會造成解碼可能失敗,起初我以為是堆棧問題,或者后面有非法指針,但是都沒得到結(jié)果,最后讓我只能懷疑編譯器了,而且

關(guān)鍵字: jpeg解碼 s3c6410

這幾天晚上一直折騰S3C6410的裸機(jī)SD卡啟動,不大想使用UBOOT,我是搞硬件的,對底層非常感興趣,不喜歡已經(jīng)寫好的,因此自己一直在嘗試,其實(shí)很早之前就試過SD卡啟動,也就是ARM11上電后會把SD卡倒數(shù)第9KB開始...

關(guān)鍵字: s3c6410 sd 裸機(jī)啟動 boot啟動

簡單地說,IAR調(diào)試S3C6410可以分成以下3個步驟:1)編寫mac文件,初始化S3C6410的看門狗、時鐘、DDRAM控制器等,為下載代碼做準(zhǔn)備。2)編寫icf文件對S3C6410的內(nèi)存空間進(jìn)行分配,配置堆棧段、...

關(guān)鍵字: iar調(diào)試 s3c6410

主要過程:. 安裝lunux 環(huán)境. 安裝編譯工具. 下載Linux kernel. 安裝AndroidSDK. 獲得root file system. 修改Linux kernel 源碼. 配置Linux kernel...

關(guān)鍵字: Android s3c6410 內(nèi)核 移植
關(guān)閉