//Based?on?Linux v3.14 source code
Linux設(shè)備樹機制(Device Tree)
一、描述
ARM Device Tree起源于OpenFirmware?(OF),在過去的Linux中,arch/arm/plat-xxx和arch/arm/mach-xxx中充斥著大量的垃圾代碼,相當(dāng)多數(shù)的代碼只是在描述板級細節(jié),而這些板級細節(jié)對于內(nèi)核來講,不過是垃圾,如板上的platform設(shè)備、resource、i2c_board_info、spi_board_info以及各種硬件的platform_data。為了改變這種局面,Linux社區(qū)的大牛們參考了PowerPC等體系架構(gòu)中使用的Flattened Device Tree(FDT),也采用了Device Tree結(jié)構(gòu),許多硬件的細節(jié)可以直接透過它傳遞給Linux,而不再需要在kernel中進行大量的冗余編碼。
Device Tree是一種描述硬件的數(shù)據(jù)結(jié)構(gòu),由一系列被命名的結(jié)點(node)和屬性(property)組成,而結(jié)點本身可包含子結(jié)點。所謂屬性,其實就是成對出現(xiàn)的name和value。在Device Tree中,可描述的信息包括(原先這些信息大多被hard code到kernel中):CPU的數(shù)量和類別,內(nèi)存基地址和大小,總線和橋,外設(shè)連接,中斷控制器和中斷使用情況,GPIO控制器和GPIO使用情況,Clock控制器和Clock使用情況。
通常由.dts文件以文本方式對系統(tǒng)設(shè)備樹進行描述,經(jīng)過Device Tree Compiler(dtc)將dts文件轉(zhuǎn)換成二進制文件binary device tree blob(dtb),.dtb文件可由Linux內(nèi)核解析,有了device tree就可以在不改動Linux內(nèi)核的情況下,對不同的平臺實現(xiàn)無差異的支持,只需更換相應(yīng)的dts文件,即可滿足。
二、相關(guān)結(jié)構(gòu)體
1.U-Boot需要將設(shè)備樹在內(nèi)存中的存儲地址傳給內(nèi)核。該樹主要由三大部分組成:頭(Header)、結(jié)構(gòu)塊(Structure block)、字符串塊(Strings block)。設(shè)備樹在內(nèi)存中的存儲布局圖。
------------------------------
base?->?|?struct boot_param_header?|
------------------------------
|?(alignment gap)?(*)?|
------------------------------
|?memory reserve map?|
------------------------------
|?(alignment gap)?|
------------------------------
|?|
|?device-tree structure?|
|?|
------------------------------
|?(alignment gap)?|
------------------------------
|?|
|?device-tree strings?|
|?|
----->?------------------------------
|
|
---?(base?+?totalsize)
1.1 頭(header)
頭主要描述設(shè)備樹的一些基本信息,例如設(shè)備樹大小,結(jié)構(gòu)塊偏移地址,字符串塊偏移地址等。偏移地址是相對于設(shè)備樹頭的起始地址計算的。
struct boot_param_header?{
__be32 magic;????????????????//設(shè)備樹魔數(shù),固定為0xd00dfeed
__be32 totalsize;????????????//整個設(shè)備樹的大小
__be32 off_dt_struct;????????//保存結(jié)構(gòu)塊在整個設(shè)備樹中的偏移
__be32 off_dt_strings;????????//保存的字符串塊在設(shè)備樹中的偏移
__be32 off_mem_rsvmap;????????//保留內(nèi)存區(qū),該區(qū)保留了不能被內(nèi)核動態(tài)分配的內(nèi)存空間
__be32 version;????????????//設(shè)備樹版本
__be32 last_comp_version;????//向下兼容版本號
__be32 boot_cpuid_phys;????//為在多核處理器中用于啟動的主cpu的物理id
__be32 dt_strings_size;????//字符串塊大小
__be32 dt_struct_size;?????//結(jié)構(gòu)塊大小
};
1.2 結(jié)構(gòu)塊(struct block)
設(shè)備樹結(jié)構(gòu)塊是一個線性化的結(jié)構(gòu)體,是設(shè)備樹的主體,以節(jié)點node的形式保存了目標(biāo)單板上的設(shè)備信息。
在結(jié)構(gòu)塊中以宏OF_DT_BEGIN_NODE標(biāo)志一個節(jié)點的開始,以宏OF_DT_END_NODE標(biāo)識一個節(jié)點的結(jié)束,整個結(jié)構(gòu)塊以宏OF_DT_END結(jié)束。一個節(jié)點主要由以下幾部分組成。
(1)節(jié)點開始標(biāo)志:一般為OF_DT_BEGIN_NODE。
(2)節(jié)點路徑或者節(jié)點的單元名(ersion<3以節(jié)點路徑表示,version>=0x10以節(jié)點單元名表示)
(3)填充字段(對齊到四字節(jié))
(4)節(jié)點屬性。每個屬性以宏OF_DT_PROP開始,后面依次為屬性值的字節(jié)長度(4字節(jié))、屬性名稱在字符串塊中的偏移量(4字節(jié))、屬性值和填充(對齊到四字節(jié))。
(5)如果存在子節(jié)點,則定義子節(jié)點。
(6)節(jié)點結(jié)束標(biāo)志OF_DT_END_NODE。
1.3 字符串塊
通過節(jié)點的定義知道節(jié)點都有若干屬性,而不同的節(jié)點的屬性又有大量相同的屬性名稱,因此將這些屬性名稱提取出一張表,當(dāng)節(jié)點需要應(yīng)用某個屬性名稱時直接在屬性名字段保存該屬性名稱在字符串塊中的偏移量。
1.4 設(shè)備樹源碼 DTS 表示
設(shè)備樹源碼文件(.dts)以可讀可編輯的文本形式描述系統(tǒng)硬件配置設(shè)備樹,支持 C/C++方式的注釋,該結(jié)構(gòu)有一個唯一的根節(jié)點“/”,每個節(jié)點都有自己的名字并可以包含多個子節(jié)點。設(shè)備樹的數(shù)據(jù)格式遵循了 Open Firmware IEEE standard 1275。這個設(shè)備樹中有很多節(jié)點,每個節(jié)點都指定了節(jié)點單元名稱。每一個屬性后面都給出相應(yīng)的值。以雙引號引出的內(nèi)容為 ASCII 字符串,以尖括號給出的是 32 位的16進制值。這個樹結(jié)構(gòu)是啟動 Linux 內(nèi)核所需節(jié)點和屬性簡化后的集合,包括了根節(jié)點的基本模式信息、CPU 和物理內(nèi)存布局,它還包括通過/chosen 節(jié)點傳遞給內(nèi)核的命令行參數(shù)信息。
1.5 machine_desc結(jié)構(gòu)
內(nèi)核提供了一個重要的結(jié)構(gòu)體struct machine_desc ,這個結(jié)構(gòu)體在內(nèi)核移植中起到相當(dāng)重要的作用,內(nèi)核通過machine_desc結(jié)構(gòu)體來控制系統(tǒng)體系架構(gòu)相關(guān)部分的初始化。machine_desc結(jié)構(gòu)體通過MACHINE_START宏來初始化,在代碼中, 通過在start_kernel->setup_arch中調(diào)用setup_machine_fdt來獲取。
struct machine_desc?{
unsigned?int?nr;?/*?architecture number?*/
const?char?*name;?/*?architecture name?*/
unsigned long atag_offset;?/*?tagged list?(relative)?*/
const?char?*const?*dt_compat;?/*?array?of device tree*?'compatible'?strings?*/
unsigned?int?nr_irqs;?/*?number of IRQs?*/
#ifdef CONFIG_ZONE_DMA
phys_addr_t dma_zone_size;?/*?size of DMA-able area?*/
#endif
unsigned?int?video_start;?/*?start of video RAM?*/
unsigned?int?video_end;?/*?end?of video RAM?*/
unsigned char reserve_lp0?:1;?/*?never has lp0?*/
unsigned char reserve_lp1?:1;?/*?never has lp1?*/
unsigned char reserve_lp2?:1;?/*?never has lp2?*/
enum reboot_mode reboot_mode;?/*?default restart mode?*/
struct smp_operations?*smp;?/*?SMP operations?*/
bool?(*smp_init)(void);
void?(*fixup)(struct tag?*,?char?**,struct meminfo?*);
void?(*init_meminfo)(void);
void?(*reserve)(void);/*?reserve mem blocks?*/
void?(*map_io)(void);/*?IO mapping?function?*/
void?(*init_early)(void);
void?(*init_irq)(void);
void?(*init_time)(void);
void?(*init_machine)(void);
void?(*init_late)(void);
#ifdef CONFIG_MULTI_IRQ_HANDLER
void?(*handle_irq)(struct pt_regs?*);
#endif
void?(*restart)(enum reboot_mode,?const?char?*);
};
1.6 設(shè)備節(jié)點結(jié)構(gòu)體
struct device_node?{
const?char?*name;????//設(shè)備name
const?char?*type;?//設(shè)備類型
phandle phandle;
const?char?*full_name;?//設(shè)備全稱,包括父設(shè)備名
struct?property?*properties;?//設(shè)備屬性鏈表
struct?property?*deadprops;?//removed properties
struct device_node?*parent;?//指向父節(jié)點
struct device_node?*child;?//指向子節(jié)點
struct device_node?*sibling;?//指向兄弟節(jié)點
struct device_node?*next;?//相同設(shè)備類型的下一個節(jié)點
struct device_node?*allnext;?//next?in?list of all nodes
struct proc_dir_entry?*pde;?//該節(jié)點對應(yīng)的proc
struct kref kref;
unsigned long _flags;
void?*data;
#if?defined(CONFIG_SPARC)
const?char?*path_component_name;
unsigned?int?unique_id;
struct of_irq_controller?*irq_trans;
#endif
};
1.7 屬性結(jié)構(gòu)體
struct?property?{
char?*name;????????//屬性名
int?length;????????//屬性值長度
void?*value;????????//屬性值
struct?property?*next;?//指向下一個屬性
unsigned long _flags;?//標(biāo)志
unsigned?int?unique_id;
};
三、設(shè)備樹初始化及解析
分析Linux內(nèi)核的源碼,可以看到其對扁平設(shè)備樹的解析流程如下:
(1)首先在內(nèi)核入口處將從u-boot傳遞過來的鏡像基地址。
(2)通過調(diào)用early_init_dt_scan()函數(shù)來獲取內(nèi)核前期初始化所需的bootargs,cmd_line等系統(tǒng)引導(dǎo)參數(shù)。
(3)根據(jù)bootargs,cmd_line等系統(tǒng)引導(dǎo)參數(shù)進入start_kernel()函數(shù),進行內(nèi)核的第二階段初始化。
(4)調(diào)用unflatten_device_tree()函數(shù)來解析dtb文件,構(gòu)建一個由device_node結(jié)構(gòu)連接而成的單項鏈表,并使用全局變量of_allnodes指針來保存這個鏈表的頭指針。
(5)內(nèi)核調(diào)用OF提供的API函數(shù)獲取of_allnodes鏈表信息來初始化內(nèi)核其他子系統(tǒng)、設(shè)備等。
//kernel 初始化的代碼(init/main.c)
asmlinkage void __init start_kernel(void)
{
...
//這個setup_arch就是各個架構(gòu)自己的設(shè)置函數(shù),哪個參與了編譯就調(diào)用哪個,arm架構(gòu)應(yīng)當(dāng)是arch/arm/kernel/setup.c中的 setup_arch。
setup_arch(&command_line);
...
}
void __init setup_arch(char?**cmdline_p)
{
const?struct machine_desc?*mdesc;
setup_processor();
//setup_machine_fdt函數(shù)獲取內(nèi)核前期初始化所需的bootargs,cmd_line等系統(tǒng)引導(dǎo)參數(shù)
mdesc?=?setup_machine_fdt(__atags_pointer);//__atags_pointer是bootloader傳遞參數(shù)的物理地址
if?(!mdesc)
mdesc?=?setup_machine_tags(__atags_pointer,?__machine_arch_type);
machine_desc?=?mdesc;
machine_name?=?mdesc->name;
if?(mdesc->reboot_mode?!=?REBOOT_HARD)
reboot_mode?=?mdesc->reboot_mode;
init_mm.start_code?=?(unsigned long)?_text;
init_mm.end_code?=?(unsigned long)?_etext;
init_mm.end_data?=?(unsigned long)?_edata;
init_mm.brk?=?(unsigned long)?_end;
strlcpy(cmd_line,?boot_command_line,?COMMAND_LINE_SIZE);
*cmdline_p?=?cmd_line;
parse_early_param();
sort(&meminfo.bank,?meminfo.nr_banks,?sizeof(meminfo.bank[0]),?meminfo_cmp,?NULL);
early_paging_init(mdesc,?lookup_processor_type(read_cpuid_id()));
setup_dma_zone(mdesc);
sanity_check_meminfo();
arm_memblock_init(&meminfo,?mdesc);
paging_init(mdesc);
request_standard_resources(mdesc);
if?(mdesc->restart)
arm_pm_restart?=?mdesc->restart;
//解析設(shè)備樹
unflatten_device_tree();
......
}
(一)函數(shù)獲取內(nèi)核前期初始化所需的bootargs,cmd_line等系統(tǒng)引導(dǎo)參數(shù)
1.?setup_machine_fdt()函數(shù)獲取內(nèi)核前期初始化所需的bootargs,cmd_line等系統(tǒng)引導(dǎo)參數(shù)。
const?struct machine_desc?*?__init setup_machine_fdt(unsigned?int?dt_phys)
{
const?struct machine_desc?*mdesc,?*mdesc_best?=?NULL;
#ifdef CONFIG_ARCH_MULTIPLATFORM
DT_MACHINE_START(GENERIC_DT,?"Generic DT based system")
MACHINE_END
mdesc_best?=?&__mach_desc_GENERIC_DT;
#endif
//bootloader傳遞參數(shù)的物理地址不為空,并將物理地址轉(zhuǎn)化為虛擬地址,
//通過函數(shù)early_init_dt_scan從設(shè)備樹中讀出bootargs,cmd_line等系統(tǒng)引導(dǎo)參數(shù)。
if?(!dt_phys?||?!early_init_dt_scan(phys_to_virt(dt_phys)))
return?NULL;
//根據(jù)設(shè)備樹中根節(jié)點屬性"compatible"的屬性描述,找到系統(tǒng)中定義的最匹配的machine_desc結(jié)構(gòu),該結(jié)構(gòu)控制系統(tǒng)體系架構(gòu)相關(guān)部分的初始化
mdesc?=?of_flat_dt_match_machine(mdesc_best,?arch_get_next_mach);
if?(!mdesc)?{
const?char?*prop;
long size;
unsigned long dt_root;
early_print(" Error: unrecognized/unsupported ""device tree compatible list: [ ");
//找到設(shè)備樹的根節(jié)點,dt_root指向根節(jié)點的屬性地址處
dt_root?=?of_get_flat_dt_root();
//讀出根節(jié)點的"compatible"屬性的屬性值
prop?=?of_get_flat_dt_prop(dt_root,?"compatible",?&size);
//將根節(jié)點的"compatible"屬性的屬性值打印出來
while?(size?>?0)?{
early_print("'%s' ",?prop);
size?-=?strlen(prop)?+?1;
prop?+=?strlen(prop)?+?1;
}
early_print("] ");
dump_machine_table();?/*?does?not?return?*/
}
//Change machine number?to?match the mdesc we're using
__machine_arch_type?=?mdesc->nr;
return mdesc;
}
struct boot_param_header?*initial_boot_params;
bool __init early_init_dt_scan(void?*params)
{
if?(!params)
return?false;
//參數(shù)params是bootloader傳遞參數(shù)的物理地址轉(zhuǎn)化為的虛擬地址,保存設(shè)備樹起始地址
initial_boot_params?=?params;
//驗證設(shè)備樹的magic?
if?(be32_to_cpu(initial_boot_params->magic)?!=?OF_DT_HEADER)?{
initial_boot_params?=?NULL;
return?false;
}
//從設(shè)備樹中讀取chosen節(jié)點的信息,包括命令行boot_command_line,initrd location及size
of_scan_flat_dt(early_init_dt_scan_chosen,?boot_command_line);
//得到根節(jié)點的{size,address}-cells信息
of_scan_flat_dt(early_init_dt_scan_root,?NULL);
//讀出設(shè)備樹的系統(tǒng)內(nèi)存設(shè)置
of_scan_flat_dt(early_init_dt_scan_memory,?NULL);
return?true;
}
int?__init of_scan_flat_dt(int?(*it)(unsigned long node,const?char?*uname,?int?depth,void?*data),void?*data)
{
//找到設(shè)備樹中結(jié)構(gòu)塊的地址
unsigned long p?=?((unsigned long)initial_boot_params)?+?be32_to_cpu(initial_boot_params->off_dt_struct);
int?rc?=?0;
int?depth?=?-1;
do?{
//獲得節(jié)點起始標(biāo)志,即OF_DT_BEGIN_NODE,OF_DT_PROP等
u32 tag?=?be32_to_cpup((__be32?*)p);
const?char?*pathp;
p?+=?4;//跳過節(jié)點起始標(biāo)志
//如果是OF_DT_END_NODE標(biāo)志,表示該節(jié)點結(jié)束,繼續(xù)下一結(jié)點
if?(tag?==?OF_DT_END_NODE)?{
depth--;
continue;
}
//OF_DT_NOP標(biāo)志代表空節(jié)點
if?(tag?==?OF_DT_NOP)
continue;
//OF_DT_END標(biāo)志整個結(jié)構(gòu)塊結(jié)束
if?(tag?==?OF_DT_END)
break;
//OF_DT_PROP標(biāo)示屬性,Property:屬性值的字節(jié)長度、屬性名稱在字符串塊中的偏移量、屬性值和填充。
if?(tag?==?OF_DT_PROP)?{
//屬性值的字節(jié)大小
u32 sz?=?be32_to_cpup((__be32?*)p);
p?+=?8;//跳過屬性值的字節(jié)長度、屬性名稱在字符串塊中的偏移量
if?(be32_to_cpu(initial_boot_params->version)?0x10)
p?=?ALIGN(p,?sz?>=?8???8?:?4);
//跳過該屬性值的大小
p?+=?sz;
//地址對齊
p?=?ALIGN(p,?4);
//表示一個屬性節(jié)點遍歷完成,因為這里并不是尋找屬性節(jié)點,而是找OF_DT_BEGIN_NODE開始的節(jié)點
continue;
}
//若都不是以上節(jié)點類型,也不是節(jié)點開始標(biāo)示(OF_DT_BEGIN_NODE),則出錯返回
if?(tag?!=?OF_DT_BEGIN_NODE)?{
pr_err("Invalid tag %x in flat device tree! ",?tag);
return?-EINVAL;
}
//執(zhí)行到這里,標(biāo)示tag=OF_DT_BEGIN_NODE,表示一個節(jié)點的開始,探索深度加1
depth++;
//節(jié)點路徑或者節(jié)點名
pathp?=?(char?*)p;
//節(jié)點地址四字節(jié)對齊,然后p指向節(jié)點屬性地址
p?=?ALIGN(p?+?strlen(pathp)?+?1,?4);
//如果是節(jié)點路徑,則返回路徑名的最后一段,假如為/root/my_root,則返回my_root,即獲得節(jié)點名
if?(*pathp?==?'/')
pathp?=?kbasename(pathp);
//調(diào)用相應(yīng)的節(jié)點處理函數(shù),p指向節(jié)點屬性地址
rc?=?it(p,?pathp,?depth,?data);
if?(rc?!=?0)
break;
}?while?(1);
return rc;
}
1.1 chosen節(jié)點
//chosen 節(jié)點并不代表一個真實的設(shè)備,只是作為一個為固件和操作系統(tǒng)之間傳遞數(shù)據(jù)的地方,比如引導(dǎo)參數(shù)。chosen 節(jié)點里的數(shù)據(jù)也不代表硬件。通常,chosen 節(jié)點在.dts 源文件中為空,并在啟動時填充。在我們的示例系統(tǒng)中,固件可以往 chosen 節(jié)點添加以下信息:
//chosen?{
//????bootargs?=?"root=/dev/nfs rw nfsroot=192.168.1.1 console=ttyS0,115200";?//節(jié)點屬性
//????linux,initrd-start?=?<0x85500000>;?//節(jié)點屬性
//????linux,initrd-end?=?<0x855a3212>;?//節(jié)點屬性
//};
int?__init early_init_dt_scan_chosen(unsigned long node,?const?char?*uname,int?depth,?void?*data)
{
unsigned long l;
char?*p;
pr_debug("search "chosen", depth: %d, uname: %s ",?depth,?uname);
//depth深度要為1,表示在根節(jié)點下(一般根節(jié)點/的depth為0)
//data表示系統(tǒng)啟動命令行boot_command_line要分配空間
//檢查節(jié)點名是否為chosen節(jié)點????
if?(depth?!=?1?||?!data?||?(strcmp(uname,?"chosen")?!=?0?&&?strcmp(uname,?"chosen@0")?!=?0))
return 0;
//從設(shè)備樹的chosen節(jié)點中讀出initrd的起始、結(jié)束地址
early_init_dt_check_for_initrd(node);
//設(shè)備樹的chosen節(jié)點中讀取bootargs屬性的屬性值,并拷貝給boot_command_line
p?=?of_get_flat_dt_prop(node,?"bootargs",?&l);
if?(p?!=?NULL?&&?l?>?0)
strlcpy(data,?p,?min((int)l,?COMMAND_LINE_SIZE));
pr_debug("Command line is: %s ",?(char*)data);
return 1;
}
static void __init early_init_dt_check_for_initrd(unsigned long node)
{
u64 start,?end;
unsigned long?len;
__be32?*prop;
pr_debug("Looking for initrd properties... ");
//返回該chosen節(jié)點中屬性名為"linux,initrd-start"的地址
prop?=?of_get_flat_dt_prop(node,?"linux,initrd-start",?&len);
if?(!prop)
return;
//從該地址讀出initrd-start的地址
start?=?of_read_number(prop,?len/4);
//返回該chosen節(jié)點中屬性名為"linux,initrd-end"的地址
prop?=?of_get_flat_dt_prop(node,?"linux,initrd-end",?&len);
if?(!prop)
return;
//從該地址讀出initrd-end的地址
end?=?of_read_number(prop,?len/4);
//將讀出的地址賦值給全局變量initrd_start和initrd_end,用于跟文件系統(tǒng)的掛載
initrd_start?=?(unsigned long)__va(start);
initrd_end?=?(unsigned long)__va(end);
initrd_below_start_ok?=?1;
pr_debug("initrd_start=0x%llx initrd_end=0x%llx ",(unsigned long long)start,?(unsigned long long)end);
}
void?*__init of_get_flat_dt_prop(unsigned long node,?const?char?*name,unsigned long?*size)
{
return of_fdt_get_property(initial_boot_params,?node,?name,?size);
}
void?*of_fdt_get_property(struct boot_param_header?*blob,unsigned long node,?const?char?*name,unsigned long?*size)
{
//p指向該chosen節(jié)點的節(jié)點屬性地址
unsigned long p?=?node;
//因為一個節(jié)點中可能包含多個屬性,所以這里遍歷chosen節(jié)點中的所有屬性,找到屬性名為name的屬性
do?{
//取得該節(jié)點屬性的起始標(biāo)志OF_DT_PROP
u32 tag?=?be32_to_cpup((__be32?*)p);
u32 sz,?noff;
const?char?*nstr;
p?+=?4;//跳過節(jié)點屬性的起始標(biāo)志
//空節(jié)點則繼續(xù)
if?(tag?==?OF_DT_NOP)
continue;
//非屬性標(biāo)志則返回NULL
if?(tag?!=?OF_DT_PROP)
return?NULL;
//運行到這里表示為屬性O(shè)F_DT_PROP
//取得該節(jié)點屬性的的屬性值size
sz?=?be32_to_cpup((__be32?*)p);
//取得該屬性名的在字符串塊中的偏移值
noff?=?be32_to_cpup((__be32?*)(p?+?4));
p?+=?8;
//跳過對齊填充字段
if?(be32_to_cpu(blob->version)?0x10)
p?=?ALIGN(p,?sz?>=?8???8?:?4);
//在字符串塊取出該屬性的名稱
nstr?=?of_fdt_get_string(blob,?noff);
if?(nstr?==?NULL)?{
pr_warning("Can't find property index name ! ");
return?NULL;
}
//若名稱一致,表示找到我們要找的屬性
if?(strcmp(name,?nstr)?==?0)?{
if?(size)
*size?=?sz;//返回該屬性的屬性值size
//返回該屬性值所在的地址
return?(void?*)p;
}
//否則繼續(xù)下一個屬性
p?+=?sz;
p?=?ALIGN(p,?4);
}?while?(1);
}
char?*of_fdt_get_string(struct boot_param_header?*blob,?u32 offset)
{
//從設(shè)備樹的字符串塊的offset處讀出name
return?((char?*)blob)?+?be32_to_cpu(blob->off_dt_strings)?+?offset;
}
static inline u64 of_read_number(const?__be32?*cell,?int?size)
{
u64 r?=?0;
//讀出屬性值,屬性值大小為size
while?(size--)
r?=?(r?<32)?|?be32_to_cpu(*(cell++));
return r;
}
1.2 根節(jié)點"/"
//設(shè)備樹有且僅有一個根節(jié)點,即“/”,根節(jié)點下包含很多子節(jié)點,例入下圖,根節(jié)點為"/",根節(jié)點的子節(jié)點為"chosen",根節(jié)點的屬性包含"compatible","#address-cells","#size-cells","interrupt-parent"等。屬性model指明了目標(biāo)板平臺或模塊的名稱,屬性compatible值指明和目標(biāo)板為同一系列的兼容的開發(fā)板名稱。對于大多數(shù)32位平臺,屬性#address-cells和#size-cells的值一般為1。#address-cells?=?<1>;?1表示地址32位,2表示地址64位。#size-cells?=?<1>;1表示rangs的每部分占一個cell,依此類推?
/*
/?{
compatible?=?"sprd,spx15";
#address-cells?=?<1>;
#size-cells?=?<1>;
interrupt-parent?=?<&gic>;
chosen?{
bootargs?=?"loglevel=8 console=ttyS1,115200n8 init=/init root=/dev/ram0 rw";
linux,initrd-start?=?<0x85500000>;
linux,initrd-end?=?<0x855a3212>;
};
}
*/
//所以本函數(shù)就是讀取根節(jié)點的"#address-cells","#size-cells"屬性
int?__init early_init_dt_scan_root(unsigned long node,?const?char?*uname,int?depth,?void?*data)
{
__be32?*prop;
//根節(jié)點的探索深度depth一定為0,否則不是根節(jié)點"/",node為根節(jié)點的屬性地址????
if?(depth?!=?0)
return 0;
dt_root_size_cells?=?OF_ROOT_NODE_SIZE_CELLS_DEFAULT;
dt_root_addr_cells?=?OF_ROOT_NODE_ADDR_CELLS_DEFAULT;
//返回根節(jié)點節(jié)點屬性名為"#size-cells"的地址
prop?=?of_get_flat_dt_prop(node,?"#size-cells",?NULL);
if?(prop)
dt_root_size_cells?=?be32_to_cpup(prop);//從該屬性獲得root size
pr_debug("dt_root_size_cells = %x ",?dt_root_size_cells);
//返回根節(jié)點節(jié)點屬性名為"#address-cells"的地址
prop?=?of_get_flat_dt_prop(node,?"#address-cells",?NULL);
if?(prop)
dt_root_addr_cells?=?be32_to_cpup(prop);//從該屬性獲得root address
pr_debug("dt_root_addr_cells = %x ",?dt_root_addr_cells);
return 1;
}
1.3 memory節(jié)點
//memory節(jié)點用于描述目標(biāo)板上物理內(nèi)存范圍,一般稱作/memory節(jié)點,可以有一個或多個。當(dāng)有多個節(jié)點時,需要后跟單元地址予以區(qū)分;只有一個單元地址時,可以不寫單元地址,默認為0。此節(jié)點包含板上物理內(nèi)存的屬性,一般要指定device_type(固定為"memory")和reg屬性。其中reg的屬性值以<起始地址 空間大小>的形式給出,如下示例中目標(biāo)板內(nèi)存起始地址為0x80000000,大小為0x20000000字節(jié)。?
//memory?{
//????device_type?=?"memory";
//????reg?=?<0x80000000 0x20000000>;
//};
int?__init early_init_dt_scan_memory(unsigned long node,?const?char?*uname,int?depth,?void?*data)
{
//獲取該節(jié)點中屬性"device_type"的屬性值
char?*type?=?of_get_flat_dt_prop(node,?"device_type",?NULL);
__be32?*reg,?*endp;
unsigned long l;
//檢查"device_type"的屬性值,確定該節(jié)點是否為memory節(jié)點
if?(type?==?NULL)?{
if?(depth?!=?1?||?strcmp(uname,?"memory@0")?!=?0)
return 0;
}?else?if?(strcmp(type,?"memory")?!=?0)
return 0;
//從該memory節(jié)點中獲取屬性"linux,usable-memory"的屬性值,及屬性值大小l
reg?=?of_get_flat_dt_prop(node,?"linux,usable-memory",?&l);
if?(reg?==?NULL)
reg?=?of_get_flat_dt_prop(node,?"reg",?&l);
if?(reg?==?NULL)
return 0;
//reg為屬性值的起始地址,endp為結(jié)束地址
endp?=?reg?+?(l?/?sizeof(__be32));
pr_debug("memory scan node %s, reg size %ld, data: %x %x %x %x, ",
uname,?l,?reg[0],?reg[1],?reg[2],?reg[3]);
while?((endp?-?reg)?>=?(dt_root_addr_cells?+?dt_root_size_cells))?{
u64 base,?size;
//讀出物理內(nèi)存的起始地址以及size
base?=?dt_mem_next_cell(dt_root_addr_cells,?®);
size?=?dt_mem_next_cell(dt_root_size_cells,?®);
if?(size?==?0)
continue;
pr_debug(" - %llx , %llx ",?(unsigned long long)base,(unsigned long long)size);
//添加物理內(nèi)存到memblock中進行管理,這里不再展開
early_init_dt_add_memory_arch(base,?size);
}
return 0;
}
2.?通過比較根節(jié)點屬性compatible值指明和目標(biāo)板為同一系列的兼容的開發(fā)板名稱
//compatible制定系統(tǒng)的名稱。它包含","格式的字符串。準(zhǔn)確地確定器件型號是非常重要的,并且我們需要包含廠商的名字來避免名字空間沖突。因為操作系統(tǒng)會使用compatible這個值來決定怎樣在這個機器上運行,所以在這個屬性中放入正確的值是非常重要的。
const?void?*?__init of_flat_dt_match_machine(const?void?*default_match,
const?void?*?(*get_next_compat)(const?char?*?const**))
{
const?void?*data?=?NULL;
const?void?*best_data?=?default_match;
const?char?*const?*compat;
unsigned long dt_root;
unsigned?int?best_score?=?~1,?score?=?0;
//讀出設(shè)備樹的根節(jié)點,dt_root指向根節(jié)點的屬性地址處
dt_root?=?of_get_flat_dt_root();
//調(diào)用arch_get_next_mach,遍歷該系統(tǒng)中的所有machine_desc結(jié)構(gòu),返回給data,并且返回該結(jié)構(gòu)的compatible
while?((data?=?get_next_compat(&compat)))?{
//將系統(tǒng)中的所有machine_desc結(jié)構(gòu)的compatible字符串與設(shè)備樹根節(jié)點的compatible屬性進行match
score?=?of_flat_dt_match(dt_root,?compat);
//返回與根節(jié)點屬性compatible屬性值最匹配的machine_desc結(jié)構(gòu)
if?(score?>?0?&&?score?
best_data?=?data;
best_score?=?score;
}
}
if?(!best_data)?{
const?char?*prop;
long size;
pr_err(" unrecognized device tree list: [ ");
prop?=?of_get_flat_dt_prop(dt_root,?"compatible",?&size);
if?(prop)?{
while?(size?>?0)?{
printk("'%s' ",?prop);
size?-=?strlen(prop)?+?1;
prop?+=?strlen(prop)?+?1;
}
}
printk("] ");
return?NULL;
}
pr_info("Machine model: %s ",?of_flat_dt_get_machine_name());
return best_data;
}
//查找設(shè)備樹的根節(jié)點
unsigned long __init of_get_flat_dt_root(void)
{
//找到設(shè)備樹的設(shè)備塊起始地址
unsigned long p?=?((unsigned long)initial_boot_params)?+
be32_to_cpu(initial_boot_params->off_dt_struct);
//跳過空節(jié)點?
while?(be32_to_cpup((__be32?*)p)?==?OF_DT_NOP)
p?+=?4;
BUG_ON(be32_to_cpup((__be32?*)p)?!=?OF_DT_BEGIN_NODE);
p?+=?4;
//第一個節(jié)點就是根節(jié)點,p指向根節(jié)點的屬性地址處
return ALIGN(p?+?strlen((char?*)p)?+?1,?4);
}
//arch/arm/kernel/devtree.c
__arch_info_begin 和 __arch_info_end是在 arch/arm/kernel/vmlinux.lds.S中:
00034:?__arch_info_begin?=?.;
00035:?*(.arch.info.init)
00036:?__arch_info_end?=?.;
這里是聲明了兩個變量:__arch_info_begin 和 __arch_info_end,其中等號后面的"."是location counter。在__arch_info_begin 的位置上,放置所有文件中的?".arch.info.init"?段的內(nèi)容,然后緊接著是 __arch_info_end 的位置.".arch.info.init"?段中定義了設(shè)備的machine_desc結(jié)構(gòu)。
//這里就是取出一個machine_desc結(jié)構(gòu)
static?const?void?*?__init arch_get_next_mach(const?char?*const?**match)
{
static?const?struct machine_desc?*mdesc?=?__arch_info_begin;
const?struct machine_desc?*m?=?mdesc;
if?(m?>=?__arch_info_end)
return?NULL;
mdesc++;//指針后移,確保下次取出下一個machine_desc結(jié)構(gòu)
*match?=?m->dt_compat;
return m;//返回當(dāng)前的machine_desc結(jié)構(gòu)
}
//與設(shè)備樹根節(jié)點進行match
int?__init of_flat_dt_match(unsigned long node,?const?char?*const?*compat)
{
//initial_boot_params指向設(shè)備樹起始地址
//node指向根節(jié)點的屬性地址
//compat為系統(tǒng)中machine_desc結(jié)構(gòu)的compatible字符串
return of_fdt_match(initial_boot_params,?node,?compat);
}
int?of_fdt_match(struct boot_param_header?*blob,?unsigned long node,const?char?*const?*compat)
{
unsigned?int?tmp,?score?=?0;
if?(!compat)
return 0;
//遍歷compatible字符串?dāng)?shù)組????
while?(*compat)?{
//返回compatible的匹配值
tmp?=?of_fdt_is_compatible(blob,?node,?*compat);
if?(tmp?&&?(score?==?0?||?(tmp?
score?=?tmp;//返回最大的匹配值
compat++;//下一個字符串
}
return score;
}
int?of_fdt_is_compatible(struct boot_param_header?*blob,unsigned long node,?const?char?*compat)
{
const?char?*cp;
unsigned long cplen,?l,?score?=?0;
//從根節(jié)點中讀出屬性"compatible"的屬性值
cp?=?of_fdt_get_property(blob,?node,?"compatible",?&cplen);
if?(cp?==?NULL)
return 0;
//比較compatible的指定的屬性字符串的一致性
while?(cplen?>?0)?{
score++;
if?(of_compat_cmp(cp,?compat,?strlen(compat))?==?0)
return score;
l?=?strlen(cp)?+?1;
cp?+=?l;
cplen?-=?l;
}
return 0;
}
(二)、解析設(shè)備樹
//unflatten_device_tree()函數(shù)來解析dtb文件,構(gòu)建一個由device_node結(jié)構(gòu)連接而成的單項鏈表,并使用全局變量of_allnodes指針來保存這個鏈表的頭指針。內(nèi)核調(diào)用OF提供的API函數(shù)獲取of_allnodes鏈表信息來初始化內(nèi)核其他子系統(tǒng)、設(shè)備。
void __init unflatten_device_tree(void)
{
//解析設(shè)備樹,將所有的設(shè)備節(jié)點鏈入全局鏈表????of_allnodes中
__unflatten_device_tree(initial_boot_params,?&of_allnodes,early_init_dt_alloc_memory_arch);
//設(shè)置內(nèi)核輸出終端,以及遍歷“/aliases”節(jié)點下的所有的屬性,掛入相應(yīng)鏈表
of_alias_scan(early_init_dt_alloc_memory_arch);
}
static void __unflatten_device_tree(struct boot_param_header?*blob,
struct device_node?**mynodes,
void?*?(*dt_alloc)(u64 size,?u64 align))
{
unsigned long size;
void?*start,?*mem;
struct device_node?**allnextp?=?mynodes;
pr_debug(" -> unflatten_device_tree() ");
if?(!blob)?{
pr_debug("No device tree pointer ");
return;
}
pr_debug("Unflattening device tree: ");
pr_debug("magic: %08x ",?be32_to_cpu(blob->magic));
pr_debug("size: %08x ",?be32_to_cpu(blob->totalsize));
pr_debug("version: %08x ",?be32_to_cpu(blob->version));
//檢查設(shè)備樹magic
if?(be32_to_cpu(blob->magic)?!=?OF_DT_HEADER)?{
pr_err("Invalid device tree blob header ");
return;
}
//找到設(shè)備樹的設(shè)備節(jié)點起始地址
start?=?((void?*)blob)?+?be32_to_cpu(blob->off_dt_struct);
//第一次調(diào)用mem傳0,allnextpp傳NULL,實際上是為了計算整個設(shè)備樹所要的空間
size?=?(unsigned long)unflatten_dt_node(blob,?0,?&start,?NULL,?NULL,?0);
size?=?ALIGN(size,?4);//4字節(jié)對齊
pr_debug(" size is %lx, allocating... ",?size);
//調(diào)用early_init_dt_alloc_memory_arch函數(shù),為設(shè)備樹分配內(nèi)存空間
mem?=?dt_alloc(size?+?4,?__alignof__(struct device_node));
memset(mem,?0,?size);
//設(shè)備樹結(jié)束處賦值0xdeadbeef,為了后邊檢查是否有數(shù)據(jù)溢出
*(__be32?*)(mem?+?size)?=?cpu_to_be32(0xdeadbeef);
pr_debug(" unflattening %p... ",?mem);
//再次獲取設(shè)備樹的設(shè)備節(jié)點起始地址
start?=?((void?*)blob)?+?be32_to_cpu(blob->off_dt_struct);
//mem為設(shè)備樹分配的內(nèi)存空間,allnextp指向全局變量of_allnodes,生成整個設(shè)備樹
unflatten_dt_node(blob,?mem,?&start,?NULL,?&allnextp,?0);
if?(be32_to_cpup(start)?!=?OF_DT_END)
pr_warning("Weird tag at end of tree: %08x ",?be32_to_cpup(start));
if?(be32_to_cpup(mem?+?size)?!=?0xdeadbeef)
pr_warning("End of tree marker overwritten: %08x ",be32_to_cpup(mem?+?size));
*allnextp?=?NULL;
pr_debug(" <- unflatten_device_tree() ");
}
static void?*?unflatten_dt_node(struct boot_param_header?*blob,
void?*mem,void?**p,
struct device_node?*dad,
struct device_node?***allnextpp,
unsigned long fpsize)
{
struct device_node?*np;
struct?property?*pp,?**prev_pp?=?NULL;
char?*pathp;
u32 tag;
unsigned?int?l,?allocl;
int?has_name?=?0;
int?new_format?=?0;
//*p指向設(shè)備樹的設(shè)備塊起始地址
tag?=?be32_to_cpup(*p);
//每個有孩子的設(shè)備節(jié)點,其tag一定是OF_DT_BEGIN_NODE
if?(tag?!=?OF_DT_BEGIN_NODE)?{
pr_err("Weird tag at start of node: %x ",?tag);
return mem;
}
*p?+=?4;//地址+4,跳過tag,這樣指向節(jié)點的名稱或者節(jié)點路徑名
pathp?=?*p;//獲得節(jié)點名或者節(jié)點路徑名
l?=?allocl?=?strlen(pathp)?+?1;//該節(jié)點名稱的長度
*p?=?PTR_ALIGN(*p?+?l,?4);//地址對齊后,*p指向該節(jié)點屬性的地址
//如果是節(jié)點名則進入,若是節(jié)點路徑名則(*pathp)?==?'/'
if?((*pathp)?!=?'/')?{
new_format?=?1;
if?(fpsize?==?0)?{//fpsize=0
fpsize?=?1;
allocl?=?2;
l?=?1;
*pathp?=?' 主站蜘蛛池模板: 韩国无遮羞禁动漫在线观看 | 在线日本高清日本免费 | 正在播放久久 | 国产精品久久久久久久人热 | 小雪奶水涨翁工帮吸的推荐语录 | 《乳色吐息》无删减版在线观看 | 欧美午夜精品久久久久久浪潮 | 哪灬你的鸣巴好大 | 精品一产品大全 | 亚洲精品成人A8198A片漫画 | 超碰97人人做人人爱亚洲尤物 | 十分钟视频影院免费 | 国产人成高清在线视频99 | 午夜看片福利在线观看 | 久久久午夜精品福利内容 | 一级毛片两人添下面 | 一级做a爰片久久毛片免费 一级做a爰片久久毛片潮喷动漫 | 恋夜秀场1234手机视频在线观看 | 青青app | 怪物高h粗暴无尽 | 国产中文字幕乱码一区 | 最新黄yyid| 国产精品亚洲精品久久国语 | 又粗又大又爽又黄的免费视频 | 久久精品久精品99热 | 国产精品毛片AV久久97 | 97成人精品视频在线播放 | 欧美丰满熟妇BBB久久久 | 亚洲精品卡2卡3卡4卡5卡区 | 久久这里只精品热在线18 | 日欧一片内射VA在线影院 | 秋霞网韩国理伦片免费看 | 国精产品一区一区三区有限公司 | 国产在线精品视频免费观看 | 国产在线观看成人免费视频 | 在线观看精品视频看看播放 | 国产成人久久AV免费看澳门 | 两个人的视频免费 | 九九热在线观看视频 | 黄子佼81岁父亲现状曝光 | 中国老太60old mantv |