我在 OpenVZ 虚拟化上购买了 VPS,并启用了 Debian 来宾操作系统。
来宾/boot
目录为空。/etc/inittab
是空的。
操作系统初始化过程是如何执行的?
reboot
OpenVZ 容器的含义是什么?
我在 OpenVZ 虚拟化上购买了 VPS,并启用了 Debian 来宾操作系统。
来宾/boot
目录为空。/etc/inittab
是空的。
操作系统初始化过程是如何执行的?
reboot
OpenVZ 容器的含义是什么?
经过研究,我将克隆vzctl
工具作为启动/停止的主要工作:
git clone https://github.com/OpenVZ/vzctl.git
git clone https://github.com/OpenVZ/libvzctl.git
vzctl
使用libvzctl
可以找到https://github.com/OpenVZ/libvzctl/blob/master/lib/env.c#L783的地方:
int exec_init(struct start_param *param)
{
char cid[STR_SIZE];
char *argv[] = {"init", "-z", " ", NULL};
char *envp[] = {"HOME=/", "TERM=linux", cid, NULL};
char **env;
int errcode = 0;
logger(1, 0, "Starting init");
if (stat_file("/sbin/init") == 0 &&
stat_file("/ertc/init") == 0 &&
stat_file("/bin/init") == 0)
errcode = VZCTL_E_BAD_TMPL;
if (write(param->err_p[1], &errcode, sizeof(errcode)) == -1)
logger(-1, errno, "exec_init: write(param->err_p[1]");
snprintf(cid, sizeof(cid), "container="SYSTEMD_CTID_FMT, EID(param->h));
env = makeenv(envp, ¶m->h->env_param->misc->ve_env);
if (env == NULL)
return VZCTL_E_NOMEM;
execve("/sbin/init", argv, env);
execve("/etc/init", argv, env);
execve("/bin/init", argv, env);
free_ar_str(env);
free(env);
return VZCTL_E_BAD_TMPL;
}
通过https://github.com/OpenVZ/libvzctl/blob/master/lib/env.c#L103停止:
int real_env_stop(int stop_mode)
{
logger(10, 0, "* stop mode %d", stop_mode);
close_fds(1, -1);
/* Disable fsync. The fsync will be done by umount() */
configure_sysctl("/proc/sys/fs/fsync-enable", "0");
switch (stop_mode) {
case M_HALT: {
char *argv[] = {"halt", NULL};
char *argv_init[] = {"init", "0", NULL};
execvep(argv[0], argv, NULL);
execvep(argv_init[0], argv_init, NULL);
break;
}
case M_REBOOT: {
char *argv[] = {"reboot", NULL};
execvep(argv[0], argv, NULL);
break;
}
case M_KILL:
return syscall(__NR_reboot, LINUX_REBOOT_MAGIC1,
LINUX_REBOOT_MAGIC2,
LINUX_REBOOT_CMD_POWER_OFF, NULL);
}
return -1;
}
在调用/sbin/init
vzctl
make 之前进行一些检查(例如系统是否已运行或已安装,是否存在某些文件等),如果需要,请停止,安装 fs,进行 chroot 的一些隔离模拟并执行/sbin/init
.
结论OpenVZ 不使用grub
/ linux-image
/initrd
来自来宾操作系统并直接调用第一个找到的对象:
"/sbin/init"
"/etc/init"
"/bin/init"
在来宾操作系统中。为了阻止它使用其中之一
halt
init 0
reboot
从来宾操作系统。从用户的角度来看,容器初始化(安全、隔离、挂载等)对来宾操作系统启动过程不感兴趣。
OpenVZ VPS 本质上是作为主机系统中的沙盒进程树运行的。因此,它没有自己的引导加载程序或内核,并且通常不会在/boot
.
OpenVZ VPS 是通过让主机系统/bin/init
在特殊环境中运行来启动的,这样它就与主机系统隔离开来,因此它认为自己是 PID 1。我不完全确定细节,因为 Parallels 有没有详细记录它。
对于使用 systemd init 而不是 SysV init 的 Linux 系统,空或丢失/etc/inittab
是正常的。当前版本的 Debian 默认使用 systemd;此行为并非特定于 OpenVZ。
我不完全确定reboot
在 OpenVZ VPS 中运行是如何工作的,但我想主机内核必须对reboot()
容器内的系统调用进行一些特殊处理,以便它导致主机内核停止或重新启动容器,而不是比整个系统。