libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
filesystem.hxx File Reference
#include <linux/close_range.h>
#include <optional>
#include <string>
#include <string_view>
#include <utility>
#include <cosmos/SysString.hxx>
#include <cosmos/dso_export.h>
#include <cosmos/error/errno.hxx>
#include <cosmos/fs/DirFD.hxx>
#include <cosmos/fs/FileDescriptor.hxx>
#include <cosmos/fs/types.hxx>
#include <cosmos/types.hxx>

Go to the source code of this file.

Classes

struct  cosmos::fs::CopyFileRangeParameters
 set of parameters for copy_file_range(). More...
 

Typedefs

using cosmos::fs::CloseRangeFlags = BitMask<CloseRangeFlag>
 Flags used in cosmos::fs::close_range().
 
using cosmos::fs::AccessChecks = BitMask<AccessCheck>
 
using cosmos::fs::AccessFlags = BitMask<AccessFlag>
 
using cosmos::fs::LockFlags = BitMask<LockFlag>
 Additional flags influencing flock behaviour.
 

Enumerations

enum class  cosmos::fs::CloseRangeFlag : unsigned int { CLOEXEC = CLOSE_RANGE_CLOEXEC , UNSHARE = CLOSE_RANGE_UNSHARE }
 
enum class  cosmos::fs::AccessCheck : int { READ_OK = R_OK , WRITE_OK = W_OK , EXEC_OK = X_OK }
 Different access checks that can be performed in check_access(). More...
 
enum class  cosmos::fs::AccessFlag : int { EFFECTIVE_CREDS = AT_EACCESS , NO_FOLLOW = AT_SYMLINK_NOFOLLOW }
 Extra flags that influence the behaviour of check_access_at(), and check_access_fd(). More...
 
enum class  cosmos::fs::LockOperation : int { LOCK_SHARED = LOCK_SH , LOCK_EXCLUSIVE = LOCK_EX , UNLOCK = LOCK_UN }
 Flags used with flock(). More...
 
enum class  cosmos::fs::LockFlag : int { LOCK_NONBLOCK = LOCK_NB }
 

Functions

FileDescriptor cosmos::fs::open (const SysString path, const OpenMode mode, const OpenFlags flags, const std::optional< FileMode > fmode={})
 Open a file using specific OpenFlags, potentially creating it first using the given fmode.
 
FileDescriptor cosmos::fs::open_at (const DirFD dir_fd, const SysString path, const OpenMode mode, const OpenFlags flags, const std::optional< FileMode > fmode={})
 Open the given path relative to the given directory file descriptor dir_fd.
 
void cosmos::fs::close_range (const FileNum first, const FileNum last=FileNum::MAX_FD, const CloseRangeFlags flags=CloseRangeFlags{})
 Close a range of file descriptor numbers in the current process.
 
std::pair< FileDescriptor, std::string > cosmos::fs::make_tempfile (const SysString _template, const OpenFlags flags=OpenFlags{OpenFlag::CLOEXEC})
 Safely create a temporary file and return it's file descriptor and path.
 
std::string cosmos::fs::make_tempdir (const SysString _template)
 Safely create a temporary directory and return it's path.
 
void cosmos::fs::make_fifo (const SysString path, const FileMode mode)
 Creates a named pipe at the given file system location.
 
void cosmos::fs::make_fifo_at (const DirFD dir_fd, const SysString path, const FileMode mode)
 Creates a named pipe relative to the given directory descriptor.
 
FileMode cosmos::fs::set_umask (const FileMode mode)
 Sets the process's file creation mask.
 
bool cosmos::fs::exists_file (const SysString path)
 Returns whether the given file system object exists.
 
void cosmos::fs::unlink_file (const SysString path)
 Removes the file object found at path.
 
void cosmos::fs::unlink_file_at (const DirFD dir_fd, const SysString path)
 Removes the file object found at path relative to dir_fd.
 
void cosmos::fs::change_dir (const SysString path)
 Change the calling process's current working directory to path.
 
std::string cosmos::fs::get_working_dir ()
 Returns the process's current working directory.
 
std::optional< std::string > cosmos::fs::which (const std::string_view exec_base) noexcept
 Find the full path to the executable program exec_base.
 
void cosmos::fs::make_dir (const SysString path, const FileMode mode)
 Creates a directory at the given location.
 
void cosmos::fs::make_dir_at (const DirFD dir_fd, const SysString path, const FileMode mode)
 Creates a directory at the location relative to dir_fd.
 
void cosmos::fs::remove_dir (const SysString path)
 Removes an empty directory at the given location.
 
void cosmos::fs::remove_dir_at (const DirFD dir_fd, const SysString path)
 Removes an empty directory relative to dir_fd.
 
Errno cosmos::fs::make_all_dirs (const SysString path, const FileMode mode)
 Creates a directory, potentially creating multiple directory components.
 
void cosmos::fs::remove_tree (const SysString path)
 Recursively removes all directory content in path.
 
void cosmos::fs::change_mode (const SysString path, const FileMode mode)
 Changes the FileMode of the given path.
 
void cosmos::fs::change_mode (const FileDescriptor fd, const FileMode mode)
 Changes the FileMode of the given open file descriptor.
 
void cosmos::fs::change_owner (const SysString path, const UserID uid, const GroupID gid=GroupID::INVALID)
 Change numerical owner and/or group ID of a file path.
 
void cosmos::fs::change_owner (const FileDescriptor fd, const UserID uid, const GroupID gid=GroupID::INVALID)
 Change numerical owner and/or group ID of the given open file descriptor.
 
void cosmos::fs::change_owner (const SysString path, const SysString user, const SysString group={})
 Change owner and/or group of the given path by user name and/or group name.
 
void cosmos::fs::change_owner (const FileDescriptor fd, const SysString user, const SysString group={})
 Change owner and/or group of the given file descriptor by user name and/or group name.
 
void cosmos::fs::change_group (const FileDescriptor fd, const GroupID id)
 Convenience wrapper of change_owner() to change only the group of a file.
 
void cosmos::fs::change_group (const FileDescriptor fd, const SysString group)
 Convenience wrapper of change_owner() to change only the group of a file.
 
void cosmos::fs::change_group (const SysString path, const GroupID id)
 Convenience wrapper of change_owner() to change only the group of a file.
 
void cosmos::fs::change_group (const SysString path, const SysString group)
 Convenience wrapper of change_owner() to change only the group of a file.
 
void cosmos::fs::change_owner_nofollow (const SysString path, const UserID uid, const GroupID gid=GroupID::INVALID)
 Changes owner and/or group of the given path while not following symlinks.
 
void cosmos::fs::change_owner_nofollow (const SysString path, const SysString user, const SysString group={})
 Changes owner and/or group of the given path while not following symlinks.
 
void cosmos::fs::make_symlink (const SysString target, const SysString path)
 Creates a symbolic link at path pointing to target.
 
void cosmos::fs::make_symlink_at (const SysString target, const DirFD dir_fd, const SysString path)
 Creates a symbolic link relative to dir_fd pointing to target.
 
std::string cosmos::fs::read_symlink (const SysString path)
 Returns the target (content) of the symbolic link at path.
 
std::string cosmos::fs::read_symlink_at (const DirFD dir_fd, const SysString path)
 Returns the target (content) of the symbolic link path relative to dir_fd.
 
void cosmos::fs::link (const SysString old_path, const SysString new_path)
 Creates a new (hard) link of the file found in old_path at new_path.
 
void cosmos::fs::linkat (const DirFD old_dir, const SysString old_path, const DirFD new_dir, const SysString new_path, const FollowSymlinks follow_old=FollowSymlinks{false})
 Creates a new (hard) link based on lookups relative to old_dir and new_dir.
 
void cosmos::fs::linkat_fd (const FileDescriptor fd, const DirFD new_dir, const SysString new_path)
 Special variant of linkat() that can link arbitrary file descriptors at a new location.
 
void cosmos::fs::linkat_proc_fd (const FileDescriptor fd, const DirFD new_dir, const SysString new_path)
 Performs the same as linkat_fd() using linkat() and the /proc file system.
 
void cosmos::fs::truncate (const FileDescriptor fd, off_t length)
 Changes the file size of the file referred to by fd to length bytes.
 
void cosmos::fs::truncate (const SysString path, off_t length)
 Changes the file size of the file found at the given path to length bytes.
 
size_t cosmos::fs::copy_file_range (CopyFileRangeParameters &pars)
 Copy data between files directly in the kernel.
 
size_t cosmos::fs::copy_file_range (const FileDescriptor fd_in, const FileDescriptor fd_out, const size_t len)
 Copy data between files directly in the kernel.
 
void cosmos::fs::check_access (const SysString path, const AccessChecks checks={})
 Check file access permissions of path.
 
void cosmos::fs::check_access_at (const DirFD dir_fd, const SysString path, const AccessChecks checks={}, const AccessFlags flags={})
 Check file access permissions of path relative to dir_fd.
 
COSMOS_API void cosmos::fs::check_access_fd (const FileDescriptor fd, const AccessChecks check={}, const AccessFlags flags={})
 Check file access permissions of the already open file descriptor fd.
 
COSMOS_API void cosmos::fs::flock (const FileDescriptor fd, const LockOperation operation, const LockFlags flags={})
 Apply or remove an advisory lock on the given file descriptor.
 

Detailed Description

File system related system calls. These is the lowest and most generic level of file system APIs that can be wrapped further in more comfortable types to use.

Definition in file filesystem.hxx.

Typedef Documentation

◆ AccessChecks

using cosmos::fs::AccessChecks = BitMask<AccessCheck>

Definition at line 687 of file filesystem.hxx.

◆ AccessFlags

using cosmos::fs::AccessFlags = BitMask<AccessFlag>

Definition at line 720 of file filesystem.hxx.

◆ CloseRangeFlags

using cosmos::fs::CloseRangeFlags = BitMask<CloseRangeFlag>

Flags used in cosmos::fs::close_range().

Definition at line 75 of file filesystem.hxx.

◆ LockFlags

using cosmos::fs::LockFlags = BitMask<LockFlag>

Additional flags influencing flock behaviour.

Definition at line 757 of file filesystem.hxx.

Enumeration Type Documentation

◆ AccessCheck

enum class cosmos::fs::AccessCheck : int
strong

Different access checks that can be performed in check_access().

Enumerator
READ_OK 

Read access is allowed.

WRITE_OK 

Write access is allowed.

EXEC_OK 

Execution is allowed.

Definition at line 681 of file filesystem.hxx.

681 : int {
682 READ_OK = R_OK,
683 WRITE_OK = W_OK,
684 EXEC_OK = X_OK
685};
@ EXEC_OK
Execution is allowed.
@ WRITE_OK
Write access is allowed.
@ READ_OK
Read access is allowed.

◆ AccessFlag

enum class cosmos::fs::AccessFlag : int
strong

Extra flags that influence the behaviour of check_access_at(), and check_access_fd().

Enumerator
EFFECTIVE_CREDS 

use the caller's effective UID and GID for the access check.

NO_FOLLOW 

don't resolve symlinks in path but check access to the link itself.

Definition at line 715 of file filesystem.hxx.

715 : int {
716 EFFECTIVE_CREDS = AT_EACCESS,
717 NO_FOLLOW = AT_SYMLINK_NOFOLLOW
718};
@ NO_FOLLOW
don't resolve symlinks in path but check access to the link itself.
@ EFFECTIVE_CREDS
use the caller's effective UID and GID for the access check.

◆ CloseRangeFlag

enum class cosmos::fs::CloseRangeFlag : unsigned int
strong
Enumerator
CLOEXEC 

Instead of closing, mark all matching file descriptors as CLOEXEC.

UNSHARE 

Unshare specified file descriptors before closing to avoid race conditions with other threads.

Definition at line 67 of file filesystem.hxx.

67 : unsigned int {
69 CLOEXEC = CLOSE_RANGE_CLOEXEC,
71 UNSHARE = CLOSE_RANGE_UNSHARE
72};
@ CLOEXEC
Instead of closing, mark all matching file descriptors as CLOEXEC.
@ UNSHARE
Unshare specified file descriptors before closing to avoid race conditions with other threads.

◆ LockFlag

enum class cosmos::fs::LockFlag : int
strong
Enumerator
LOCK_NONBLOCK 

don't block if a lock cannot be placed, throw ApiError with Errno::WOULD_BLOCK instead.

Definition at line 752 of file filesystem.hxx.

752 : int {
753 LOCK_NONBLOCK = LOCK_NB
754};
@ LOCK_NONBLOCK
don't block if a lock cannot be placed, throw ApiError with Errno::WOULD_BLOCK instead.

◆ LockOperation

enum class cosmos::fs::LockOperation : int
strong

Flags used with flock().

Enumerator
LOCK_SHARED 

place a shared lock of which multiple may exist at the same time (for reading)

LOCK_EXCLUSIVE 

place an exclusive lock of which only one may exist at the same time (for writing)

UNLOCK 

remove an existing lock (regardless of shared or exclusive)

Definition at line 746 of file filesystem.hxx.

746 : int {
747 LOCK_SHARED = LOCK_SH,
748 LOCK_EXCLUSIVE = LOCK_EX,
749 UNLOCK = LOCK_UN
750};
@ UNLOCK
remove an existing lock (regardless of shared or exclusive)
@ LOCK_SHARED
place a shared lock of which multiple may exist at the same time (for reading)
@ LOCK_EXCLUSIVE
place an exclusive lock of which only one may exist at the same time (for writing)

Function Documentation

◆ change_dir()

COSMOS_API void cosmos::fs::change_dir ( const SysString path)

Change the calling process's current working directory to path.

On error a FileError exception is thrown.

Definition at line 198 of file filesystem.cxx.

198 {
199 if (::chdir(path.raw()) != 0) {
200 cosmos_throw (FileError(path, "chdir()"));
201 }
202}

◆ change_group() [1/4]

void cosmos::fs::change_group ( const FileDescriptor fd,
const GroupID id )
inline

Convenience wrapper of change_owner() to change only the group of a file.

Definition at line 445 of file filesystem.hxx.

445 {
446 change_owner(fd, UserID::INVALID, id);
447}

◆ change_group() [2/4]

void cosmos::fs::change_group ( const FileDescriptor fd,
const SysString group )
inline

Convenience wrapper of change_owner() to change only the group of a file.

Definition at line 450 of file filesystem.hxx.

450 {
451 change_owner(fd, {}, group);
452}

◆ change_group() [3/4]

void cosmos::fs::change_group ( const SysString path,
const GroupID id )
inline

Convenience wrapper of change_owner() to change only the group of a file.

Definition at line 455 of file filesystem.hxx.

455 {
456 change_owner(path, UserID::INVALID, id);
457}

◆ change_group() [4/4]

void cosmos::fs::change_group ( const SysString path,
const SysString group )
inline

Convenience wrapper of change_owner() to change only the group of a file.

Definition at line 460 of file filesystem.hxx.

460 {
461 change_owner(path, {}, group);
462}

◆ change_mode() [1/2]

COSMOS_API void cosmos::fs::change_mode ( const FileDescriptor fd,
const FileMode mode )

Changes the FileMode of the given open file descriptor.

This behaves the same as change_mode(const SysString, FileMode) with the exception that the file associated with the given file descriptor will be affected instead of a to-be-opened path.

Additional common errors that can occur:

  • bad file descriptor (Errno::BAD_FD)

Definition at line 397 of file filesystem.cxx.

397 {
398 if (::fchmod(to_integral(fd.raw()), to_integral(mode.raw())) != 0) {
399 cosmos_throw (FileError("(fd)", "fchmod()"));
400 }
401}

◆ change_mode() [2/2]

COSMOS_API void cosmos::fs::change_mode ( const SysString path,
const FileMode mode )

Changes the FileMode of the given path.

Attempts to change the FileMode associated with the file object found at the given path. Symlinks will be followed.

To change the mode of symlinks use change_owner_nofollow(). To avoid symlinks use File::open() with OpenFlag::NOFOLLOW. Obtain a FileStatus() for the open FileDescriptor and then use change_mode(FileDescriptor, FileMode) if the file isn't a symlink.

Note
On Linux there is no way to change the mode symlinks.

Various errors can occur which will cause a FileError to be thrown. The most common errors are:

  • file does not exist (Errno::NO_ENTRY)
  • no permission to change the mode (Errno::PERMISSION)
  • read only file system (Errno::READ_ONLY_FS)

Definition at line 391 of file filesystem.cxx.

391 {
392 if (::chmod(path.raw(), to_integral(mode.raw())) != 0) {
393 cosmos_throw (FileError(path, "chmod()"));
394 }
395}

◆ change_owner() [1/4]

COSMOS_API void cosmos::fs::change_owner ( const FileDescriptor fd,
const SysString user,
const SysString group = {} )

Change owner and/or group of the given file descriptor by user name and/or group name.

This is a convenience function on top of change_owner(FileDescriptor, UserID, GroupID). The description of change_owner(const SysString, const SysString, const SysString) applies here as well.

Definition at line 452 of file filesystem.cxx.

452 {
453 const UserID uid = resolve_user(user);
454 const GroupID gid = resolve_group(group);
455 change_owner(fd, uid, gid);
456}

◆ change_owner() [2/4]

COSMOS_API void cosmos::fs::change_owner ( const FileDescriptor fd,
const UserID uid,
const GroupID gid = GroupID::INVALID )

Change numerical owner and/or group ID of the given open file descriptor.

This behaves the same as change_owner(const SysString, UserID, GroupID) with the exception that the file associated with the given file descriptor will be affected instead of a to-be-opened path.

Additional common errors that can occur:

  • bad file descriptor (Errno::BAD_FD)

Definition at line 409 of file filesystem.cxx.

409 {
410 if (::fchown(to_integral(fd.raw()), to_integral(uid), to_integral(gid)) != 0) {
411 cosmos_throw (FileError("(fd)", "fchown()"));
412 }
413}

◆ change_owner() [3/4]

COSMOS_API void cosmos::fs::change_owner ( const SysString path,
const SysString user,
const SysString group = {} )

Change owner and/or group of the given path by user name and/or group name.

This is a convenience function on top of change_owner(const SysString, UserID, GroupID). It looks up the numerical UserID of user and the numerical GroupID of group.

To skip changing the owner of user or group simply pass an empty SysString as respective parameter.

Additional errors that can occur are the ones described in PasswdInfo(), GroupInfo() and RuntimeError() in case the username or group do not exist.

Definition at line 445 of file filesystem.cxx.

445 {
446
447 const UserID uid = resolve_user(user);
448 const GroupID gid = resolve_group(group);
449 change_owner(path, uid, gid);
450}

◆ change_owner() [4/4]

COSMOS_API void cosmos::fs::change_owner ( const SysString path,
const UserID uid,
const GroupID gid = GroupID::INVALID )

Change numerical owner and/or group ID of a file path.

Attempts to change the owner and/or the group of the file object found at path. If path refers to a symbolic link then the target of the link is affected. UserID::INVALID and GroupID::INVALID will cause the respective item not to be touched.

Various errors can occur which will cause a FileError to be thrown. The most common errors are:

  • file does not exist (Errno::NO_ENTRY)
  • no permission to change the owner (Errno::PERMISSION)
  • read only file system (Errno::READ_ONLY_FS)

Definition at line 403 of file filesystem.cxx.

403 {
404 if (::chown(path.raw(), to_integral(uid), to_integral(gid)) != 0) {
405 cosmos_throw (FileError(path, "chown()"));
406 }
407}

◆ change_owner_nofollow() [1/2]

COSMOS_API void cosmos::fs::change_owner_nofollow ( const SysString path,
const SysString user,
const SysString group = {} )

Changes owner and/or group of the given path while not following symlinks.

See also
change_owner_nofollow(const SysString, const UserID, const GroupID)

Definition at line 464 of file filesystem.cxx.

464 {
465 const UserID uid = resolve_user(user);
466 const GroupID gid = resolve_group(group);
467 change_owner_nofollow(path, uid, gid);
468}

◆ change_owner_nofollow() [2/2]

COSMOS_API void cosmos::fs::change_owner_nofollow ( const SysString path,
const UserID uid,
const GroupID gid = GroupID::INVALID )

Changes owner and/or group of the given path while not following symlinks.

This behaves the same as change_owner(const SysString, const UserID, const GroupID) with the exception that if the final path component refers to a symbolic link that the ownership of the link is changed, not that of the target.

If path is not a symlink then it's owner will still be changed and no error is thrown.

Definition at line 458 of file filesystem.cxx.

458 {
459 if (::lchown(path.raw(), to_integral(uid), to_integral(gid)) != 0) {
460 cosmos_throw (FileError(path, "lchown()"));
461 }
462}

◆ check_access()

COSMOS_API void cosmos::fs::check_access ( const SysString path,
const AccessChecks checks = {} )

Check file access permissions of path.

This checks whether path can be accessed by the calling process, depending on the provided checks bitmask. If checks has no bits set, then this only checks for file existence. Otherwise it is checked whether path is accessible for read, write and/or execution, depending on the bits set in checks.

For determining access permissions the caller's real UID and GID are used.

If the check succeeds, then call returns normally. Otherwise a FileError is thrown, containing the specific error reason. Typical errors will be one of the following:

  • Errno::ACCESS: either the requested checks would be denied to the file, or search permission is denied for one of the components of the prefix of path.
  • Errno::NO_ENTRY: a component of path does not exist or is a dangling symbolic link.

Note that this call is often better replaced by direct open of a path, to prevent any time-of-check/time-of-use race conditions.

Definition at line 627 of file filesystem.cxx.

627 {
628 if (::access(path.raw(), checks.raw()) == 0) {
629 return;
630 }
631
632 cosmos_throw (ApiError("access()"));
633}

◆ check_access_at()

COSMOS_API void cosmos::fs::check_access_at ( const DirFD dir_fd,
const SysString path,
const AccessChecks checks = {},
const AccessFlags flags = {} )

Check file access permissions of path relative to dir_fd.

This behaves similar to check_access() with the following differences:

  • if path is a relative path, then it is interpreted relative to dir_fd, instead of relative to the caller's CWD.
  • if path is absolute then dir_fd is ignored.
  • if dir_fd is cosmos::AT_CWD, then path is again interpreted relative to the caller's CWD, like check_access() does.

The flags argument further influences the behaviour of the check.

Definition at line 635 of file filesystem.cxx.

636 {
637 if (::faccessat(to_integral(dir_fd.raw()), path.raw(), checks.raw(), flags.raw()) == 0) {
638 return;
639 }
640
641 cosmos_throw (ApiError("faccessat()"));
642}

◆ check_access_fd()

COSMOS_API void cosmos::fs::check_access_fd ( const FileDescriptor fd,
const AccessChecks check = {},
const AccessFlags flags = {} )

Check file access permissions of the already open file descriptor fd.

This call may be used on a file descriptor opened with OpenFlag::PATH, which is likely also the main purpose of this variant of check_access().

Definition at line 644 of file filesystem.cxx.

645 {
646
647 if (::faccessat(to_integral(fd.raw()), "", checks.raw(), flags.raw() | AT_EMPTY_PATH) == 0) {
648 return;
649 }
650
651 cosmos_throw (ApiError("faccessat()"));
652}

◆ close_range()

COSMOS_API void cosmos::fs::close_range ( const FileNum first,
const FileNum last = FileNum::MAX_FD,
const CloseRangeFlags flags = CloseRangeFlags{} )

Close a range of file descriptor numbers in the current process.

This call closes all file descriptor numbers in the range [first, last], where last is included in the range.

This is mostly useful before executing a child process to get rid of any file descriptors not marked with the close-on-exec file descriptor flag. The call is a lot more efficient then a loop in userspace calling close() on every single file descriptor.

You can specify FileNum::MAX_FD for last if you want to close all file descriptors starting from a given range. This is also the default for last.

This system call can fail on out of memory conditions or if the maximum number of file descriptors is exceeded in combination with CloseRangeFlag::UNSHARE.

Definition at line 74 of file filesystem.cxx.

74 {
75 // NOTE: close_range() uses unsigned int for file descriptor numbers,
76 // inconsistent with all other system calls. Using FileNum::MAX_FD
77 // (int) as the maximum file descriptor should still work I guess.
78 if (::close_range(to_integral(first), to_integral(last), flags.raw()) != 0) {
79 cosmos_throw (ApiError("close_range()"));
80 }
81}

◆ copy_file_range() [1/2]

COSMOS_API size_t cosmos::fs::copy_file_range ( const FileDescriptor fd_in,
const FileDescriptor fd_out,
const size_t len )

Copy data between files directly in the kernel.

This is a simplified version of copy_file_range(CopyFileRangeParameters&) that does not use explicit offsets. Instead the current file descriptor offsets will be used and updated.

Definition at line 606 of file filesystem.cxx.

608 {
609 return copy_file_range(fd_in, nullptr, fd_out, nullptr, len);
610}

◆ copy_file_range() [2/2]

COSMOS_API size_t cosmos::fs::copy_file_range ( CopyFileRangeParameters & pars)

Copy data between files directly in the kernel.

This operation allows for an efficient copy operation between two file descriptors, without routing data through userspace. Additionally, if both file descriptors belong to the same file system, this system call allows to optimize the copy on lower levels e.g. by employing copy-on-write techniques.

The output file descriptor must not be opened with OpenFlag::APPEND, otherwise an error is thrown.

As usual partial I/O can occur. The call attempts to copy pars.len bytes and the number of actually copied bytes is returned from this call. If the input file descriptor reached the end of file then 0 is returned. The remaining number of bytes is updated in pars.len. This way the same parameter structure can be used to continue the operation until it is complete.

If an input offset is supplied then reading from the input file descriptor starts at this position and the file descriptor's offset it not altered. The provided offset is updated by the number of bytes copied. If no input offset is supplied then reading starts from the file descriptor's offset and the offset is altered accordingly. The same holds true for the output offset.

There are two other system calls on Linux that perform similar tasks, but have restrictions:

  • splice(): this is limited to one of the FDs being a pipe
  • sendfile(): this used to be limited to writing to sockets FDs. Also it only supports an input FD offset, no output FD offset.

Definition at line 612 of file filesystem.cxx.

612 {
613 auto copied = copy_file_range(
614 pars.in, pars.off_in ? &pars.off_in.value() : nullptr,
615 pars.out, pars.off_out ? &pars.off_out.value() : nullptr,
616 pars.len);
617
618 pars.len -= copied;
619
620 return copied;
621}

◆ exists_file()

COSMOS_API bool cosmos::fs::exists_file ( const SysString path)

Returns whether the given file system object exists.

The information returned is only a snapshot in time. It is subject to race conditions i.e. when trying to open the file afterwards it may already be gone or have been replaced by a different file system object.

For safely testing for existence and opening a file, you should open the file right away and test for an according error condition.

This function will not follow symlinks i.e. if path refers to a dangling symlink then it will still return true.

If the condition cannot be exactly determined, because an error different than "ENOENT" is returned by the operating system then an exception is thrown.

This function does not determine the file type i.e. it could be also a directory, socket etc.

Definition at line 176 of file filesystem.cxx.

176 {
177 struct stat s;
178 if (::lstat(path.raw(), &s) == 0)
179 return true;
180 else if (get_errno() != Errno::NO_ENTRY)
181 cosmos_throw (FileError(path, "lstat()"));
182
183 return false;
184}
Errno get_errno()
Wrapper that returns the Errno strongly typed representation of the current errno
Definition errno.hxx:111

◆ flock()

COSMOS_API void cosmos::fs::flock ( const FileDescriptor fd,
const LockOperation operation,
const LockFlags flags = {} )

Apply or remove an advisory lock on the given file descriptor.

This type of locking is advisory only i.e. the participating processes need to cooperate with each other. A process with sufficient privileges can still modify the file without owning a lock.

The lock is associated with the open file description, which means the lock is inherited to child processes via fork() and to duplicated file descriptors. Only once all file descriptors referring to the lock have been closed, will the lock automatically be released. An UNLOCK operation on any file descriptor referring to the open file description will release the lock for all the other file descriptors as well.

A process may only hold one type of lock on a file. If the file is already locked, then a subsequent lock can be used to convert the lock type into a different one (i.e. shared to exclusive and vice versa).

Locks from this call are preserved across execve() (if a file descriptor referring to the lock is passed through execve()).

The OpenMode of the file descriptor is not relevant for locking the file using this API.

This API has restrictions and side effects on network file systems like CIFS and NFS:

  • on older kernels the locking happened only for the local machine
  • on newer kernels the lock can be transparently implemented via the fcntl() byte-range locking, which has different semantics (e.g. for placing write locks, the file descriptor has to be open for writing).

Definition at line 654 of file filesystem.cxx.

654 {
655 if (::flock(to_integral(fd.raw()), cosmos::to_integral(operation) | flags.raw()) != 0) {
656 cosmos_throw (ApiError("flock()"));
657 }
658}

◆ get_working_dir()

COSMOS_API std::string cosmos::fs::get_working_dir ( )

Returns the process's current working directory.

This call can fail e.g. on out of memory conditions or if the CWD has been unlinked. An ApiError is thrown in such cases.

Definition at line 204 of file filesystem.cxx.

204 {
205 std::string ret;
206 ret.resize(128);
207
208 while(true) {
209 if (auto res = ::getcwd(ret.data(), ret.size()); res == nullptr) {
210 switch (get_errno()) {
211 case Errno::RANGE: {
212 // double the size and retry
213 ret.resize(ret.size() * 2);
214 continue;
215 }
216 default: cosmos_throw (ApiError("getcwd()"));
217 }
218 }
219
220 // can be npos if the CWD is of exactly the size we have
221 if (auto termpos = ret.find('\0'); termpos != ret.npos) {
222 ret.resize(termpos);
223 }
224
225 return ret;
226 }
227}

◆ link()

COSMOS_API void cosmos::fs::link ( const SysString old_path,
const SysString new_path )

Creates a new (hard) link of the file found in old_path at new_path.

Hard links only work on the same file system. An attempt to create a hard link across different mounts will result in an ApiError with Errno::CROSS_DEVICE.

If new_path already exists then it will not be overwritten but Errno::EXISTS will be thrown.

Hard links don't work for directories, if old_path refers to one then Errno::PERMISSION will be thrown.

Furthermore a range of errors related to lack of permissions, lack of memory, lack of disk space, problems in path resolution and so on can occur.

On success both names will refer to the same file and it cannot be determined any more which was was the "original".

Definition at line 528 of file filesystem.cxx.

528 {
529 if (::link(old_path.raw(), new_path.raw()) != 0) {
530 cosmos_throw (FileError(new_path, std::string{"link() for "} + std::string{old_path}));
531 }
532}

◆ linkat()

COSMOS_API void cosmos::fs::linkat ( const DirFD old_dir,
const SysString old_path,
const DirFD new_dir,
const SysString new_path,
const FollowSymlinks follow_old = FollowSymlinks{false} )

Creates a new (hard) link based on lookups relative to old_dir and new_dir.

This behaves similar to link(). For old_path and new_path the usual at() API rules apply:

  • if the path is absolute then related DirFD is ignored.
  • else if the DirFD has the special value cosmos::AT_CWD then path is interpreted relative to the current working directory.
  • else the path is interpreted relative to the directory represented by DirFD.

follow_old determines whether symlink's encountered at old_path will be resolved or not.

Definition at line 534 of file filesystem.cxx.

536 {
537 if (::linkat(
538 to_integral(old_dir.raw()), old_path.raw(),
539 to_integral(new_dir.raw()), new_path.raw(),
540 follow_old ? AT_SYMLINK_FOLLOW : 0) != 0) {
541 cosmos_throw (FileError(new_path, std::string{"linkat() for "} + std::string{old_path}));
542 }
543}

◆ linkat_fd()

COSMOS_API void cosmos::fs::linkat_fd ( const FileDescriptor fd,
const DirFD new_dir,
const SysString new_path )

Special variant of linkat() that can link arbitrary file descriptors at a new location.

Contrary to linkat() this call can give the specified file descriptor a new name at new_dir / new_path, without specifying a source name. This generally only works for files that have a non-zero link count. This does not work for directory file descriptors.

As a special case this does work for file descriptors opened with OpenFlag::PATH and for temporary files opened with OpenFlag::TMPFILE but without OpenFlag::EXCLUSIVE (in this case a link count of 0 is accepted).

Note, however, that this call requires the CAP_DAC_READ_SEARCH capability, for security reasons. Starting with kernel version 6.10 the requirements are relaxed. If fd has been opened by the same process using the same credentials as the caller has, then the linkat will succeed. The credentials must not have been changed (and changed back) in the meantime.

An alternative is to use regular linkat() and the /proc file system, see man 2 linkat.

Definition at line 545 of file filesystem.cxx.

545 {
546 if (::linkat(
547 to_integral(fd.raw()), "",
548 to_integral(new_dir.raw()), new_path.raw(),
549 AT_EMPTY_PATH) != 0) {
550
551 cosmos_throw (FileError(new_path, std::string{"linkat(AT_EMPTY_PATH)"}));
552 }
553}

◆ linkat_proc_fd()

COSMOS_API void cosmos::fs::linkat_proc_fd ( const FileDescriptor fd,
const DirFD new_dir,
const SysString new_path )

Performs the same as linkat_fd() using linkat() and the /proc file system.

To avoid the permission issues that linkat_fd() has this variant of the linkat call uses a workaround based on the /proc file system to achieve the same result, see man 2 linkat.

Definition at line 555 of file filesystem.cxx.

555 {
556 // the exact security reasons why linkat_fd() isn't allowed without
557 // CAP_DAC_READ_SEARCH are a bit unclear. It seems the concern is that
558 // a process get's hold of a file descriptor for which it wouldn't
559 // have permissions to change ownership etc.
560 //
561 // By linking the FD into a directory controlled by the unprivileged
562 // process it would become possible to manipulate the ownership after
563 // all.
564 //
565 // It looks like this variant of linkat() does some checks that
566 // prevent this.
567 linkat(
568 AT_CWD, cosmos::sprintf("/proc/self/fd/%d", to_integral(fd.raw())),
569 new_dir, new_path, FollowSymlinks{true});
570}

◆ make_all_dirs()

COSMOS_API Errno cosmos::fs::make_all_dirs ( const SysString path,
const FileMode mode )

Creates a directory, potentially creating multiple directory components.

This is similar to make_dir() but also creates possibly missing parent directory elements in path. All created directory components will receive the given mode.

If any of the path components cannot be created (or accessed) then an FileError is thrown. This could mean that some of the paths have been created in the error case, but not the full path.

Returns
Errno::NO_ERROR if the full path (and thus at least the final directory component) was created, Errno::EXISTS if the directory was already existing.

Definition at line 306 of file filesystem.cxx.

306 {
307 const auto normpath = normalize_path(path);
308 size_t sep_pos = 0;
309 std::string prefix;
310 Errno ret{Errno::EXISTS};
311
312 if (path.empty()) {
313 cosmos_throw (UsageError("empty string passed in"));
314 }
315
316 while (sep_pos != normpath.npos) {
317 sep_pos = normpath.find('/', sep_pos + 1);
318 prefix = normpath.substr(0, sep_pos);
319
320 if (prefix.back() == '/') {
321 // root directory "/" or a trailing or duplicate slash
322 continue;
323 } else if (prefix == ".") {
324 // leading "." component, no sense in trying to create it
325 continue;
326 }
327
328 if (::mkdir(prefix.data(), to_integral(mode.raw())) != 0) {
329 if (get_errno() == Errno::EXISTS) {
330 continue;
331 }
332
333 cosmos_throw (FileError(prefix, "mkdir()"));
334 }
335
336 // at least one directory was created
337 ret = Errno::NO_ERROR;
338 }
339
340 return ret;
341}
Errno
Strong enum type representing errno error constants.
Definition errno.hxx:29

◆ make_dir()

COSMOS_API void cosmos::fs::make_dir ( const SysString path,
const FileMode mode )

Creates a directory at the given location.

This attempts to create a single new directory at the given path. The parent directory must already exist, otherwise a FileError is thrown.

The given mode determines the permissions of the newly created directory. The permissions are modified by the process's umask (mode = mode * & ~umask)

Definition at line 282 of file filesystem.cxx.

282 {
283 if (::mkdir(path.raw(), to_integral(mode.raw())) != 0) {
284 cosmos_throw (FileError(path, "mkdir()"));
285 }
286}

◆ make_dir_at()

COSMOS_API void cosmos::fs::make_dir_at ( const DirFD dir_fd,
const SysString path,
const FileMode mode )

Creates a directory at the location relative to dir_fd.

This relates to make_dir() the same way unlink_file_at() relates to unlink_file().

See also
unlink_file_at().

Definition at line 288 of file filesystem.cxx.

288 {
289 if (::mkdirat(to_integral(dir_fd.raw()), path.raw(), to_integral(mode.raw())) != 0) {
290 cosmos_throw (FileError(path, "mkdirat()"));
291 }
292}

◆ make_fifo()

COSMOS_API void cosmos::fs::make_fifo ( const SysString path,
const FileMode mode )

Creates a named pipe at the given file system location.

A named pipe is identical to a cosmos::Pipe, only that is has a visible name in the file system. Unrelated processes can exchange data via the pipe this way.

The file system entry carries the usual UNIX permissions that will be initialized by combining mode with the calling process's umask.

Opening named pipes will block both for reading and writing as long as no communication partner exists. When opening a named pipe in non-blocking mode will succeed for OpenMode::READ_ONLY even if no writer is present yet. For OpenMode::WRITE_ONLY an error Errno::NXIO is returned in the non-blocking case. As a special case on Linux when opening the pipe with OpenMode::READ_WRITE will succeed regardless of blocking or non-blocking mode. This allows to open the pipe for writing while there are no readers available.

The underlying system call for this is mknod() which will never follow symlinks or reuse existing files. Instead an error is thrown if the path already exists in any form.

Definition at line 147 of file filesystem.cxx.

147 {
148 auto res = ::mkfifo(path.raw(), to_integral(mode.raw()));
149
150 if (res != 0) {
151 cosmos_throw (FileError(path, "mkfifo()"));
152 }
153}

◆ make_fifo_at()

COSMOS_API void cosmos::fs::make_fifo_at ( const DirFD dir_fd,
const SysString path,
const FileMode mode )

Creates a named pipe relative to the given directory descriptor.

This behaves like make_fifo with the usual _at semantics:

  • if path is absolute then dir_fd is ignored.
  • if path is relative then it is interpreted relative to dir_fd.
  • if path is relative and dir_fd has the special value cosmos::AT_CWD then it is interpreted relative to the current working directory.

Definition at line 155 of file filesystem.cxx.

156 {
157 auto res = ::mkfifoat(to_integral(dir_fd.raw()), path.raw(), to_integral(mode.raw()));
158
159 if (res != 0) {
160 cosmos_throw (FileError(path, "mkfifoat()"));
161 }
162}

◆ make_symlink()

COSMOS_API void cosmos::fs::make_symlink ( const SysString target,
const SysString path )

Creates a symbolic link at path pointing to target.

A symbolic link is simply a pointer to another file system location as provided in target. Leading ".." components in target will refer to the parent directories of path.

The permission bits (FileMode) of a symlink are ignored by Linux. The user and group ownership are ignored when dereferencing a symlink, except for when the protected_symlinks feature is enabled (see proc(5); the feature is enabled on most Linux systems these days). The ownership is also relevant for renaming or removing a symlink.

If an error occurs then a FileError is thrown. Common errors are:

  • access denied to path. Either search access in the initial components or write access to the final directory component (Errno::ACCESS).
  • the file system does not allow the creation of symbolic links (Errno::PERMISSION).
  • Earlier parts of path do not exist or target is empty (Errno::NO_ENTRY).
  • path already exists (Errno::EXISTS).

Definition at line 470 of file filesystem.cxx.

470 {
471 if (::symlink(target.raw(), path.raw()) != 0) {
472 cosmos_throw (FileError(path, "symlink()"));
473 }
474}

◆ make_symlink_at()

COSMOS_API void cosmos::fs::make_symlink_at ( const SysString target,
const DirFD dir_fd,
const SysString path )

Creates a symbolic link relative to dir_fd pointing to target.

This behaves just like make_symlink(). It relates to make_symlink() the same way unlink_file_at() relates to unlink_file().

Definition at line 476 of file filesystem.cxx.

477 {
478 if (::symlinkat(target.raw(), to_integral(dir_fd.raw()), path.raw()) != 0) {
479 cosmos_throw (FileError(path, "symlinkat()"));
480 }
481}

◆ make_tempdir()

COSMOS_API std::string cosmos::fs::make_tempdir ( const SysString _template)

Safely create a temporary directory and return it's path.

This is similar to make_tempfile(). _template is the relative or absolute path to use as a template for the to be created directory. The basename of the path will be expanded with a random string. An empty basename is not allowed.

The created directory will receive a mode of 0700.

The returned string contains the expanded _template. It is the caller's responsibility to remove the directory from the file system again at the appropriate time.

See also
TempDir for a convenience type that transparently handles removal of the temporary directory.

Definition at line 134 of file filesystem.cxx.

134 {
135 // there's no way to have the X's in the middle of the basename like
136 // with mkostemps().
137 std::string expanded{_template};
138 expanded += "XXXXXX";
139
140 if (::mkdtemp(expanded.data()) == nullptr) {
141 cosmos_throw (ApiError("mkdtemp()"));
142 }
143
144 return expanded;
145}

◆ make_tempfile()

COSMOS_API std::pair< FileDescriptor, std::string > cosmos::fs::make_tempfile ( const SysString _template,
const OpenFlags flags = OpenFlags{OpenFlag::CLOEXEC} )

Safely create a temporary file and return it's file descriptor and path.

_template needs to be a template for the path to use for the temporary file. It can be an absolute or a relative path. The path determines the directory where the temporary file will be created in. The template also needs to contain a basename on which the actual file path will be based. You can place a pair of "{}" in the basename to mark the position in the path where a unique random still will be inserted. The last occurrence of "{}" will be used for this. If no such substring is found in the basename then the unique random still will be added as a suffix to the basename.

It is an error to use a zero-length basename which will cause an exception to be thrown.

The flags argument can specify optional additional open flags to be applied when opening the temporary file. The implementation will implicitly use OpenMode::READ_WRITE and OpenFlag::CREATE and OpenFlag::EXCLUSIVE. These should not be set in flags. The file will have the permissions ModeT{0600}.

On success this call returns a pair consisting of the newly opened file descriptor corresponding to the temporary file, and the expanded filename under which the temporary file has been created.

It is the caller's responsibility to correctly close the returned FileDescriptor and to delete the temporary file (using unlink_file()) once it is no longer needed. The cosmos::TempFile class takes care of these tasks, if necessary.

Definition at line 120 of file filesystem.cxx.

121 {
122
123 auto [path, suffixlen] = expand_temp_path(_template);
124
125 const auto fd = ::mkostemps(path.data(), suffixlen, flags.raw());
126
127 if (fd == -1) {
128 cosmos_throw (ApiError("mkostemps()"));
129 }
130
131 return {FileDescriptor{FileNum{fd}}, path};
132}
FileNum
Primitive file descriptor.
Definition types.hxx:32

◆ open()

COSMOS_API FileDescriptor cosmos::fs::open ( const SysString path,
const OpenMode mode,
const OpenFlags flags,
const std::optional< FileMode > fmode = {} )

Open a file using specific OpenFlags, potentially creating it first using the given fmode.

Warning
If used for creating a file, then you need to specify also the FileMode in that case. An exception will the thrown if this condition is violated.

The returned FileDescriptor object does not manage the lifetime of the file descriptor, you have to take care of closing it at the appropriate time yourself!

Definition at line 34 of file filesystem.cxx.

36 {
37
38 if (flags.anyOf({OpenFlag::CREATE, OpenFlag::TMPFILE}) && !fmode) {
39 cosmos_throw (UsageError("the given open flags require an fmode argument"));
40 }
41
42 int raw_flags = flags.raw() | to_integral(mode);
43
44
45 auto fd = ::open(path.raw(), raw_flags, fmode ? to_integral(fmode.value().raw()) : 0);
46
47 if (fd == -1) {
48 cosmos_throw (FileError(path, "open"));
49 }
50
51 return FileDescriptor{FileNum{fd}};
52}

◆ open_at()

COSMOS_API FileDescriptor cosmos::fs::open_at ( const DirFD dir_fd,
const SysString path,
const OpenMode mode,
const OpenFlags flags,
const std::optional< FileMode > fmode = {} )

Open the given path relative to the given directory file descriptor dir_fd.

This open variant behaves similar to open(const SysString,, const OpenMode, const OpenFlags, const std::optional<FileMode>). The following differences exist:

  • if path is an absolute path then dir_fd is ignored and the behaviour is identical to the other open variants.
  • if path is a relative path and dir_fd is an invalid file descriptor then the open fails (this can explicitly be used to enforce absolute path specifications.
  • if path is a relative path and dir_fd is a valid directory file descriptor, then the path is looked up relative to dir_fd. dir_fd needs to be opened with OpenMode::READ_ONLY or with OpenFlag::PATH. The special DirFD value cosmos::AT_CWD can be used to open files relative to the current working directory.

Definition at line 54 of file filesystem.cxx.

57 {
58 int raw_flags = flags.raw() | to_integral(mode);
59
60 if (flags.anyOf({OpenFlag::CREATE, OpenFlag::TMPFILE}) && !fmode) {
61 cosmos_throw (UsageError("the given open flags require an fmode argument"));
62 }
63
64 auto fd = ::openat(to_integral(dir_fd.raw()), path.raw(),
65 raw_flags, fmode ? to_integral(fmode.value().raw()) : 0);
66
67 if (fd == -1) {
68 cosmos_throw (FileError(path, "openat"));
69 }
70
71 return FileDescriptor{FileNum{fd}};
72}

◆ read_symlink()

COSMOS_API std::string cosmos::fs::read_symlink ( const SysString path)

Returns the target (content) of the symbolic link at path.

This returns the target path of the symlink present at the given path.

If an error occurs then a FileError is thrown. Common errors are:

  • access denied (Errno::ACCESS)
  • invalid argument if path is not a symlink (Errno::INVALID_ARG)
  • no entry if path does not exist (Errno::NO_ENTRY)

Definition at line 515 of file filesystem.cxx.

515 {
516 return read_symlink(path, "readlink()", ::readlink);
517}

◆ read_symlink_at()

COSMOS_API std::string cosmos::fs::read_symlink_at ( const DirFD dir_fd,
const SysString path )

Returns the target (content) of the symbolic link path relative to dir_fd.

This relates to read_symlink() the same way unlink_file_at() relates to unlink_file().

See also
unlink_file_at().

Definition at line 519 of file filesystem.cxx.

519 {
520
521 auto readlink_func = [&](const char *p, char *buf, size_t size) {
522 return ::readlinkat(to_integral(dir_fd.raw()), p, buf, size);
523 };
524
525 return read_symlink(path, "readlinkat()", readlink_func);
526}

◆ remove_dir()

COSMOS_API void cosmos::fs::remove_dir ( const SysString path)

Removes an empty directory at the given location.

The directory must exist and must be empty for the call to succeed. On error a FileError is thrown.

Definition at line 294 of file filesystem.cxx.

294 {
295 if (::rmdir(path.raw()) != 0) {
296 cosmos_throw (FileError(path, "rmdir()"));
297 }
298}

◆ remove_dir_at()

COSMOS_API void cosmos::fs::remove_dir_at ( const DirFD dir_fd,
const SysString path )

Removes an empty directory relative to dir_fd.

This relates to remove_dir() the same way unlink_file_at() relates to unlink_file().

See also
unlink_file_at().

Definition at line 300 of file filesystem.cxx.

300 {
301 if (::unlinkat(to_integral(dir_fd.raw()), path.raw(), AT_REMOVEDIR) != 0) {
302 cosmos_throw (FileError(path, "unlinkat(AT_REMOVEDIR)"));
303 }
304}

◆ remove_tree()

COSMOS_API void cosmos::fs::remove_tree ( const SysString path)

Recursively removes all directory content in path.

This function recursively removes all content in path. It is expected that as least path itself exists and is a directory, otherwise an exception is thrown.

If an error occurs while removing any of the descendant path elements then a FileError is thrown and the state of the directory tree is undefined. Note that using this function with concurrently file system access from within the calling or another process in the system can cause race conditions that leads to undefined behaviour.

Definition at line 385 of file filesystem.cxx.

385 {
386 DirStream dir{path};
387 remove_tree(dir);
388 remove_dir(path);
389}

◆ set_umask()

COSMOS_API FileMode cosmos::fs::set_umask ( const FileMode mode)

Sets the process's file creation mask.

The file creation mask is a process wide attribute that determines an upper limit of the file permission bits that are set on newly created files and directories. Most prominently this affects files created via open() and directories created via mkdir(). Even if more open permissions are specified in these system calls, the bits will be switched off if they are also present in the process's umask. The file permission bits are calculated as (<mode> & ~umask) i.e. bits that are set in the umask will be set to zero during file creation.

Since this is a process wide attribute it will affect all threads in the process and can thus cause race conditions. If necessary, you should set the umask in the main thread of a program early on.

Only the lower 9 bits of mode will be taken into account (i.e. owner/group/other permissions bits). If any other bits are set then an UsageError exception is thrown.

The umask is inherited across fork() and is not changed during execve().

Other system calls that make use of the umask are the creation of POSIX IPC objects (message queues, semaphores, shared memory), named pipes, and UNIX domain sockets. It is not used by SYSV IPC objects.

Returns
The umask that was previously in effect. To only read the current process's umask you need to read the proc file system in /proc/<pid>/status (umask is available there since Linux 4.7).

Definition at line 164 of file filesystem.cxx.

164 {
165 auto raw_mode = to_integral(mode.raw());
166
167 if ((raw_mode & ~0777) != 0) {
168 cosmos_throw (UsageError("invalid bits set in umask"));
169 }
170
171 auto old_mode = ::umask(raw_mode);
172
173 return FileMode{ModeT{old_mode}};
174}
ModeT
Combined file type and mode bits of a file (as found in st_mode struct stat).
Definition types.hxx:106

◆ truncate() [1/2]

COSMOS_API void cosmos::fs::truncate ( const FileDescriptor fd,
off_t length )

Changes the file size of the file referred to by fd to length bytes.

If length is smaller than the current file size, then the extra data stored in the file is lost. If length is larger than the current file size, then the extra data reads as zero bytes.

This operation requires the file to be opened for writing. A successful operation causes the inode of the file to be updated.

Definition at line 572 of file filesystem.cxx.

572 {
573 if (::ftruncate(to_integral(fd.raw()), length) != 0) {
574 cosmos_throw (ApiError("ftruncate()"));
575 }
576}

◆ truncate() [2/2]

COSMOS_API void cosmos::fs::truncate ( const SysString path,
off_t length )

Changes the file size of the file found at the given path to length bytes.

Behaves just like truncate(const FileDescriptor, off_t), with the difference that the operation is performed on the given path. The file must be writable.

Definition at line 578 of file filesystem.cxx.

578 {
579 if (::truncate(path.raw(), length) != 0) {
580 cosmos_throw (ApiError("truncate()"));
581 }
582}

◆ unlink_file()

COSMOS_API void cosmos::fs::unlink_file ( const SysString path)

Removes the file object found at path.

This function removes the file object found at path from the file system. Processes that already have opened the file can continue using it, but the name is removed from the file system.

This call does not work with directories, use remove_dir() for them instead.

If the path does not exist then this is considered an error and an exception is thrown.

Definition at line 186 of file filesystem.cxx.

186 {
187 if (::unlink(path.raw()) != 0) {
188 cosmos_throw (FileError(path, "unlink()"));
189 }
190}

◆ unlink_file_at()

COSMOS_API void cosmos::fs::unlink_file_at ( const DirFD dir_fd,
const SysString path )

Removes the file object found at path relative to dir_fd.

This behaves similar to unlink_file():

  • if path is an absolute path then dir_fd is ignored and the call is equivalent to unlink_file().
  • if path is a relative path and dir_fd has the special value cosmos::AT_CWD, then path is looked up relative to the current working directory, equivalent to unlink_file().
  • if path is a relative path and dir_fd is a valid open directory file descriptor then path is looked up relative to that directory.

Definition at line 192 of file filesystem.cxx.

192 {
193 if (::unlinkat(to_integral(dir_fd.raw()), path.raw(), 0) != 0) {
194 cosmos_throw (FileError(path, "unlinkat()"));
195 }
196}

◆ which()

COSMOS_API std::optional< std::string > cosmos::fs::which ( const std::string_view exec_base)
noexcept

Find the full path to the executable program exec_base.

This function looks in all directories listed in the PATH environment variable for an executable named exec_base. If one is found then the full path to this executable is returned.

If exec_base is an absolute path then it is only checked whether the path (symlinks are followed) is accessible and executable and returns it unmodified if this is the case.

If the program cannot be found or is not accessible then {} is returned. This function does not throw exceptions, on error {} is returned.

Definition at line 229 of file filesystem.cxx.

229 {
230
231 auto checkExecutable = [](const std::string &path) -> bool {
232 try {
233 File f{path, OpenMode::READ_ONLY};
234 FileStatus status;
235
236 try {
237 status.updateFrom(f.fd());
238 } catch (const CosmosError &) {
239 return false;
240 }
241
242 if (!status.type().isRegular())
243 return false;
244 else if (!status.mode().canAnyExec())
245 return false;
246
247 return true;
248 } catch (...) {
249 // probably a permission, I/O error, or NO_ENTRY error
250 return false;
251 }
252 };
253
254 if (exec_base.empty())
255 return {};
256
257 if (exec_base.front() == '/') {
258 // check absolute path and be done with it
259 if (checkExecutable(std::string{exec_base})) {
260 return {std::string{exec_base}};
261 }
262 return {};
263 }
264
265 const auto pathvar = proc::get_env_var("PATH");
266 if (!pathvar)
267 return {};
268
269 const auto paths = split(*pathvar, ":");
270
271 for (const auto &dir: paths) {
272 auto path = dir + "/" + std::string{exec_base};
273
274 if (checkExecutable(path)) {
275 return {path};
276 }
277 }
278
279 return {};
280}