forked from mirrors/gecko-dev
That is how it is actually defined in fcntl.h, and thus how C code calls it. It so happens that in many cases, it just works out fine because variadic arguments end up at the same place as if they weren't variadic in registers/stack, but that's not actually true on all platforms. Most notably, that's not true on Apple Silicon. Differential Revision: https://phabricator.services.mozilla.com/D87558
1069 lines
29 KiB
JavaScript
1069 lines
29 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/* eslint-env mozilla/chrome-worker, node */
|
|
/* global OS */
|
|
|
|
// eslint-disable-next-line no-lone-blocks
|
|
{
|
|
if (typeof Components != "undefined") {
|
|
// We do not wish osfile_unix_back.jsm to be used directly as a main thread
|
|
// module yet. When time comes, it will be loaded by a combination of
|
|
// a main thread front-end/worker thread implementation that makes sure
|
|
// that we are not executing synchronous IO code in the main thread.
|
|
|
|
throw new Error(
|
|
"osfile_unix_back.jsm cannot be used from the main thread yet"
|
|
);
|
|
}
|
|
(function(exports) {
|
|
"use strict";
|
|
if (exports.OS && exports.OS.Unix && exports.OS.Unix.File) {
|
|
return; // Avoid double initialization
|
|
}
|
|
|
|
let SharedAll = require("resource://gre/modules/osfile/osfile_shared_allthreads.jsm");
|
|
let SysAll = require("resource://gre/modules/osfile/osfile_unix_allthreads.jsm");
|
|
SharedAll.LOG.bind(SharedAll, "Unix", "back");
|
|
let libc = SysAll.libc;
|
|
let Const = SharedAll.Constants.libc;
|
|
|
|
/**
|
|
* Initialize the Unix module.
|
|
*
|
|
* @param {function=} declareFFI
|
|
*/
|
|
// FIXME: Both |init| and |aDeclareFFI| are deprecated, we should remove them
|
|
let init = function init(aDeclareFFI) {
|
|
if (aDeclareFFI) {
|
|
aDeclareFFI.bind(null, libc);
|
|
} else {
|
|
SysAll.declareFFI;
|
|
}
|
|
SharedAll.declareLazyFFI;
|
|
|
|
// Initialize types that require additional OS-specific
|
|
// support - either finalization or matching against
|
|
// OS-specific constants.
|
|
let Type = Object.create(SysAll.Type);
|
|
let SysFile = (exports.OS.Unix.File = { Type });
|
|
|
|
/**
|
|
* A file descriptor.
|
|
*/
|
|
Type.fd = Type.int.withName("fd");
|
|
Type.fd.importFromC = function importFromC(fd_int) {
|
|
return ctypes.CDataFinalizer(fd_int, SysFile._close);
|
|
};
|
|
|
|
/**
|
|
* A C integer holding -1 in case of error or a file descriptor
|
|
* in case of success.
|
|
*/
|
|
Type.negativeone_or_fd = Type.fd.withName("negativeone_or_fd");
|
|
Type.negativeone_or_fd.importFromC = function importFromC(fd_int) {
|
|
if (fd_int == -1) {
|
|
return -1;
|
|
}
|
|
return ctypes.CDataFinalizer(fd_int, SysFile._close);
|
|
};
|
|
|
|
/**
|
|
* A C integer holding -1 in case of error or a meaningless value
|
|
* in case of success.
|
|
*/
|
|
Type.negativeone_or_nothing = Type.int.withName("negativeone_or_nothing");
|
|
|
|
/**
|
|
* A C integer holding -1 in case of error or a positive integer
|
|
* in case of success.
|
|
*/
|
|
Type.negativeone_or_ssize_t = Type.ssize_t.withName(
|
|
"negativeone_or_ssize_t"
|
|
);
|
|
|
|
/**
|
|
* Various libc integer types
|
|
*/
|
|
Type.mode_t = Type.intn_t(Const.OSFILE_SIZEOF_MODE_T).withName("mode_t");
|
|
Type.uid_t = Type.intn_t(Const.OSFILE_SIZEOF_UID_T).withName("uid_t");
|
|
Type.gid_t = Type.intn_t(Const.OSFILE_SIZEOF_GID_T).withName("gid_t");
|
|
|
|
/**
|
|
* Type |time_t|
|
|
*/
|
|
Type.time_t = Type.intn_t(Const.OSFILE_SIZEOF_TIME_T).withName("time_t");
|
|
|
|
// Structure |dirent|
|
|
// Building this type is rather complicated, as its layout varies between
|
|
// variants of Unix. For this reason, we rely on a number of constants
|
|
// (computed in C from the C data structures) that give us the layout.
|
|
// The structure we compute looks like
|
|
// { int8_t[...] before_d_type; // ignored content
|
|
// int8_t d_type ;
|
|
// int8_t[...] before_d_name; // ignored content
|
|
// char[...] d_name;
|
|
// };
|
|
{
|
|
let d_name_extra_size = 0;
|
|
if (Const.OSFILE_SIZEOF_DIRENT_D_NAME < 8) {
|
|
// d_name is defined like "char d_name[1];" on some platforms
|
|
// (e.g. Solaris), we need to give it more size for our structure.
|
|
d_name_extra_size = 256;
|
|
}
|
|
|
|
let dirent = new SharedAll.HollowStructure(
|
|
"dirent",
|
|
Const.OSFILE_SIZEOF_DIRENT + d_name_extra_size
|
|
);
|
|
if (Const.OSFILE_OFFSETOF_DIRENT_D_TYPE != undefined) {
|
|
// |dirent| doesn't have d_type on some platforms (e.g. Solaris).
|
|
dirent.add_field_at(
|
|
Const.OSFILE_OFFSETOF_DIRENT_D_TYPE,
|
|
"d_type",
|
|
ctypes.uint8_t
|
|
);
|
|
}
|
|
dirent.add_field_at(
|
|
Const.OSFILE_OFFSETOF_DIRENT_D_NAME,
|
|
"d_name",
|
|
ctypes.ArrayType(
|
|
ctypes.char,
|
|
Const.OSFILE_SIZEOF_DIRENT_D_NAME + d_name_extra_size
|
|
)
|
|
);
|
|
|
|
// We now have built |dirent|.
|
|
Type.dirent = dirent.getType();
|
|
}
|
|
Type.null_or_dirent_ptr = new SharedAll.Type(
|
|
"null_of_dirent",
|
|
Type.dirent.out_ptr.implementation
|
|
);
|
|
|
|
// Structure |stat|
|
|
// Same technique
|
|
{
|
|
let stat = new SharedAll.HollowStructure(
|
|
"stat",
|
|
Const.OSFILE_SIZEOF_STAT
|
|
);
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_MODE,
|
|
"st_mode",
|
|
Type.mode_t.implementation
|
|
);
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_UID,
|
|
"st_uid",
|
|
Type.uid_t.implementation
|
|
);
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_GID,
|
|
"st_gid",
|
|
Type.gid_t.implementation
|
|
);
|
|
|
|
// Here, things get complicated with different data structures.
|
|
// Some platforms have |time_t st_atime| and some platforms have
|
|
// |timespec st_atimespec|. However, since |timespec| starts with
|
|
// a |time_t|, followed by nanoseconds, we just cheat and pretend
|
|
// that everybody has |time_t st_atime|, possibly followed by padding
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_ATIME,
|
|
"st_atime",
|
|
Type.time_t.implementation
|
|
);
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_MTIME,
|
|
"st_mtime",
|
|
Type.time_t.implementation
|
|
);
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_CTIME,
|
|
"st_ctime",
|
|
Type.time_t.implementation
|
|
);
|
|
|
|
// To complicate further, MacOS and some BSDs have a field |birthtime|
|
|
if ("OSFILE_OFFSETOF_STAT_ST_BIRTHTIME" in Const) {
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_BIRTHTIME,
|
|
"st_birthtime",
|
|
Type.time_t.implementation
|
|
);
|
|
}
|
|
|
|
stat.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STAT_ST_SIZE,
|
|
"st_size",
|
|
Type.off_t.implementation
|
|
);
|
|
Type.stat = stat.getType();
|
|
}
|
|
|
|
// Structure |DIR|
|
|
if ("OSFILE_SIZEOF_DIR" in Const) {
|
|
// On platforms for which we need to access the fields of DIR
|
|
// directly (e.g. because certain functions are implemented
|
|
// as macros), we need to define DIR as a hollow structure.
|
|
let DIR = new SharedAll.HollowStructure("DIR", Const.OSFILE_SIZEOF_DIR);
|
|
|
|
DIR.add_field_at(
|
|
Const.OSFILE_OFFSETOF_DIR_DD_FD,
|
|
"dd_fd",
|
|
Type.fd.implementation
|
|
);
|
|
|
|
Type.DIR = DIR.getType();
|
|
} else {
|
|
// On other platforms, we keep DIR as a blackbox
|
|
Type.DIR = new SharedAll.Type("DIR", ctypes.StructType("DIR"));
|
|
}
|
|
|
|
Type.null_or_DIR_ptr = Type.DIR.out_ptr.withName("null_or_DIR*");
|
|
Type.null_or_DIR_ptr.importFromC = function importFromC(dir) {
|
|
if (dir == null || dir.isNull()) {
|
|
return null;
|
|
}
|
|
return ctypes.CDataFinalizer(dir, SysFile._close_dir);
|
|
};
|
|
|
|
// Structure |timeval|
|
|
{
|
|
let timeval = new SharedAll.HollowStructure(
|
|
"timeval",
|
|
Const.OSFILE_SIZEOF_TIMEVAL
|
|
);
|
|
timeval.add_field_at(
|
|
Const.OSFILE_OFFSETOF_TIMEVAL_TV_SEC,
|
|
"tv_sec",
|
|
Type.long.implementation
|
|
);
|
|
timeval.add_field_at(
|
|
Const.OSFILE_OFFSETOF_TIMEVAL_TV_USEC,
|
|
"tv_usec",
|
|
Type.long.implementation
|
|
);
|
|
Type.timeval = timeval.getType();
|
|
Type.timevals = new SharedAll.Type(
|
|
"two timevals",
|
|
ctypes.ArrayType(Type.timeval.implementation, 2)
|
|
);
|
|
}
|
|
|
|
// Types fsblkcnt_t and fsfilcnt_t, used by structure |statvfs|
|
|
Type.fsblkcnt_t = Type.uintn_t(Const.OSFILE_SIZEOF_FSBLKCNT_T).withName(
|
|
"fsblkcnt_t"
|
|
);
|
|
// There is no guarantee of the size or order of members in sys-header structs
|
|
// It mostly is "unsigned long", but can be "unsigned int" as well.
|
|
// So it has its own "type".
|
|
// NOTE: This is still only partially correct, as signedness is also not guaranteed,
|
|
// so assuming an unsigned int might still be wrong here.
|
|
// But unsigned seems to have worked all those years, even though its signed
|
|
// on various platforms.
|
|
Type.statvfs_f_frsize = Type.uintn_t(
|
|
Const.OSFILE_SIZEOF_STATVFS_F_FRSIZE
|
|
).withName("statvfs_f_rsize");
|
|
|
|
// Structure |statvfs|
|
|
// Use an hollow structure
|
|
{
|
|
let statvfs = new SharedAll.HollowStructure(
|
|
"statvfs",
|
|
Const.OSFILE_SIZEOF_STATVFS
|
|
);
|
|
|
|
statvfs.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STATVFS_F_FRSIZE,
|
|
"f_frsize",
|
|
Type.statvfs_f_frsize.implementation
|
|
);
|
|
statvfs.add_field_at(
|
|
Const.OSFILE_OFFSETOF_STATVFS_F_BAVAIL,
|
|
"f_bavail",
|
|
Type.fsblkcnt_t.implementation
|
|
);
|
|
|
|
Type.statvfs = statvfs.getType();
|
|
}
|
|
|
|
// Declare libc functions as functions of |OS.Unix.File|
|
|
|
|
// Finalizer-related functions
|
|
libc.declareLazy(
|
|
SysFile,
|
|
"_close",
|
|
"close",
|
|
ctypes.default_abi,
|
|
/* return */ ctypes.int,
|
|
ctypes.int
|
|
);
|
|
|
|
SysFile.close = function close(fd) {
|
|
// Detach the finalizer and call |_close|.
|
|
return fd.dispose();
|
|
};
|
|
|
|
libc.declareLazy(
|
|
SysFile,
|
|
"_close_dir",
|
|
"closedir",
|
|
ctypes.default_abi,
|
|
/* return */ ctypes.int,
|
|
Type.DIR.in_ptr.implementation
|
|
);
|
|
|
|
SysFile.closedir = function closedir(fd) {
|
|
// Detach the finalizer and call |_close_dir|.
|
|
return fd.dispose();
|
|
};
|
|
|
|
{
|
|
// Symbol free() is special.
|
|
// We override the definition of free() on several platforms.
|
|
let default_lib = new SharedAll.Library("default_lib", "a.out");
|
|
|
|
// On platforms for which we override free(), nspr defines
|
|
// a special library name "a.out" that will resolve to the
|
|
// correct implementation free().
|
|
// If it turns out we don't have an a.out library or a.out
|
|
// doesn't contain free, use the ordinary libc free.
|
|
|
|
default_lib.declareLazyWithFallback(
|
|
libc,
|
|
SysFile,
|
|
"free",
|
|
"free",
|
|
ctypes.default_abi,
|
|
/* return*/ ctypes.void_t,
|
|
ctypes.voidptr_t
|
|
);
|
|
}
|
|
|
|
// Other functions
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"access",
|
|
"access",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.int
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"chdir",
|
|
"chdir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"chmod",
|
|
"chmod",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.mode_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"chown",
|
|
"chown",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.uid_t,
|
|
Type.gid_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"copyfile",
|
|
"copyfile",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
/* source*/ Type.path,
|
|
Type.path,
|
|
Type.void_t.in_ptr,
|
|
Type.uint32_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"dup",
|
|
"dup",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_fd,
|
|
Type.fd
|
|
);
|
|
|
|
if ("OSFILE_SIZEOF_DIR" in Const) {
|
|
// On platforms for which |dirfd| is a macro
|
|
SysFile.dirfd = function dirfd(DIRp) {
|
|
return Type.DIR.in_ptr.implementation(DIRp).contents.dd_fd;
|
|
};
|
|
} else {
|
|
// On platforms for which |dirfd| is a function
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"dirfd",
|
|
"dirfd",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_fd,
|
|
Type.DIR.in_ptr
|
|
);
|
|
}
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"chdir",
|
|
"chdir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fchdir",
|
|
"fchdir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.fd
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fchmod",
|
|
"fchmod",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.fd,
|
|
Type.mode_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fchown",
|
|
"fchown",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.fd,
|
|
Type.uid_t,
|
|
Type.gid_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fsync",
|
|
"fsync",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.fd
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"getcwd",
|
|
"getcwd",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.out_path,
|
|
Type.out_path,
|
|
Type.size_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"getwd",
|
|
"getwd",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.out_path,
|
|
Type.out_path
|
|
);
|
|
|
|
// Two variants of |getwd| which allocate the memory
|
|
// dynamically.
|
|
|
|
// Linux/Android version
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"get_current_dir_name",
|
|
"get_current_dir_name",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.out_path.releaseWithLazy(() => SysFile.free)
|
|
);
|
|
|
|
// MacOS/BSD version (will return NULL on Linux/Android)
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"getwd_auto",
|
|
"getwd",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.out_path.releaseWithLazy(() => SysFile.free),
|
|
Type.void_t.out_ptr
|
|
);
|
|
|
|
if (OS.Constants.Sys.Name == "Darwin") {
|
|
// At the time of writing we only need this on MacOS. If we generalize
|
|
// this, be sure to do so with the other xattr functions also.
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"getxattr",
|
|
"getxattr",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.int,
|
|
Type.path,
|
|
Type.cstring,
|
|
Type.void_t.out_ptr,
|
|
Type.size_t,
|
|
Type.uint32_t,
|
|
Type.int
|
|
);
|
|
}
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fdatasync",
|
|
"fdatasync",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.fd
|
|
); // Note: MacOS/BSD-specific
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"ftruncate",
|
|
"ftruncate",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.fd,
|
|
/* length*/ Type.off_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"lchown",
|
|
"lchown",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.uid_t,
|
|
Type.gid_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"link",
|
|
"link",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
/* source*/ Type.path,
|
|
Type.path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"lseek",
|
|
"lseek",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.off_t,
|
|
Type.fd,
|
|
/* offset*/ Type.off_t,
|
|
/* whence*/ Type.int
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"mkdir",
|
|
"mkdir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.int,
|
|
/* path*/ Type.path,
|
|
/* mode*/ Type.int
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"mkstemp",
|
|
"mkstemp",
|
|
ctypes.default_abi,
|
|
Type.fd,
|
|
/* template*/ Type.out_path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"open",
|
|
"open",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_fd,
|
|
Type.path,
|
|
/* oflags*/ Type.int,
|
|
"..."
|
|
);
|
|
|
|
if (OS.Constants.Sys.Name == "NetBSD") {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"opendir",
|
|
"__opendir30",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.null_or_DIR_ptr,
|
|
Type.path
|
|
);
|
|
} else {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"opendir",
|
|
"opendir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.null_or_DIR_ptr,
|
|
Type.path
|
|
);
|
|
}
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"pread",
|
|
"pread",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_ssize_t,
|
|
Type.fd,
|
|
Type.void_t.out_ptr,
|
|
/* nbytes*/ Type.size_t,
|
|
/* offset*/ Type.off_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"pwrite",
|
|
"pwrite",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_ssize_t,
|
|
Type.fd,
|
|
Type.void_t.in_ptr,
|
|
/* nbytes*/ Type.size_t,
|
|
/* offset*/ Type.off_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"read",
|
|
"read",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_ssize_t,
|
|
Type.fd,
|
|
Type.void_t.out_ptr,
|
|
/* nbytes*/ Type.size_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"posix_fadvise",
|
|
"posix_fadvise",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.int,
|
|
Type.fd,
|
|
/* offset*/ Type.off_t,
|
|
Type.off_t,
|
|
/* advise*/ Type.int
|
|
);
|
|
|
|
if (Const._DARWIN_INODE64_SYMBOLS) {
|
|
// Special case for MacOS X 10.5+
|
|
// Symbol name "readdir" still exists but is used for a
|
|
// deprecated function that does not match the
|
|
// constants of |Const|.
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"readdir",
|
|
"readdir$INODE64",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.null_or_dirent_ptr,
|
|
Type.DIR.in_ptr
|
|
); // For MacOS X
|
|
} else if (OS.Constants.Sys.Name == "NetBSD") {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"readdir",
|
|
"__readdir30",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.null_or_dirent_ptr,
|
|
Type.DIR.in_ptr
|
|
); // Other Unices
|
|
} else {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"readdir",
|
|
"readdir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.null_or_dirent_ptr,
|
|
Type.DIR.in_ptr
|
|
); // Other Unices
|
|
}
|
|
|
|
if (OS.Constants.Sys.Name == "Darwin") {
|
|
// At the time of writing we only need this on MacOS. If we generalize
|
|
// this, be sure to do so with the other xattr functions also.
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"removexattr",
|
|
"removexattr",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.cstring,
|
|
Type.int
|
|
);
|
|
}
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"rename",
|
|
"rename",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"rmdir",
|
|
"rmdir",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.int,
|
|
Type.path
|
|
);
|
|
|
|
if (OS.Constants.Sys.Name == "Darwin") {
|
|
// At the time of writing we only need this on MacOS. If we generalize
|
|
// this, be sure to do so with the other xattr functions also.
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"setxattr",
|
|
"setxattr",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.cstring,
|
|
Type.void_t.in_ptr,
|
|
Type.size_t,
|
|
Type.uint32_t,
|
|
Type.int
|
|
);
|
|
}
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"splice",
|
|
"splice",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.long,
|
|
Type.fd,
|
|
/* off_in*/ Type.off_t.in_ptr,
|
|
/* fd_out*/ Type.fd,
|
|
/* off_out*/ Type.off_t.in_ptr,
|
|
Type.size_t,
|
|
Type.unsigned_int
|
|
); // Linux/Android-specific
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"statfs",
|
|
"statfs",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.statvfs.out_ptr
|
|
); // Android,B2G
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"statvfs",
|
|
"statvfs",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
Type.statvfs.out_ptr
|
|
); // Other platforms
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"symlink",
|
|
"symlink",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
/* source*/ Type.path,
|
|
Type.path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"truncate",
|
|
"truncate",
|
|
ctypes.default_abi,
|
|
/* return*/ Type.negativeone_or_nothing,
|
|
Type.path,
|
|
/* length*/ Type.off_t
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"unlink",
|
|
"unlink",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path
|
|
);
|
|
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"write",
|
|
"write",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_ssize_t,
|
|
/* fd */ Type.fd,
|
|
/* buf */ Type.void_t.in_ptr,
|
|
/* nbytes */ Type.size_t
|
|
);
|
|
|
|
// Weird cases that require special treatment
|
|
|
|
// OSes use a variety of hacks to differentiate between
|
|
// 32-bits and 64-bits versions of |stat|, |lstat|, |fstat|.
|
|
if (Const._DARWIN_INODE64_SYMBOLS) {
|
|
// MacOS X 64-bits
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"stat",
|
|
"stat$INODE64",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"lstat",
|
|
"lstat$INODE64",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fstat",
|
|
"fstat$INODE64",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.fd,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
} else if (Const._STAT_VER != undefined) {
|
|
const ver = Const._STAT_VER;
|
|
let xstat_name, lxstat_name, fxstat_name;
|
|
if (OS.Constants.Sys.Name == "SunOS") {
|
|
// Solaris
|
|
xstat_name = "_xstat";
|
|
lxstat_name = "_lxstat";
|
|
fxstat_name = "_fxstat";
|
|
} else {
|
|
// Linux, all widths
|
|
xstat_name = "__xstat";
|
|
lxstat_name = "__lxstat";
|
|
fxstat_name = "__fxstat";
|
|
}
|
|
|
|
let Stat = {};
|
|
libc.declareLazyFFI(
|
|
Stat,
|
|
"xstat",
|
|
xstat_name,
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* _stat_ver */ Type.int,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
Stat,
|
|
"lxstat",
|
|
lxstat_name,
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* _stat_ver */ Type.int,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
Stat,
|
|
"fxstat",
|
|
fxstat_name,
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* _stat_ver */ Type.int,
|
|
/* fd */ Type.fd,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
|
|
SysFile.stat = function stat(path, buf) {
|
|
return Stat.xstat(ver, path, buf);
|
|
};
|
|
|
|
SysFile.lstat = function lstat(path, buf) {
|
|
return Stat.lxstat(ver, path, buf);
|
|
};
|
|
|
|
SysFile.fstat = function fstat(fd, buf) {
|
|
return Stat.fxstat(ver, fd, buf);
|
|
};
|
|
} else if (OS.Constants.Sys.Name == "NetBSD") {
|
|
// NetBSD 5.0 and newer
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"stat",
|
|
"__stat50",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"lstat",
|
|
"__lstat50",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fstat",
|
|
"__fstat50",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* fd */ Type.fd,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
} else {
|
|
// Mac OS X 32-bits, other Unix
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"stat",
|
|
"stat",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"lstat",
|
|
"lstat",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"fstat",
|
|
"fstat",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* fd */ Type.fd,
|
|
/* buf */ Type.stat.out_ptr
|
|
);
|
|
}
|
|
|
|
// We cannot make a C array of CDataFinalizer, so
|
|
// pipe cannot be directly defined as a C function.
|
|
|
|
let Pipe = {};
|
|
libc.declareLazyFFI(
|
|
Pipe,
|
|
"_pipe",
|
|
"pipe",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* fds */ new SharedAll.Type(
|
|
"two file descriptors",
|
|
ctypes.ArrayType(ctypes.int, 2)
|
|
)
|
|
);
|
|
|
|
// A shared per-thread buffer used to communicate with |pipe|
|
|
let _pipebuf = new (ctypes.ArrayType(ctypes.int, 2))();
|
|
|
|
SysFile.pipe = function pipe(array) {
|
|
let result = Pipe._pipe(_pipebuf);
|
|
if (result == -1) {
|
|
return result;
|
|
}
|
|
array[0] = ctypes.CDataFinalizer(_pipebuf[0], SysFile._close);
|
|
array[1] = ctypes.CDataFinalizer(_pipebuf[1], SysFile._close);
|
|
return result;
|
|
};
|
|
|
|
if (OS.Constants.Sys.Name == "NetBSD") {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"utimes",
|
|
"__utimes50",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* timeval[2] */ Type.timevals.out_ptr
|
|
);
|
|
} else {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"utimes",
|
|
"utimes",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* path */ Type.path,
|
|
/* timeval[2] */ Type.timevals.out_ptr
|
|
);
|
|
}
|
|
if (OS.Constants.Sys.Name == "NetBSD") {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"futimes",
|
|
"__futimes50",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* fd */ Type.fd,
|
|
/* timeval[2] */ Type.timevals.out_ptr
|
|
);
|
|
} else {
|
|
libc.declareLazyFFI(
|
|
SysFile,
|
|
"futimes",
|
|
"futimes",
|
|
ctypes.default_abi,
|
|
/* return */ Type.negativeone_or_nothing,
|
|
/* fd */ Type.fd,
|
|
/* timeval[2] */ Type.timevals.out_ptr
|
|
);
|
|
}
|
|
};
|
|
|
|
exports.OS.Unix = {
|
|
File: {
|
|
_init: init,
|
|
},
|
|
};
|
|
})(this);
|
|
}
|