summaryrefslogtreecommitdiff
path: root/chromium/net/base/file_stream_context.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/net/base/file_stream_context.cc')
-rw-r--r--chromium/net/base/file_stream_context.cc260
1 files changed, 260 insertions, 0 deletions
diff --git a/chromium/net/base/file_stream_context.cc b/chromium/net/base/file_stream_context.cc
new file mode 100644
index 00000000000..abc058a9ca8
--- /dev/null
+++ b/chromium/net/base/file_stream_context.cc
@@ -0,0 +1,260 @@
+// 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<int>(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::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<net::Error>(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
+