// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/files/file.h" #include <io.h> #include "base/files/file_path.h" #include "base/logging.h" #include "base/metrics/sparse_histogram.h" #include "base/threading/thread_restrictions.h" namespace base { void File::CreateBaseFileUnsafe(const FilePath& name, uint32 flags) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(!IsValid()); DWORD disposition = 0; if (flags & FLAG_OPEN) disposition = OPEN_EXISTING; if (flags & FLAG_CREATE) { DCHECK(!disposition); disposition = CREATE_NEW; } if (flags & FLAG_OPEN_ALWAYS) { DCHECK(!disposition); disposition = OPEN_ALWAYS; } if (flags & FLAG_CREATE_ALWAYS) { DCHECK(!disposition); disposition = CREATE_ALWAYS; } if (flags & FLAG_OPEN_TRUNCATED) { DCHECK(!disposition); DCHECK(flags & FLAG_WRITE); disposition = TRUNCATE_EXISTING; } if (!disposition) { NOTREACHED(); return; } DWORD access = 0; if (flags & FLAG_WRITE) access = GENERIC_WRITE; if (flags & FLAG_APPEND) { DCHECK(!access); access = FILE_APPEND_DATA; } if (flags & FLAG_READ) access |= GENERIC_READ; if (flags & FLAG_WRITE_ATTRIBUTES) access |= FILE_WRITE_ATTRIBUTES; if (flags & FLAG_EXECUTE) access |= GENERIC_EXECUTE; DWORD sharing = (flags & FLAG_EXCLUSIVE_READ) ? 0 : FILE_SHARE_READ; if (!(flags & FLAG_EXCLUSIVE_WRITE)) sharing |= FILE_SHARE_WRITE; if (flags & FLAG_SHARE_DELETE) sharing |= FILE_SHARE_DELETE; DWORD create_flags = 0; if (flags & FLAG_ASYNC) create_flags |= FILE_FLAG_OVERLAPPED; if (flags & FLAG_TEMPORARY) create_flags |= FILE_ATTRIBUTE_TEMPORARY; if (flags & FLAG_HIDDEN) create_flags |= FILE_ATTRIBUTE_HIDDEN; if (flags & FLAG_DELETE_ON_CLOSE) create_flags |= FILE_FLAG_DELETE_ON_CLOSE; if (flags & FLAG_BACKUP_SEMANTICS) create_flags |= FILE_FLAG_BACKUP_SEMANTICS; file_.Set(CreateFile(name.value().c_str(), access, sharing, NULL, disposition, create_flags, NULL)); if (file_.IsValid()) { error_ = FILE_OK; async_ = ((flags & FLAG_ASYNC) == FLAG_ASYNC); if (flags & (FLAG_OPEN_ALWAYS)) created_ = (ERROR_ALREADY_EXISTS != GetLastError()); else if (flags & (FLAG_CREATE_ALWAYS | FLAG_CREATE)) created_ = true; } else { error_ = OSErrorToFileError(GetLastError()); } } bool File::IsValid() const { return file_.IsValid(); } PlatformFile File::TakePlatformFile() { return file_.Take(); } void File::Close() { base::ThreadRestrictions::AssertIOAllowed(); file_.Close(); } int64 File::Seek(Whence whence, int64 offset) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); if (offset < 0) return -1; LARGE_INTEGER distance, res; distance.QuadPart = offset; DWORD move_method = static_cast<DWORD>(whence); if (!SetFilePointerEx(file_, distance, &res, move_method)) return -1; return res.QuadPart; } int File::Read(int64 offset, char* data, int size) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); DCHECK(!async_); if (size < 0) return -1; LARGE_INTEGER offset_li; offset_li.QuadPart = offset; OVERLAPPED overlapped = {0}; overlapped.Offset = offset_li.LowPart; overlapped.OffsetHigh = offset_li.HighPart; DWORD bytes_read; if (::ReadFile(file_, data, size, &bytes_read, &overlapped) != 0) return bytes_read; if (ERROR_HANDLE_EOF == GetLastError()) return 0; return -1; } int File::ReadAtCurrentPos(char* data, int size) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); DCHECK(!async_); if (size < 0) return -1; DWORD bytes_read; if (::ReadFile(file_, data, size, &bytes_read, NULL) != 0) return bytes_read; if (ERROR_HANDLE_EOF == GetLastError()) return 0; return -1; } int File::ReadNoBestEffort(int64 offset, char* data, int size) { return Read(offset, data, size); } int File::ReadAtCurrentPosNoBestEffort(char* data, int size) { return ReadAtCurrentPos(data, size); } int File::Write(int64 offset, const char* data, int size) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); DCHECK(!async_); LARGE_INTEGER offset_li; offset_li.QuadPart = offset; OVERLAPPED overlapped = {0}; overlapped.Offset = offset_li.LowPart; overlapped.OffsetHigh = offset_li.HighPart; DWORD bytes_written; if (::WriteFile(file_, data, size, &bytes_written, &overlapped) != 0) return bytes_written; return -1; } int File::WriteAtCurrentPos(const char* data, int size) { NOTREACHED(); return -1; } int File::WriteAtCurrentPosNoBestEffort(const char* data, int size) { return WriteAtCurrentPos(data, size); } bool File::Truncate(int64 length) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); // Get the current file pointer. LARGE_INTEGER file_pointer; LARGE_INTEGER zero; zero.QuadPart = 0; if (::SetFilePointerEx(file_, zero, &file_pointer, FILE_CURRENT) == 0) return false; LARGE_INTEGER length_li; length_li.QuadPart = length; // If length > file size, SetFilePointerEx() should extend the file // with zeroes on all Windows standard file systems (NTFS, FATxx). if (!::SetFilePointerEx(file_, length_li, NULL, FILE_BEGIN)) return false; // Set the new file length and move the file pointer to its old position. // This is consistent with ftruncate()'s behavior, even when the file // pointer points to a location beyond the end of the file. return ((::SetEndOfFile(file_) != 0) && (::SetFilePointerEx(file_, file_pointer, NULL, FILE_BEGIN) != 0)); } bool File::Flush() { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); return ::FlushFileBuffers(file_) != FALSE; } bool File::SetTimes(Time last_access_time, Time last_modified_time) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); FILETIME last_access_filetime = last_access_time.ToFileTime(); FILETIME last_modified_filetime = last_modified_time.ToFileTime(); return (::SetFileTime(file_, NULL, &last_access_filetime, &last_modified_filetime) != 0); } bool File::GetInfo(Info* info) { base::ThreadRestrictions::AssertIOAllowed(); DCHECK(IsValid()); BY_HANDLE_FILE_INFORMATION file_info; if (GetFileInformationByHandle(file_, &file_info) == 0) return false; LARGE_INTEGER size; size.HighPart = file_info.nFileSizeHigh; size.LowPart = file_info.nFileSizeLow; info->size = size.QuadPart; info->is_directory = (file_info.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0; info->is_symbolic_link = false; // Windows doesn't have symbolic links. info->last_modified = base::Time::FromFileTime(file_info.ftLastWriteTime); info->last_accessed = base::Time::FromFileTime(file_info.ftLastAccessTime); info->creation_time = base::Time::FromFileTime(file_info.ftCreationTime); return true; } File::Error base::File::Lock() { DCHECK(IsValid()); BOOL result = LockFile(file_, 0, 0, MAXDWORD, MAXDWORD); if (!result) return OSErrorToFileError(GetLastError()); return FILE_OK; } File::Error File::Unlock() { DCHECK(IsValid()); BOOL result = UnlockFile(file_, 0, 0, MAXDWORD, MAXDWORD); if (!result) return OSErrorToFileError(GetLastError()); return FILE_OK; } // Static. File::Error File::OSErrorToFileError(DWORD last_error) { switch (last_error) { case ERROR_SHARING_VIOLATION: return FILE_ERROR_IN_USE; case ERROR_FILE_EXISTS: return FILE_ERROR_EXISTS; case ERROR_FILE_NOT_FOUND: case ERROR_PATH_NOT_FOUND: return FILE_ERROR_NOT_FOUND; case ERROR_ACCESS_DENIED: return FILE_ERROR_ACCESS_DENIED; case ERROR_TOO_MANY_OPEN_FILES: return FILE_ERROR_TOO_MANY_OPENED; case ERROR_OUTOFMEMORY: case ERROR_NOT_ENOUGH_MEMORY: return FILE_ERROR_NO_MEMORY; case ERROR_HANDLE_DISK_FULL: case ERROR_DISK_FULL: case ERROR_DISK_RESOURCES_EXHAUSTED: return FILE_ERROR_NO_SPACE; case ERROR_USER_MAPPED_FILE: return FILE_ERROR_INVALID_OPERATION; case ERROR_NOT_READY: case ERROR_SECTOR_NOT_FOUND: case ERROR_DEV_NOT_EXIST: case ERROR_IO_DEVICE: case ERROR_FILE_CORRUPT: case ERROR_DISK_CORRUPT: return FILE_ERROR_IO; default: UMA_HISTOGRAM_SPARSE_SLOWLY("PlatformFile.UnknownErrors.Windows", last_error); return FILE_ERROR_FAILED; } } void File::SetPlatformFile(PlatformFile file) { file_.Set(file); } } // namespace base