Upstream version 10.39.225.0
[platform/framework/web/crosswalk.git] / src / content / renderer / pepper / message_channel.h
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
3 // found in the LICENSE file.
4
5 #ifndef CONTENT_RENDERER_PEPPER_MESSAGE_CHANNEL_H_
6 #define CONTENT_RENDERER_PEPPER_MESSAGE_CHANNEL_H_
7
8 #include <deque>
9 #include <list>
10 #include <map>
11
12 #include "base/basictypes.h"
13 #include "base/memory/weak_ptr.h"
14 #include "gin/handle.h"
15 #include "gin/interceptor.h"
16 #include "gin/wrappable.h"
17 #include "ppapi/proxy/host_dispatcher.h"
18 #include "ppapi/shared_impl/resource.h"
19 #include "third_party/WebKit/public/web/WebSerializedScriptValue.h"
20 #include "v8/include/v8.h"
21
22 struct PP_Var;
23
24 namespace gin {
25 class Arguments;
26 }  // namespace gin
27
28 namespace ppapi {
29 class ScopedPPVar;
30 }  // namespace ppapi
31
32 namespace content {
33
34 class PepperPluginInstanceImpl;
35 class PluginObject;
36
37 // MessageChannel implements bidirectional postMessage functionality, allowing
38 // calls from JavaScript to plugins and vice-versa. See
39 // PPB_Messaging::PostMessage and PPP_Messaging::HandleMessage for more
40 // information.
41 //
42 // Currently, only 1 MessageChannel can exist, to implement postMessage
43 // functionality for the instance interfaces.  In the future, when we create a
44 // MessagePort type in PPAPI, those may be implemented here as well with some
45 // refactoring.
46 //   - Separate message ports won't require the passthrough object.
47 //   - The message target won't be limited to instance, and should support
48 //     either plugin-provided or JS objects.
49 // TODO(dmichael):  Add support for separate MessagePorts.
50 class MessageChannel :
51     public gin::Wrappable<MessageChannel>,
52     public gin::NamedPropertyInterceptor,
53     public ppapi::proxy::HostDispatcher::SyncMessageStatusObserver {
54  public:
55   static gin::WrapperInfo kWrapperInfo;
56
57   // Creates a MessageChannel, returning a pointer to it and sets |result| to
58   // the v8 object which is backed by the message channel. The returned pointer
59   // is only valid as long as the object in |result| is alive.
60   static MessageChannel* Create(PepperPluginInstanceImpl* instance,
61                                 v8::Persistent<v8::Object>* result);
62
63   virtual ~MessageChannel();
64
65   // Called when the instance is deleted. The MessageChannel might outlive the
66   // plugin instance because it is garbage collected.
67   void InstanceDeleted();
68
69   // Post a message to the onmessage handler for this channel's instance
70   // asynchronously.
71   void PostMessageToJavaScript(PP_Var message_data);
72
73   // Messages are queued initially. After the PepperPluginInstanceImpl is ready
74   // to send and handle messages, users of MessageChannel should call
75   // Start().
76   void Start();
77
78   // Set the V8Object to which we should forward any calls which aren't
79   // related to postMessage. Note that this can be empty; it only gets set if
80   // there is a scriptable 'InstanceObject' associated with this channel's
81   // instance.
82   void SetPassthroughObject(v8::Handle<v8::Object> passthrough);
83
84   PepperPluginInstanceImpl* instance() { return instance_; }
85
86   void SetReadOnlyProperty(PP_Var key, PP_Var value);
87
88  private:
89   // Struct for storing the result of a v8 object being converted to a PP_Var.
90   struct VarConversionResult;
91
92   explicit MessageChannel(PepperPluginInstanceImpl* instance);
93
94   // gin::NamedPropertyInterceptor
95   virtual v8::Local<v8::Value> GetNamedProperty(
96       v8::Isolate* isolate,
97       const std::string& property) OVERRIDE;
98   virtual bool SetNamedProperty(v8::Isolate* isolate,
99                                 const std::string& property,
100                                 v8::Local<v8::Value> value) OVERRIDE;
101   virtual std::vector<std::string> EnumerateNamedProperties(
102       v8::Isolate* isolate) OVERRIDE;
103
104   // gin::Wrappable
105   virtual gin::ObjectTemplateBuilder GetObjectTemplateBuilder(
106       v8::Isolate* isolate) OVERRIDE;
107
108   // ppapi::proxy::HostDispatcher::SyncMessageStatusObserver
109   virtual void BeginBlockOnSyncMessage() OVERRIDE;
110   virtual void EndBlockOnSyncMessage() OVERRIDE;
111
112   // Post a message to the plugin's HandleMessage function for this channel's
113   // instance.
114   void PostMessageToNative(gin::Arguments* args);
115   // Post a message to the plugin's HandleBlocking Message function for this
116   // channel's instance synchronously, and return a result.
117   void PostBlockingMessageToNative(gin::Arguments* args);
118
119   // Post a message to the onmessage handler for this channel's instance
120   // synchronously.  This is used by PostMessageToJavaScript.
121   void PostMessageToJavaScriptImpl(
122       const blink::WebSerializedScriptValue& message_data);
123
124   PluginObject* GetPluginObject(v8::Isolate* isolate);
125
126   void EnqueuePluginMessage(v8::Handle<v8::Value> v8_value);
127
128   void FromV8ValueComplete(VarConversionResult* result_holder,
129                            const ppapi::ScopedPPVar& result_var,
130                            bool success);
131
132   // Drain the queue of messages that are going to the plugin. All "completed"
133   // messages at the head of the queue will be sent; any messages awaiting
134   // conversion as well as messages after that in the queue will not be sent.
135   void DrainCompletedPluginMessages();
136   // Drain the queue of messages that are going to JavaScript.
137   void DrainJSMessageQueue();
138   // PostTask to call DrainJSMessageQueue() soon. Use this when you want to
139   // send the messages, but can't immediately (e.g., because the instance is
140   // not ready or JavaScript is on the stack).
141   void DrainJSMessageQueueSoon();
142
143   void UnregisterSyncMessageStatusObserver();
144
145   PepperPluginInstanceImpl* instance_;
146
147   // We pass all non-postMessage calls through to the passthrough_object_.
148   // This way, a plugin can use PPB_Class or PPP_Class_Deprecated and also
149   // postMessage.  This is necessary to support backwards-compatibility, and
150   // also trusted plugins for which we will continue to support synchronous
151   // scripting.
152   v8::Persistent<v8::Object> passthrough_object_;
153
154   enum MessageQueueState {
155     WAITING_TO_START,  // Waiting for Start() to be called. Queue messages.
156     QUEUE_MESSAGES,  // Queue messages temporarily.
157     SEND_DIRECTLY,   // Post messages directly.
158   };
159
160   // This queue stores values being posted to JavaScript.
161   std::deque<blink::WebSerializedScriptValue> js_message_queue_;
162   MessageQueueState js_message_queue_state_;
163   // When the renderer is sending a blocking message to the plugin, we will
164   // queue Plugin->JS messages temporarily to avoid re-entering JavaScript. This
165   // counts how many blocking renderer->plugin messages are on the stack so that
166   // we only begin sending messages to JavaScript again when the depth reaches
167   // zero.
168   int blocking_message_depth_;
169
170   // This queue stores vars that are being sent to the plugin. Because
171   // conversion can happen asynchronously for object types, the queue stores
172   // the var until all previous vars have been converted and sent. This
173   // preserves the order in which JS->plugin messages are processed.
174   //
175   // Note we rely on raw VarConversionResult* pointers remaining valid after
176   // calls to push_back or pop_front; hence why we're using list. (deque would
177   // probably also work, but is less clearly specified).
178   std::list<VarConversionResult> plugin_message_queue_;
179   MessageQueueState plugin_message_queue_state_;
180
181   std::map<std::string, ppapi::ScopedPPVar> internal_named_properties_;
182
183   // A callback to invoke at shutdown to ensure we unregister ourselves as
184   // Observers for sync messages.
185   base::Closure unregister_observer_callback_;
186
187   // This is used to ensure pending tasks will not fire after this object is
188   // destroyed.
189   base::WeakPtrFactory<MessageChannel> weak_ptr_factory_;
190
191   DISALLOW_COPY_AND_ASSIGN(MessageChannel);
192 };
193
194 }  // namespace content
195
196 #endif  // CONTENT_RENDERER_PEPPER_MESSAGE_CHANNEL_H_