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