• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 PPAPI_PROXY_RESOURCE_MESSAGE_PARAMS_H_
6 #define PPAPI_PROXY_RESOURCE_MESSAGE_PARAMS_H_
7 
8 #include <vector>
9 
10 #include "base/memory/ref_counted.h"
11 #include "ipc/ipc_message_utils.h"
12 #include "ppapi/c/pp_resource.h"
13 #include "ppapi/proxy/ppapi_proxy_export.h"
14 #include "ppapi/proxy/serialized_handle.h"
15 
16 namespace ppapi {
17 namespace proxy {
18 
19 // Common parameters for resource call and reply params structures below.
20 class PPAPI_PROXY_EXPORT ResourceMessageParams {
21  public:
22   virtual ~ResourceMessageParams();
23 
pp_resource()24   PP_Resource pp_resource() const { return pp_resource_; }
sequence()25   int32_t sequence() const { return sequence_; }
26 
27   // Note that the caller doesn't take ownership of the returned handles.
handles()28   const std::vector<SerializedHandle>& handles() const {
29     return handles_->data();
30   }
31 
32   // Makes ResourceMessageParams leave its handles open, even if they weren't
33   // taken using a Take.* function. After this call, no Take.* calls are
34   // allowed.
35   void ConsumeHandles() const;
36 
37   // Returns the handle at the given index if it exists and is of the given
38   // type. The corresponding slot in the list is set to an invalid handle.
39   // If the index doesn't exist or the handle isn't of the given type, returns
40   // an invalid handle.
41   // Note that the caller is responsible for closing the returned handle, if it
42   // is valid.
43   SerializedHandle TakeHandleOfTypeAtIndex(size_t index,
44                                            SerializedHandle::Type type) const;
45 
46   // Helper functions to return shared memory, socket or file handles passed in
47   // the params struct.
48   // If the index has a valid handle of the given type, it will be placed in the
49   // output parameter, the corresponding slot in the list will be set to an
50   // invalid handle, and the function will return true. If the handle doesn't
51   // exist or is a different type, the functions will return false and the
52   // output parameter will be untouched.
53   //
54   // Note: 1) the handle could still be a "null" or invalid handle of the right
55   //          type and the functions will succeed.
56   //       2) the caller is responsible for closing the returned handle, if it
57   //          is valid.
58   bool TakeSharedMemoryHandleAtIndex(size_t index,
59                                      base::SharedMemoryHandle* handle) const;
60   bool TakeSocketHandleAtIndex(size_t index,
61                                IPC::PlatformFileForTransit* handle) const;
62   bool TakeFileHandleAtIndex(size_t index,
63                              IPC::PlatformFileForTransit* handle) const;
64   void TakeAllSharedMemoryHandles(
65       std::vector<base::SharedMemoryHandle>* handles) const;
66 
67   // Appends the given handle to the list of handles sent with the call or
68   // reply.
69   void AppendHandle(const SerializedHandle& handle) const;
70 
71  protected:
72   ResourceMessageParams();
73   ResourceMessageParams(PP_Resource resource, int32_t sequence);
74 
75   virtual void Serialize(IPC::Message* msg) const;
76   virtual bool Deserialize(const IPC::Message* msg, PickleIterator* iter);
77 
78   // Writes everything except the handles to |msg|.
79   void WriteHeader(IPC::Message* msg) const;
80   // Writes the handles to |msg|.
81   void WriteHandles(IPC::Message* msg) const;
82   // Matching deserialize helpers.
83   bool ReadHeader(const IPC::Message* msg, PickleIterator* iter);
84   bool ReadHandles(const IPC::Message* msg, PickleIterator* iter);
85 
86  private:
87   class SerializedHandles
88       : public base::RefCountedThreadSafe<SerializedHandles> {
89    public:
90     SerializedHandles();
91     ~SerializedHandles();
92 
set_should_close(bool value)93     void set_should_close(bool value) { should_close_ = value; }
data()94     std::vector<SerializedHandle>& data() { return data_; }
95 
96    private:
97     friend class base::RefCountedThreadSafe<SerializedHandles>;
98 
99     // Whether the handles stored in |data_| should be closed when this object
100     // goes away.
101     //
102     // It is set to true by ResourceMessageParams::Deserialize(), so that the
103     // receiving side of the params (the host side for
104     // ResourceMessageCallParams; the plugin side for
105     // ResourceMessageReplyParams) will close those handles which haven't been
106     // taken using any of the Take*() methods.
107     bool should_close_;
108     std::vector<SerializedHandle> data_;
109   };
110 
111   PP_Resource pp_resource_;
112 
113   // Identifier for this message. Sequence numbers are quasi-unique within a
114   // resource, but will overlap between different resource objects.
115   //
116   // If you send a lot of messages, the ID may wrap around. This is OK. All IDs
117   // are valid and 0 and -1 aren't special, so those cases won't confuse us.
118   // In practice, if you send more than 4 billion messages for a resource, the
119   // old ones will be long gone and there will be no collisions.
120   //
121   // If there is a malicious plugin (or exceptionally bad luck) that causes a
122   // wraparound and collision the worst that will happen is that we can get
123   // confused between different callbacks. But since these can only cause
124   // confusion within the plugin and within callbacks on the same resource,
125   // there shouldn't be a security problem.
126   int32_t sequence_;
127 
128   // A list of all handles transferred in the message. Handles go here so that
129   // the NaCl adapter can extract them generally when it rewrites them to
130   // go between Windows and NaCl (Posix) apps.
131   // TODO(yzshen): Mark it as mutable so that we can take/append handles using a
132   // const reference. We need to change all the callers and make it not mutable.
133   mutable scoped_refptr<SerializedHandles> handles_;
134 };
135 
136 // Parameters common to all ResourceMessage "Call" requests.
137 class PPAPI_PROXY_EXPORT ResourceMessageCallParams
138     : public ResourceMessageParams {
139  public:
140   ResourceMessageCallParams();
141   ResourceMessageCallParams(PP_Resource resource, int32_t sequence);
142   virtual ~ResourceMessageCallParams();
143 
set_has_callback()144   void set_has_callback() { has_callback_ = true; }
has_callback()145   bool has_callback() const { return has_callback_; }
146 
147   virtual void Serialize(IPC::Message* msg) const OVERRIDE;
148   virtual bool Deserialize(const IPC::Message* msg,
149                            PickleIterator* iter) OVERRIDE;
150 
151  private:
152   bool has_callback_;
153 };
154 
155 // Parameters common to all ResourceMessage "Reply" requests.
156 class PPAPI_PROXY_EXPORT ResourceMessageReplyParams
157     : public ResourceMessageParams {
158  public:
159   ResourceMessageReplyParams();
160   ResourceMessageReplyParams(PP_Resource resource, int32_t sequence);
161   virtual ~ResourceMessageReplyParams();
162 
set_result(int32_t r)163   void set_result(int32_t r) { result_ = r; }
result()164   int32_t result() const { return result_; }
165 
166   virtual void Serialize(IPC::Message* msg) const OVERRIDE;
167   virtual bool Deserialize(const IPC::Message* msg,
168                            PickleIterator* iter) OVERRIDE;
169 
170   // Writes everything except the handles to |msg|.
171   void WriteReplyHeader(IPC::Message* msg) const;
172 
173  private:
174   // Pepper "result code" for the callback.
175   int32_t result_;
176 };
177 
178 }  // namespace proxy
179 }  // namespace ppapi
180 
181 namespace IPC {
182 
183 template <> struct PPAPI_PROXY_EXPORT
184 ParamTraits<ppapi::proxy::ResourceMessageCallParams> {
185   typedef ppapi::proxy::ResourceMessageCallParams param_type;
186   static void Write(Message* m, const param_type& p) {
187     p.Serialize(m);
188   }
189   static bool Read(const Message* m, PickleIterator* iter, param_type* r) {
190     return r->Deserialize(m, iter);
191   }
192   static void Log(const param_type& p, std::string* l) {
193   }
194 };
195 
196 template <> struct PPAPI_PROXY_EXPORT
197 ParamTraits<ppapi::proxy::ResourceMessageReplyParams> {
198   typedef ppapi::proxy::ResourceMessageReplyParams param_type;
199   static void Write(Message* m, const param_type& p) {
200     p.Serialize(m);
201   }
202   static bool Read(const Message* m, PickleIterator* iter, param_type* r) {
203     return r->Deserialize(m, iter);
204   }
205   static void Log(const param_type& p, std::string* l) {
206   }
207 };
208 
209 }  // namespace IPC
210 
211 #endif  // PPAPI_PROXY_RESOURCE_MESSAGE_PARAMS_H_
212