forked from mirrors/gecko-dev
1051 lines
28 KiB
JavaScript
1051 lines
28 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.js 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.js 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
|
|
);
|
|
|
|
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,
|
|
"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);
|
|
}
|