In
software development
Software development is the process of conceiving, specifying, designing, programming, documenting, testing, and bug fixing involved in creating and maintaining applications, frameworks, or other software components. Software development inv ...
, time-of-check to time-of-use (TOCTOU, TOCTTOU or TOC/TOU) is a class of
software bugs caused by a
race condition
A race condition or race hazard is the condition of an electronics, software, or other system where the system's substantive behavior is dependent on the sequence or timing of other uncontrollable events. It becomes a bug when one or more of t ...
involving the ''checking'' of the state of a part of a system (such as a security credential) and the ''use'' of the results of that check.
TOCTOU race conditions are common in
Unix
Unix (; trademarked as UNIX) is a family of multitasking, multiuser computer operating systems that derive from the original AT&T Unix, whose development started in 1969 at the Bell Labs research center by Ken Thompson, Dennis Ritchie, a ...
between operations on the
file system
In computing, file system or filesystem (often abbreviated to fs) is a method and data structure that the operating system uses to control how data is stored and retrieved. Without a file system, data placed in a storage medium would be one lar ...
, but can occur in other contexts, including local
sockets and improper use of
database transaction
A database transaction symbolizes a unit of work, performed within a database management system (or similar system) against a database, that is treated in a coherent and reliable way independent of other transactions. A transaction generally repr ...
s. In the early 1990s, the mail utility of BSD 4.3 UNIX had an
exploitable race condition for temporary files because it used the
mktemp()
function.
Early versions of
OpenSSH
OpenSSH (also known as OpenBSD Secure Shell) is a suite of secure networking utilities based on the Secure Shell (SSH) protocol, which provides a secure channel over an unsecured network in a client–server architecture.
Network Working G ...
had an exploitable race condition for
Unix domain sockets. They remain a problem in modern systems; as of 2019, a TOCTOU race condition in
Docker allows root access to the filesystem of the host platform.
Examples
In
Unix
Unix (; trademarked as UNIX) is a family of multitasking, multiuser computer operating systems that derive from the original AT&T Unix, whose development started in 1969 at the Bell Labs research center by Ken Thompson, Dennis Ritchie, a ...
, the following
C code, when used in a
setuid
The Unix access rights flags setuid and setgid (short for ''set user identity'' and ''set group identity'') allow users to run an executable with the file system permissions of the executable's owner or group respectively and to change behaviour ...
program, has a TOCTOU bug:
if (access("file", W_OK) != 0)
fd = open("file", O_WRONLY);
write(fd, buffer, sizeof(buffer));
Here, ''access'' is intended to check whether the real user who executed the
setuid
program would normally be allowed to write the file (i.e.,
''access''
checks the
real userid
Unix-like operating systems identify a user by a value called a user identifier, often abbreviated to user ID or UID. The UID, along with the group identifier (GID) and other access control criteria, is used to determine which system resources a us ...
rather than
effective userid).
This race condition is vulnerable to an attack:
In this example, an attacker can exploit the race condition between the
access
and
open
to trick the
setuid
victim into overwriting an entry in the system password database. TOCTOU races can be used for
privilege escalation to get administrative access to a machine.
Although this sequence of events requires precise timing, it is possible for an attacker to arrange such conditions without too much difficulty.
The implication is that applications cannot assume the state managed by the operating system (in this case the file system namespace) will not change between system calls.
Reliably timing TOCTOU
Exploiting a TOCTOU race condition requires precise timing to ensure that the attacker's operations interleave properly with the victim's. In the example above, the attacker must execute the
symlink
system call precisely between the
access
and
open
. For the most general attack, the attacker must be scheduled for execution after each operation by the victim, also known as "single-stepping" the victim.
In the case of BSD 4.3 mail utility and
mktemp()
,
the attacker can simply keep launching mail utility in one process, and keep guessing the temporary file names and keep making symlinks in another process. The attack can usually succeed in less than one minute.
Techniques for single-stepping a victim program include file system mazes and algorithmic complexity attacks. In both cases, the attacker manipulates the OS state to control scheduling of the victim.
File system mazes force the victim to read a directory entry that is not in the OS cache, and the OS puts the victim to sleep while it is reading the directory from disk. Algorithmic complexity attacks force the victim to spend its entire scheduling quantum inside a single system call traversing the kernel's hash table of cached file names. The attacker creates a very large number of files with names that hash to the same value as the file the victim will look up.
Preventing TOCTOU
Despite conceptual simplicity, TOCTOU race conditions are difficult to avoid and eliminate. One general technique is to use error handling instead of pre-checking, under the philosophy of EAFP – "It is easier to ask for forgiveness than permission" rather than LBYL – "look before you leap" – in this case there is no check, and failure of assumptions to hold are signaled by an error being returned.
In the context of file system TOCTOU race conditions, the fundamental challenge is ensuring that the file system cannot be changed between two system calls. In 2004, an impossibility result was published, showing that there was no portable, deterministic technique for avoiding TOCTOU race conditions when using the UNIX
access
and
open
filesystem calls.
Since this impossibility result, libraries for tracking
file descriptor
In Unix and Unix-like computer operating systems, a file descriptor (FD, less frequently fildes) is a process-unique identifier ( handle) for a file or other input/output resource, such as a pipe or network socket.
File descriptors typically ...
s and ensuring correctness have been proposed by researchers.
An alternative solution proposed in the research community is for UNIX systems to adopt
transaction
Transaction or transactional may refer to:
Commerce
* Financial transaction, an agreement, communication, or movement carried out between a buyer and a seller to exchange an asset for payment
*Debits and credits in a Double-entry bookkeeping sys ...
s in the file system or the OS kernel. Transactions provide a
concurrency control
In information technology and computer science, especially in the fields of computer programming, operating systems, multiprocessors, and databases, concurrency control ensures that correct results for concurrent operations are generated, while ...
abstraction for the OS, and can be used to prevent TOCTOU races. While no production UNIX kernel has yet adopted transactions, proof-of-concept research prototypes have been developed for Linux, including the Valor file system and the TxOS kernel.
Microsoft Windows has added transactions to its
NTFS
New Technology File System (NTFS) is a proprietary journaling file system developed by Microsoft. Starting with Windows NT 3.1, it is the default file system of the Windows NT family. It superseded File Allocation Table (FAT) as the preferred f ...
file system, but Microsoft discourages their use, and has indicated that they may be removed in a future version of Windows.
File locking
File locking is a mechanism that restricts access to a computer file, or to a region of a file, by allowing only one user (computing), user or computer process, process to modify or delete it at a specific time and to prevent reading of the file w ...
is a common technique for preventing race conditions for a single file, but it does not extend to the file system namespace and other metadata, nor does locking work well with networked filesystems, and cannot prevent TOCTOU race conditions.
For setuid binaries a possible solution is to use the
seteuid()
system call to change the effective user and then perform the
open()
. Differences in
setuid()
between operating systems can be problematic.
See also
*
Linearizability
References
Further reading
*
* {{cite web , last1=Tsafrir , first1=Dan , last2=Hertz , first2=Tomer , last3=Wagner , first3=David , last4=Da Silva , first4=Dilma , year=2008 , title=Portably Solving File TOCTTOU Races with Hardness Amplification , work=Proceedings of the 6th USENIX Conference on File and Storage Technologies (FAST '08), San Jose (CA), February 26–29, 2008 , pages=189–206 , url=https://people.eecs.berkeley.edu/~daw/papers/tocttou-fast08.pdf
Computer security exploits
software bugs