| Index: ppapi/native_client/src/shared/ppapi_proxy/browser_ppb_file_io_rpc_server.cc
|
| ===================================================================
|
| --- ppapi/native_client/src/shared/ppapi_proxy/browser_ppb_file_io_rpc_server.cc (revision 0)
|
| +++ ppapi/native_client/src/shared/ppapi_proxy/browser_ppb_file_io_rpc_server.cc (revision 0)
|
| @@ -0,0 +1,310 @@
|
| +// Copyright (c) 2011 The Native Client 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 <string.h>
|
| +#include <limits>
|
| +
|
| +#include "native_client/src/include/nacl_macros.h"
|
| +#include "native_client/src/shared/ppapi_proxy/browser_callback.h"
|
| +#include "native_client/src/shared/ppapi_proxy/browser_globals.h"
|
| +#include "native_client/src/shared/ppapi_proxy/utility.h"
|
| +#include "ppapi/c/pp_file_info.h"
|
| +#include "ppapi/c/pp_completion_callback.h"
|
| +#include "ppapi/c/pp_errors.h"
|
| +#include "ppapi/c/ppb_file_io.h"
|
| +#include "srpcgen/ppb_rpc.h"
|
| +
|
| +using ppapi_proxy::DebugPrintf;
|
| +using ppapi_proxy::DeleteRemoteCallbackInfo;
|
| +using ppapi_proxy::MakeRemoteCompletionCallback;
|
| +using ppapi_proxy::PPBFileIOInterface;
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Create(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + // input
|
| + PP_Instance instance,
|
| + // output
|
| + PP_Resource* resource) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +
|
| + *resource = PPBFileIOInterface()->Create(instance);
|
| + DebugPrintf("PPB_FileIO::Create: resource=%"NACL_PRIu32"\n", *resource);
|
| +}
|
| +
|
| +// TODO(sanga): Use a caching resource tracker instead of going over the proxy
|
| +// to determine if the given resource is a file io.
|
| +void PpbFileIORpcServer::PPB_FileIO_IsFileIO(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + // inputs
|
| + PP_Resource resource,
|
| + // output
|
| + int32_t* success) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +
|
| + PP_Bool pp_success = PPBFileIOInterface()->IsFileIO(resource);
|
| + DebugPrintf("PPB_FileIO::IsFileIO: pp_success=%d\n", pp_success);
|
| +
|
| + *success = (pp_success == PP_TRUE);
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Open(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + // inputs
|
| + PP_Resource file_io,
|
| + PP_Resource file_ref,
|
| + int32_t open_flags,
|
| + int32_t callback_id,
|
| + // output
|
| + int32_t* pp_error) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| +
|
| + PP_CompletionCallback remote_callback =
|
| + MakeRemoteCompletionCallback(rpc->channel, callback_id);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + *pp_error = PPBFileIOInterface()->Open(
|
| + file_io,
|
| + file_ref,
|
| + open_flags,
|
| + remote_callback);
|
| + DebugPrintf("PPB_FileIO::Open: pp_error=%"NACL_PRId32"\n", *pp_error);
|
| +
|
| + if (*pp_error != PP_OK_COMPLETIONPENDING) // Async error.
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +namespace {
|
| +
|
| +bool IsResultPP_OK(int32_t result) {
|
| + return (result == PP_OK);
|
| +}
|
| +
|
| +nacl_abi_size_t SizeOfPP_FileInfo(int32_t /*query_callback_result*/) {
|
| + return sizeof(PP_FileInfo);
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Query(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + PP_Resource file_io,
|
| + int32_t bytes_to_read,
|
| + int32_t callback_id,
|
| + nacl_abi_size_t* info_bytes, char* info,
|
| + int32_t* pp_error) {
|
| + // Since PPB_FileIO::Query does not complete synchronously, and we use remote
|
| + // completion callback with completion callback table on the untrusted side
|
| + // (see browser_callback.h and plugin_callback.h), so the actual file info
|
| + // parameter is not used.
|
| + UNREFERENCED_PARAMETER(info);
|
| +
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| +
|
| + // TODO(sanga): Drop bytes_to_read for parameters since it is not part of the
|
| + // interface, and just use the buffer size itself.
|
| + CHECK(bytes_to_read == sizeof(PP_FileInfo));
|
| + char* callback_buffer = NULL;
|
| + PP_CompletionCallback remote_callback =
|
| + MakeRemoteCompletionCallback(rpc->channel, callback_id, bytes_to_read,
|
| + &callback_buffer, IsResultPP_OK,
|
| + SizeOfPP_FileInfo);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + // callback_buffer has been assigned to a buffer on the heap, the size
|
| + // of PP_FileInfo.
|
| + PP_FileInfo* file_info = reinterpret_cast<PP_FileInfo*>(callback_buffer);
|
| + *pp_error = PPBFileIOInterface()->Query(file_io, file_info, remote_callback);
|
| + DebugPrintf("PPB_FileIO::Query: pp_error=%"NACL_PRId32"\n", *pp_error);
|
| + CHECK(*pp_error != PP_OK); // Query should not complete synchronously
|
| +
|
| + *info_bytes = 0;
|
| + if (*pp_error != PP_OK_COMPLETIONPENDING) {
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + }
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Touch(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + PP_Resource file_io,
|
| + double last_access_time,
|
| + double last_modified_time,
|
| + int32_t callback_id,
|
| + int32_t* pp_error) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| +
|
| + PP_CompletionCallback remote_callback = MakeRemoteCompletionCallback(
|
| + rpc->channel,
|
| + callback_id);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + *pp_error = PPBFileIOInterface()->Touch(file_io, last_access_time,
|
| + last_modified_time, remote_callback);
|
| + DebugPrintf("PPB_FileIO::Touch: pp_error=%"NACL_PRId32"\n", *pp_error);
|
| + CHECK(*pp_error != PP_OK); // Touch should not complete synchronously
|
| +
|
| + if (*pp_error != PP_OK_COMPLETIONPENDING)
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Read(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + // inputs
|
| + PP_Resource file_io,
|
| + int64_t offset,
|
| + int32_t bytes_to_read,
|
| + int32_t callback_id,
|
| + // outputs
|
| + nacl_abi_size_t* buffer_size,
|
| + char* buffer,
|
| + int32_t* pp_error_or_bytes) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| + CHECK(*buffer_size <=
|
| + static_cast<nacl_abi_size_t>(std::numeric_limits<int32_t>::max()));
|
| + CHECK(*buffer_size == static_cast<nacl_abi_size_t>(bytes_to_read));
|
| +
|
| + char* callback_buffer = NULL;
|
| + PP_CompletionCallback remote_callback =
|
| + MakeRemoteCompletionCallback(rpc->channel, callback_id, bytes_to_read,
|
| + &callback_buffer);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + *pp_error_or_bytes = PPBFileIOInterface()->Read(
|
| + file_io,
|
| + offset,
|
| + callback_buffer,
|
| + bytes_to_read,
|
| + remote_callback);
|
| + DebugPrintf("PPB_FileIO::Read: pp_error_or_bytes=%"NACL_PRId32"\n",
|
| + *pp_error_or_bytes);
|
| + CHECK(*pp_error_or_bytes <= bytes_to_read);
|
| +
|
| + if (*pp_error_or_bytes > 0) { // Bytes read into |callback_buffer|.
|
| + // No callback scheduled.
|
| + CHECK(static_cast<nacl_abi_size_t>(*pp_error_or_bytes) <= *buffer_size);
|
| + *buffer_size = static_cast<nacl_abi_size_t>(*pp_error_or_bytes);
|
| + memcpy(buffer, callback_buffer, *buffer_size);
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + } else if (*pp_error_or_bytes != PP_OK_COMPLETIONPENDING) { // Async error.
|
| + // No callback scheduled.
|
| + *buffer_size = 0;
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + } else {
|
| + // Callback scheduled.
|
| + *buffer_size = 0;
|
| + }
|
| +
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Write(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + PP_Resource file_io,
|
| + int64_t offset,
|
| + nacl_abi_size_t buffer_bytes, char* buffer,
|
| + int32_t bytes_to_write,
|
| + int32_t callback_id,
|
| + int32_t* pp_error_or_bytes) {
|
| + // TODO(sanga): Add comments for the parameters.
|
| +
|
| + CHECK(buffer_bytes <=
|
| + static_cast<nacl_abi_size_t>(std::numeric_limits<int32_t>::max()));
|
| + CHECK(static_cast<nacl_abi_size_t>(bytes_to_write) <= buffer_bytes);
|
| +
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| +
|
| + PP_CompletionCallback remote_callback = MakeRemoteCompletionCallback(
|
| + rpc->channel, callback_id);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + *pp_error_or_bytes = PPBFileIOInterface()->Write(file_io, offset, buffer,
|
| + bytes_to_write,
|
| + remote_callback);
|
| + DebugPrintf("PPB_FileIO::Write: pp_error_or_bytes=%"NACL_PRId32"\n",
|
| + *pp_error_or_bytes);
|
| +
|
| + // Bytes must be written asynchronously.
|
| + if (*pp_error_or_bytes != PP_OK_COMPLETIONPENDING) {
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + }
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_SetLength(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + PP_Resource file_io,
|
| + int64_t length,
|
| + int32_t callback_id,
|
| + int32_t* pp_error) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| +
|
| + PP_CompletionCallback remote_callback = MakeRemoteCompletionCallback(
|
| + rpc->channel, callback_id);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + *pp_error = PPBFileIOInterface()->SetLength(file_io, length, remote_callback);
|
| + DebugPrintf("PPB_FileIO::SetLength: pp_error=%"NACL_PRId32"\n", *pp_error);
|
| + CHECK(*pp_error != PP_OK); // SetLength should not complete synchronously
|
| +
|
| + if (*pp_error != PP_OK_COMPLETIONPENDING)
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Flush(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + PP_Resource file_io,
|
| + int32_t callback_id,
|
| + int32_t* pp_error) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_APP_ERROR;
|
| +
|
| + PP_CompletionCallback remote_callback = MakeRemoteCompletionCallback(
|
| + rpc->channel, callback_id);
|
| + if (NULL == remote_callback.func)
|
| + return;
|
| +
|
| + *pp_error = PPBFileIOInterface()->Flush(file_io, remote_callback);
|
| + DebugPrintf("PPB_FileIO::Flush: pp_error=%"NACL_PRId32"\n", *pp_error);
|
| + CHECK(*pp_error != PP_OK); // Flush should not complete synchronously
|
| +
|
| + if (*pp_error != PP_OK_COMPLETIONPENDING)
|
| + DeleteRemoteCallbackInfo(remote_callback);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| +}
|
| +
|
| +void PpbFileIORpcServer::PPB_FileIO_Close(
|
| + NaClSrpcRpc* rpc,
|
| + NaClSrpcClosure* done,
|
| + PP_Resource file_io) {
|
| + NaClSrpcClosureRunner runner(done);
|
| + rpc->result = NACL_SRPC_RESULT_OK;
|
| + DebugPrintf("PPB_FileIO::Close: file_io=%"NACL_PRIu32"\n", file_io);
|
| + PPBFileIOInterface()->Close(file_io);
|
| +}
|
|
|