首頁 > 後端開發 > Golang > 為什麼從 Go 呼叫 Setns 會為 Mnt 命名空間回傳 EINVAL?

為什麼從 Go 呼叫 Setns 會為 Mnt 命名空間回傳 EINVAL?

Susan Sarandon
發布: 2024-11-01 05:35:02
原創
700 人瀏覽過

Why Does Calling Setns from Go Return EINVAL for the Mnt Namespace?

從Go 調用Setns 會為Mnt 命名空間返回EINVAL

背景

目標是使用C 或Go 程式碼輸入容器的mnt 命名空間。然而,當嘗試進入 mnt 命名空間時,Go 程式碼始終從 setns 呼叫傳回 EINVAL。

工作C 代碼

以下C 代碼成功進入所有指定的命名空間:

<code class="c">#include <sched.h>

main() {
    // ...
    for (i=0; i<5; i++) {
        setns(fd, 0); // Join the provided namespace
    }
}</code>
登入後複製

Go 代碼失敗

另一方面,等效的Go 代碼為mnt 命名空間返回EINVAL 錯誤:

<code class="go">import (
    "syscall"
)

func main() {
    // ...
    err := syscall.RawSyscall(308, fd, 0, 0) // Calling setns
    if err != 0 {
        fmt.Println("setns on", namespaces[i], "namespace failed")
    }
}</code>
登入後複製

答案

問題在於Go 的多線程特性。當 Go 從多執行緒上下文呼叫 setns 時,它對於 mnt 命名空間會失敗,因為它需要單執行緒呼叫者。要解決此問題,必須在 Go 運行時創建任何其他執行緒之前進行 setns 呼叫。

解決方案:單執行緒建構函式技巧

實現此目的的一種方法是使用CGO 建構子技巧,其中在Go 啟動之前執行C 函數:

<code class="c">__attribute__((constructor)) void enter_namespace(void) { setns(...); }</code>
登入後複製

將此建構子加入Go 程式碼中,並硬編碼PID,允許成功進入mnt 命名空間:

<code class="go">/*
__attribute__((constructor)) void enter_namespace(void) { ... }
*/
import "C"</code>
登入後複製

但是,這種方法需要對PID 進行硬編碼,這並不理想。

替代方案:Linux Kernel 4.16 以上

在Linux 核心中4.16 及更高版本中,為setns 引入了一種新模式,允許從多執行緒上下文中安全地調用它。透過使用帶有附加參數 (CLONE_IOCLONE_COMMON) 的 setns 呼叫的修改版本,可以從 Go 進入 mnt 命名空間,即使它是多執行緒的。

以上是為什麼從 Go 呼叫 Setns 會為 Mnt 命名空間回傳 EINVAL?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板