1*f949cb75SHu Haowen.. SPDX-License-Identifier: GPL-2.0 2*f949cb75SHu Haowen 3*f949cb75SHu Haowen.. include:: ../disclaimer-zh_TW.rst 4*f949cb75SHu Haowen 5*f949cb75SHu Haowen:Original: Documentation/admin-guide/bootconfig.rst 6*f949cb75SHu Haowen 7*f949cb75SHu Haowen:譯者: 吳想成 Wu XiangCheng <bobwxc@email.cn> 8*f949cb75SHu Haowen 9*f949cb75SHu Haowen======== 10*f949cb75SHu Haowen引導配置 11*f949cb75SHu Haowen======== 12*f949cb75SHu Haowen 13*f949cb75SHu Haowen:作者: Masami Hiramatsu <mhiramat@kernel.org> 14*f949cb75SHu Haowen 15*f949cb75SHu Haowen概述 16*f949cb75SHu Haowen==== 17*f949cb75SHu Haowen 18*f949cb75SHu Haowen引導配置擴展了現有的內核命令行,以一種更有效率的方式在引導內核時進一步支持 19*f949cb75SHu Haowen鍵值數據。這允許管理員傳遞一份結構化關鍵字的配置文件。 20*f949cb75SHu Haowen 21*f949cb75SHu Haowen配置文件語法 22*f949cb75SHu Haowen============ 23*f949cb75SHu Haowen 24*f949cb75SHu Haowen引導配置文件的語法採用非常簡單的鍵值結構。每個關鍵字由點連接的單詞組成,鍵 25*f949cb75SHu Haowen和值由 ``=`` 連接。值以分號( ``;`` )或換行符( ``\n`` )結尾。數組值中每 26*f949cb75SHu Haowen個元素由逗號( ``,`` )分隔。:: 27*f949cb75SHu Haowen 28*f949cb75SHu Haowen KEY[.WORD[...]] = VALUE[, VALUE2[...]][;] 29*f949cb75SHu Haowen 30*f949cb75SHu Haowen與內核命令行語法不同,逗號和 ``=`` 周圍允許有空格。 31*f949cb75SHu Haowen 32*f949cb75SHu Haowen關鍵字只允許包含字母、數字、連字符( ``-`` )和下劃線( ``_`` )。值可包含 33*f949cb75SHu Haowen可打印字符和空格,但分號( ``;`` )、換行符( ``\n`` )、逗號( ``,`` )、 34*f949cb75SHu Haowen井號( ``#`` )和右大括號( ``}`` )等分隔符除外。 35*f949cb75SHu Haowen 36*f949cb75SHu Haowen如果你需要在值中使用這些分隔符,可以用雙引號( ``"VALUE"`` )或單引號 37*f949cb75SHu Haowen( ``'VALUE'`` )括起來。注意,引號無法轉義。 38*f949cb75SHu Haowen 39*f949cb75SHu Haowen鍵的值可以爲空或不存在。這些鍵用於檢查該鍵是否存在(類似布爾值)。 40*f949cb75SHu Haowen 41*f949cb75SHu Haowen鍵值語法 42*f949cb75SHu Haowen-------- 43*f949cb75SHu Haowen 44*f949cb75SHu Haowen引導配置文件語法允許用戶通過大括號合併鍵名部分相同的關鍵字。例如:: 45*f949cb75SHu Haowen 46*f949cb75SHu Haowen foo.bar.baz = value1 47*f949cb75SHu Haowen foo.bar.qux.quux = value2 48*f949cb75SHu Haowen 49*f949cb75SHu Haowen也可以寫成:: 50*f949cb75SHu Haowen 51*f949cb75SHu Haowen foo.bar { 52*f949cb75SHu Haowen baz = value1 53*f949cb75SHu Haowen qux.quux = value2 54*f949cb75SHu Haowen } 55*f949cb75SHu Haowen 56*f949cb75SHu Haowen或者更緊湊一些,寫成:: 57*f949cb75SHu Haowen 58*f949cb75SHu Haowen foo.bar { baz = value1; qux.quux = value2 } 59*f949cb75SHu Haowen 60*f949cb75SHu Haowen在這兩種樣式中,引導解析時相同的關鍵字都會自動合併。因此可以追加類似的樹或 61*f949cb75SHu Haowen鍵值。 62*f949cb75SHu Haowen 63*f949cb75SHu Haowen相同關鍵字的值 64*f949cb75SHu Haowen-------------- 65*f949cb75SHu Haowen 66*f949cb75SHu Haowen禁止兩個或多個值或數組共享同一個關鍵字。例如:: 67*f949cb75SHu Haowen 68*f949cb75SHu Haowen foo = bar, baz 69*f949cb75SHu Haowen foo = qux # !錯誤! 我們不可以重定義相同的關鍵字 70*f949cb75SHu Haowen 71*f949cb75SHu Haowen如果你想要更新值,必須顯式使用覆蓋操作符 ``:=`` 。例如:: 72*f949cb75SHu Haowen 73*f949cb75SHu Haowen foo = bar, baz 74*f949cb75SHu Haowen foo := qux 75*f949cb75SHu Haowen 76*f949cb75SHu Haowen這樣 ``foo`` 關鍵字的值就變成了 ``qux`` 。這對於通過添加(部分)自定義引導 77*f949cb75SHu Haowen配置來覆蓋默認值非常有用,免於解析默認引導配置。 78*f949cb75SHu Haowen 79*f949cb75SHu Haowen如果你想對現有關鍵字追加值作爲數組成員,可以使用 ``+=`` 操作符。例如:: 80*f949cb75SHu Haowen 81*f949cb75SHu Haowen foo = bar, baz 82*f949cb75SHu Haowen foo += qux 83*f949cb75SHu Haowen 84*f949cb75SHu Haowen這樣, ``foo`` 關鍵字就同時擁有了 ``bar`` , ``baz`` 和 ``qux`` 。 85*f949cb75SHu Haowen 86*f949cb75SHu Haowen此外,父關鍵字下可同時存在值和子關鍵字。 87*f949cb75SHu Haowen例如,下列配置是可行的。:: 88*f949cb75SHu Haowen 89*f949cb75SHu Haowen foo = value1 90*f949cb75SHu Haowen foo.bar = value2 91*f949cb75SHu Haowen foo := value3 # 這會更新foo的值。 92*f949cb75SHu Haowen 93*f949cb75SHu Haowen注意,裸值不能直接放進結構化關鍵字中,必須在大括號外定義它。例如:: 94*f949cb75SHu Haowen 95*f949cb75SHu Haowen foo { 96*f949cb75SHu Haowen bar = value1 97*f949cb75SHu Haowen bar { 98*f949cb75SHu Haowen baz = value2 99*f949cb75SHu Haowen qux = value3 100*f949cb75SHu Haowen } 101*f949cb75SHu Haowen } 102*f949cb75SHu Haowen 103*f949cb75SHu Haowen同時,關鍵字下值節點的順序是固定的。如果值和子關鍵字同時存在,值永遠是該關 104*f949cb75SHu Haowen鍵字的第一個子節點。因此如果用戶先指定子關鍵字,如:: 105*f949cb75SHu Haowen 106*f949cb75SHu Haowen foo.bar = value1 107*f949cb75SHu Haowen foo = value2 108*f949cb75SHu Haowen 109*f949cb75SHu Haowen則在程序(和/proc/bootconfig)中,它會按如下顯示:: 110*f949cb75SHu Haowen 111*f949cb75SHu Haowen foo = value2 112*f949cb75SHu Haowen foo.bar = value1 113*f949cb75SHu Haowen 114*f949cb75SHu Haowen註釋 115*f949cb75SHu Haowen---- 116*f949cb75SHu Haowen 117*f949cb75SHu Haowen配置語法接受shell腳本風格的註釋。註釋以井號( ``#`` )開始,到換行符 118*f949cb75SHu Haowen( ``\n`` )結束。 119*f949cb75SHu Haowen 120*f949cb75SHu Haowen:: 121*f949cb75SHu Haowen 122*f949cb75SHu Haowen # comment line 123*f949cb75SHu Haowen foo = value # value is set to foo. 124*f949cb75SHu Haowen bar = 1, # 1st element 125*f949cb75SHu Haowen 2, # 2nd element 126*f949cb75SHu Haowen 3 # 3rd element 127*f949cb75SHu Haowen 128*f949cb75SHu Haowen會被解析爲:: 129*f949cb75SHu Haowen 130*f949cb75SHu Haowen foo = value 131*f949cb75SHu Haowen bar = 1, 2, 3 132*f949cb75SHu Haowen 133*f949cb75SHu Haowen注意你不能把註釋放在值和分隔符( ``,`` 或 ``;`` )之間。如下配置語法是錯誤的:: 134*f949cb75SHu Haowen 135*f949cb75SHu Haowen key = 1 # comment 136*f949cb75SHu Haowen ,2 137*f949cb75SHu Haowen 138*f949cb75SHu Haowen 139*f949cb75SHu Haowen/proc/bootconfig 140*f949cb75SHu Haowen================ 141*f949cb75SHu Haowen 142*f949cb75SHu Haowen/proc/bootconfig是引導配置的用戶空間接口。與/proc/cmdline不同,此文件內容以 143*f949cb75SHu Haowen鍵值列表樣式顯示。 144*f949cb75SHu Haowen每個鍵值對一行,樣式如下:: 145*f949cb75SHu Haowen 146*f949cb75SHu Haowen KEY[.WORDS...] = "[VALUE]"[,"VALUE2"...] 147*f949cb75SHu Haowen 148*f949cb75SHu Haowen 149*f949cb75SHu Haowen用引導配置引導內核 150*f949cb75SHu Haowen================== 151*f949cb75SHu Haowen 152*f949cb75SHu Haowen用引導配置引導內核有兩種方法:將引導配置附加到initrd鏡像或直接嵌入內核中。 153*f949cb75SHu Haowen 154*f949cb75SHu Haowen*initrd: initial RAM disk,初始內存磁盤* 155*f949cb75SHu Haowen 156*f949cb75SHu Haowen將引導配置附加到initrd 157*f949cb75SHu Haowen---------------------- 158*f949cb75SHu Haowen 159*f949cb75SHu Haowen由於默認情況下引導配置文件是用initrd加載的,因此它將被添加到initrd(initramfs) 160*f949cb75SHu Haowen鏡像文件的末尾,其中包含填充、大小、校驗值和12字節幻數,如下所示:: 161*f949cb75SHu Haowen 162*f949cb75SHu Haowen [initrd][bootconfig][padding][size(le32)][checksum(le32)][#BOOTCONFIG\n] 163*f949cb75SHu Haowen 164*f949cb75SHu Haowen大小和校驗值爲小端序存放的32位無符號值。 165*f949cb75SHu Haowen 166*f949cb75SHu Haowen當引導配置被加到initrd鏡像時,整個文件大小會對齊到4字節。空字符( ``\0`` ) 167*f949cb75SHu Haowen會填補對齊空隙。因此 ``size`` 就是引導配置文件的長度+填充的字節。 168*f949cb75SHu Haowen 169*f949cb75SHu HaowenLinux內核在內存中解碼initrd鏡像的最後部分以獲取引導配置數據。由於這種“揹負式” 170*f949cb75SHu Haowen的方法,只要引導加載器傳遞了正確的initrd文件大小,就無需更改或更新引導加載器 171*f949cb75SHu Haowen和內核鏡像本身。如果引導加載器意外傳遞了更長的大小,內核將無法找到引導配置數 172*f949cb75SHu Haowen據。 173*f949cb75SHu Haowen 174*f949cb75SHu HaowenLinux內核在tools/bootconfig下提供了 ``bootconfig`` 命令來完成此操作,管理員 175*f949cb75SHu Haowen可以用它從initrd鏡像中刪除或追加配置文件。你可以用以下命令來構建它:: 176*f949cb75SHu Haowen 177*f949cb75SHu Haowen # make -C tools/bootconfig 178*f949cb75SHu Haowen 179*f949cb75SHu Haowen要向initrd鏡像添加你的引導配置文件,請按如下命令操作(舊數據會自動移除):: 180*f949cb75SHu Haowen 181*f949cb75SHu Haowen # tools/bootconfig/bootconfig -a your-config /boot/initrd.img-X.Y.Z 182*f949cb75SHu Haowen 183*f949cb75SHu Haowen要從鏡像中移除配置,可以使用-d選項:: 184*f949cb75SHu Haowen 185*f949cb75SHu Haowen # tools/bootconfig/bootconfig -d /boot/initrd.img-X.Y.Z 186*f949cb75SHu Haowen 187*f949cb75SHu Haowen然後在內核命令行上添加 ``bootconfig`` 告訴內核去initrd文件末尾尋找內核配置。 188*f949cb75SHu Haowen 189*f949cb75SHu Haowen將引導配置嵌入內核 190*f949cb75SHu Haowen------------------ 191*f949cb75SHu Haowen 192*f949cb75SHu Haowen如果你不能使用initrd,也可以通過Kconfig選項將引導配置文件嵌入內核中。在此情 193*f949cb75SHu Haowen況下,你需要用以下選項重新編譯內核:: 194*f949cb75SHu Haowen 195*f949cb75SHu Haowen CONFIG_BOOT_CONFIG_EMBED=y 196*f949cb75SHu Haowen CONFIG_BOOT_CONFIG_EMBED_FILE="/引導配置/文件/的/路徑" 197*f949cb75SHu Haowen 198*f949cb75SHu Haowen``CONFIG_BOOT_CONFIG_EMBED_FILE`` 需要從源碼樹或對象樹開始的引導配置文件的 199*f949cb75SHu Haowen絕對/相對路徑。內核會將其嵌入作爲默認引導配置。 200*f949cb75SHu Haowen 201*f949cb75SHu Haowen與將引導配置附加到initrd一樣,你也需要在內核命令行上添加 ``bootconfig`` 告訴 202*f949cb75SHu Haowen內核去啓用內嵌的引導配置。 203*f949cb75SHu Haowen 204*f949cb75SHu Haowen注意,即使你已經設置了此選項,仍可用附加到initrd的其他引導配置覆蓋內嵌的引導 205*f949cb75SHu Haowen配置。 206*f949cb75SHu Haowen 207*f949cb75SHu Haowen通過引導配置傳遞內核參數 208*f949cb75SHu Haowen======================== 209*f949cb75SHu Haowen 210*f949cb75SHu Haowen除了內核命令行,引導配置也可以用於傳遞內核參數。所有 ``kernel`` 關鍵字下的鍵 211*f949cb75SHu Haowen值對都將直接傳遞給內核命令行。此外, ``init`` 下的鍵值對將通過命令行傳遞給 212*f949cb75SHu Haoweninit進程。參數按以下順序與用戶給定的內核命令行字符串相連,因此命令行參數可以 213*f949cb75SHu Haowen覆蓋引導配置參數(這取決於子系統如何處理參數,但通常前面的參數將被後面的參數 214*f949cb75SHu Haowen覆蓋):: 215*f949cb75SHu Haowen 216*f949cb75SHu Haowen [bootconfig params][cmdline params] -- [bootconfig init params][cmdline init params] 217*f949cb75SHu Haowen 218*f949cb75SHu Haowen如果引導配置文件給出的kernel/init參數是:: 219*f949cb75SHu Haowen 220*f949cb75SHu Haowen kernel { 221*f949cb75SHu Haowen root = 01234567-89ab-cdef-0123-456789abcd 222*f949cb75SHu Haowen } 223*f949cb75SHu Haowen init { 224*f949cb75SHu Haowen splash 225*f949cb75SHu Haowen } 226*f949cb75SHu Haowen 227*f949cb75SHu Haowen這將被複制到內核命令行字符串中,如下所示:: 228*f949cb75SHu Haowen 229*f949cb75SHu Haowen root="01234567-89ab-cdef-0123-456789abcd" -- splash 230*f949cb75SHu Haowen 231*f949cb75SHu Haowen如果用戶給出的其他命令行是:: 232*f949cb75SHu Haowen 233*f949cb75SHu Haowen ro bootconfig -- quiet 234*f949cb75SHu Haowen 235*f949cb75SHu Haowen則最後的內核命令行如下:: 236*f949cb75SHu Haowen 237*f949cb75SHu Haowen root="01234567-89ab-cdef-0123-456789abcd" ro bootconfig -- splash quiet 238*f949cb75SHu Haowen 239*f949cb75SHu Haowen 240*f949cb75SHu Haowen配置文件的限制 241*f949cb75SHu Haowen============== 242*f949cb75SHu Haowen 243*f949cb75SHu Haowen當前最大的配置大小是32KB,關鍵字總數(不是鍵值條目)必須少於1024個節點。 244*f949cb75SHu Haowen注意:這不是條目數而是節點數,條目必須消耗超過2個節點(一個關鍵字和一個值)。 245*f949cb75SHu Haowen所以從理論上講最多512個鍵值對。如果關鍵字平均包含3個單詞,則可有256個鍵值對。 246*f949cb75SHu Haowen在大多數情況下,配置項的數量將少於100個條目,小於8KB,因此這應該足夠了。如果 247*f949cb75SHu Haowen節點數超過1024,解析器將返回錯誤,即使文件大小小於32KB。(請注意,此最大尺寸 248*f949cb75SHu Haowen不包括填充的空字符。) 249*f949cb75SHu Haowen無論如何,因爲 ``bootconfig`` 命令在附加啓動配置到initrd映像時會驗證它,用戶 250*f949cb75SHu Haowen可以在引導之前注意到它。 251*f949cb75SHu Haowen 252*f949cb75SHu Haowen 253*f949cb75SHu Haowen引導配置API 254*f949cb75SHu Haowen=========== 255*f949cb75SHu Haowen 256*f949cb75SHu Haowen用戶可以查詢或遍歷鍵值對,也可以查找(前綴)根關鍵字節點,並在查找該節點下的 257*f949cb75SHu Haowen鍵值。 258*f949cb75SHu Haowen 259*f949cb75SHu Haowen如果您有一個關鍵字字符串,則可以直接使用 xbc_find_value() 查詢該鍵的值。如果 260*f949cb75SHu Haowen你想知道引導配置裏有哪些關鍵字,可以使用 xbc_for_each_key_value() 迭代鍵值對。 261*f949cb75SHu Haowen請注意,您需要使用 xbc_array_for_each_value() 訪問數組的值,例如:: 262*f949cb75SHu Haowen 263*f949cb75SHu Haowen vnode = NULL; 264*f949cb75SHu Haowen xbc_find_value("key.word", &vnode); 265*f949cb75SHu Haowen if (vnode && xbc_node_is_array(vnode)) 266*f949cb75SHu Haowen xbc_array_for_each_value(vnode, value) { 267*f949cb75SHu Haowen printk("%s ", value); 268*f949cb75SHu Haowen } 269*f949cb75SHu Haowen 270*f949cb75SHu Haowen如果您想查找具有前綴字符串的鍵,可以使用 xbc_find_node() 通過前綴字符串查找 271*f949cb75SHu Haowen節點,然後用 xbc_node_for_each_key_value() 迭代前綴節點下的鍵。 272*f949cb75SHu Haowen 273*f949cb75SHu Haowen但最典型的用法是獲取前綴下的命名值或前綴下的命名數組,例如:: 274*f949cb75SHu Haowen 275*f949cb75SHu Haowen root = xbc_find_node("key.prefix"); 276*f949cb75SHu Haowen value = xbc_node_find_value(root, "option", &vnode); 277*f949cb75SHu Haowen ... 278*f949cb75SHu Haowen xbc_node_for_each_array_value(root, "array-option", value, anode) { 279*f949cb75SHu Haowen ... 280*f949cb75SHu Haowen } 281*f949cb75SHu Haowen 282*f949cb75SHu Haowen這將訪問值“key.prefix.option”的值和“key.prefix.array-option”的數組。 283*f949cb75SHu Haowen 284*f949cb75SHu Haowen鎖是不需要的,因爲在初始化之後配置只讀。如果需要修改,必須複製所有數據和關鍵字。 285*f949cb75SHu Haowen 286*f949cb75SHu Haowen 287*f949cb75SHu Haowen函數與結構體 288*f949cb75SHu Haowen============ 289*f949cb75SHu Haowen 290*f949cb75SHu Haowen相關定義的kernel-doc參見: 291*f949cb75SHu Haowen 292*f949cb75SHu Haowen - include/linux/bootconfig.h 293*f949cb75SHu Haowen - lib/bootconfig.c 294*f949cb75SHu Haowen 295