15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved. 25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be 35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file. 45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/proxy/ppb_buffer_proxy.h" 65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <vector> 85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/logging.h" 105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "build/build_config.h" 115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_completion_callback.h" 125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_errors.h" 135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_resource.h" 145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/dev/ppb_buffer_dev.h" 155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/proxy/host_dispatcher.h" 165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/proxy/plugin_dispatcher.h" 175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/proxy/ppapi_messages.h" 185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/thunk/enter.h" 195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/thunk/resource_creation_api.h" 205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/thunk/thunk.h" 215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace ppapi { 235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace proxy { 245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)Buffer::Buffer(const HostResource& resource, 265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const base::SharedMemoryHandle& shm_handle, 275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) uint32_t size) 285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) : Resource(OBJECT_IS_PROXY, resource), 295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) shm_(shm_handle, false), 305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) size_(size), 315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) map_count_(0) { 325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)Buffer::~Buffer() { 355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Unmap(); 365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)thunk::PPB_Buffer_API* Buffer::AsPPB_Buffer_API() { 395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return this; 405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PP_Bool Buffer::Describe(uint32_t* size_in_bytes) { 435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *size_in_bytes = size_; 445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return PP_TRUE; 455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PP_Bool Buffer::IsMapped() { 485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return PP_FromBool(map_count_ > 0); 495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void* Buffer::Map() { 525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (map_count_++ == 0) 535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) shm_.Map(size_); 545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return shm_.memory(); 555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void Buffer::Unmap() { 585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (--map_count_ == 0) 595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) shm_.Unmap(); 605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)int32_t Buffer::GetSharedMemory(int* out_handle) { 632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) NOTREACHED(); 642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return PP_ERROR_NOTSUPPORTED; 652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PPB_Buffer_Proxy::PPB_Buffer_Proxy(Dispatcher* dispatcher) 685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) : InterfaceProxy(dispatcher) { 695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PPB_Buffer_Proxy::~PPB_Buffer_Proxy() { 725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PP_Resource PPB_Buffer_Proxy::CreateProxyResource(PP_Instance instance, 765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) uint32_t size) { 775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginDispatcher* dispatcher = PluginDispatcher::GetForInstance(instance); 785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!dispatcher) 795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return 0; 805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HostResource result; 825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ppapi::proxy::SerializedHandle shm_handle; 835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) dispatcher->Send(new PpapiHostMsg_PPBBuffer_Create( 845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) API_ID_PPB_BUFFER, instance, size, &result, &shm_handle)); 855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (result.is_null() || !shm_handle.IsHandleValid() || 865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !shm_handle.is_shmem()) 875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return 0; 885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return AddProxyResource(result, shm_handle.shmem(), size); 905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PP_Resource PPB_Buffer_Proxy::AddProxyResource( 945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const HostResource& resource, 955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::SharedMemoryHandle shm_handle, 965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) uint32_t size) { 975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return (new Buffer(resource, shm_handle, size))->GetReference(); 985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool PPB_Buffer_Proxy::OnMessageReceived(const IPC::Message& msg) { 1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool handled = true; 1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IPC_BEGIN_MESSAGE_MAP(PPB_Buffer_Proxy, msg) 1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IPC_MESSAGE_HANDLER(PpapiHostMsg_PPBBuffer_Create, OnMsgCreate) 1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IPC_MESSAGE_UNHANDLED(handled = false) 1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IPC_END_MESSAGE_MAP() 1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(brettw) handle bad messages! 1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return handled; 1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PPB_Buffer_Proxy::OnMsgCreate( 1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PP_Instance instance, 1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) uint32_t size, 1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HostResource* result_resource, 1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ppapi::proxy::SerializedHandle* result_shm_handle) { 1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Overwritten below on success. 1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) result_shm_handle->set_null_shmem(); 1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HostDispatcher* dispatcher = HostDispatcher::GetForInstance(instance); 1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!dispatcher) 1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 1202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!dispatcher->permissions().HasPermission(ppapi::PERMISSION_DEV)) 1212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return; 1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) thunk::EnterResourceCreation enter(instance); 1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (enter.failed()) 1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PP_Resource local_buffer_resource = enter.functions()->CreateBuffer(instance, 1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) size); 1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (local_buffer_resource == 0) 1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) thunk::EnterResourceNoLock<thunk::PPB_Buffer_API> trusted_buffer( 1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) local_buffer_resource, false); 1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (trusted_buffer.failed()) 1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int local_fd; 1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (trusted_buffer.object()->GetSharedMemory(&local_fd) != PP_OK) 1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) result_resource->SetHostResource(instance, local_buffer_resource); 1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(piman/brettw): Change trusted interface to return a PP_FileHandle, 1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // those casts are ugly. 1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::PlatformFile platform_file = 1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_WIN) 1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) reinterpret_cast<HANDLE>(static_cast<intptr_t>(local_fd)); 1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(OS_POSIX) 1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) local_fd; 1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#else 1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) #error Not implemented. 1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) result_shm_handle->set_shmem( 1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) dispatcher->ShareHandleWithRemote(platform_file, false), size); 1535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} // namespace proxy 1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} // namespace ppapi 157