- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
C 代码工作正常并正确进入命名空间,但 Go 代码似乎总是从 setns
返回 EINVAL调用电话进入mnt
命名空间。我在 Go .so
上尝试了许多排列(包括带有 cgo 和外部 1.2
的嵌入式 C 代码) , 1.3
和当前的提示。
单步执行gdb
中的代码表明两个序列都在调用 setns
在 libc
完全相同的方式(或者在我看来)。
我将问题归结为下面的代码。我做错了什么?
我有一个用于启动快速busybox容器的shell别名:
alias startbb='docker inspect --format "{{ .State.Pid }}" $(docker run -d busybox sleep 1000000)'
运行后,startbb
将启动一个容器并输出它的 PID。
lxc-checkconfig
输出:
Found kernel config file /boot/config-3.8.0-44-generic
--- Namespaces ---
Namespaces: enabled
Utsname namespace: enabled
Ipc namespace: enabled
Pid namespace: enabled
User namespace: missing
Network namespace: enabled
Multiple /dev/pts instances: enabled
--- Control groups ---
Cgroup: enabled
Cgroup clone_children flag: enabled
Cgroup device: enabled
Cgroup sched: enabled
Cgroup cpu account: enabled
Cgroup memory controller: missing
Cgroup cpuset: enabled
--- Misc ---
Veth pair device: enabled
Macvlan: enabled
Vlan: enabled
File capabilities: enabled
uname -a
产生:
Linux gecko 3.8.0-44-generic #66~precise1-Ubuntu SMP Tue Jul 15 04:01:04 UTC 2014 x86_64 x86_64 x86_64 GNU/Linux
以下 C 代码可以正常工作:
#include <errno.h>
#include <sched.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
main(int argc, char* argv[]) {
int i;
char nspath[1024];
char *namespaces[] = { "ipc", "uts", "net", "pid", "mnt" };
if (geteuid()) { fprintf(stderr, "%s\n", "abort: you want to run this as root"); exit(1); }
if (argc != 2) { fprintf(stderr, "%s\n", "abort: you must provide a PID as the sole argument"); exit(2); }
for (i=0; i<5; i++) {
sprintf(nspath, "/proc/%s/ns/%s", argv[1], namespaces[i]);
int fd = open(nspath, O_RDONLY);
if (setns(fd, 0) == -1) {
fprintf(stderr, "setns on %s namespace failed: %s\n", namespaces[i], strerror(errno));
} else {
fprintf(stdout, "setns on %s namespace succeeded\n", namespaces[i]);
}
close(fd);
}
}
使用 gcc -o checkns checkns.c
编译后,sudo ./checkns <PID>
的输出是:
setns on ipc namespace succeeded
setns on uts namespace succeeded
setns on net namespace succeeded
setns on pid namespace succeeded
setns on mnt namespace succeeded
相反,以下 Go 代码(应该是相同的)不能很好地工作:
package main
import (
"fmt"
"os"
"path/filepath"
"syscall"
)
func main() {
if syscall.Geteuid() != 0 {
fmt.Println("abort: you want to run this as root")
os.Exit(1)
}
if len(os.Args) != 2 {
fmt.Println("abort: you must provide a PID as the sole argument")
os.Exit(2)
}
namespaces := []string{"ipc", "uts", "net", "pid", "mnt"}
for i := range namespaces {
fd, _ := syscall.Open(filepath.Join("/proc", os.Args[1], "ns", namespaces[i]), syscall.O_RDONLY, 0644)
err, _, msg := syscall.RawSyscall(308, uintptr(fd), 0, 0) // 308 == setns
if err != 0 {
fmt.Println("setns on", namespaces[i], "namespace failed:", msg)
} else {
fmt.Println("setns on", namespaces[i], "namespace succeeded")
}
}
}
相反,运行 sudo go run main.go <PID>
产生:
setns on ipc namespace succeeded
setns on uts namespace succeeded
setns on net namespace succeeded
setns on pid namespace succeeded
setns on mnt namespace failed: invalid argument
最佳答案
(有an issue filed on the Go project)
所以,这个问题的答案是您必须从单线程上下文中调用 setns
。这是有道理的,因为 setns
应该将当前线程加入命名空间。由于 Go 是多线程的,因此您需要在 Go 运行时线程启动之前进行 setns
调用。
我认为这是因为执行syscall.RawSyscall
调用的线程不是主线程——甚至withruntime.LockOSThread
结果不是你所期望的(即,goroutine 被“锁定”到主 C 线程,因此相当于下面解释的构造函数技巧)。
我在提交问题后得到的回复建议使用“cgo
构造函数技巧”。我在这个“技巧”上找不到任何“正确”的文档,但是 Docker/Michael Crosby 在 nsinit
中使用了它,即使我逐行查看了该代码,但我没有尝试以这种方式运行它(请参阅下面的挫败感)。
“诀窍”基本上是您可以让 cgo
在启动 Go 运行时之前执行 C 函数。
为此,您添加 __attribute__((constructor))
宏来装饰您要在 Go 启动之前运行的函数:
/*
__attribute__((constructor)) void init() {
// this code will execute before Go starts up
// in runs in a single-threaded C context
// before Go's threads start running
}
*/
import "C"
使用这个作为模板,我修改了 checkns.go
如下:
/*
#include <sched.h>
#include <stdio.h>
#include <fcntl.h>
__attribute__((constructor)) void enter_namespace(void) {
setns(open("/proc/<PID>/ns/mnt", O_RDONLY, 0644), 0);
}
*/
import "C"
... rest of file is unchanged ...
此代码有效,但需要对 PID
进行硬编码,因为它没有从命令行输入正确读取,但它说明了这个想法(如果您提供 PID
来自如上所述启动的容器)。
这很令人沮丧,因为我想多次调用 setns
但由于此 C 代码在 Go 运行时开始之前执行,因此没有可用的 Go 代码。
更新: 在内核邮件列表中闲逛提供了 this link到记录这一点的对话。我似乎无法在任何实际发布的联机帮助页中找到它,但这是 setns(2)
补丁中的引述,由 Eric Biederman 确认:
A process may not be reassociated with a new mount namespace if it is multi-threaded. Changing the mount namespace requires that the caller possess both CAP_SYS_CHROOT and CAP_SYS_ADMIN capabilities in its own user namespace and CAP_SYS_ADMIN in the target mount namespace.
关于从 Go 调用 setns 为 mnt 命名空间返回 EINVAL,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25704661/
我在极少数情况下看到 pthread_cond_timedwait() 返回 EINVAL 并导致我们的系统发生致命崩溃。我知道这意味着传入的参数之一必须无效,但是 mutex 或 cond 变量如何
我正在尝试使用 aio_write(1) 函数。下面的代码是我的程序。但是,aio_write() 总是返回 EINVAL。 看了EINVAL的错误原因:aio_offset、aio_reqprio、
我们正在从 ARM 嵌入式 Ubuntu 14.04 更新到 ARM 嵌入式 Ubuntu 16.04。在第一个平台上,我们可以毫无问题地访问使用 SPIDEV 的芯片。在 Ubuntu 平台上,我在
我有以下两个功能: static int MessAttrManager_inotifyUnregister(MessPropertyManager* base, int wd){ MessAtt
我正在尝试弄清楚如何正确使用 tee。在我的应用程序中,tee 出于某种原因总是返回 EINVAL。我感到绝望,并尝试运行 tee 手册页中列出的示例应用程序(例如:https://linux.die
我无法使 mmap 函数工作。它返回 EINVAL 错误代码。 void* mapped = mmap((void*)(map_addr + slide),
我正在尝试调试一个问题,但我不明白。 func BackgroundProcess(lpCommandLine string) (error) { var lpProcessAttrs win
我正在使用为以下问题提供的代码 numa+mbind+segfault ,每次调用 mbind 都会返回 EINVAL。我怎样才能得到什么是完全错误的?我问这个是因为有很多原因可以返回 EINVAL。
我正在尝试在 C 中发送一个 UDP 数据包。我有以下 sendto(): char* msg = "Hello"; //ret is the return value of getaddrinfo,
我正在尝试下载一个 ~2GB 的文件并将其写入本地文件,但我遇到了这个问题: 这是适用的代码: File.open(local_file, "wb") do |tempfile| puts
我在闪存驱动器 E 上使用 Node v11.5 和 npm 6.4.1,在 win 7 上,我正在尝试安装最新的 netlify cli。关注 https://cli.netlify.com/get
我是新人,我有一个问题。在android中,我无法通过TCP套接字连接任何远程地址。当我尝试连接时,调试器显示错误: Exception: failed to connect to /23.20.47
我修改了this C数据报(UDP)套接字客户端示例: #include #include #include #include #include #include #include #i
我有一个 C-ZMQ 客户端,它接收两个随机端口(来自 pyzmq 服务器),然后连接到它们。 通常情况下,一切正常,但有时第二次连接会失败,errno 设置为 EINVAL。 (即使我在连接调用之间
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我正在尝试使用 setsockopt 函数将地址添加到接口(interface)的多播地址列表中,但失败并显示错误号 EINVAL。我在内核代码中放置了一些 printk,看起来 errno 最终是在
我正在尝试绑定(bind)服务器套接字,以便我可以接收和监听来自其他客户端的传入消息。但我无法绑定(bind),它返回错误 - EINVAL(无效参数)。我已经完成了之前提出的与此错误相关的问题,Li
我正在尝试对 IO 密集型代码使用预取提示。我根据我对 gpfs_fcntl() 手册页的理解设置了代码,但确实得到了一个 EINVAL。我现在有点迷茫,我做错了什么 - 任何提示表示赞赏。 挂载
我想用以下格式的文件名创建一个文件:DAY-MONTH-YEAR--HOUR:MINUTE 但是当我使用 -- 或/and : 我遇到了 open failed: EINVAL 异常。我试图摆脱这些焦
几个小时以来我一直遇到 shmget 的问题,现在我似乎无法弄清楚。每次我尝试调用 shmget() 时,它总是返回 EINVAL: "Invalid Argument"错误。 代码的重要部分如下:
我是一名优秀的程序员,十分优秀!