Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chroot because pivot_root manpage says so #12

Open
Michcioperz opened this issue Dec 6, 2018 · 6 comments
Open

chroot because pivot_root manpage says so #12

Michcioperz opened this issue Dec 6, 2018 · 6 comments

Comments

@Michcioperz
Copy link
Member

// chroot because pivot_root manpage says so
withErrnoCheck("chroot", chroot, ".");

no.

@Wolf480pl
Copy link
Member

pivot_root(2):

[...] the implementation of pivot_root() may change in the future. At the time of writing, pivot_root() changes root and current working directory of each process or thread to new_root if they point to the old root directory. This is necessary in order to prevent kernel threads from keeping the old root directory busy with their root and current working directory, even if they never access the filesystem in any way. In the future, there may be a mechanism for kernel threads to explicitly relinquish any access to the filesystem, such that this fairly intrusive mechanism can be removed from pivot_root().

This means that using pivot_root alone is not enough, as it may break in the future.

chroot alone is not enough, because it leaves the old root mounted in our mount namespace, which makes it possible to escape the chroot eg. with another call to chroot or pivot_root. This requires CAP_SYS_ADMIN, which the sandboxed does not have, but for the sake of defense-in-depth, we want to stay secure even if the sandboxed process somehow acquires CAP_SYS_ADMIN.

TODO: verify the above claim experimentally

The manpage for pivot_root(2) says that we should chroot before pivot_root instead of after. We should verify if this works as good or better than what we do right now. We should also check how other tools use pivot_root, considering that future changes to the kernel will probably take into account what popular existing tools do.

@Michcioperz
Copy link
Member Author

Apologies for my earlier laconic explanation.

What I meant was that pivot_root(2) man page recommends chdir("/") and not chroot.

By the way, pivot_root(2):

   The following restrictions apply to new_root and put_old:

   -  They must be directories.

   -  new_root  and  put_old must not be on the same filesystem as the current
      root.

   -  put_old must be underneath new_root, that is, adding a nonzero number of
      /.. to the string pointed to by put_old must yield the same directory as
      new_root.

@Michcioperz
Copy link
Member Author

Just found out lxc does the pivot_root thing:

https://github.com/lxc/lxc/blob/4faaed332b99e60c0e1eed7dcc697f4fbf4f9441/src/lxc/conf.c#L1496

I still don't approve of this personally.

@Wolf480pl
Copy link
Member

What I meant was that pivot_root(2) man page recommends chdir("/") and not chroot.

Yeah, but pivot_root(8) does. I'm not sure why there is a difference, will need to look into that.

Just found out lxc does the pivot_root thing:
[...]
I still don't approve of this personally.

I am aware that the use of . as old_root is not compliant with the manpage, that it is tricky to understand, and that it is ugly. However, this way we don't need an empty directory in the new root.
Because we allow arbitrary filesystem to be bind-mounted on /, and because it can be mounted read-only, and mounted concurrently by multiple instances of sio2jail, we'd need to make sure every such filesystem contains said empty directory from the very beginning, and fail if it doesn't.

This would break some things, require changes elsewhere in the system, and AFAIK would not make any observable behaviour better. Unless I'm missing something.

@Michcioperz
Copy link
Member Author

I guess a little uncompliance is fine given how often we update the kernel on prod

@Michcioperz
Copy link
Member Author

Yeah, but pivot_root(8) does.

I can't help but notice we're not using pivot_root(8) here.

I'm not sure why there is a difference, will need to look into that.

One reason I can think of is that pivot_root(8) is a binary and its execution of chdir("/") is futile since it doesn't exec at the end. But assuming this first-result-on-ddg repo is a straight mirror of upstream:

https://github.com/karelzak/util-linux/blob/917f53cf13c36d32c175f80f2074576595830573/sys-utils/pivot_root.8#L13-L34

this seems to be for the reason mentioned and out of convenience, to break up whatever Links to the Past are left.

iceboy233 referenced this issue in vijos/sandbox Aug 30, 2019
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants