// 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. #ifndef PPAPI_PROXY_RAW_VAR_DATA_H_ #define PPAPI_PROXY_RAW_VAR_DATA_H_ #include <vector> #include "base/callback.h" #include "base/memory/scoped_ptr.h" #include "base/memory/scoped_vector.h" #include "ppapi/c/pp_instance.h" #include "ppapi/c/pp_var.h" #include "ppapi/proxy/ppapi_param_traits.h" #include "ppapi/proxy/ppapi_proxy_export.h" #include "ppapi/proxy/serialized_handle.h" class PickleIterator; namespace IPC { class Message; } namespace ppapi { namespace proxy { class RawVarData; typedef base::Callback<void(IPC::Message*, const SerializedHandle&)> HandleWriter; // Contains the data associated with a graph of connected PP_Vars. Useful for // serializing/deserializing a graph of PP_Vars. First we compute the transitive // closure of the given PP_Var to find all PP_Vars which are referenced by that // var. A RawVarData object is created for each of these vars. We then write // data contained in each RawVarData to the message. The format looks like this: // idx | size | (number of vars in the graph) // 0 | var type | // | var data | // 1 | var type | // | var data | // 2 | var type | // | var data | // | .... | // // Vars that reference other vars (such as Arrays or Dictionaries) use indices // into the message to denote which PP_Var is pointed to. class PPAPI_PROXY_EXPORT RawVarDataGraph { public: // Construct a RawVarDataGraph from a given root PP_Var. A null pointer // is returned upon failure. static scoped_ptr<RawVarDataGraph> Create(const PP_Var& var, PP_Instance instance); // Constructs an empty RawVarDataGraph. RawVarDataGraph(); ~RawVarDataGraph(); // Construct a new PP_Var from the graph. All of the PP_Vars referenced by // the returned PP_Var are also constructed. Each PP_Var created has a // ref-count equal to the number of references it has in the graph of vars. // The returned var (the "root") has one additional reference. PP_Var CreatePPVar(PP_Instance instance); // Write the graph to a message using the given HandleWriter. void Write(IPC::Message* m, const HandleWriter& handle_writer); // Create a RawVarDataGraph from the given message. static scoped_ptr<RawVarDataGraph> Read(const IPC::Message* m, PickleIterator* iter); // Returns a vector of SerializedHandles associated with this RawVarDataGraph. // Ownership of the pointers remains with the elements of the RawVarDataGraph. std::vector<SerializedHandle*> GetHandles(); // Sets the threshold size at which point we switch from transmitting // array buffers in IPC messages to using shared memory. This is only used // for testing purposes where we need to transmit small buffers using shmem // (in order to have fast tests). static void SetMinimumArrayBufferSizeForShmemForTest(uint32 threshold); // A list of the nodes in the graph. ScopedVector<RawVarData> data_; }; // Abstract base class for the data contained in a PP_Var. class RawVarData { public: // Create a new, empty RawVarData for the given type. static RawVarData* Create(PP_VarType type); RawVarData(); virtual ~RawVarData(); // Returns the type of the PP_Var represented by the RawVarData. virtual PP_VarType Type() = 0; // Initializes a RawVarData from a PP_Var. Returns true on success. virtual bool Init(const PP_Var& var, PP_Instance instance) = 0; // Create a PP_Var from the raw data contained in this object. virtual PP_Var CreatePPVar(PP_Instance instance) = 0; // Some PP_Vars may require 2-step initialization. For example, they may // reference other PP_Vars which had not yet been created when |CreatePPVar| // was called. The original var created with |CreatePPVar| is passed back in, // along with the graph it is a part of to be initialized. virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) = 0; // Writes the RawVarData to a message. virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) = 0; // Reads the RawVarData from a message. Returns true on success. virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) = 0; // Returns a SerializedHandle associated with this RawVarData or NULL if none // exists. Ownership of the pointer remains with the RawVarData. virtual SerializedHandle* GetHandle(); bool initialized() { return initialized_; } protected: bool initialized_; }; // A RawVarData class for PP_Vars which are value types. class BasicRawVarData : public RawVarData { public: BasicRawVarData(); virtual ~BasicRawVarData(); // RawVarData implementation. virtual PP_VarType Type() OVERRIDE; virtual bool Init(const PP_Var& var, PP_Instance instance) OVERRIDE; virtual PP_Var CreatePPVar(PP_Instance instance) OVERRIDE; virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) OVERRIDE; virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) OVERRIDE; virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) OVERRIDE; private: PP_Var var_; }; // A RawVarData class for string PP_Vars. class StringRawVarData : public RawVarData { public: StringRawVarData(); virtual ~StringRawVarData(); // RawVarData implementation. virtual PP_VarType Type() OVERRIDE; virtual bool Init(const PP_Var& var, PP_Instance instance) OVERRIDE; virtual PP_Var CreatePPVar(PP_Instance instance) OVERRIDE; virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) OVERRIDE; virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) OVERRIDE; virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) OVERRIDE; private: // The data in the string. std::string data_; }; // A RawVarData class for array buffer PP_Vars. class ArrayBufferRawVarData : public RawVarData { public: // Enum for array buffer message types. enum ShmemType { ARRAY_BUFFER_NO_SHMEM, ARRAY_BUFFER_SHMEM_HOST, ARRAY_BUFFER_SHMEM_PLUGIN, }; ArrayBufferRawVarData(); virtual ~ArrayBufferRawVarData(); // RawVarData implementation. virtual PP_VarType Type() OVERRIDE; virtual bool Init(const PP_Var& var, PP_Instance instance) OVERRIDE; virtual PP_Var CreatePPVar(PP_Instance instance) OVERRIDE; virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) OVERRIDE; virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) OVERRIDE; virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) OVERRIDE; virtual SerializedHandle* GetHandle() OVERRIDE; private: // The type of the storage underlying the array buffer. ShmemType type_; // The data in the buffer. Valid for |type_| == ARRAY_BUFFER_NO_SHMEM. std::string data_; // Host shmem handle. Valid for |type_| == ARRAY_BUFFER_SHMEM_HOST. int host_shm_handle_id_; // Plugin shmem handle. Valid for |type_| == ARRAY_BUFFER_SHMEM_PLUGIN. SerializedHandle plugin_shm_handle_; }; // A RawVarData class for array PP_Vars. class ArrayRawVarData : public RawVarData { public: ArrayRawVarData(); virtual ~ArrayRawVarData(); void AddChild(size_t element); // RawVarData implementation. virtual PP_VarType Type() OVERRIDE; virtual bool Init(const PP_Var& var, PP_Instance instance) OVERRIDE; virtual PP_Var CreatePPVar(PP_Instance instance) OVERRIDE; virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) OVERRIDE; virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) OVERRIDE; virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) OVERRIDE; private: std::vector<size_t> children_; }; // A RawVarData class for dictionary PP_Vars. class DictionaryRawVarData : public RawVarData { public: DictionaryRawVarData(); virtual ~DictionaryRawVarData(); void AddChild(const std::string& key, size_t value); // RawVarData implementation. virtual PP_VarType Type() OVERRIDE; virtual bool Init(const PP_Var& var, PP_Instance instance) OVERRIDE; virtual PP_Var CreatePPVar(PP_Instance instance) OVERRIDE; virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) OVERRIDE; virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) OVERRIDE; virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) OVERRIDE; private: std::vector<std::pair<std::string, size_t> > children_; }; // A RawVarData class for resource PP_Vars. // This class does not hold a reference on the PP_Resource that is being // serialized. If sending a resource from the plugin to the host, the plugin // should not release the ResourceVar before sending the serialized message to // the host, and the host should immediately consume the ResourceVar before // processing further messages. class ResourceRawVarData : public RawVarData { public: ResourceRawVarData(); virtual ~ResourceRawVarData(); // RawVarData implementation. virtual PP_VarType Type() OVERRIDE; virtual bool Init(const PP_Var& var, PP_Instance instance) OVERRIDE; virtual PP_Var CreatePPVar(PP_Instance instance) OVERRIDE; virtual void PopulatePPVar(const PP_Var& var, const std::vector<PP_Var>& graph) OVERRIDE; virtual void Write(IPC::Message* m, const HandleWriter& handle_writer) OVERRIDE; virtual bool Read(PP_VarType type, const IPC::Message* m, PickleIterator* iter) OVERRIDE; private: // Resource ID in the plugin. If one has not yet been created, this is 0. // This is a borrowed reference; the resource's refcount is not incremented. PP_Resource pp_resource_; // Pending resource host ID in the renderer. int pending_renderer_host_id_; // Pending resource host ID in the browser. int pending_browser_host_id_; // A message containing information about how to create a plugin-side // resource. The message type will vary based on the resource type, and will // usually contain a pending resource host ID, and other required information. // If the resource was created directly, this is NULL. scoped_ptr<IPC::Message> creation_message_; }; } // namespace proxy } // namespace ppapi #endif // PPAPI_PROXY_RAW_VAR_DATA_H_