// 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 "net/base/file_stream_context.h" #include "base/location.h" #include "base/message_loop/message_loop_proxy.h" #include "base/task_runner_util.h" #include "base/threading/thread_restrictions.h" #include "net/base/file_stream_net_log_parameters.h" #include "net/base/net_errors.h" namespace { void CallInt64ToInt(const net::CompletionCallback& callback, int64 result) { callback.Run(static_cast(result)); } } namespace net { FileStream::Context::IOResult::IOResult() : result(OK), os_error(0) { } FileStream::Context::IOResult::IOResult(int64 result, int os_error) : result(result), os_error(os_error) { } // static FileStream::Context::IOResult FileStream::Context::IOResult::FromOSError( int64 os_error) { return IOResult(MapSystemError(os_error), os_error); } FileStream::Context::OpenResult::OpenResult() : file(base::kInvalidPlatformFileValue) { } FileStream::Context::OpenResult::OpenResult(base::PlatformFile file, IOResult error_code) : file(file), error_code(error_code) { } void FileStream::Context::Orphan() { DCHECK(!orphaned_); orphaned_ = true; if (file_ != base::kInvalidPlatformFileValue) bound_net_log_.EndEvent(NetLog::TYPE_FILE_STREAM_OPEN); if (!async_in_progress_) { CloseAndDelete(); } else if (file_ != base::kInvalidPlatformFileValue) { CancelIo(file_); } } void FileStream::Context::OpenAsync(const base::FilePath& path, int open_flags, const CompletionCallback& callback) { DCHECK(!async_in_progress_); BeginOpenEvent(path); const bool posted = base::PostTaskAndReplyWithResult( task_runner_.get(), FROM_HERE, base::Bind( &Context::OpenFileImpl, base::Unretained(this), path, open_flags), base::Bind(&Context::OnOpenCompleted, base::Unretained(this), callback)); DCHECK(posted); async_in_progress_ = true; } int FileStream::Context::OpenSync(const base::FilePath& path, int open_flags) { DCHECK(!async_in_progress_); BeginOpenEvent(path); OpenResult result = OpenFileImpl(path, open_flags); file_ = result.file; if (file_ == base::kInvalidPlatformFileValue) { ProcessOpenError(result.error_code); } else { // TODO(satorux): Remove this once all async clients are migrated to use // Open(). crbug.com/114783 if (open_flags & base::PLATFORM_FILE_ASYNC) OnAsyncFileOpened(); } return result.error_code.result; } void FileStream::Context::CloseSync() { DCHECK(!async_in_progress_); if (file_ != base::kInvalidPlatformFileValue) { base::ClosePlatformFile(file_); file_ = base::kInvalidPlatformFileValue; bound_net_log_.EndEvent(NetLog::TYPE_FILE_STREAM_OPEN); } } void FileStream::Context::CloseAsync(const CompletionCallback& callback) { DCHECK(!async_in_progress_); const bool posted = base::PostTaskAndReplyWithResult( task_runner_.get(), FROM_HERE, base::Bind(&Context::CloseFileImpl, base::Unretained(this)), base::Bind(&Context::ProcessAsyncResult, base::Unretained(this), IntToInt64(callback), FILE_ERROR_SOURCE_CLOSE)); DCHECK(posted); async_in_progress_ = true; } void FileStream::Context::SeekAsync(Whence whence, int64 offset, const Int64CompletionCallback& callback) { DCHECK(!async_in_progress_); const bool posted = base::PostTaskAndReplyWithResult( task_runner_.get(), FROM_HERE, base::Bind( &Context::SeekFileImpl, base::Unretained(this), whence, offset), base::Bind(&Context::ProcessAsyncResult, base::Unretained(this), callback, FILE_ERROR_SOURCE_SEEK)); DCHECK(posted); async_in_progress_ = true; } int64 FileStream::Context::SeekSync(Whence whence, int64 offset) { IOResult result = SeekFileImpl(whence, offset); RecordError(result, FILE_ERROR_SOURCE_SEEK); return result.result; } void FileStream::Context::FlushAsync(const CompletionCallback& callback) { DCHECK(!async_in_progress_); const bool posted = base::PostTaskAndReplyWithResult( task_runner_.get(), FROM_HERE, base::Bind(&Context::FlushFileImpl, base::Unretained(this)), base::Bind(&Context::ProcessAsyncResult, base::Unretained(this), IntToInt64(callback), FILE_ERROR_SOURCE_FLUSH)); DCHECK(posted); async_in_progress_ = true; } int FileStream::Context::FlushSync() { IOResult result = FlushFileImpl(); RecordError(result, FILE_ERROR_SOURCE_FLUSH); return result.result; } void FileStream::Context::RecordError(const IOResult& result, FileErrorSource source) const { if (result.result >= 0) { // |result| is not an error. return; } // The following check is against incorrect use or bug. File descriptor // shouldn't ever be closed outside of FileStream while it still tries to do // something with it. DCHECK_NE(result.result, ERR_INVALID_HANDLE); if (!orphaned_) { bound_net_log_.AddEvent( NetLog::TYPE_FILE_STREAM_ERROR, base::Bind(&NetLogFileStreamErrorCallback, source, result.os_error, static_cast(result.result))); } RecordFileError(result.os_error, source, record_uma_); } void FileStream::Context::BeginOpenEvent(const base::FilePath& path) { std::string file_name = path.AsUTF8Unsafe(); bound_net_log_.BeginEvent(NetLog::TYPE_FILE_STREAM_OPEN, NetLog::StringCallback("file_name", &file_name)); } FileStream::Context::OpenResult FileStream::Context::OpenFileImpl( const base::FilePath& path, int open_flags) { // FileStream::Context actually closes the file asynchronously, independently // from FileStream's destructor. It can cause problems for users wanting to // delete the file right after FileStream deletion. Thus we are always // adding SHARE_DELETE flag to accommodate such use case. open_flags |= base::PLATFORM_FILE_SHARE_DELETE; base::PlatformFile file = base::CreatePlatformFile(path, open_flags, NULL, NULL); if (file == base::kInvalidPlatformFileValue) return OpenResult(file, IOResult::FromOSError(GetLastErrno())); return OpenResult(file, IOResult(OK, 0)); } void FileStream::Context::ProcessOpenError(const IOResult& error_code) { bound_net_log_.EndEvent(NetLog::TYPE_FILE_STREAM_OPEN); RecordError(error_code, FILE_ERROR_SOURCE_OPEN); } void FileStream::Context::OnOpenCompleted(const CompletionCallback& callback, OpenResult open_result) { file_ = open_result.file; if (file_ == base::kInvalidPlatformFileValue) ProcessOpenError(open_result.error_code); else if (!orphaned_) OnAsyncFileOpened(); OnAsyncCompleted(IntToInt64(callback), open_result.error_code.result); } void FileStream::Context::CloseAndDelete() { DCHECK(!async_in_progress_); if (file_ == base::kInvalidPlatformFileValue) { delete this; } else { const bool posted = task_runner_->PostTaskAndReply( FROM_HERE, base::Bind(base::IgnoreResult(&base::ClosePlatformFile), file_), base::Bind(&Context::OnCloseCompleted, base::Unretained(this))); DCHECK(posted); file_ = base::kInvalidPlatformFileValue; } } void FileStream::Context::OnCloseCompleted() { delete this; } Int64CompletionCallback FileStream::Context::IntToInt64( const CompletionCallback& callback) { return base::Bind(&CallInt64ToInt, callback); } void FileStream::Context::ProcessAsyncResult( const Int64CompletionCallback& callback, FileErrorSource source, const IOResult& result) { RecordError(result, source); OnAsyncCompleted(callback, result.result); } void FileStream::Context::OnAsyncCompleted( const Int64CompletionCallback& callback, int64 result) { // Reset this before Run() as Run() may issue a new async operation. Also it // should be reset before CloseAsync() because it shouldn't run if any async // operation is in progress. async_in_progress_ = false; if (orphaned_) CloseAndDelete(); else callback.Run(result); } } // namespace net