Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(917)

Side by Side Diff: chrome/nacl/nacl_ipc_adapter.cc

Issue 11894003: PPAPI/NaCl: Move handle extraction code to ppapi/proxy (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: merge Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « chrome/nacl/nacl_ipc_adapter.h ('k') | ppapi/ppapi_ipc.gypi » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 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 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "chrome/nacl/nacl_ipc_adapter.h" 5 #include "chrome/nacl/nacl_ipc_adapter.h"
6 6
7 #include <limits.h> 7 #include <limits.h>
8 #include <string.h> 8 #include <string.h>
9 9
10 #include "base/basictypes.h" 10 #include "base/basictypes.h"
11 #include "base/bind.h" 11 #include "base/bind.h"
12 #include "base/location.h" 12 #include "base/location.h"
13 #include "base/memory/scoped_ptr.h" 13 #include "base/memory/scoped_ptr.h"
14 #include "base/shared_memory.h" 14 #include "base/shared_memory.h"
15 #include "build/build_config.h" 15 #include "build/build_config.h"
16 #include "ipc/ipc_channel.h" 16 #include "ipc/ipc_channel.h"
17 #include "ipc/ipc_message_macros.h"
18 #include "ipc/ipc_platform_file.h" 17 #include "ipc/ipc_platform_file.h"
19 #include "native_client/src/trusted/desc/nacl_desc_custom.h" 18 #include "native_client/src/trusted/desc/nacl_desc_custom.h"
20 #include "native_client/src/trusted/desc/nacl_desc_wrapper.h" 19 #include "native_client/src/trusted/desc/nacl_desc_wrapper.h"
21 #include "ppapi/proxy/ppapi_messages.h" 20 #include "ppapi/proxy/ppapi_messages.h"
22 #include "ppapi/proxy/resource_message_params.h" 21 #include "ppapi/proxy/serialized_handle.h"
23 22
24 namespace { 23 namespace {
25 24
26 enum BufferSizeStatus { 25 enum BufferSizeStatus {
27 // The buffer contains a full message with no extra bytes. 26 // The buffer contains a full message with no extra bytes.
28 MESSAGE_IS_COMPLETE, 27 MESSAGE_IS_COMPLETE,
29 28
30 // The message doesn't fit and the buffer contains only some of it. 29 // The message doesn't fit and the buffer contains only some of it.
31 MESSAGE_IS_TRUNCATED, 30 MESSAGE_IS_TRUNCATED,
32 31
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after
84 funcs.SendMsg = NaClDescCustomSendMsg; 83 funcs.SendMsg = NaClDescCustomSendMsg;
85 funcs.RecvMsg = NaClDescCustomRecvMsg; 84 funcs.RecvMsg = NaClDescCustomRecvMsg;
86 // NaClDescMakeCustomDesc gives us a reference on the returned NaClDesc. 85 // NaClDescMakeCustomDesc gives us a reference on the returned NaClDesc.
87 return NaClDescMakeCustomDesc(new DescThunker(adapter), &funcs); 86 return NaClDescMakeCustomDesc(new DescThunker(adapter), &funcs);
88 } 87 }
89 88
90 void DeleteChannel(IPC::Channel* channel) { 89 void DeleteChannel(IPC::Channel* channel) {
91 delete channel; 90 delete channel;
92 } 91 }
93 92
94 // TODO(dmichael): Move all this handle conversion code to ppapi/proxy.
95 // crbug.com/165201
96 void WriteHandle(int handle_index,
97 const ppapi::proxy::SerializedHandle& handle,
98 IPC::Message* message) {
99 ppapi::proxy::SerializedHandle::WriteHeader(handle.header(), message);
100
101 // Now write the handle itself in POSIX style.
102 message->WriteBool(true); // valid == true
103 message->WriteInt(handle_index);
104 }
105
106 typedef std::vector<ppapi::proxy::SerializedHandle> Handles;
107
108 // We define overload for catching SerializedHandles, so that we can share
109 // them correctly to the untrusted side, and another for handling all other
110 // parameters. See ConvertHandlesImpl for how these get used.
111 void ConvertHandlesInParam(const ppapi::proxy::SerializedHandle& handle,
112 Handles* handles,
113 IPC::Message* msg,
114 int* handle_index) {
115 handles->push_back(handle);
116 if (msg)
117 WriteHandle((*handle_index)++, handle, msg);
118 }
119
120 // For PpapiMsg_ResourceReply and the reply to PpapiHostMsg_ResourceSyncCall,
121 // the handles are carried inside the ResourceMessageReplyParams.
122 // NOTE: We only translate handles from host->NaCl. The only kind of
123 // ResourceMessageParams that travels this direction is
124 // ResourceMessageReplyParams, so that's the only one we need to handle.
125 void ConvertHandlesInParam(
126 const ppapi::proxy::ResourceMessageReplyParams& params,
127 Handles* handles,
128 IPC::Message* msg,
129 int* handle_index) {
130 // First, if we need to rewrite the message parameters, write everything
131 // before the handles (there's nothing after the handles).
132 if (msg) {
133 params.WriteReplyHeader(msg);
134 // IPC writes the vector length as an int before the contents of the
135 // vector.
136 msg->WriteInt(static_cast<int>(params.handles().size()));
137 }
138 for (Handles::const_iterator iter = params.handles().begin();
139 iter != params.handles().end();
140 ++iter) {
141 // ConvertHandle will write each handle to |msg|, if necessary.
142 ConvertHandlesInParam(*iter, handles, msg, handle_index);
143 }
144 // Tell ResourceMessageReplyParams that we have taken the handles, so it
145 // shouldn't close them. The NaCl runtime will take ownership of them.
146 params.ConsumeHandles();
147 }
148
149 // This overload is to catch all types other than SerializedHandle or
150 // ResourceMessageReplyParams. On Windows, |msg| will be a valid pointer, and we
151 // must write |param| to it.
152 template <class T>
153 void ConvertHandlesInParam(const T& param,
154 Handles* /* handles */,
155 IPC::Message* msg,
156 int* /* handle_index */) {
157 // It's not a handle, so just write to the output message, if necessary.
158 if (msg)
159 IPC::WriteParam(msg, param);
160 }
161
162 // These just break apart the given tuple and run ConvertHandle over each param.
163 // The idea is to extract any handles in the tuple, while writing all data to
164 // msg (if msg is valid). The msg will only be valid on Windows, where we need
165 // to re-write all of the message parameters, writing the handles in POSIX style
166 // for NaCl.
167 template <class A>
168 void ConvertHandlesImpl(const Tuple1<A>& t1, Handles* handles,
169 IPC::Message* msg) {
170 int handle_index = 0;
171 ConvertHandlesInParam(t1.a, handles, msg, &handle_index);
172 }
173 template <class A, class B>
174 void ConvertHandlesImpl(const Tuple2<A, B>& t1, Handles* handles,
175 IPC::Message* msg) {
176 int handle_index = 0;
177 ConvertHandlesInParam(t1.a, handles, msg, &handle_index);
178 ConvertHandlesInParam(t1.b, handles, msg, &handle_index);
179 }
180 template <class A, class B, class C>
181 void ConvertHandlesImpl(const Tuple3<A, B, C>& t1, Handles* handles,
182 IPC::Message* msg) {
183 int handle_index = 0;
184 ConvertHandlesInParam(t1.a, handles, msg, &handle_index);
185 ConvertHandlesInParam(t1.b, handles, msg, &handle_index);
186 ConvertHandlesInParam(t1.c, handles, msg, &handle_index);
187 }
188 template <class A, class B, class C, class D>
189 void ConvertHandlesImpl(const Tuple4<A, B, C, D>& t1, Handles* handles,
190 IPC::Message* msg) {
191 int handle_index = 0;
192 ConvertHandlesInParam(t1.a, handles, msg, &handle_index);
193 ConvertHandlesInParam(t1.b, handles, msg, &handle_index);
194 ConvertHandlesInParam(t1.c, handles, msg, &handle_index);
195 ConvertHandlesInParam(t1.d, handles, msg, &handle_index);
196 }
197
198 template <class MessageType>
199 class HandleConverter {
200 public:
201 explicit HandleConverter(const IPC::Message* msg)
202 : msg_(static_cast<const MessageType*>(msg)) {
203 }
204 bool ConvertMessage(Handles* handles, IPC::Message* out_msg) {
205 typename TupleTypes<typename MessageType::Schema::Param>::ValueTuple params;
206 if (!MessageType::Read(msg_, &params))
207 return false;
208 ConvertHandlesImpl(params, handles, out_msg);
209 return true;
210 }
211
212 bool ConvertReply(Handles* handles, IPC::SyncMessage* out_msg) {
213 typename TupleTypes<typename MessageType::Schema::ReplyParam>::ValueTuple
214 params;
215 if (!MessageType::ReadReplyParam(msg_, &params))
216 return false;
217 // If we need to rewrite the message (i.e., on Windows), we need to make
218 // sure we write the message id first.
219 if (out_msg) {
220 out_msg->set_reply();
221 int id = IPC::SyncMessage::GetMessageId(*msg_);
222 out_msg->WriteInt(id);
223 }
224 ConvertHandlesImpl(params, handles, out_msg);
225 return true;
226 }
227 // TODO(dmichael): Add ConvertSyncMessage for outgoing sync messages, if we
228 // ever pass handles in one of those.
229
230 private:
231 const MessageType* msg_;
232 };
233
234 } // namespace 93 } // namespace
235 94
236 class NaClIPCAdapter::RewrittenMessage 95 class NaClIPCAdapter::RewrittenMessage
237 : public base::RefCounted<RewrittenMessage> { 96 : public base::RefCounted<RewrittenMessage> {
238 public: 97 public:
239 RewrittenMessage(); 98 RewrittenMessage();
240 99
241 bool is_consumed() const { return data_read_cursor_ == data_len_; } 100 bool is_consumed() const { return data_read_cursor_ == data_len_; }
242 101
243 void SetData(const NaClIPCAdapter::NaClMessageHeader& header, 102 void SetData(const NaClIPCAdapter::NaClMessageHeader& header,
(...skipping 215 matching lines...) Expand 10 before | Expand all | Expand 10 after
459 NaClDesc* NaClIPCAdapter::MakeNaClDesc() { 318 NaClDesc* NaClIPCAdapter::MakeNaClDesc() {
460 return MakeNaClDescCustom(this); 319 return MakeNaClDescCustom(this);
461 } 320 }
462 321
463 #if defined(OS_POSIX) 322 #if defined(OS_POSIX)
464 int NaClIPCAdapter::TakeClientFileDescriptor() { 323 int NaClIPCAdapter::TakeClientFileDescriptor() {
465 return io_thread_data_.channel_->TakeClientFileDescriptor(); 324 return io_thread_data_.channel_->TakeClientFileDescriptor();
466 } 325 }
467 #endif 326 #endif
468 327
469 #define CASE_FOR_MESSAGE(MESSAGE_TYPE) \
470 case MESSAGE_TYPE::ID: { \
471 HandleConverter<MESSAGE_TYPE> extractor(&msg); \
472 if (!extractor.ConvertMessage(&handles, new_msg_ptr)) \
473 return false; \
474 break; \
475 }
476 #define CASE_FOR_REPLY(MESSAGE_TYPE) \
477 case MESSAGE_TYPE::ID: { \
478 HandleConverter<MESSAGE_TYPE> extractor(&msg); \
479 if (!extractor.ConvertReply( \
480 &handles, \
481 static_cast<IPC::SyncMessage*>(new_msg_ptr))) \
482 return false; \
483 break; \
484 }
485
486 bool NaClIPCAdapter::OnMessageReceived(const IPC::Message& msg) { 328 bool NaClIPCAdapter::OnMessageReceived(const IPC::Message& msg) {
487 { 329 {
488 base::AutoLock lock(lock_); 330 base::AutoLock lock(lock_);
489 331
490 scoped_refptr<RewrittenMessage> rewritten_msg(new RewrittenMessage); 332 scoped_refptr<RewrittenMessage> rewritten_msg(new RewrittenMessage);
491 333
492 // Pointer to the "new" message we will rewrite on Windows. On posix, this 334 typedef std::vector<ppapi::proxy::SerializedHandle> Handles;
493 // isn't necessary, so it will stay NULL. 335 Handles handles;
494 IPC::Message* new_msg_ptr = NULL; 336 scoped_ptr<IPC::Message> new_msg_ptr;
495 IPC::Message new_msg(msg.routing_id(), msg.type(), msg.priority()); 337 bool success = locked_data_.handle_converter_.ConvertNativeHandlesToPosix(
496 #if defined(OS_WIN) 338 msg, &handles, &new_msg_ptr);
497 new_msg_ptr = &new_msg; 339 if (!success)
498 #else 340 return false;
499 // Even on POSIX, we have to rewrite messages to create channels, because
500 // these contain a handle with an invalid (place holder) descriptor. The
501 // message sending code sees this and doesn't pass the descriptor over
502 // correctly.
503 if (msg.type() == PpapiMsg_CreateNaClChannel::ID)
504 new_msg_ptr = &new_msg;
505 #endif
506 341
507 Handles handles;
508 switch (msg.type()) {
509 CASE_FOR_MESSAGE(PpapiMsg_CreateNaClChannel)
510 CASE_FOR_MESSAGE(PpapiMsg_PPBAudio_NotifyAudioStreamCreated)
511 CASE_FOR_MESSAGE(PpapiPluginMsg_ResourceReply)
512 case IPC_REPLY_ID: {
513 int id = IPC::SyncMessage::GetMessageId(msg);
514 LockedData::PendingSyncMsgMap::iterator iter(
515 locked_data_.pending_sync_msgs_.find(id));
516 if (iter == locked_data_.pending_sync_msgs_.end()) {
517 NOTREACHED();
518 return false;
519 }
520 uint32_t type = iter->second;
521 locked_data_.pending_sync_msgs_.erase(iter);
522 switch (type) {
523 CASE_FOR_REPLY(PpapiHostMsg_PPBGraphics3D_GetTransferBuffer)
524 CASE_FOR_REPLY(PpapiHostMsg_PPBImageData_CreateNaCl)
525 CASE_FOR_REPLY(PpapiHostMsg_ResourceSyncCall)
526 default:
527 // Do nothing for messages we don't know.
528 break;
529 }
530 break;
531 }
532 default:
533 // Do nothing for messages we don't know.
534 break;
535 }
536 // Now add any descriptors we found to rewritten_msg. |handles| is usually 342 // Now add any descriptors we found to rewritten_msg. |handles| is usually
537 // empty, unless we read a message containing a FD or handle. 343 // empty, unless we read a message containing a FD or handle.
538 nacl::DescWrapperFactory factory; 344 nacl::DescWrapperFactory factory;
539 for (Handles::const_iterator iter = handles.begin(); 345 for (Handles::const_iterator iter = handles.begin();
540 iter != handles.end(); 346 iter != handles.end();
541 ++iter) { 347 ++iter) {
542 scoped_ptr<nacl::DescWrapper> nacl_desc; 348 scoped_ptr<nacl::DescWrapper> nacl_desc;
543 switch (iter->type()) { 349 switch (iter->type()) {
544 case ppapi::proxy::SerializedHandle::SHARED_MEMORY: { 350 case ppapi::proxy::SerializedHandle::SHARED_MEMORY: {
545 const base::SharedMemoryHandle& shm_handle = iter->shmem(); 351 const base::SharedMemoryHandle& shm_handle = iter->shmem();
(...skipping 123 matching lines...) Expand 10 before | Expand all | Expand 10 after
669 475
670 // Technically we didn't have to do any of the previous work in the lock. But 476 // Technically we didn't have to do any of the previous work in the lock. But
671 // sometimes our buffer will point to the to_be_sent_ string which is 477 // sometimes our buffer will point to the to_be_sent_ string which is
672 // protected by the lock, and it's messier to factor Send() such that it can 478 // protected by the lock, and it's messier to factor Send() such that it can
673 // unlock for us. Holding the lock for the message construction, which is 479 // unlock for us. Holding the lock for the message construction, which is
674 // just some memcpys, shouldn't be a big deal. 480 // just some memcpys, shouldn't be a big deal.
675 lock_.AssertAcquired(); 481 lock_.AssertAcquired();
676 if (locked_data_.channel_closed_) 482 if (locked_data_.channel_closed_)
677 return false; // TODO(brettw) clean up handles here when we add support! 483 return false; // TODO(brettw) clean up handles here when we add support!
678 484
679 // Store the type of all sync messages so that later we can translate the
680 // reply if necessary.
681 if (msg->is_sync()) { 485 if (msg->is_sync()) {
682 int id = IPC::SyncMessage::GetMessageId(*msg); 486 locked_data_.handle_converter_.RegisterSyncMessageForReply(*msg);
683 locked_data_.pending_sync_msgs_[id] = msg->type();
684 } 487 }
685 // Actual send must be done on the I/O thread. 488 // Actual send must be done on the I/O thread.
686 task_runner_->PostTask(FROM_HERE, 489 task_runner_->PostTask(FROM_HERE,
687 base::Bind(&NaClIPCAdapter::SendMessageOnIOThread, this, 490 base::Bind(&NaClIPCAdapter::SendMessageOnIOThread, this,
688 base::Passed(&msg))); 491 base::Passed(&msg)));
689 return true; 492 return true;
690 } 493 }
691 494
692 void NaClIPCAdapter::ClearToBeSent() { 495 void NaClIPCAdapter::ClearToBeSent() {
693 lock_.AssertAcquired(); 496 lock_.AssertAcquired();
(...skipping 28 matching lines...) Expand all
722 header.payload_size = static_cast<uint32>(msg.payload_size()); 525 header.payload_size = static_cast<uint32>(msg.payload_size());
723 header.routing = msg.routing_id(); 526 header.routing = msg.routing_id();
724 header.type = msg.type(); 527 header.type = msg.type();
725 header.flags = msg.flags(); 528 header.flags = msg.flags();
726 header.num_fds = static_cast<int>(rewritten_msg->desc_count()); 529 header.num_fds = static_cast<int>(rewritten_msg->desc_count());
727 530
728 rewritten_msg->SetData(header, msg.payload(), msg.payload_size()); 531 rewritten_msg->SetData(header, msg.payload(), msg.payload_size());
729 locked_data_.to_be_received_.push(rewritten_msg); 532 locked_data_.to_be_received_.push(rewritten_msg);
730 } 533 }
731 534
OLDNEW
« no previous file with comments | « chrome/nacl/nacl_ipc_adapter.h ('k') | ppapi/ppapi_ipc.gypi » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698