// 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 "base/stl_util.h" #include "ipc/ipc_message.h" #include "ppapi/c/pp_errors.h" #include "ppapi/proxy/ppapi_messages.h" #include "ppapi/shared_impl/file_growth.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(!ContainsKey(files_, file_io)); files_.insert(file_io); } void FileSystemResource::CloseQuotaFile(PP_Resource file_io) { DCHECK(ContainsKey(files_, file_io)); files_.erase(file_io); } 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; FileGrowthMap file_growths; for (std::set::iterator it = files_.begin(); it != files_.end(); ++it) { EnterResourceNoLock enter(*it, true); if (enter.failed()) { NOTREACHED(); continue; } PPB_FileIO_API* file_io_api = enter.object(); file_growths[*it] = FileGrowth( file_io_api->GetMaxWrittenOffset(), file_io_api->GetAppendModeWriteAmount()); } Call(BROWSER, PpapiHostMsg_FileSystem_ReserveQuota(amount, file_growths), base::Bind(&FileSystemResource::ReserveQuotaComplete, this)); } void FileSystemResource::ReserveQuotaComplete( const ResourceMessageReplyParams& params, int64_t amount, const FileSizeMap& file_sizes) { DCHECK(reserving_quota_); reserving_quota_ = false; reserved_quota_ = amount; for (FileSizeMap::const_iterator it = file_sizes.begin(); it != file_sizes.end(); ++it) { EnterResourceNoLock enter(it->first, true); // It is possible that the host has sent an offset for a file that has been // destroyed in the plugin. Ignore it. if (enter.failed()) continue; PPB_FileIO_API* file_io_api = enter.object(); file_io_api->SetMaxWrittenOffset(it->second); file_io_api->SetAppendModeWriteAmount(0); } 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