// Copyright (c) 2013 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 "ppapi/proxy/file_system_resource.h" #include "base/bind.h" #include "ipc/ipc_message.h" #include "ppapi/c/pp_errors.h" #include "ppapi/proxy/ppapi_messages.h" #include "ppapi/shared_impl/tracked_callback.h" #include "ppapi/thunk/enter.h" #include "ppapi/thunk/ppb_file_io_api.h" using ppapi::thunk::EnterResourceNoLock; using ppapi::thunk::PPB_FileIO_API; using ppapi::thunk::PPB_FileSystem_API; namespace ppapi { namespace proxy { FileSystemResource::QuotaRequest::QuotaRequest( int64_t amount_arg, const RequestQuotaCallback& callback_arg) : amount(amount_arg), callback(callback_arg) { } FileSystemResource::QuotaRequest::~QuotaRequest() { } FileSystemResource::FileSystemResource(Connection connection, PP_Instance instance, PP_FileSystemType type) : PluginResource(connection, instance), type_(type), called_open_(false), callback_count_(0), callback_result_(PP_OK), reserved_quota_(0), reserving_quota_(false) { DCHECK(type_ != PP_FILESYSTEMTYPE_INVALID); SendCreate(RENDERER, PpapiHostMsg_FileSystem_Create(type_)); SendCreate(BROWSER, PpapiHostMsg_FileSystem_Create(type_)); } FileSystemResource::FileSystemResource(Connection connection, PP_Instance instance, int pending_renderer_id, int pending_browser_id, PP_FileSystemType type) : PluginResource(connection, instance), type_(type), called_open_(true), callback_count_(0), callback_result_(PP_OK), reserved_quota_(0), reserving_quota_(false) { DCHECK(type_ != PP_FILESYSTEMTYPE_INVALID); AttachToPendingHost(RENDERER, pending_renderer_id); AttachToPendingHost(BROWSER, pending_browser_id); } FileSystemResource::~FileSystemResource() { } PPB_FileSystem_API* FileSystemResource::AsPPB_FileSystem_API() { return this; } int32_t FileSystemResource::Open(int64_t expected_size, scoped_refptr callback) { DCHECK(type_ != PP_FILESYSTEMTYPE_ISOLATED); if (called_open_) return PP_ERROR_FAILED; called_open_ = true; Call(RENDERER, PpapiHostMsg_FileSystem_Open(expected_size), base::Bind(&FileSystemResource::OpenComplete, this, callback)); Call(BROWSER, PpapiHostMsg_FileSystem_Open(expected_size), base::Bind(&FileSystemResource::OpenComplete, this, callback)); return PP_OK_COMPLETIONPENDING; } PP_FileSystemType FileSystemResource::GetType() { return type_; } void FileSystemResource::OpenQuotaFile(PP_Resource file_io) { DCHECK(max_written_offsets_.find(file_io) == max_written_offsets_.end()); EnterResourceNoLock enter(file_io, true); DCHECK(!enter.failed()); PPB_FileIO_API* file_io_api = enter.object(); max_written_offsets_[file_io] = file_io_api->GetMaxWrittenOffset(); } void FileSystemResource::CloseQuotaFile(PP_Resource file_io) { OffsetMap::iterator it = max_written_offsets_.find(file_io); DCHECK(it != max_written_offsets_.end()); max_written_offsets_.erase(it); } int64_t FileSystemResource::RequestQuota( int64_t amount, const RequestQuotaCallback& callback) { DCHECK(amount >= 0); if (!reserving_quota_ && reserved_quota_ >= amount) { reserved_quota_ -= amount; return amount; } // Queue up a pending quota request. pending_quota_requests_.push(QuotaRequest(amount, callback)); // Reserve more quota if we haven't already. if (!reserving_quota_) ReserveQuota(amount); return PP_OK_COMPLETIONPENDING; } int32_t FileSystemResource::InitIsolatedFileSystem( const std::string& fsid, PP_IsolatedFileSystemType_Private type, const base::Callback& callback) { // This call is mutually exclusive with Open() above, so we can reuse the // called_open state. DCHECK(type_ == PP_FILESYSTEMTYPE_ISOLATED); if (called_open_) return PP_ERROR_FAILED; called_open_ = true; Call(RENDERER, PpapiHostMsg_FileSystem_InitIsolatedFileSystem(fsid, type), base::Bind(&FileSystemResource::InitIsolatedFileSystemComplete, this, callback)); Call(BROWSER, PpapiHostMsg_FileSystem_InitIsolatedFileSystem(fsid, type), base::Bind(&FileSystemResource::InitIsolatedFileSystemComplete, this, callback)); return PP_OK_COMPLETIONPENDING; } void FileSystemResource::OpenComplete( scoped_refptr callback, const ResourceMessageReplyParams& params) { ++callback_count_; // Prioritize worse result since only one status can be returned. if (params.result() != PP_OK) callback_result_ = params.result(); // Received callback from browser and renderer. if (callback_count_ == 2) callback->Run(callback_result_); } void FileSystemResource::InitIsolatedFileSystemComplete( const base::Callback& callback, const ResourceMessageReplyParams& params) { ++callback_count_; // Prioritize worse result since only one status can be returned. if (params.result() != PP_OK) callback_result_ = params.result(); // Received callback from browser and renderer. if (callback_count_ == 2) callback.Run(callback_result_); } void FileSystemResource::ReserveQuota(int64_t amount) { DCHECK(!reserving_quota_); reserving_quota_ = true; for (OffsetMap::iterator it = max_written_offsets_.begin(); it != max_written_offsets_.end(); ++it) { EnterResourceNoLock enter(it->first, true); DCHECK(!enter.failed()); PPB_FileIO_API* file_io_api = enter.object(); it->second = file_io_api->GetMaxWrittenOffset(); } Call(BROWSER, PpapiHostMsg_FileSystem_ReserveQuota(amount, max_written_offsets_), base::Bind(&FileSystemResource::ReserveQuotaComplete, this)); } void FileSystemResource::ReserveQuotaComplete( const ResourceMessageReplyParams& params, int64_t amount, const OffsetMap& max_written_offsets) { DCHECK(reserving_quota_); reserving_quota_ = false; reserved_quota_ = amount; for (OffsetMap::const_iterator it = max_written_offsets.begin(); it != max_written_offsets.end(); ++it) { EnterResourceNoLock enter(it->first, true); DCHECK(!enter.failed()); PPB_FileIO_API* file_io_api = enter.object(); file_io_api->SetMaxWrittenOffset(it->second); } DCHECK(!pending_quota_requests_.empty()); // If we can't grant the first request after refreshing reserved_quota_, then // fail all pending quota requests to avoid an infinite refresh/fail loop. bool fail_all = reserved_quota_ < pending_quota_requests_.front().amount; while (!pending_quota_requests_.empty()) { QuotaRequest& request = pending_quota_requests_.front(); if (fail_all) { request.callback.Run(0); pending_quota_requests_.pop(); } else if (reserved_quota_ >= request.amount) { reserved_quota_ -= request.amount; request.callback.Run(request.amount); pending_quota_requests_.pop(); } else { // Refresh the quota reservation for the first pending request that we // can't satisfy. ReserveQuota(request.amount); break; } } } } // namespace proxy } // namespace ppapi