2014-06-30 15:39:45 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
2012-05-21 11:12:37 +00:00
|
|
|
* 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/. */
|
2010-08-18 21:30:09 +00:00
|
|
|
|
|
|
|
#ifndef mozilla_FileUtils_h
|
|
|
|
#define mozilla_FileUtils_h
|
2010-11-09 02:48:59 +00:00
|
|
|
|
2012-07-30 14:20:58 +00:00
|
|
|
#include "nscore.h" // nullptr
|
2012-04-17 14:11:53 +00:00
|
|
|
|
2014-02-10 22:57:01 +00:00
|
|
|
#if defined(XP_UNIX)
|
2010-11-09 02:48:59 +00:00
|
|
|
# include <unistd.h>
|
|
|
|
#elif defined(XP_WIN)
|
|
|
|
# include <io.h>
|
|
|
|
#endif
|
|
|
|
#include "prio.h"
|
|
|
|
|
2012-04-12 10:21:24 +00:00
|
|
|
#include "mozilla/Scoped.h"
|
2013-03-04 01:56:03 +00:00
|
|
|
#include "nsIFile.h"
|
2013-01-04 02:46:47 +00:00
|
|
|
#include <errno.h>
|
2013-03-04 01:56:03 +00:00
|
|
|
#include <limits.h>
|
2012-04-12 10:21:24 +00:00
|
|
|
|
2010-08-18 21:30:09 +00:00
|
|
|
namespace mozilla {
|
|
|
|
|
2013-03-04 01:56:03 +00:00
|
|
|
#if defined(XP_WIN)
|
|
|
|
typedef void* filedesc_t;
|
|
|
|
typedef const wchar_t* pathstr_t;
|
|
|
|
#else
|
|
|
|
typedef int filedesc_t;
|
|
|
|
typedef const char* pathstr_t;
|
|
|
|
#endif
|
2010-08-18 21:30:09 +00:00
|
|
|
|
2010-11-09 02:48:59 +00:00
|
|
|
/**
|
2012-04-12 10:21:24 +00:00
|
|
|
* ScopedCloseFD is a RAII wrapper for POSIX file descriptors
|
|
|
|
*
|
|
|
|
* Instances |close()| their fds when they go out of scope.
|
2010-11-09 02:48:59 +00:00
|
|
|
*/
|
2012-04-12 10:21:24 +00:00
|
|
|
struct ScopedCloseFDTraits
|
2010-11-09 02:48:59 +00:00
|
|
|
{
|
2012-04-12 10:21:24 +00:00
|
|
|
typedef int type;
|
|
|
|
static type empty() { return -1; }
|
2014-06-27 01:35:39 +00:00
|
|
|
static void release(type aFd)
|
|
|
|
{
|
|
|
|
if (aFd != -1) {
|
|
|
|
while (close(aFd) == -1 && errno == EINTR) {
|
2013-01-04 02:46:47 +00:00
|
|
|
}
|
2010-11-09 02:48:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2012-04-12 10:21:24 +00:00
|
|
|
typedef Scoped<ScopedCloseFDTraits> ScopedClose;
|
2010-11-09 02:48:59 +00:00
|
|
|
|
2013-03-04 01:56:03 +00:00
|
|
|
#if !defined(XPCOM_GLUE)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* AutoFDClose is a RAII wrapper for PRFileDesc.
|
|
|
|
*
|
|
|
|
* Instances |PR_Close| their fds when they go out of scope.
|
|
|
|
**/
|
|
|
|
struct ScopedClosePRFDTraits
|
|
|
|
{
|
|
|
|
typedef PRFileDesc* type;
|
2013-10-10 20:41:39 +00:00
|
|
|
static type empty() { return nullptr; }
|
2014-06-27 01:35:39 +00:00
|
|
|
static void release(type aFd)
|
|
|
|
{
|
|
|
|
if (aFd) {
|
|
|
|
PR_Close(aFd);
|
2013-03-04 01:56:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
typedef Scoped<ScopedClosePRFDTraits> AutoFDClose;
|
|
|
|
|
2013-07-23 12:30:55 +00:00
|
|
|
/* RAII wrapper for FILE descriptors */
|
|
|
|
struct ScopedCloseFileTraits
|
|
|
|
{
|
2014-06-27 01:35:39 +00:00
|
|
|
typedef FILE* type;
|
2013-07-23 12:30:55 +00:00
|
|
|
static type empty() { return nullptr; }
|
2014-06-27 01:35:39 +00:00
|
|
|
static void release(type aFile)
|
|
|
|
{
|
|
|
|
if (aFile) {
|
|
|
|
fclose(aFile);
|
2013-07-23 12:30:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
typedef Scoped<ScopedCloseFileTraits> ScopedCloseFile;
|
|
|
|
|
2010-09-16 20:21:12 +00:00
|
|
|
/**
|
|
|
|
* Fallocate efficiently and continuously allocates files via fallocate-type APIs.
|
|
|
|
* This is useful for avoiding fragmentation.
|
|
|
|
* On sucess the file be padded with zeros to grow to aLength.
|
|
|
|
*
|
|
|
|
* @param aFD file descriptor.
|
|
|
|
* @param aLength length of file to grow to.
|
|
|
|
* @return true on success.
|
|
|
|
*/
|
2014-08-27 22:47:27 +00:00
|
|
|
bool fallocate(PRFileDesc* aFD, int64_t aLength);
|
2010-09-16 20:21:12 +00:00
|
|
|
|
2013-03-04 01:56:03 +00:00
|
|
|
/**
|
|
|
|
* Use readahead to preload shared libraries into the file cache before loading.
|
2014-06-27 01:35:39 +00:00
|
|
|
* WARNING: This function should not be used without a telemetry field trial
|
2013-03-04 01:56:03 +00:00
|
|
|
* demonstrating a clear performance improvement!
|
|
|
|
*
|
|
|
|
* @param aFile nsIFile representing path to shared library
|
|
|
|
*/
|
2014-08-27 22:47:27 +00:00
|
|
|
void ReadAheadLib(nsIFile* aFile);
|
2013-03-04 01:56:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Use readahead to preload a file into the file cache before reading.
|
2014-06-27 01:35:39 +00:00
|
|
|
* WARNING: This function should not be used without a telemetry field trial
|
2013-03-04 01:56:03 +00:00
|
|
|
* demonstrating a clear performance improvement!
|
|
|
|
*
|
|
|
|
* @param aFile nsIFile representing path to shared library
|
|
|
|
* @param aOffset Offset into the file to begin preloading
|
|
|
|
* @param aCount Number of bytes to preload (SIZE_MAX implies file size)
|
|
|
|
* @param aOutFd Pointer to file descriptor. If specified, ReadAheadFile will
|
|
|
|
* return its internal, opened file descriptor instead of closing it.
|
|
|
|
*/
|
2014-08-27 22:47:27 +00:00
|
|
|
void ReadAheadFile(nsIFile* aFile, const size_t aOffset = 0,
|
|
|
|
const size_t aCount = SIZE_MAX,
|
|
|
|
filedesc_t* aOutFd = nullptr);
|
2013-03-04 01:56:03 +00:00
|
|
|
|
|
|
|
#endif // !defined(XPCOM_GLUE)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use readahead to preload shared libraries into the file cache before loading.
|
2014-06-27 01:35:39 +00:00
|
|
|
* WARNING: This function should not be used without a telemetry field trial
|
2013-03-04 01:56:03 +00:00
|
|
|
* demonstrating a clear performance improvement!
|
|
|
|
*
|
|
|
|
* @param aFilePath path to shared library
|
|
|
|
*/
|
2014-08-27 22:47:27 +00:00
|
|
|
void ReadAheadLib(pathstr_t aFilePath);
|
2013-03-04 01:56:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Use readahead to preload a file into the file cache before loading.
|
2014-06-27 01:35:39 +00:00
|
|
|
* WARNING: This function should not be used without a telemetry field trial
|
2013-03-04 01:56:03 +00:00
|
|
|
* demonstrating a clear performance improvement!
|
|
|
|
*
|
|
|
|
* @param aFilePath path to shared library
|
|
|
|
* @param aOffset Offset into the file to begin preloading
|
|
|
|
* @param aCount Number of bytes to preload (SIZE_MAX implies file size)
|
|
|
|
* @param aOutFd Pointer to file descriptor. If specified, ReadAheadFile will
|
|
|
|
* return its internal, opened file descriptor instead of closing it.
|
|
|
|
*/
|
2014-08-27 22:47:27 +00:00
|
|
|
void ReadAheadFile(pathstr_t aFilePath, const size_t aOffset = 0,
|
|
|
|
const size_t aCount = SIZE_MAX,
|
|
|
|
filedesc_t* aOutFd = nullptr);
|
2013-03-04 01:56:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Use readahead to preload a file into the file cache before reading.
|
|
|
|
* When this function exits, the file pointer is guaranteed to be in the same
|
|
|
|
* position it was in before this function was called.
|
2014-06-27 01:35:39 +00:00
|
|
|
* WARNING: This function should not be used without a telemetry field trial
|
2013-03-04 01:56:03 +00:00
|
|
|
* demonstrating a clear performance improvement!
|
|
|
|
*
|
|
|
|
* @param aFd file descriptor opened for read access
|
|
|
|
* (on Windows, file must be opened with FILE_FLAG_SEQUENTIAL_SCAN)
|
|
|
|
* @param aOffset Offset into the file to begin preloading
|
|
|
|
* @param aCount Number of bytes to preload (SIZE_MAX implies file size)
|
|
|
|
*/
|
2014-08-27 22:47:27 +00:00
|
|
|
void ReadAhead(filedesc_t aFd, const size_t aOffset = 0,
|
|
|
|
const size_t aCount = SIZE_MAX);
|
2013-03-04 01:56:03 +00:00
|
|
|
|
|
|
|
|
2015-01-14 23:22:00 +00:00
|
|
|
#if defined(MOZ_WIDGET_GONK) || defined(XP_UNIX)
|
|
|
|
#define MOZ_TEMP_FAILURE_RETRY(exp) (__extension__({ \
|
|
|
|
typeof (exp) _rc; \
|
|
|
|
do { \
|
|
|
|
_rc = (exp); \
|
|
|
|
} while (_rc == -1 && errno == EINTR); \
|
|
|
|
_rc; \
|
|
|
|
}))
|
|
|
|
#endif
|
|
|
|
|
2013-04-23 17:55:46 +00:00
|
|
|
/* Define ReadSysFile() only on GONK to avoid unnecessary lubxul bloat.
|
|
|
|
Also define it in debug builds, so that unit tests for it can be written
|
|
|
|
and run in non-GONK builds. */
|
|
|
|
#if (defined(MOZ_WIDGET_GONK) || defined(DEBUG)) && defined(XP_UNIX)
|
|
|
|
|
|
|
|
#ifndef ReadSysFile_PRESENT
|
|
|
|
#define ReadSysFile_PRESENT
|
|
|
|
#endif /* ReadSysFile_PRESENT */
|
2013-03-27 15:19:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Read the contents of a file.
|
|
|
|
* This function is intended for reading a single-lined text files from
|
|
|
|
* /sys/. If the file ends with a newline ('\n') then it will be discarded.
|
|
|
|
* The output buffer will always be '\0'-terminated on successful completion.
|
|
|
|
* If aBufSize == 0, then this function will return true if the file exists
|
|
|
|
* and is readable (it will not attempt to read anything from it).
|
|
|
|
* On failure the contents of aBuf after this call will be undefined and the
|
|
|
|
* value of the global variable errno will be set accordingly.
|
|
|
|
* @return true on success, notice that less than requested bytes could have
|
|
|
|
* been read if the file was smaller
|
|
|
|
*/
|
2014-06-27 01:35:39 +00:00
|
|
|
bool ReadSysFile(const char* aFilename, char* aBuf, size_t aBufSize);
|
2013-03-27 15:19:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse the contents of a file, assuming it contains a decimal integer.
|
|
|
|
* @return true on success
|
|
|
|
*/
|
2014-06-27 01:35:39 +00:00
|
|
|
bool ReadSysFile(const char* aFilename, int* aVal);
|
2013-03-27 15:19:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse the contents of a file, assuming it contains a boolean value
|
|
|
|
* (either 0 or 1).
|
|
|
|
* @return true on success
|
|
|
|
*/
|
2014-06-27 01:35:39 +00:00
|
|
|
bool ReadSysFile(const char* aFilename, bool* aVal);
|
2013-03-27 15:19:09 +00:00
|
|
|
|
2013-04-23 17:55:46 +00:00
|
|
|
#endif /* (MOZ_WIDGET_GONK || DEBUG) && XP_UNIX */
|
2013-03-27 15:19:09 +00:00
|
|
|
|
2010-08-18 21:30:09 +00:00
|
|
|
} // namespace mozilla
|
|
|
|
#endif
|