diff options
author | dumi@chromium.org <dumi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-08-20 22:42:50 +0000 |
---|---|---|
committer | dumi@chromium.org <dumi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-08-20 22:42:50 +0000 |
commit | b90c93fff90803a61d25a71a64c6bb06a73de358 (patch) | |
tree | a2fd7f283f041e6f770481f5b44569b233a6d9eb /base/file_util_proxy.cc | |
parent | 826cb81e81488453b7c125531f086ca42e563861 (diff) | |
download | chromium_src-b90c93fff90803a61d25a71a64c6bb06a73de358.zip chromium_src-b90c93fff90803a61d25a71a64c6bb06a73de358.tar.gz chromium_src-b90c93fff90803a61d25a71a64c6bb06a73de358.tar.bz2 |
Moving file_system_proxy to base/ and changing it to work with MessageLoopProxies instead of ChromeThreads.
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/3131026
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@56923 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'base/file_util_proxy.cc')
-rw-r--r-- | base/file_util_proxy.cc | 245 |
1 files changed, 245 insertions, 0 deletions
diff --git a/base/file_util_proxy.cc b/base/file_util_proxy.cc new file mode 100644 index 0000000..62d150e --- /dev/null +++ b/base/file_util_proxy.cc @@ -0,0 +1,245 @@ +// Copyright (c) 2010 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/file_util_proxy.h" + +#include "base/file_util.h" +#include "base/message_loop_proxy.h" + +namespace { + +class MessageLoopRelay + : public base::RefCountedThreadSafe<MessageLoopRelay> { + public: + MessageLoopRelay() + : origin_message_loop_proxy_( + base::MessageLoopProxy::CreateForCurrentThread()) { + } + + void Start(scoped_refptr<base::MessageLoopProxy> message_loop_proxy, + const tracked_objects::Location& from_here) { + message_loop_proxy->PostTask( + from_here, + NewRunnableMethod(this, &MessageLoopRelay::ProcessOnTargetThread)); + } + + protected: + friend class base::RefCountedThreadSafe<MessageLoopRelay>; + virtual ~MessageLoopRelay() {} + + // Called to perform work on the FILE thread. + virtual void RunWork() = 0; + + // Called to notify the callback on the origin thread. + virtual void RunCallback() = 0; + + private: + void ProcessOnTargetThread() { + RunWork(); + origin_message_loop_proxy_->PostTask( + FROM_HERE, + NewRunnableMethod(this, &MessageLoopRelay::RunCallback)); + } + + scoped_refptr<base::MessageLoopProxy> origin_message_loop_proxy_; +}; + +class RelayCreateOrOpen : public MessageLoopRelay { + public: + RelayCreateOrOpen( + scoped_refptr<base::MessageLoopProxy> message_loop_proxy, + const FilePath& file_path, + int file_flags, + base::FileUtilProxy::CreateOrOpenCallback* callback) + : message_loop_proxy_(message_loop_proxy), + file_path_(file_path), + file_flags_(file_flags), + callback_(callback), + file_handle_(base::kInvalidPlatformFileValue), + created_(false) { + DCHECK(callback); + } + + protected: + virtual ~RelayCreateOrOpen() { + if (file_handle_ != base::kInvalidPlatformFileValue) + base::FileUtilProxy::Close(message_loop_proxy_, file_handle_, NULL); + } + + virtual void RunWork() { + file_handle_ = base::CreatePlatformFile(file_path_, file_flags_, &created_); + } + + virtual void RunCallback() { + callback_->Run(base::PassPlatformFile(&file_handle_), created_); + delete callback_; + } + + private: + scoped_refptr<base::MessageLoopProxy> message_loop_proxy_; + FilePath file_path_; + int file_flags_; + base::FileUtilProxy::CreateOrOpenCallback* callback_; + base::PlatformFile file_handle_; + bool created_; +}; + +class RelayCreateTemporary : public MessageLoopRelay { + public: + RelayCreateTemporary( + scoped_refptr<base::MessageLoopProxy> message_loop_proxy, + base::FileUtilProxy::CreateTemporaryCallback* callback) + : message_loop_proxy_(message_loop_proxy), + callback_(callback), + file_handle_(base::kInvalidPlatformFileValue) { + DCHECK(callback); + } + + protected: + virtual ~RelayCreateTemporary() { + if (file_handle_ != base::kInvalidPlatformFileValue) + base::FileUtilProxy::Close(message_loop_proxy_, file_handle_, NULL); + } + + virtual void RunWork() { + // TODO(darin): file_util should have a variant of CreateTemporaryFile + // that returns a FilePath and a PlatformFile. + file_util::CreateTemporaryFile(&file_path_); + + // Use a fixed set of flags that are appropriate for writing to a temporary + // file from the IO thread using a net::FileStream. + int file_flags = + base::PLATFORM_FILE_CREATE_ALWAYS | + base::PLATFORM_FILE_WRITE | + base::PLATFORM_FILE_ASYNC | + base::PLATFORM_FILE_TEMPORARY; + file_handle_ = base::CreatePlatformFile(file_path_, file_flags, NULL); + } + + virtual void RunCallback() { + callback_->Run(base::PassPlatformFile(&file_handle_), file_path_); + delete callback_; + } + + private: + scoped_refptr<base::MessageLoopProxy> message_loop_proxy_; + base::FileUtilProxy::CreateTemporaryCallback* callback_; + base::PlatformFile file_handle_; + FilePath file_path_; +}; + +class RelayWithStatusCallback : public MessageLoopRelay { + public: + explicit RelayWithStatusCallback( + base::FileUtilProxy::StatusCallback* callback) + : callback_(callback), + succeeded_(false) { + // It is OK for callback to be NULL. + } + + protected: + virtual void RunCallback() { + // The caller may not have been interested in the result. + if (callback_) { + callback_->Run(succeeded_); + delete callback_; + } + } + + void SetStatus(bool succeeded) { succeeded_ = succeeded; } + + private: + base::FileUtilProxy::StatusCallback* callback_; + bool succeeded_; +}; + +class RelayClose : public RelayWithStatusCallback { + public: + RelayClose(base::PlatformFile file_handle, + base::FileUtilProxy::StatusCallback* callback) + : RelayWithStatusCallback(callback), + file_handle_(file_handle) { + } + + protected: + virtual void RunWork() { + SetStatus(base::ClosePlatformFile(file_handle_)); + } + + private: + base::PlatformFile file_handle_; +}; + +class RelayDelete : public RelayWithStatusCallback { + public: + RelayDelete(const FilePath& file_path, + bool recursive, + base::FileUtilProxy::StatusCallback* callback) + : RelayWithStatusCallback(callback), + file_path_(file_path), + recursive_(recursive) { + } + + protected: + virtual void RunWork() { + SetStatus(file_util::Delete(file_path_, recursive_)); + } + + private: + FilePath file_path_; + bool recursive_; +}; + +void Start(const tracked_objects::Location& from_here, + scoped_refptr<base::MessageLoopProxy> message_loop_proxy, + scoped_refptr<MessageLoopRelay> relay) { + relay->Start(message_loop_proxy, from_here); +} + +} // namespace + +namespace base { + +// static +void FileUtilProxy::CreateOrOpen( + scoped_refptr<MessageLoopProxy> message_loop_proxy, + const FilePath& file_path, int file_flags, + CreateOrOpenCallback* callback) { + Start(FROM_HERE, message_loop_proxy, new RelayCreateOrOpen( + message_loop_proxy, file_path, file_flags, callback)); +} + +// static +void FileUtilProxy::CreateTemporary( + scoped_refptr<MessageLoopProxy> message_loop_proxy, + CreateTemporaryCallback* callback) { + Start(FROM_HERE, message_loop_proxy, + new RelayCreateTemporary(message_loop_proxy, callback)); +} + +// static +void FileUtilProxy::Close(scoped_refptr<MessageLoopProxy> message_loop_proxy, + base::PlatformFile file_handle, + StatusCallback* callback) { + Start(FROM_HERE, message_loop_proxy, new RelayClose(file_handle, callback)); +} + +// static +void FileUtilProxy::Delete(scoped_refptr<MessageLoopProxy> message_loop_proxy, + const FilePath& file_path, + StatusCallback* callback) { + Start(FROM_HERE, message_loop_proxy, + new RelayDelete(file_path, false, callback)); +} + +// static +void FileUtilProxy::RecursiveDelete( + scoped_refptr<MessageLoopProxy> message_loop_proxy, + const FilePath& file_path, + StatusCallback* callback) { + Start(FROM_HERE, message_loop_proxy, + new RelayDelete(file_path, true, callback)); +} + +} // namespace base |