333 lines
13 KiB
ReStructuredText
333 lines
13 KiB
ReStructuredText
|
unshare system call
|
||
|
===================
|
||
|
|
||
|
This document describes the new system call, unshare(). The document
|
||
|
provides an overview of the feature, why it is needed, how it can
|
||
|
be used, its interface specification, design, implementation and
|
||
|
how it can be tested.
|
||
|
|
||
|
Change Log
|
||
|
----------
|
||
|
version 0.1 Initial document, Janak Desai (janak@us.ibm.com), Jan 11, 2006
|
||
|
|
||
|
Contents
|
||
|
--------
|
||
|
1) Overview
|
||
|
2) Benefits
|
||
|
3) Cost
|
||
|
4) Requirements
|
||
|
5) Functional Specification
|
||
|
6) High Level Design
|
||
|
7) Low Level Design
|
||
|
8) Test Specification
|
||
|
9) Future Work
|
||
|
|
||
|
1) Overview
|
||
|
-----------
|
||
|
|
||
|
Most legacy operating system kernels support an abstraction of threads
|
||
|
as multiple execution contexts within a process. These kernels provide
|
||
|
special resources and mechanisms to maintain these "threads". The Linux
|
||
|
kernel, in a clever and simple manner, does not make distinction
|
||
|
between processes and "threads". The kernel allows processes to share
|
||
|
resources and thus they can achieve legacy "threads" behavior without
|
||
|
requiring additional data structures and mechanisms in the kernel. The
|
||
|
power of implementing threads in this manner comes not only from
|
||
|
its simplicity but also from allowing application programmers to work
|
||
|
outside the confinement of all-or-nothing shared resources of legacy
|
||
|
threads. On Linux, at the time of thread creation using the clone system
|
||
|
call, applications can selectively choose which resources to share
|
||
|
between threads.
|
||
|
|
||
|
unshare() system call adds a primitive to the Linux thread model that
|
||
|
allows threads to selectively 'unshare' any resources that were being
|
||
|
shared at the time of their creation. unshare() was conceptualized by
|
||
|
Al Viro in the August of 2000, on the Linux-Kernel mailing list, as part
|
||
|
of the discussion on POSIX threads on Linux. unshare() augments the
|
||
|
usefulness of Linux threads for applications that would like to control
|
||
|
shared resources without creating a new process. unshare() is a natural
|
||
|
addition to the set of available primitives on Linux that implement
|
||
|
the concept of process/thread as a virtual machine.
|
||
|
|
||
|
2) Benefits
|
||
|
-----------
|
||
|
|
||
|
unshare() would be useful to large application frameworks such as PAM
|
||
|
where creating a new process to control sharing/unsharing of process
|
||
|
resources is not possible. Since namespaces are shared by default
|
||
|
when creating a new process using fork or clone, unshare() can benefit
|
||
|
even non-threaded applications if they have a need to disassociate
|
||
|
from default shared namespace. The following lists two use-cases
|
||
|
where unshare() can be used.
|
||
|
|
||
|
2.1 Per-security context namespaces
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
unshare() can be used to implement polyinstantiated directories using
|
||
|
the kernel's per-process namespace mechanism. Polyinstantiated directories,
|
||
|
such as per-user and/or per-security context instance of /tmp, /var/tmp or
|
||
|
per-security context instance of a user's home directory, isolate user
|
||
|
processes when working with these directories. Using unshare(), a PAM
|
||
|
module can easily setup a private namespace for a user at login.
|
||
|
Polyinstantiated directories are required for Common Criteria certification
|
||
|
with Labeled System Protection Profile, however, with the availability
|
||
|
of shared-tree feature in the Linux kernel, even regular Linux systems
|
||
|
can benefit from setting up private namespaces at login and
|
||
|
polyinstantiating /tmp, /var/tmp and other directories deemed
|
||
|
appropriate by system administrators.
|
||
|
|
||
|
2.2 unsharing of virtual memory and/or open files
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
Consider a client/server application where the server is processing
|
||
|
client requests by creating processes that share resources such as
|
||
|
virtual memory and open files. Without unshare(), the server has to
|
||
|
decide what needs to be shared at the time of creating the process
|
||
|
which services the request. unshare() allows the server an ability to
|
||
|
disassociate parts of the context during the servicing of the
|
||
|
request. For large and complex middleware application frameworks, this
|
||
|
ability to unshare() after the process was created can be very
|
||
|
useful.
|
||
|
|
||
|
3) Cost
|
||
|
-------
|
||
|
|
||
|
In order to not duplicate code and to handle the fact that unshare()
|
||
|
works on an active task (as opposed to clone/fork working on a newly
|
||
|
allocated inactive task) unshare() had to make minor reorganizational
|
||
|
changes to copy_* functions utilized by clone/fork system call.
|
||
|
There is a cost associated with altering existing, well tested and
|
||
|
stable code to implement a new feature that may not get exercised
|
||
|
extensively in the beginning. However, with proper design and code
|
||
|
review of the changes and creation of an unshare() test for the LTP
|
||
|
the benefits of this new feature can exceed its cost.
|
||
|
|
||
|
4) Requirements
|
||
|
---------------
|
||
|
|
||
|
unshare() reverses sharing that was done using clone(2) system call,
|
||
|
so unshare() should have a similar interface as clone(2). That is,
|
||
|
since flags in clone(int flags, void \*stack) specifies what should
|
||
|
be shared, similar flags in unshare(int flags) should specify
|
||
|
what should be unshared. Unfortunately, this may appear to invert
|
||
|
the meaning of the flags from the way they are used in clone(2).
|
||
|
However, there was no easy solution that was less confusing and that
|
||
|
allowed incremental context unsharing in future without an ABI change.
|
||
|
|
||
|
unshare() interface should accommodate possible future addition of
|
||
|
new context flags without requiring a rebuild of old applications.
|
||
|
If and when new context flags are added, unshare() design should allow
|
||
|
incremental unsharing of those resources on an as needed basis.
|
||
|
|
||
|
5) Functional Specification
|
||
|
---------------------------
|
||
|
|
||
|
NAME
|
||
|
unshare - disassociate parts of the process execution context
|
||
|
|
||
|
SYNOPSIS
|
||
|
#include <sched.h>
|
||
|
|
||
|
int unshare(int flags);
|
||
|
|
||
|
DESCRIPTION
|
||
|
unshare() allows a process to disassociate parts of its execution
|
||
|
context that are currently being shared with other processes. Part
|
||
|
of execution context, such as the namespace, is shared by default
|
||
|
when a new process is created using fork(2), while other parts,
|
||
|
such as the virtual memory, open file descriptors, etc, may be
|
||
|
shared by explicit request to share them when creating a process
|
||
|
using clone(2).
|
||
|
|
||
|
The main use of unshare() is to allow a process to control its
|
||
|
shared execution context without creating a new process.
|
||
|
|
||
|
The flags argument specifies one or bitwise-or'ed of several of
|
||
|
the following constants.
|
||
|
|
||
|
CLONE_FS
|
||
|
If CLONE_FS is set, file system information of the caller
|
||
|
is disassociated from the shared file system information.
|
||
|
|
||
|
CLONE_FILES
|
||
|
If CLONE_FILES is set, the file descriptor table of the
|
||
|
caller is disassociated from the shared file descriptor
|
||
|
table.
|
||
|
|
||
|
CLONE_NEWNS
|
||
|
If CLONE_NEWNS is set, the namespace of the caller is
|
||
|
disassociated from the shared namespace.
|
||
|
|
||
|
CLONE_VM
|
||
|
If CLONE_VM is set, the virtual memory of the caller is
|
||
|
disassociated from the shared virtual memory.
|
||
|
|
||
|
RETURN VALUE
|
||
|
On success, zero returned. On failure, -1 is returned and errno is
|
||
|
|
||
|
ERRORS
|
||
|
EPERM CLONE_NEWNS was specified by a non-root process (process
|
||
|
without CAP_SYS_ADMIN).
|
||
|
|
||
|
ENOMEM Cannot allocate sufficient memory to copy parts of caller's
|
||
|
context that need to be unshared.
|
||
|
|
||
|
EINVAL Invalid flag was specified as an argument.
|
||
|
|
||
|
CONFORMING TO
|
||
|
The unshare() call is Linux-specific and should not be used
|
||
|
in programs intended to be portable.
|
||
|
|
||
|
SEE ALSO
|
||
|
clone(2), fork(2)
|
||
|
|
||
|
6) High Level Design
|
||
|
--------------------
|
||
|
|
||
|
Depending on the flags argument, the unshare() system call allocates
|
||
|
appropriate process context structures, populates it with values from
|
||
|
the current shared version, associates newly duplicated structures
|
||
|
with the current task structure and releases corresponding shared
|
||
|
versions. Helper functions of clone (copy_*) could not be used
|
||
|
directly by unshare() because of the following two reasons.
|
||
|
|
||
|
1) clone operates on a newly allocated not-yet-active task
|
||
|
structure, where as unshare() operates on the current active
|
||
|
task. Therefore unshare() has to take appropriate task_lock()
|
||
|
before associating newly duplicated context structures
|
||
|
|
||
|
2) unshare() has to allocate and duplicate all context structures
|
||
|
that are being unshared, before associating them with the
|
||
|
current task and releasing older shared structures. Failure
|
||
|
do so will create race conditions and/or oops when trying
|
||
|
to backout due to an error. Consider the case of unsharing
|
||
|
both virtual memory and namespace. After successfully unsharing
|
||
|
vm, if the system call encounters an error while allocating
|
||
|
new namespace structure, the error return code will have to
|
||
|
reverse the unsharing of vm. As part of the reversal the
|
||
|
system call will have to go back to older, shared, vm
|
||
|
structure, which may not exist anymore.
|
||
|
|
||
|
Therefore code from copy_* functions that allocated and duplicated
|
||
|
current context structure was moved into new dup_* functions. Now,
|
||
|
copy_* functions call dup_* functions to allocate and duplicate
|
||
|
appropriate context structures and then associate them with the
|
||
|
task structure that is being constructed. unshare() system call on
|
||
|
the other hand performs the following:
|
||
|
|
||
|
1) Check flags to force missing, but implied, flags
|
||
|
|
||
|
2) For each context structure, call the corresponding unshare()
|
||
|
helper function to allocate and duplicate a new context
|
||
|
structure, if the appropriate bit is set in the flags argument.
|
||
|
|
||
|
3) If there is no error in allocation and duplication and there
|
||
|
are new context structures then lock the current task structure,
|
||
|
associate new context structures with the current task structure,
|
||
|
and release the lock on the current task structure.
|
||
|
|
||
|
4) Appropriately release older, shared, context structures.
|
||
|
|
||
|
7) Low Level Design
|
||
|
-------------------
|
||
|
|
||
|
Implementation of unshare() can be grouped in the following 4 different
|
||
|
items:
|
||
|
|
||
|
a) Reorganization of existing copy_* functions
|
||
|
|
||
|
b) unshare() system call service function
|
||
|
|
||
|
c) unshare() helper functions for each different process context
|
||
|
|
||
|
d) Registration of system call number for different architectures
|
||
|
|
||
|
7.1) Reorganization of copy_* functions
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
Each copy function such as copy_mm, copy_namespace, copy_files,
|
||
|
etc, had roughly two components. The first component allocated
|
||
|
and duplicated the appropriate structure and the second component
|
||
|
linked it to the task structure passed in as an argument to the copy
|
||
|
function. The first component was split into its own function.
|
||
|
These dup_* functions allocated and duplicated the appropriate
|
||
|
context structure. The reorganized copy_* functions invoked
|
||
|
their corresponding dup_* functions and then linked the newly
|
||
|
duplicated structures to the task structure with which the
|
||
|
copy function was called.
|
||
|
|
||
|
7.2) unshare() system call service function
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
* Check flags
|
||
|
Force implied flags. If CLONE_THREAD is set force CLONE_VM.
|
||
|
If CLONE_VM is set, force CLONE_SIGHAND. If CLONE_SIGHAND is
|
||
|
set and signals are also being shared, force CLONE_THREAD. If
|
||
|
CLONE_NEWNS is set, force CLONE_FS.
|
||
|
|
||
|
* For each context flag, invoke the corresponding unshare_*
|
||
|
helper routine with flags passed into the system call and a
|
||
|
reference to pointer pointing the new unshared structure
|
||
|
|
||
|
* If any new structures are created by unshare_* helper
|
||
|
functions, take the task_lock() on the current task,
|
||
|
modify appropriate context pointers, and release the
|
||
|
task lock.
|
||
|
|
||
|
* For all newly unshared structures, release the corresponding
|
||
|
older, shared, structures.
|
||
|
|
||
|
7.3) unshare_* helper functions
|
||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
For unshare_* helpers corresponding to CLONE_SYSVSEM, CLONE_SIGHAND,
|
||
|
and CLONE_THREAD, return -EINVAL since they are not implemented yet.
|
||
|
For others, check the flag value to see if the unsharing is
|
||
|
required for that structure. If it is, invoke the corresponding
|
||
|
dup_* function to allocate and duplicate the structure and return
|
||
|
a pointer to it.
|
||
|
|
||
|
7.4) Finally
|
||
|
~~~~~~~~~~~~
|
||
|
|
||
|
Appropriately modify architecture specific code to register the
|
||
|
new system call.
|
||
|
|
||
|
8) Test Specification
|
||
|
---------------------
|
||
|
|
||
|
The test for unshare() should test the following:
|
||
|
|
||
|
1) Valid flags: Test to check that clone flags for signal and
|
||
|
signal handlers, for which unsharing is not implemented
|
||
|
yet, return -EINVAL.
|
||
|
|
||
|
2) Missing/implied flags: Test to make sure that if unsharing
|
||
|
namespace without specifying unsharing of filesystem, correctly
|
||
|
unshares both namespace and filesystem information.
|
||
|
|
||
|
3) For each of the four (namespace, filesystem, files and vm)
|
||
|
supported unsharing, verify that the system call correctly
|
||
|
unshares the appropriate structure. Verify that unsharing
|
||
|
them individually as well as in combination with each
|
||
|
other works as expected.
|
||
|
|
||
|
4) Concurrent execution: Use shared memory segments and futex on
|
||
|
an address in the shm segment to synchronize execution of
|
||
|
about 10 threads. Have a couple of threads execute execve,
|
||
|
a couple _exit and the rest unshare with different combination
|
||
|
of flags. Verify that unsharing is performed as expected and
|
||
|
that there are no oops or hangs.
|
||
|
|
||
|
9) Future Work
|
||
|
--------------
|
||
|
|
||
|
The current implementation of unshare() does not allow unsharing of
|
||
|
signals and signal handlers. Signals are complex to begin with and
|
||
|
to unshare signals and/or signal handlers of a currently running
|
||
|
process is even more complex. If in the future there is a specific
|
||
|
need to allow unsharing of signals and/or signal handlers, it can
|
||
|
be incrementally added to unshare() without affecting legacy
|
||
|
applications using unshare().
|
||
|
|