Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ppapi/proxy/file_io_resource.h" | 5 #include "ppapi/proxy/file_io_resource.h" |
| 6 | 6 |
| 7 #include "base/bind.h" | 7 #include "base/bind.h" |
| 8 #include "base/files/file_util_proxy.h" | |
| 8 #include "ipc/ipc_message.h" | 9 #include "ipc/ipc_message.h" |
| 9 #include "ppapi/c/pp_errors.h" | 10 #include "ppapi/c/pp_errors.h" |
| 11 #include "ppapi/proxy/plugin_globals.h" | |
| 10 #include "ppapi/proxy/ppapi_messages.h" | 12 #include "ppapi/proxy/ppapi_messages.h" |
| 11 #include "ppapi/shared_impl/array_writer.h" | 13 #include "ppapi/shared_impl/array_writer.h" |
| 14 #include "ppapi/shared_impl/file_type_conversion.h" | |
| 12 #include "ppapi/shared_impl/ppapi_globals.h" | 15 #include "ppapi/shared_impl/ppapi_globals.h" |
| 16 #include "ppapi/shared_impl/proxy_lock.h" | |
| 13 #include "ppapi/shared_impl/resource_tracker.h" | 17 #include "ppapi/shared_impl/resource_tracker.h" |
| 14 #include "ppapi/thunk/enter.h" | 18 #include "ppapi/thunk/enter.h" |
| 15 #include "ppapi/thunk/ppb_file_ref_api.h" | 19 #include "ppapi/thunk/ppb_file_ref_api.h" |
| 16 | 20 |
| 17 using ppapi::thunk::EnterResourceNoLock; | 21 using ppapi::thunk::EnterResourceNoLock; |
| 18 using ppapi::thunk::PPB_FileIO_API; | 22 using ppapi::thunk::PPB_FileIO_API; |
| 19 using ppapi::thunk::PPB_FileRef_API; | 23 using ppapi::thunk::PPB_FileRef_API; |
| 20 | 24 |
| 21 namespace { | 25 namespace { |
| 22 | 26 |
| 23 // An adapter to let Read() share the same implementation with ReadToArray(). | 27 // An adapter to let Read() share the same implementation with ReadToArray(). |
| 24 void* DummyGetDataBuffer(void* user_data, uint32_t count, uint32_t size) { | 28 void* DummyGetDataBuffer(void* user_data, uint32_t count, uint32_t size) { |
| 25 return user_data; | 29 return user_data; |
| 26 } | 30 } |
| 27 | 31 |
| 32 // All potentially blocking file tasks should be done on the file thread. In | |
| 33 // most cases, we can use base::FileUtilProxy to do the task and callback to the | |
| 34 // original thread. This won't work for blocking PPAPI callbacks, since we | |
| 35 // block the calling thread, which will cause FileUtilProxy to hang. Instead, | |
|
dmichael (off chromium)
2013/08/02 15:44:29
how about instead of "which will cause FileUtilPro
bbudge
2013/08/02 20:56:44
n/a now that we've eliminated FileUtilProxy.
| |
| 36 // do the same work in these tasks, which correctly wake the original thread. | |
| 37 // However, we can't use these tasks for non-blocking callbacks because the | |
| 38 // in-process proxy has no lock, which would lead to unsafe access to the | |
| 39 // resource by both threads. | |
| 40 | |
| 41 void DoQuery(base::PlatformFile file, | |
| 42 PP_FileInfo* info, | |
| 43 const base::FileUtilProxy::GetFileInfoCallback& callback) { | |
| 44 base::PlatformFileError error = base::PLATFORM_FILE_OK; | |
| 45 base::PlatformFileInfo file_info; | |
| 46 if (!GetPlatformFileInfo(file, &file_info)) | |
| 47 error = base::PLATFORM_FILE_ERROR_FAILED; | |
| 48 | |
| 49 callback.Run(error, file_info); | |
| 50 } | |
| 51 | |
| 52 void DoRead(base::PlatformFile file, | |
| 53 int64 offset, | |
| 54 int bytes_to_read, | |
| 55 const base::FileUtilProxy::ReadCallback& callback) { | |
| 56 scoped_ptr<char[]> buffer(new char[bytes_to_read]); | |
|
dmichael (off chromium)
2013/08/02 15:44:29
We shouldn't have to make a copy for the "Read" ca
bbudge
2013/08/02 20:56:44
Looks like we can't avoid the copy due to Abort se
| |
| 57 int bytes_read = | |
| 58 base::ReadPlatformFile(file, offset, buffer.get(), bytes_to_read); | |
| 59 base::PlatformFileError error = (bytes_read < 0) ? | |
| 60 base::PLATFORM_FILE_ERROR_FAILED : base::PLATFORM_FILE_OK; | |
| 61 | |
| 62 callback.Run(error, buffer.get(), bytes_read); | |
| 63 } | |
| 64 | |
| 65 void DoClose(base::PlatformFile file) { | |
| 66 base::ClosePlatformFile(file); | |
| 67 } | |
| 68 | |
| 28 } // namespace | 69 } // namespace |
| 29 | 70 |
| 30 namespace ppapi { | 71 namespace ppapi { |
| 31 namespace proxy { | 72 namespace proxy { |
| 32 | 73 |
| 33 FileIOResource::FileIOResource(Connection connection, PP_Instance instance) | 74 FileIOResource::FileIOResource(Connection connection, PP_Instance instance) |
| 34 : PluginResource(connection, instance) { | 75 : PluginResource(connection, instance), |
| 76 file_handle_(PP_kInvalidFileHandle), | |
| 77 file_system_type_(PP_FILESYSTEMTYPE_INVALID) { | |
| 35 SendCreate(RENDERER, PpapiHostMsg_FileIO_Create()); | 78 SendCreate(RENDERER, PpapiHostMsg_FileIO_Create()); |
| 36 } | 79 } |
| 37 | 80 |
| 38 FileIOResource::~FileIOResource() { | 81 FileIOResource::~FileIOResource() { |
| 82 CloseFileHandle(); | |
| 39 } | 83 } |
| 40 | 84 |
| 41 PPB_FileIO_API* FileIOResource::AsPPB_FileIO_API() { | 85 PPB_FileIO_API* FileIOResource::AsPPB_FileIO_API() { |
| 42 return this; | 86 return this; |
| 43 } | 87 } |
| 44 | 88 |
| 45 int32_t FileIOResource::Open(PP_Resource file_ref, | 89 int32_t FileIOResource::Open(PP_Resource file_ref, |
| 46 int32_t open_flags, | 90 int32_t open_flags, |
| 47 scoped_refptr<TrackedCallback> callback) { | 91 scoped_refptr<TrackedCallback> callback) { |
| 48 EnterResourceNoLock<PPB_FileRef_API> enter(file_ref, true); | 92 EnterResourceNoLock<PPB_FileRef_API> enter(file_ref, true); |
| 49 if (enter.failed()) | 93 if (enter.failed()) |
| 50 return PP_ERROR_BADRESOURCE; | 94 return PP_ERROR_BADRESOURCE; |
| 51 | 95 |
| 96 PPB_FileRef_API* file_ref_api = enter.object(); | |
| 97 PP_FileSystemType type = file_ref_api->GetFileSystemType(); | |
| 98 if (type != PP_FILESYSTEMTYPE_LOCALPERSISTENT && | |
| 99 type != PP_FILESYSTEMTYPE_LOCALTEMPORARY && | |
| 100 type != PP_FILESYSTEMTYPE_EXTERNAL && | |
| 101 type != PP_FILESYSTEMTYPE_ISOLATED) { | |
| 102 NOTREACHED(); | |
| 103 return PP_ERROR_FAILED; | |
| 104 } | |
| 105 file_system_type_ = type; | |
| 106 | |
| 52 int32_t rv = state_manager_.CheckOperationState( | 107 int32_t rv = state_manager_.CheckOperationState( |
| 53 FileIOStateManager::OPERATION_EXCLUSIVE, false); | 108 FileIOStateManager::OPERATION_EXCLUSIVE, false); |
| 54 if (rv != PP_OK) | 109 if (rv != PP_OK) |
| 55 return rv; | 110 return rv; |
| 56 | 111 |
| 57 Call<PpapiPluginMsg_FileIO_OpenReply>(RENDERER, | 112 Call<PpapiPluginMsg_FileIO_OpenReply>(RENDERER, |
| 58 PpapiHostMsg_FileIO_Open( | 113 PpapiHostMsg_FileIO_Open( |
| 59 enter.resource()->host_resource().host_resource(), | 114 enter.resource()->host_resource().host_resource(), |
| 60 open_flags), | 115 open_flags), |
| 61 base::Bind(&FileIOResource::OnPluginMsgOpenFileComplete, this, | 116 base::Bind(&FileIOResource::OnPluginMsgOpenFileComplete, this, |
| 62 callback)); | 117 callback)); |
| 63 | 118 |
| 64 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | 119 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); |
| 65 return PP_OK_COMPLETIONPENDING; | 120 return PP_OK_COMPLETIONPENDING; |
| 66 } | 121 } |
| 67 | 122 |
| 68 int32_t FileIOResource::Query(PP_FileInfo* info, | 123 int32_t FileIOResource::Query(PP_FileInfo* info, |
| 69 scoped_refptr<TrackedCallback> callback) { | 124 scoped_refptr<TrackedCallback> callback) { |
| 70 int32_t rv = state_manager_.CheckOperationState( | 125 int32_t rv = state_manager_.CheckOperationState( |
| 71 FileIOStateManager::OPERATION_EXCLUSIVE, true); | 126 FileIOStateManager::OPERATION_EXCLUSIVE, true); |
| 72 if (rv != PP_OK) | 127 if (rv != PP_OK) |
| 73 return rv; | 128 return rv; |
| 74 | 129 |
| 75 Call<PpapiPluginMsg_FileIO_QueryReply>(RENDERER, | 130 if (file_handle_ == base::kInvalidPlatformFileValue) |
| 76 PpapiHostMsg_FileIO_Query(), | 131 return PP_ERROR_FAILED; |
| 77 base::Bind(&FileIOResource::OnPluginMsgQueryComplete, this, | 132 |
| 78 callback, info)); | 133 base::TaskRunner* file_task_runner = |
| 134 PpapiGlobals::Get()->GetFileTaskRunner(pp_instance()); | |
| 135 base::FileUtilProxy::GetFileInfoCallback file_task = RunWhileLocked( | |
| 136 base::Bind(&FileIOResource::OnQueryComplete, this, callback, info)); | |
| 137 if (callback->is_blocking()) { | |
| 138 file_task_runner->PostTask( | |
| 139 FROM_HERE, | |
| 140 Bind(&DoQuery, file_handle_, info, file_task)); | |
| 141 } else { | |
| 142 if (!base::FileUtilProxy::GetFileInfoFromPlatformFile(file_task_runner, | |
| 143 file_handle_, | |
| 144 file_task)) { | |
| 145 return PP_ERROR_FAILED; | |
| 146 } | |
| 147 } | |
| 79 | 148 |
| 80 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | 149 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); |
| 81 return PP_OK_COMPLETIONPENDING; | 150 return PP_OK_COMPLETIONPENDING; |
| 82 } | 151 } |
| 83 | 152 |
| 84 int32_t FileIOResource::Touch(PP_Time last_access_time, | 153 int32_t FileIOResource::Touch(PP_Time last_access_time, |
| 85 PP_Time last_modified_time, | 154 PP_Time last_modified_time, |
| 86 scoped_refptr<TrackedCallback> callback) { | 155 scoped_refptr<TrackedCallback> callback) { |
| 87 int32_t rv = state_manager_.CheckOperationState( | 156 int32_t rv = state_manager_.CheckOperationState( |
| 88 FileIOStateManager::OPERATION_EXCLUSIVE, true); | 157 FileIOStateManager::OPERATION_EXCLUSIVE, true); |
| (...skipping 14 matching lines...) Expand all Loading... | |
| 103 int32_t bytes_to_read, | 172 int32_t bytes_to_read, |
| 104 scoped_refptr<TrackedCallback> callback) { | 173 scoped_refptr<TrackedCallback> callback) { |
| 105 int32_t rv = state_manager_.CheckOperationState( | 174 int32_t rv = state_manager_.CheckOperationState( |
| 106 FileIOStateManager::OPERATION_READ, true); | 175 FileIOStateManager::OPERATION_READ, true); |
| 107 if (rv != PP_OK) | 176 if (rv != PP_OK) |
| 108 return rv; | 177 return rv; |
| 109 | 178 |
| 110 PP_ArrayOutput output_adapter; | 179 PP_ArrayOutput output_adapter; |
| 111 output_adapter.GetDataBuffer = &DummyGetDataBuffer; | 180 output_adapter.GetDataBuffer = &DummyGetDataBuffer; |
| 112 output_adapter.user_data = buffer; | 181 output_adapter.user_data = buffer; |
| 113 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_READ); | |
| 114 return ReadValidated(offset, bytes_to_read, output_adapter, callback); | 182 return ReadValidated(offset, bytes_to_read, output_adapter, callback); |
| 115 } | 183 } |
| 116 | 184 |
| 117 int32_t FileIOResource::ReadToArray(int64_t offset, | 185 int32_t FileIOResource::ReadToArray(int64_t offset, |
| 118 int32_t max_read_length, | 186 int32_t max_read_length, |
| 119 PP_ArrayOutput* array_output, | 187 PP_ArrayOutput* array_output, |
| 120 scoped_refptr<TrackedCallback> callback) { | 188 scoped_refptr<TrackedCallback> callback) { |
| 121 DCHECK(array_output); | 189 DCHECK(array_output); |
| 122 int32_t rv = state_manager_.CheckOperationState( | 190 int32_t rv = state_manager_.CheckOperationState( |
| 123 FileIOStateManager::OPERATION_READ, true); | 191 FileIOStateManager::OPERATION_READ, true); |
| 124 if (rv != PP_OK) | 192 if (rv != PP_OK) |
| 125 return rv; | 193 return rv; |
| 126 | 194 |
| 127 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_READ); | |
| 128 return ReadValidated(offset, max_read_length, *array_output, callback); | 195 return ReadValidated(offset, max_read_length, *array_output, callback); |
| 129 } | 196 } |
| 130 | 197 |
| 131 int32_t FileIOResource::Write(int64_t offset, | 198 int32_t FileIOResource::Write(int64_t offset, |
| 132 const char* buffer, | 199 const char* buffer, |
| 133 int32_t bytes_to_write, | 200 int32_t bytes_to_write, |
| 134 scoped_refptr<TrackedCallback> callback) { | 201 scoped_refptr<TrackedCallback> callback) { |
| 135 int32_t rv = state_manager_.CheckOperationState( | 202 int32_t rv = state_manager_.CheckOperationState( |
| 136 FileIOStateManager::OPERATION_WRITE, true); | 203 FileIOStateManager::OPERATION_WRITE, true); |
| 137 if (rv != PP_OK) | 204 if (rv != PP_OK) |
| (...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 174 Call<PpapiPluginMsg_FileIO_GeneralReply>(RENDERER, | 241 Call<PpapiPluginMsg_FileIO_GeneralReply>(RENDERER, |
| 175 PpapiHostMsg_FileIO_Flush(), | 242 PpapiHostMsg_FileIO_Flush(), |
| 176 base::Bind(&FileIOResource::OnPluginMsgGeneralComplete, this, | 243 base::Bind(&FileIOResource::OnPluginMsgGeneralComplete, this, |
| 177 callback)); | 244 callback)); |
| 178 | 245 |
| 179 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | 246 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); |
| 180 return PP_OK_COMPLETIONPENDING; | 247 return PP_OK_COMPLETIONPENDING; |
| 181 } | 248 } |
| 182 | 249 |
| 183 void FileIOResource::Close() { | 250 void FileIOResource::Close() { |
| 251 CloseFileHandle(); | |
| 184 Post(RENDERER, PpapiHostMsg_FileIO_Close()); | 252 Post(RENDERER, PpapiHostMsg_FileIO_Close()); |
| 185 } | 253 } |
| 186 | 254 |
| 187 int32_t FileIOResource::GetOSFileDescriptor() { | 255 int32_t FileIOResource::GetOSFileDescriptor() { |
| 188 int32_t file_descriptor; | 256 int32_t file_descriptor; |
| 189 // Only available when running in process. | 257 // Only available when running in process. |
| 190 SyncCall<PpapiPluginMsg_FileIO_GetOSFileDescriptorReply>( | 258 SyncCall<PpapiPluginMsg_FileIO_GetOSFileDescriptorReply>( |
| 191 RENDERER, PpapiHostMsg_FileIO_GetOSFileDescriptor(), &file_descriptor); | 259 RENDERER, PpapiHostMsg_FileIO_GetOSFileDescriptor(), &file_descriptor); |
| 192 return file_descriptor; | 260 return file_descriptor; |
| 193 } | 261 } |
| 194 | 262 |
| 263 int32_t FileIOResource::RequestOSFileHandle( | |
| 264 PP_FileHandle* handle, | |
| 265 scoped_refptr<TrackedCallback> callback) { | |
| 266 int32_t rv = state_manager_.CheckOperationState( | |
| 267 FileIOStateManager::OPERATION_EXCLUSIVE, true); | |
| 268 if (rv != PP_OK) | |
| 269 return rv; | |
| 270 | |
| 271 Call<PpapiPluginMsg_FileIO_RequestOSFileHandleReply>(RENDERER, | |
| 272 PpapiHostMsg_FileIO_RequestOSFileHandle(), | |
| 273 base::Bind(&FileIOResource::OnPluginMsgRequestOSFileHandleComplete, this, | |
| 274 callback, handle)); | |
| 275 | |
| 276 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | |
| 277 return PP_OK_COMPLETIONPENDING; | |
| 278 } | |
| 279 | |
| 195 int32_t FileIOResource::WillWrite(int64_t offset, | 280 int32_t FileIOResource::WillWrite(int64_t offset, |
| 196 int32_t bytes_to_write, | 281 int32_t bytes_to_write, |
| 197 scoped_refptr<TrackedCallback> callback) { | 282 scoped_refptr<TrackedCallback> callback) { |
| 198 Call<PpapiPluginMsg_FileIO_GeneralReply>(RENDERER, | 283 Call<PpapiPluginMsg_FileIO_GeneralReply>(RENDERER, |
| 199 PpapiHostMsg_FileIO_WillWrite(offset, bytes_to_write), | 284 PpapiHostMsg_FileIO_WillWrite(offset, bytes_to_write), |
| 200 base::Bind(&FileIOResource::OnPluginMsgGeneralComplete, this, | 285 base::Bind(&FileIOResource::OnPluginMsgGeneralComplete, this, |
| 201 callback)); | 286 callback)); |
| 287 | |
| 202 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | 288 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); |
| 203 return PP_OK_COMPLETIONPENDING; | 289 return PP_OK_COMPLETIONPENDING; |
| 204 } | 290 } |
| 205 | 291 |
| 206 int32_t FileIOResource::WillSetLength(int64_t length, | 292 int32_t FileIOResource::WillSetLength(int64_t length, |
| 207 scoped_refptr<TrackedCallback> callback) { | 293 scoped_refptr<TrackedCallback> callback) { |
| 208 Call<PpapiPluginMsg_FileIO_GeneralReply>(RENDERER, | 294 Call<PpapiPluginMsg_FileIO_GeneralReply>(RENDERER, |
| 209 PpapiHostMsg_FileIO_WillSetLength(length), | 295 PpapiHostMsg_FileIO_WillSetLength(length), |
| 210 base::Bind(&FileIOResource::OnPluginMsgGeneralComplete, this, | 296 base::Bind(&FileIOResource::OnPluginMsgGeneralComplete, this, |
| 211 callback)); | 297 callback)); |
| 298 | |
| 212 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | 299 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); |
| 213 return PP_OK_COMPLETIONPENDING; | 300 return PP_OK_COMPLETIONPENDING; |
| 214 } | 301 } |
| 215 | 302 |
| 216 int32_t FileIOResource::ReadValidated(int64_t offset, | 303 int32_t FileIOResource::ReadValidated(int64_t offset, |
| 217 int32_t bytes_to_read, | 304 int32_t bytes_to_read, |
| 218 const PP_ArrayOutput& array_output, | 305 const PP_ArrayOutput& array_output, |
| 219 scoped_refptr<TrackedCallback> callback) { | 306 scoped_refptr<TrackedCallback> callback) { |
| 220 Call<PpapiPluginMsg_FileIO_ReadReply>(RENDERER, | 307 if (file_handle_ == base::kInvalidPlatformFileValue) |
| 221 PpapiHostMsg_FileIO_Read(offset, bytes_to_read), | 308 return PP_ERROR_FAILED; |
| 222 base::Bind(&FileIOResource::OnPluginMsgReadComplete, this, | 309 if (bytes_to_read < 0) |
| 223 callback, array_output)); | 310 return PP_ERROR_FAILED; |
| 311 | |
| 312 base::TaskRunner* file_task_runner = | |
| 313 PpapiGlobals::Get()->GetFileTaskRunner(pp_instance()); | |
| 314 base::FileUtilProxy::ReadCallback file_task = RunWhileLocked( | |
| 315 base::Bind(&FileIOResource::OnReadComplete, this, callback, | |
| 316 array_output)); | |
| 317 if (callback->is_blocking()) { | |
| 318 file_task_runner->PostTask( | |
| 319 FROM_HERE, | |
| 320 Bind(&DoRead, file_handle_, offset, bytes_to_read, file_task)); | |
| 321 } else { | |
| 322 if (!base::FileUtilProxy::Read(file_task_runner, | |
| 323 file_handle_, | |
| 324 offset, | |
| 325 bytes_to_read, | |
| 326 file_task)) { | |
| 327 return PP_ERROR_FAILED; | |
| 328 } | |
| 329 } | |
| 330 | |
| 331 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_READ); | |
| 224 return PP_OK_COMPLETIONPENDING; | 332 return PP_OK_COMPLETIONPENDING; |
| 225 } | 333 } |
| 226 | 334 |
| 227 int32_t FileIOResource::RequestOSFileHandle( | 335 void FileIOResource::CloseFileHandle() { |
| 228 PP_FileHandle* handle, | 336 if (file_handle_ != base::kInvalidPlatformFileValue) { |
| 229 scoped_refptr<TrackedCallback> callback) { | 337 base::TaskRunner* file_task_runner = |
| 230 int32_t rv = state_manager_.CheckOperationState( | 338 PpapiGlobals::Get()->GetFileTaskRunner(pp_instance()); |
| 231 FileIOStateManager::OPERATION_EXCLUSIVE, true); | 339 file_task_runner->PostTask(FROM_HERE, |
| 232 if (rv != PP_OK) | 340 base::Bind(&DoClose, file_handle_)); |
| 233 return rv; | |
| 234 | 341 |
| 235 Call<PpapiPluginMsg_FileIO_RequestOSFileHandleReply>(RENDERER, | 342 file_handle_ = base::kInvalidPlatformFileValue; |
| 236 PpapiHostMsg_FileIO_RequestOSFileHandle(), | 343 } |
| 237 base::Bind(&FileIOResource::OnPluginMsgRequestOSFileHandleComplete, this, | 344 } |
| 238 callback, handle)); | |
| 239 | 345 |
| 240 state_manager_.SetPendingOperation(FileIOStateManager::OPERATION_EXCLUSIVE); | 346 void FileIOResource::OnQueryComplete( |
| 241 return PP_OK_COMPLETIONPENDING; | 347 scoped_refptr<TrackedCallback> callback, |
| 348 PP_FileInfo* output_info, | |
| 349 base::PlatformFileError error_code, | |
| 350 const base::PlatformFileInfo& file_info) { | |
| 351 DCHECK(state_manager_.get_pending_operation() == | |
| 352 FileIOStateManager::OPERATION_EXCLUSIVE); | |
| 353 | |
| 354 if (!TrackedCallback::IsPending(callback)) { | |
| 355 state_manager_.SetOperationFinished(); | |
| 356 return; | |
| 357 } | |
| 358 | |
| 359 int32_t result = ::ppapi::PlatformFileErrorToPepperError(error_code); | |
| 360 if (result == PP_OK) { | |
| 361 ppapi::PlatformFileInfoToPepperFileInfo(file_info, file_system_type_, | |
| 362 output_info); | |
| 363 } | |
| 364 | |
| 365 // End this operation now, so the user's callback can execute another FileIO | |
| 366 // operation, assuming there are no other pending operations. | |
| 367 state_manager_.SetOperationFinished(); | |
| 368 callback->Run(result); | |
| 369 } | |
| 370 | |
| 371 void FileIOResource::OnReadComplete( | |
| 372 scoped_refptr<TrackedCallback> callback, | |
| 373 PP_ArrayOutput array_output, | |
| 374 base::PlatformFileError error_code, | |
| 375 const char* data, int bytes_read) { | |
| 376 DCHECK(state_manager_.get_pending_operation() == | |
| 377 FileIOStateManager::OPERATION_READ); | |
| 378 | |
| 379 if (!TrackedCallback::IsPending(callback)) { | |
| 380 state_manager_.SetOperationFinished(); | |
| 381 return; | |
| 382 } | |
| 383 | |
| 384 int32_t result = ::ppapi::PlatformFileErrorToPepperError(error_code); | |
| 385 if (result == PP_OK) { | |
| 386 result = std::max(0, bytes_read); | |
| 387 ArrayWriter output; | |
| 388 output.set_pp_array_output(array_output); | |
| 389 if (output.is_valid()) | |
| 390 output.StoreArray(data, result); | |
| 391 else | |
| 392 result = PP_ERROR_FAILED; | |
| 393 } | |
| 394 | |
| 395 // End this operation now, so the user's callback can execute another FileIO | |
| 396 // operation, assuming there are no other pending operations. | |
| 397 state_manager_.SetOperationFinished(); | |
| 398 callback->Run(result); | |
| 242 } | 399 } |
| 243 | 400 |
| 244 void FileIOResource::OnPluginMsgGeneralComplete( | 401 void FileIOResource::OnPluginMsgGeneralComplete( |
| 245 scoped_refptr<TrackedCallback> callback, | 402 scoped_refptr<TrackedCallback> callback, |
| 246 const ResourceMessageReplyParams& params) { | 403 const ResourceMessageReplyParams& params) { |
| 247 DCHECK(state_manager_.get_pending_operation() == | 404 DCHECK(state_manager_.get_pending_operation() == |
| 248 FileIOStateManager::OPERATION_EXCLUSIVE || | 405 FileIOStateManager::OPERATION_EXCLUSIVE || |
| 249 state_manager_.get_pending_operation() == | 406 state_manager_.get_pending_operation() == |
| 250 FileIOStateManager::OPERATION_WRITE); | 407 FileIOStateManager::OPERATION_WRITE); |
| 251 // End the operation now. The callback may perform another file operation. | 408 // End this operation now, so the user's callback can execute another FileIO |
| 409 // operation, assuming there are no other pending operations. | |
| 252 state_manager_.SetOperationFinished(); | 410 state_manager_.SetOperationFinished(); |
| 253 callback->Run(params.result()); | 411 callback->Run(params.result()); |
| 254 } | 412 } |
| 255 | 413 |
| 256 void FileIOResource::OnPluginMsgOpenFileComplete( | 414 void FileIOResource::OnPluginMsgOpenFileComplete( |
| 257 scoped_refptr<TrackedCallback> callback, | 415 scoped_refptr<TrackedCallback> callback, |
| 258 const ResourceMessageReplyParams& params) { | 416 const ResourceMessageReplyParams& params) { |
| 259 DCHECK(state_manager_.get_pending_operation() == | 417 DCHECK(state_manager_.get_pending_operation() == |
| 260 FileIOStateManager::OPERATION_EXCLUSIVE); | 418 FileIOStateManager::OPERATION_EXCLUSIVE); |
| 261 if (params.result() == PP_OK) | 419 if (params.result() == PP_OK) |
| 262 state_manager_.SetOpenSucceed(); | 420 state_manager_.SetOpenSucceed(); |
| 263 // End the operation now. The callback may perform another file operation. | 421 |
| 422 int32_t result = params.result(); | |
| 423 IPC::PlatformFileForTransit transit_file; | |
| 424 if (result == PP_OK && !params.TakeFileHandleAtIndex(0, &transit_file)) | |
| 425 result = PP_ERROR_FAILED; | |
| 426 file_handle_ = IPC::PlatformFileForTransitToPlatformFile(transit_file); | |
| 427 | |
| 428 // End this operation now, so the user's callback can execute another FileIO | |
| 429 // operation, assuming there are no other pending operations. | |
| 264 state_manager_.SetOperationFinished(); | 430 state_manager_.SetOperationFinished(); |
| 265 callback->Run(params.result()); | 431 callback->Run(params.result()); |
| 266 } | 432 } |
| 267 | 433 |
| 268 void FileIOResource::OnPluginMsgQueryComplete( | |
| 269 scoped_refptr<TrackedCallback> callback, | |
| 270 PP_FileInfo* output_info, | |
| 271 const ResourceMessageReplyParams& params, | |
| 272 const PP_FileInfo& info) { | |
| 273 DCHECK(state_manager_.get_pending_operation() == | |
| 274 FileIOStateManager::OPERATION_EXCLUSIVE); | |
| 275 *output_info = info; | |
| 276 // End the operation now. The callback may perform another file operation. | |
| 277 state_manager_.SetOperationFinished(); | |
| 278 callback->Run(params.result()); | |
| 279 } | |
| 280 | |
| 281 void FileIOResource::OnPluginMsgReadComplete( | |
| 282 scoped_refptr<TrackedCallback> callback, | |
| 283 PP_ArrayOutput array_output, | |
| 284 const ResourceMessageReplyParams& params, | |
| 285 const std::string& data) { | |
| 286 DCHECK(state_manager_.get_pending_operation() == | |
| 287 FileIOStateManager::OPERATION_READ); | |
| 288 | |
| 289 // The result code should contain the data size if it's positive. | |
| 290 int32_t result = params.result(); | |
| 291 DCHECK((result < 0 && data.size() == 0) || | |
| 292 result == static_cast<int32_t>(data.size())); | |
| 293 | |
| 294 ArrayWriter output; | |
| 295 output.set_pp_array_output(array_output); | |
| 296 if (output.is_valid()) | |
| 297 output.StoreArray(data.data(), std::max(0, result)); | |
| 298 else | |
| 299 result = PP_ERROR_FAILED; | |
| 300 | |
| 301 // End the operation now. The callback may perform another file operation. | |
| 302 state_manager_.SetOperationFinished(); | |
| 303 callback->Run(result); | |
| 304 } | |
| 305 | |
| 306 void FileIOResource::OnPluginMsgRequestOSFileHandleComplete( | 434 void FileIOResource::OnPluginMsgRequestOSFileHandleComplete( |
| 307 scoped_refptr<TrackedCallback> callback, | 435 scoped_refptr<TrackedCallback> callback, |
| 308 PP_FileHandle* output_handle, | 436 PP_FileHandle* output_handle, |
| 309 const ResourceMessageReplyParams& params) { | 437 const ResourceMessageReplyParams& params) { |
| 310 DCHECK(state_manager_.get_pending_operation() == | 438 DCHECK(state_manager_.get_pending_operation() == |
| 311 FileIOStateManager::OPERATION_EXCLUSIVE); | 439 FileIOStateManager::OPERATION_EXCLUSIVE); |
| 312 | 440 |
| 313 if (!TrackedCallback::IsPending(callback)) { | 441 if (!TrackedCallback::IsPending(callback)) { |
| 314 state_manager_.SetOperationFinished(); | 442 state_manager_.SetOperationFinished(); |
| 315 return; | 443 return; |
| 316 } | 444 } |
| 317 | 445 |
| 318 int32_t result = params.result(); | 446 int32_t result = params.result(); |
| 319 IPC::PlatformFileForTransit transit_file; | 447 IPC::PlatformFileForTransit transit_file; |
| 320 if (!params.TakeFileHandleAtIndex(0, &transit_file)) | 448 if (!params.TakeFileHandleAtIndex(0, &transit_file)) |
| 321 result = PP_ERROR_FAILED; | 449 result = PP_ERROR_FAILED; |
| 322 *output_handle = IPC::PlatformFileForTransitToPlatformFile(transit_file); | 450 *output_handle = IPC::PlatformFileForTransitToPlatformFile(transit_file); |
| 323 | 451 |
| 324 // End the operation now. The callback may perform another file operation. | 452 // End this operation now, so the user's callback can execute another FileIO |
| 453 // operation, assuming there are no other pending operations. | |
| 325 state_manager_.SetOperationFinished(); | 454 state_manager_.SetOperationFinished(); |
| 326 callback->Run(result); | 455 callback->Run(result); |
| 327 } | 456 } |
| 328 | 457 |
| 329 } // namespace proxy | 458 } // namespace proxy |
| 330 } // namespace ppapi | 459 } // namespace ppapi |
| OLD | NEW |