Chromium Code Reviews
|
| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "ppapi/proxy/ppb_file_system_proxy.h" | |
| 6 | |
| 7 #include "base/message_loop.h" | |
| 8 #include "base/task.h" | |
| 9 #include "ppapi/c/dev/ppb_file_system_dev.h" | |
| 10 #include "ppapi/c/pp_errors.h" | |
| 11 #include "ppapi/proxy/host_dispatcher.h" | |
| 12 #include "ppapi/proxy/plugin_dispatcher.h" | |
| 13 #include "ppapi/proxy/plugin_resource.h" | |
| 14 #include "ppapi/proxy/ppapi_messages.h" | |
| 15 #include "ppapi/proxy/serialized_var.h" | |
| 16 | |
| 17 namespace pp { | |
| 18 namespace proxy { | |
| 19 | |
| 20 // This object maintains most of the state of the ref in the plugin for fast | |
| 21 // querying. It's all set in the constructor from the "create info" sent from | |
| 22 // the host. | |
| 23 class FileSystem : public PluginResource { | |
| 24 public: | |
| 25 FileSystem(const HostResource& host_resource, PP_FileSystemType_Dev type); | |
| 26 virtual ~FileSystem(); | |
| 27 | |
| 28 virtual FileSystem* AsFileSystem(); | |
| 29 | |
| 30 PP_FileSystemType_Dev type_; | |
| 31 bool opened_; | |
| 32 PP_CompletionCallback current_open_callback_; | |
| 33 | |
| 34 private: | |
| 35 DISALLOW_COPY_AND_ASSIGN(FileSystem); | |
| 36 }; | |
| 37 | |
| 38 FileSystem::FileSystem(const HostResource& host_resource, | |
| 39 PP_FileSystemType_Dev type) | |
| 40 : PluginResource(host_resource), | |
| 41 type_(type), | |
| 42 opened_(false), | |
| 43 current_open_callback_(PP_MakeCompletionCallback(NULL, NULL)) { | |
| 44 } | |
| 45 | |
| 46 // TODO(brettw) this logic is duplicated with some other resource objects | |
| 47 // like FileChooser. It would be nice to look at all of the different resources | |
| 48 // that need callback tracking and design something that they can all re-use. | |
| 49 FileSystem::~FileSystem() { | |
| 50 // Ensure the callback is always fired. | |
| 51 if (current_open_callback_.func) { | |
| 52 // TODO(brettw) the callbacks at this level should be refactored with a | |
| 53 // more automatic tracking system like we have in the renderer. | |
| 54 MessageLoop::current()->PostTask(FROM_HERE, NewRunnableFunction( | |
| 55 current_open_callback_.func, current_open_callback_.user_data, | |
| 56 static_cast<int32_t>(PP_ERROR_ABORTED))); | |
| 57 } | |
| 58 } | |
| 59 | |
| 60 FileSystem* FileSystem::AsFileSystem() { | |
| 61 return this; | |
| 62 } | |
| 63 | |
| 64 namespace { | |
| 65 | |
| 66 PP_Resource Create(PP_Instance instance, PP_FileSystemType_Dev type) { | |
| 67 PluginDispatcher* dispatcher = PluginDispatcher::GetForInstance(instance); | |
| 68 if (!dispatcher) | |
| 69 return PP_ERROR_BADARGUMENT; | |
| 70 | |
| 71 HostResource result; | |
| 72 dispatcher->Send(new PpapiHostMsg_PPBFileSystem_Create( | |
| 73 INTERFACE_ID_PPB_FILE_SYSTEM, instance, type, &result)); | |
| 74 if (result.is_null()) | |
| 75 return 0; | |
| 76 | |
| 77 linked_ptr<FileSystem> object(new FileSystem(result, type)); | |
| 78 return PluginResourceTracker::GetInstance()->AddResource(object); | |
| 79 } | |
| 80 | |
| 81 PP_Bool IsFileSystem(PP_Resource resource) { | |
| 82 FileSystem* object = PluginResource::GetAs<FileSystem>(resource); | |
| 83 return BoolToPPBool(!!object); | |
| 84 } | |
| 85 | |
| 86 int32_t Open(PP_Resource file_system, | |
| 87 int64_t expected_size, | |
| 88 struct PP_CompletionCallback callback) { | |
| 89 FileSystem* object = PluginResource::GetAs<FileSystem>(file_system); | |
| 90 if (!object) | |
| 91 return PP_ERROR_BADRESOURCE; | |
| 92 if (object->opened_) | |
|
vtl
2011/02/21 19:06:00
I'm a little surprised that we allow multiple Open
| |
| 93 return PP_OK; | |
| 94 | |
| 95 Dispatcher* dispatcher = PluginDispatcher::GetForInstance(object->instance()); | |
| 96 if (!dispatcher) | |
| 97 return PP_ERROR_BADARGUMENT; | |
|
vtl
2011/02/21 19:06:00
Is "bad argument" really right? (Probably a preced
brettw
2011/02/21 19:19:10
I was setting this elsewhere. In many places I end
| |
| 98 | |
| 99 if (object->current_open_callback_.func) | |
| 100 return PP_ERROR_INPROGRESS; | |
| 101 object->current_open_callback_ = callback; | |
| 102 | |
| 103 dispatcher->Send(new PpapiHostMsg_PPBFileSystem_Open( | |
| 104 INTERFACE_ID_PPB_FILE_SYSTEM, object->host_resource(), expected_size)); | |
| 105 return PP_ERROR_WOULDBLOCK; | |
| 106 } | |
| 107 | |
| 108 PP_FileSystemType_Dev GetType(PP_Resource resource) { | |
| 109 FileSystem* object = PluginResource::GetAs<FileSystem>(resource); | |
| 110 if (!object) | |
| 111 return PP_FILESYSTEMTYPE_NONE; | |
| 112 return object->type_; | |
| 113 } | |
| 114 | |
| 115 const PPB_FileSystem_Dev file_system_interface = { | |
| 116 &Create, | |
| 117 &IsFileSystem, | |
| 118 &Open, | |
| 119 &GetType | |
| 120 }; | |
| 121 | |
| 122 InterfaceProxy* CreateFileSystemProxy(Dispatcher* dispatcher, | |
| 123 const void* target_interface) { | |
| 124 return new PPB_FileSystem_Proxy(dispatcher, target_interface); | |
| 125 } | |
| 126 | |
| 127 } // namespace | |
| 128 | |
| 129 PPB_FileSystem_Proxy::PPB_FileSystem_Proxy(Dispatcher* dispatcher, | |
| 130 const void* target_interface) | |
| 131 : InterfaceProxy(dispatcher, target_interface), | |
| 132 callback_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) { | |
| 133 } | |
| 134 | |
| 135 PPB_FileSystem_Proxy::~PPB_FileSystem_Proxy() { | |
| 136 } | |
| 137 | |
| 138 const InterfaceProxy::Info* PPB_FileSystem_Proxy::GetInfo() { | |
| 139 static const Info info = { | |
| 140 &file_system_interface, | |
| 141 PPB_FILESYSTEM_DEV_INTERFACE, | |
| 142 INTERFACE_ID_PPB_FILE_SYSTEM, | |
| 143 false, | |
| 144 &CreateFileSystemProxy, | |
| 145 }; | |
| 146 return &info; | |
| 147 } | |
| 148 | |
| 149 bool PPB_FileSystem_Proxy::OnMessageReceived(const IPC::Message& msg) { | |
| 150 bool handled = true; | |
| 151 IPC_BEGIN_MESSAGE_MAP(PPB_FileSystem_Proxy, msg) | |
| 152 IPC_MESSAGE_HANDLER(PpapiHostMsg_PPBFileSystem_Create, OnMsgCreate) | |
| 153 IPC_MESSAGE_HANDLER(PpapiHostMsg_PPBFileSystem_Open, OnMsgOpen) | |
| 154 IPC_MESSAGE_HANDLER(PpapiMsg_PPBFileSystem_OpenComplete, OnMsgOpenComplete) | |
| 155 IPC_MESSAGE_UNHANDLED(handled = false) | |
| 156 IPC_END_MESSAGE_MAP() | |
| 157 return handled; | |
| 158 } | |
| 159 | |
| 160 void PPB_FileSystem_Proxy::OnMsgCreate(PP_Instance instance, | |
| 161 int type, | |
| 162 HostResource* result) { | |
| 163 PP_Resource resource = ppb_file_system_target()->Create( | |
| 164 instance, static_cast<PP_FileSystemType_Dev>(type)); | |
| 165 if (!resource) | |
| 166 return; // CreateInfo default constructor initializes to 0. | |
| 167 result->SetHostResource(instance, resource); | |
| 168 } | |
| 169 | |
| 170 void PPB_FileSystem_Proxy::OnMsgOpen(const HostResource& host_resource, | |
| 171 int64_t expected_size) { | |
| 172 CompletionCallback callback = callback_factory_.NewCallback( | |
| 173 &PPB_FileSystem_Proxy::OpenCompleteInHost, host_resource); | |
| 174 | |
| 175 int32_t result = ppb_file_system_target()->Open( | |
| 176 host_resource.host_resource(), expected_size, | |
| 177 callback.pp_completion_callback()); | |
| 178 if (result != PP_ERROR_WOULDBLOCK) | |
| 179 callback.Run(result); | |
| 180 } | |
| 181 | |
| 182 // Called in the plugin to handle the open callback. | |
| 183 void PPB_FileSystem_Proxy::OnMsgOpenComplete(const HostResource& filesystem, | |
| 184 int32_t result) { | |
| 185 FileSystem* object = PluginResource::GetAs<FileSystem>( | |
| 186 PluginResourceTracker::GetInstance()->PluginResourceForHostResource( | |
| 187 filesystem)); | |
| 188 if (!object || !object->current_open_callback_.func) | |
| 189 return; | |
| 190 | |
| 191 PP_CompletionCallback callback = object->current_open_callback_; | |
| 192 object->current_open_callback_ = PP_MakeCompletionCallback(NULL, NULL); | |
| 193 PP_RunCompletionCallback(&callback, result); | |
| 194 } | |
| 195 | |
| 196 void PPB_FileSystem_Proxy::OpenCompleteInHost( | |
| 197 int32_t result, | |
| 198 const HostResource& host_resource) { | |
| 199 dispatcher()->Send(new PpapiMsg_PPBFileSystem_OpenComplete( | |
| 200 INTERFACE_ID_PPB_FILE_SYSTEM, host_resource, result)); | |
| 201 } | |
| 202 | |
| 203 } // namespace proxy | |
| 204 } // namespace pp | |
| OLD | NEW |