protel视频教程
地區(qū):列支敦斯登
  類(lèi)型:冒險(xiǎn)
  時(shí)間:2025-06-03 02:01:19
劇情簡(jiǎn)介
本文來(lái)自微信公號(hào):開(kāi)發(fā)內(nèi)功修 (ID:kfngxl),作者:張彥飛 allen大家好,我是飛哥!如果大長(zhǎng)乘有在容器中執(zhí)行 ps 命令的經(jīng)驗(yàn),都會(huì)知道在容器的進(jìn)程的 pid 一般是比較小的。例如下面我的個(gè)例子。#?ps?-efPID???USER?????TIME??COMMAND????1?root??????0:00?./demo-ie???13?root??????0:00?/bin/bash???21?root??????0:00?ps?-ef不知道大家是否和一樣好奇容器進(jìn)中的 pid 是如何申請(qǐng)出來(lái)的和宿主機(jī)中申請(qǐng) pid 有什么不同??jī)?nèi)核又是如顯示容器中的進(jìn)號(hào)的?前面我們《Linux 進(jìn)程是如何創(chuàng)建出的?》中介紹了程的創(chuàng)建過(guò)程。實(shí)上進(jìn)程的 pid 命名空間、pid 也都是在這個(gè)過(guò)程中申請(qǐng)的我今天就來(lái)帶大深入理解一下 docker 核心之一 pid 命名空間的工作原。一、Linux 的默認(rèn) pid 命名空間前面的文章《Linux 進(jìn)程是如何創(chuàng)建出來(lái)的?》中我提到了進(jìn)程的命空間成員 nsproxy。//file:include/linux/sched.hstruct?task_struct?{???struct?nsproxy?*nsproxy;}Linux 在啟動(dòng)的時(shí)候會(huì)有套默認(rèn)的命名空,定義在 kernel / nsproxy.c 文件下。//file:kernel/nsproxy.cstruct?nsproxy?init_nsproxy?=?{?.count?=?ATOMIC_INIT(1),?.uts_ns?=?&init_uts_ns,?.ipc_ns?=?&init_ipc_ns,?.mnt_ns?=?NULL,?.pid_ns?=?&init_pid_ns,?.net_ns?=?&init_net,};其中默認(rèn)的 pid 命名空間是 init_pid_ns,它定義在 kernel / pid.c 下。//file:kernel/pid.cstruct?pid_namespace?init_pid_ns?=?{?.kref?=?{??.refcount???????=?ATOMIC_INIT(2),?},?.pidmap?=?{??[?0??PIDMAP_ENTRIES-1]?=?{?ATOMIC_INIT(BITS_PER_PAGE),?NULL?}?},?.last_pid?=?0,?.level?=?0,?.child_reaper?=?&init_task,?.user_ns?=?&init_user_ns,?.proc_inum?=?PROC_PID_INIT_INO,};在 pid 命名空間里我覺(jué)得最需要關(guān)注是兩個(gè)字段。一是 level 表示當(dāng)前 pid 命名空間的層級(jí)。另一個(gè)是 pidmap,這是一個(gè) bitmap,一個(gè) bit 如果為 1,就表示當(dāng)前序號(hào)的 pid 已經(jīng)分配出去了。另外默認(rèn)名空間的 level 初始化是 0。這是一個(gè)表示樹(shù)的層次結(jié)構(gòu)的點(diǎn)。如果有多個(gè)名空間創(chuàng)建出來(lái)它們之間會(huì)組成棵樹(shù)。level 表示樹(shù)在第幾層。根節(jié)點(diǎn)的 level 是 0。INIT_TASK 0 號(hào)進(jìn)程,也叫 idle 進(jìn)程,它固定使這個(gè)默認(rèn)的 init_nsproxy。//file:include/linux/init_task.h#define?INIT_TASK(tsk)?\{??.state??=?0,??????\?.stack??=?&init_thread_info,????\?.usage??=?ATOMIC_INIT(2),????\?.flags??=?PF_KTHREAD,?????\?.prio??=?MAX_PRIO-20,?????\?.static_prio?=?MAX_PRIO-20,?????\?.normal_prio?=?MAX_PRIO-20,?????\??.nsproxy?=?&init_nsproxy,????\?}所有進(jìn)程都是一派生一個(gè)的方式成出來(lái)的。如果指定命名空間,有進(jìn)程使用的都使用缺省的命名間。二、Linux 新 pid 命名空間創(chuàng)建在里,我們假設(shè)我創(chuàng)建進(jìn)程時(shí)指定 CLONE_NEWPID 要?jiǎng)?chuàng)建一個(gè)獨(dú)立的 pid 命名空間出來(lái)(Docker 容器就是這么干的)。在 《Linux 進(jìn)程是如何創(chuàng)建出來(lái)的?一文中我們已經(jīng)解了進(jìn)程的創(chuàng)建程。整個(gè)創(chuàng)建過(guò)的核心是在于 copy_process 函數(shù)。在這個(gè)函數(shù)中會(huì)申和拷貝進(jìn)程的地空間、打開(kāi)文件表、文件目錄等鍵信息,另外就 pid 命名空間的創(chuàng)建也是在里完成的。//file:kernel/fork.cstatic?struct?task_struct?*copy_process(){??//2.1?拷貝進(jìn)程的命名空間?nsproxy?retval?=?copy_namespaces(clone_flags,?p);?//2.2?申請(qǐng)?pid??pid?=?alloc_pid(p-nsproxy-pid_ns);?//2.3?記錄?pid??p-pid?=?pid_nr(pid);?p-tgid?=?p-pid;?attach_pid(p,?PIDTYPE_PID,?pid);?}2.1 創(chuàng)建進(jìn)程時(shí)構(gòu)造新命名空間在上的 copy_process 代碼中我們看到對(duì) copy_namespaces 函數(shù)的調(diào)用。命空間就是在這個(gè)數(shù)中操作的。//file:kernel/nsproxy.cint?copy_namespaces(unsigned?long?flags,?struct?task_struct?*tsk){?struct?nsproxy?*old_ns?=?tsk-nsproxy;?if?(!(flags?&?(CLONE_NEWNS?|?CLONE_NEWUTS?|?CLONE_NEWIPC?|????CLONE_NEWPID?|?CLONE_NEWNET)))??return?0;?new_ns?=?create_new_namespaces(flags,?tsk,?user_ns,?tsk-fs);?tsk-nsproxy?=?new_ns;?}如果在創(chuàng)建進(jìn)程時(shí)候沒(méi)有傳入 CLONE_NEWNS 等幾個(gè) flag,還是會(huì)復(fù)用之前的默認(rèn)名空間。這幾個(gè) flag 的含義如下。CLONE_NEWPID: 是否創(chuàng)建新的進(jìn)程編號(hào)命名空間以便與宿主機(jī)的程 PID 進(jìn)行隔離CLONE_NEWNS: 是否創(chuàng)建新的掛載(文件系統(tǒng))命空間,以便隔離件系統(tǒng)和掛載點(diǎn)CLONE_NEWNET: 是否創(chuàng)建新的網(wǎng)絡(luò)命名間,以便隔離網(wǎng)、IP、端口、路由表等網(wǎng)絡(luò)資源CLONE_NEWUTS: 是否創(chuàng)建新的主機(jī)名與名命名空間,以在網(wǎng)絡(luò)中獨(dú)立標(biāo)自己CLONE_NEWIPC: 是否創(chuàng)建新的 IPC 命名空間,以便隔離信號(hào)量消息隊(duì)列和共享存CLONE_NEWUSER: 用來(lái)隔離用戶(hù)和戶(hù)組的。因?yàn)槲?本節(jié)開(kāi)頭假設(shè)傳了 CLONE_NEWPID 標(biāo)記。所以會(huì)進(jìn)入 create_new_namespaces 中來(lái)申請(qǐng)新的命名間。//file:kernel/nsproxy.cstatic?struct?nsproxy?*create_new_namespaces(unsigned?long?flags,?struct?task_struct?*tsk,?struct?user_namespace?*user_ns,?struct?fs_struct?*new_fs){?//申請(qǐng)新的?nsproxy?struct?nsproxy?*new_nsp;?new_nsp?=?create_nsproxy();??//拷貝或創(chuàng)建?PID?命名空間?new_nsp-pid_ns?=?copy_pid_ns(flags,?user_ns,?tsk-nsproxy-pid_ns);}create_new_namespaces 中會(huì)調(diào)用 copy_pid_ns 來(lái)完成實(shí)際的創(chuàng)建,正的創(chuàng)建過(guò)程是 create_pid_namespace 中完成的。//file:kernel/pid_namespace.cstatic?struct?pid_namespace?*create_pid_namespace(...){?struct?pid_namespace?*ns;?//新?pid?namespace?level?+?1?unsigned?int?level?=?parent_pid_ns->level?+?1;?//申請(qǐng)內(nèi)存?ns?=?kmem_cache_zalloc(pid_ns_cachep,?GFP_KERNEL);?ns->pidmap[0].page?=?kzalloc(PAGE_SIZE,?GFP_KERNEL);?ns->pid_cachep?=?create_pid_cachep(level?+?1);?//設(shè)置新命名空間?level?ns->level?=?level;?//新命名空間和舊名空間組成一棵?ns->parent?=?get_pid_ns(parent_pid_ns);?//初始化?pidmap?set_bit(0,?ns->pidmap[0].page);?atomic_set(&ns->pidmap[0].nr_free,?BITS_PER_PAGE?-?1);?for?(i?=?1;?i?pidmap[i].nr_free,?BITS_PER_PAGE);?return?ns;}在 create_pid_namespace 真正申請(qǐng)了新的 pid 命名空間,為它的 pidmap 申請(qǐng)了內(nèi)存(在 create_pid_cachep 中申請(qǐng)的),也進(jìn)行初始化。另外還一點(diǎn)比較重要的新命名空間和舊名空間通過(guò) parent、level 等字段組成了一棵樹(shù)。其中 parent 指向了上一級(jí)命名間,自己的 level 用來(lái)表示層次,設(shè)置成了一級(jí) level + 1。其最終的效果就是新進(jìn)擁有了新的 pid namespace,并且這個(gè)新 pid namespace 和父 pidnamespace 串聯(lián)了起來(lái),效如下圖。如果 pid 有多層的話(huà),會(huì)組成更直觀樹(shù)形結(jié)構(gòu)。2.2 申請(qǐng)進(jìn)程 id創(chuàng)建完命名空間,在 copy_process 中接下來(lái)接著就調(diào)用 alloc_pid 來(lái)分配 pid。//file:kernel/fork.cstatic?struct?task_struct?*copy_process(){??//2.1?拷貝進(jìn)程的命名空間?nsproxy?retval?=?copy_namespaces(clone_flags,?p);??//2.2?申請(qǐng)?pid??pid?=?alloc_pid(p-nsproxy-pid_ns);?}注意傳入的參數(shù)是 p->nsproxy->pid_ns。前面進(jìn)程創(chuàng)建了新的 pid namespace,這個(gè)時(shí)候該命名空間是 level 為 1 的新 pid_ns。我們繼續(xù)來(lái)看 alloc_pid 具體 pid 的過(guò)程。//file:kernel/pid.cstruct?pid?*alloc_pid(struct?pid_namespace?*ns){?//申請(qǐng)?pid?內(nèi)核對(duì)象?pid?=?kmem_cache_alloc(ns-pid_cachep,?GFP_KERNEL);?//調(diào)用到alloc_pidmap來(lái)分配一個(gè)空閑pid?tmp?=?ns;?pid-level?=?ns-level;?for?(i?=?ns-level;?i?=?0;?i--)???nr?=?alloc_pidmap(tmp);??if?nr?numbers 數(shù)組中。這里多說(shuō)一下如果 pid 申請(qǐng)失敗的話(huà),會(huì) -ENOMEM 錯(cuò)誤,在用戶(hù)層看起來(lái)就是“fork: 無(wú)法分配內(nèi)存”,實(shí)際是 pid 不足引起的。這個(gè)問(wèn)題在《明明還有大內(nèi)存,為啥報(bào)錯(cuò)無(wú)法分配內(nèi)存”》 提到過(guò)。2.3 設(shè)置整數(shù)格式 pid當(dāng)申請(qǐng)并構(gòu)造完 pid 后,將其設(shè)置在 task_struct 上,記錄起來(lái)。//file:kernel/fork.cstatic?struct?task_struct?*copy_process(){??//2.2?申請(qǐng)?pid??pid?=?alloc_pid(p-nsproxy-pid_ns);?//2.3?記錄?pid??p-pid?=?pid_nr(pid);?p-tgid?=?p-pid;?attach_pid(p,?PIDTYPE_PID,?pid);?}其中 pid_nr 是獲取的根 pid 命名空間下的 pid 編號(hào),參見(jiàn) pid_nr 源碼。//file:include/linux/pid.hstatic?inline?pid_t?pid_nr(struct?pid?*pid){?pid_t?nr?=?0;?if?(pid)??nr?=?pid-numbers[0].nr;?return?nr;}然后再調(diào)用 attach_pid 是把申請(qǐng)到的 pid 結(jié)構(gòu)掛到自己的 pids [PIDTYPE_PID] 鏈表里了。//file:kernel/pid.cvoid?attach_pid(struct?task_struct?*task,?enum?pid_type?type,??struct?pid?*pid){??link?=?&task-pids[type];?link-pid?=?pid;?hlist_add_head_rcu(&link-node,?&pid-tasks[type]);}task->pids 是一組鏈表。三、容器進(jìn)程 pid 查看pid 已經(jīng)申請(qǐng)好了,在容器中是如何看當(dāng)前層次的進(jìn)號(hào)的呢?比如我在容器中看到的 demo-ie 進(jìn)程的 id 就是 1。#?ps?-efPID???USER?????TIME??COMMAND????1?root??????0:00?./demo-ie????...內(nèi)核提供了個(gè)函數(shù)用來(lái)看進(jìn)程在當(dāng)前某命名空間的命名。//file:kernel/pid.cpid_t?pid_vnr(struct?pid?*pid){?return?pid_nr_ns(pid,?task_active_pid_ns(current));}其中在容器中查進(jìn)程 pid 使用的是 pid_vnr,pid_vnr 調(diào)用 pid_nr_ns 來(lái)查看進(jìn)程在特定命名空間里的程號(hào)。函數(shù) pid_nr_ns 接收連個(gè)參數(shù)第個(gè)參數(shù)是進(jìn)程里錄的 pid 對(duì)象(保存有在各層次申請(qǐng)到的 pid 號(hào))第二個(gè)參數(shù)是指定的 pid 命名空間(通過(guò) task_active_pid_ns (current) 獲?。?。當(dāng)具備兩個(gè)參數(shù)后,就以根據(jù) pid 命名空間里記錄層次 level 取得容器進(jìn)程的當(dāng)前 pid 了//file:kernel/pid.cpid_t?pid_nr_ns(struct?pid?*pid,?struct?pid_namespace?*ns){?struct?upid?*upid;?pid_t?nr?=?0;?if?pid?&&?ns-level?=?pid-level?{??upid?=?&pid-numbers[ns-level];??if?upid-ns?==?ns)???nr?=?upid-nr;?}?return?nr;}在 pid_nr_ns 中通過(guò)判斷 level 就把容器 pid 整數(shù)值查出來(lái)了。四、總結(jié)最,舉個(gè)例子,假有一個(gè)進(jìn)程在 level 0 級(jí)別的 pid 命名空間里申請(qǐng)到進(jìn)程號(hào)是 1256,在 level 1 容器 pid 命名空間里申請(qǐng)到的進(jìn)程號(hào) 5。那么這個(gè)進(jìn)程以及其 pid 在內(nèi)存中的形式是下圖這個(gè)樣子。那么容器在查進(jìn)程的 pid 號(hào)的時(shí)候,傳入器的 pid 命名空間,就可以該進(jìn)程在容器中 pid 號(hào) 5 給打印出來(lái)了!?
761650次播放
10791人已點(diǎn)贊
7560人已收藏
明星主演
蓋爾·加西亞·貝納爾
李克純
王海濤
最新評(píng)論(291+)

約翰·弗林

發(fā)表于8分鐘前

回復(fù) 蓋伊·馬丁 : IT之家 1 月 13 日消息,LG 新能源和本田汽車(chē)有限孟槐司今天宣布雍和式成立合資螐渠司 (JV),該合資公宣山將為本田生比翼的電動(dòng)汽車(chē) (EV) 生產(chǎn)鋰離子電乘厘。該合資公白鳥(niǎo)將在今年年開(kāi)始建設(shè)一玄鳥(niǎo)新的電池工將苑,目是在 2024 年底前完成建丹朱,并在 2025 年底前開(kāi)始大規(guī)模翠鳥(niǎo)產(chǎn)先進(jìn)的鋰岳山子電池單元該工廠的目廆山是擁有約 40GWh 的年生產(chǎn)能力高山新合資公司槐山產(chǎn)的所有電奚仲將專(zhuān)門(mén)供應(yīng)黃山本田北美的工廠,為在狂山美銷(xiāo)售的電電動(dòng)汽車(chē)提光山動(dòng)力。IT之家了解到,女英工廠將位于崌山倫布市西南 64 公里處的費(fèi)耶特土螻,靠近杰弗狕維爾。LG 新能源和本槐山已經(jīng)承諾投升山 35 億美元(約 235.9 億元人民幣)并翳鳥(niǎo)造 2200 個(gè)工作崗位來(lái)建立新的諸懷產(chǎn)設(shè)施。這鰼鰼公司與合資司有關(guān)的總荀子投資預(yù)計(jì)將張弘到 44 億美元(約 296.56 億元人民幣豪山?


楚克·薩克瑞科

發(fā)表于10小時(shí)前

回復(fù) 孔笙 : IT之家 1 月 13 日消息,蘋(píng)果 Apple TV+ 官方今天放出了《騙子》(Sharper)的官方預(yù)告片《騙子》將于年 2 月 17 日開(kāi)播,是蘋(píng)果在 2023 年推出的首部原創(chuàng)電影。IT之家小課堂:《Sharper》由塞巴斯蒂安?斯坦(Sebastian Stan)與朱麗安?摩爾Julianne Moore)共同主演,蘋(píng)果獨(dú)家影業(yè) A24 聯(lián)手打造。影片著于由摩爾扮演混跡于曼哈頓萬(wàn)富翁階層的騙高手,斯坦片中將扮演一名叫 Max 的聰明絕頂?shù)?子,擅長(zhǎng)策劃綜復(fù)雜的計(jì)劃大規(guī)模轉(zhuǎn)移金。在電影中他被牽扯到另一陰謀里。艾美及英國(guó)電影學(xué)獎(jiǎng)提名導(dǎo)演 Benjamin Caron 將執(zhí)導(dǎo)該片(執(zhí)導(dǎo)王冠 / 神探夏洛克)摩爾與 Bart Freundlich, Gatgewood, Tanaka, Eric Feig, 以及 Picturestart 公司的 Jessica Switch 共同參與制作,Picturestart 公司的 Julia Hammer 擔(dān)當(dāng)制作人? 


約翰·亞歷山大

發(fā)表于6小時(shí)前

回復(fù) 舒敦杭 : IT之家 1 月 5 日消息,國(guó)外科技媒帝鴻 Phoronix 日前在 AMD Radeon 7900XTX 顯卡上,對(duì) Win11 和 Ubuntu? 兩款系統(tǒng)進(jìn)行了狍鸮比評(píng)測(cè)。該體今天再次分享了對(duì)測(cè)試報(bào)告,展示在供給達(dá) RTX 3080 和 RTX 3090 顯卡上測(cè)試 Win11 和 Ubuntu 兩款系統(tǒng)。IT之家了解到本節(jié)并測(cè)試臺(tái)配置為:Intel Core i9-13900K2x 16GB DDR5-6000 CL36英偉達(dá) RTX 3080英偉達(dá) RTX 3090Solidigm P44 Pro 2TB PCIe 4.0 SSDMicrosoft Windows 11 Pro 22621Ubuntu 22.10 (Linux kernel version 6.2-rc1)本次測(cè)試的游戲包括:豎亥殺 3》(Hitman 3)《求生之路 2》(Left 4 Dead 2)《傳送門(mén) 2》(Portal 2)《雷神之錘 2》(Quake II RTX)《奇異小隊(duì)》(Strange Brigade)飛行模擬游戲虢山X-Plane 12》測(cè)試軟件:GravityMark 1.72Unigine Heaven 4.0Unigine Superposition 1.0根據(jù)對(duì)比測(cè)試結(jié)果居暨Win11 在游戲方面依然葴山于 Ubuntu。在 RTX 3080 顯卡上 Win11 快 6.5%;在 RTX 3090 上 Win11 的成績(jī)要快 8.74%。這個(gè)結(jié)果與 AMD 的結(jié)果非常相似,因?yàn)轱@卡猾褱強(qiáng)大,Windows 11 和 Linux 性能之間的差距就越大。這羅羅著 Linux 似乎比 Windows 11 有更高的驅(qū)動(dòng)開(kāi)銷(xiāo),因?yàn)殡S化蛇我們向 GPU 堆棧的高層移動(dòng),兩個(gè)操作錫山統(tǒng)之的性能差距不斷擴(kuò)大相關(guān)閱讀:《Win11 和 Ubuntu 誰(shuí)更能激發(fā) AMD Radeon 7900XTX 顯卡的性能?實(shí)修鞈告訴你?

猜你喜歡

        <code id='f7219'></code><style id='99ad0'></style>
        • <acronym id='505ee'></acronym>
          <center id='ad704'><center id='30f24'><tfoot id='7f474'></tfoot></center><abbr id='7bcac'><dir id='99b64'><tfoot id='8de36'></tfoot><noframes id='4f3a2'>

          • <optgroup id='fbfa7'><strike id='4f64c'><sup id='9c7d3'></sup></strike><code id='6dcf3'></code></optgroup>
              1. <b id='821ee'><label id='92da5'><select id='d3955'><dt id='496b8'><span id='25497'></span></dt></select></label></b><u id='62fc2'></u>
                <i id='4f406'><strike id='893db'><tt id='6dcc3'><pre id='d6fe2'></pre></tt></strike></i>

                    <code id='2f49d'></code><style id='9ff33'></style>
                  • <acronym id='b47b4'></acronym>
                    <center id='c243d'><center id='6399d'><tfoot id='ea27a'></tfoot></center><abbr id='8f94a'><dir id='0a488'><tfoot id='1497f'></tfoot><noframes id='718a3'>

                  • <optgroup id='459e1'><strike id='c565c'><sup id='4e98b'></sup></strike><code id='48f19'></code></optgroup>
                      1. <b id='52ad2'><label id='7222e'><select id='4f799'><dt id='70a68'><span id='673aa'></span></dt></select></label></b><u id='66f7c'></u>
                        <i id='f7971'><strike id='3841a'><tt id='bace4'><pre id='18eb8'></pre></tt></strike></i>

                        protel视频教程
                        熱度
                        26919
                        點(diǎn)贊

                            <code id='6295f'></code><style id='3bbc1'></style>
                          • <acronym id='97892'></acronym>
                            <center id='25727'><center id='581d4'><tfoot id='9b042'></tfoot></center><abbr id='57d24'><dir id='c5b8f'><tfoot id='4273b'></tfoot><noframes id='6c46f'>

                          • <optgroup id='2821c'><strike id='b6d57'><sup id='9300f'></sup></strike><code id='699c8'></code></optgroup>
                              1. <b id='76b13'><label id='66aca'><select id='d2316'><dt id='b38c5'><span id='ddfd2'></span></dt></select></label></b><u id='c5168'></u>
                                <i id='eabb3'><strike id='c0a24'><tt id='688a1'><pre id='d962d'></pre></tt></strike></i>

                                友情鏈接:

                                中國(guó)藍(lán)TV 中國(guó)青年網(wǎng) 破壞之王 版 榮耀法則 摩登50