fune/toolkit/components/osfile/modules/osfile_unix_back.jsm
Mike Hommey d739152bcb Bug 1659904 - Define UnixFile.open as a variadic function. r=Yoric
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
2020-08-22 21:51:04 +00:00

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);
}