OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2013 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 "ppapi/proxy/udp_socket_private_resource.h" | 5 #include "ppapi/proxy/udp_socket_shared.h" |
6 | 6 |
7 #include <algorithm> | 7 #include <algorithm> |
8 #include <cstring> | 8 #include <cstring> |
9 | 9 |
10 #include "base/basictypes.h" | |
11 #include "base/logging.h" | 10 #include "base/logging.h" |
12 #include "ppapi/c/pp_bool.h" | 11 #include "ppapi/c/pp_bool.h" |
13 #include "ppapi/c/pp_completion_callback.h" | 12 #include "ppapi/c/pp_completion_callback.h" |
14 #include "ppapi/c/pp_errors.h" | 13 #include "ppapi/c/pp_errors.h" |
15 #include "ppapi/proxy/ppapi_messages.h" | 14 #include "ppapi/proxy/ppapi_messages.h" |
15 #include "ppapi/thunk/enter.h" | |
16 #include "ppapi/thunk/resource_creation_api.h" | |
16 | 17 |
17 namespace ppapi { | 18 namespace ppapi { |
18 namespace proxy { | 19 namespace proxy { |
19 | 20 |
20 const int32_t UDPSocketPrivateResource::kMaxReadSize = 1024 * 1024; | 21 const int32_t UDPSocketShared::kMaxReadSize = 1024 * 1024; |
21 const int32_t UDPSocketPrivateResource::kMaxWriteSize = 1024 * 1024; | 22 const int32_t UDPSocketShared::kMaxWriteSize = 1024 * 1024; |
22 | 23 |
23 UDPSocketPrivateResource::UDPSocketPrivateResource(Connection connection, | 24 UDPSocketShared::UDPSocketShared(Connection connection, PP_Instance instance) |
24 PP_Instance instance) | |
25 : PluginResource(connection, instance), | 25 : PluginResource(connection, instance), |
26 bound_(false), | 26 bound_(false), |
27 closed_(false), | 27 closed_(false), |
28 read_buffer_(NULL), | 28 read_buffer_(NULL), |
29 bytes_to_read_(-1) { | 29 bytes_to_read_(-1) { |
30 recvfrom_addr_.size = 0; | 30 recvfrom_addr_.size = 0; |
31 memset(recvfrom_addr_.data, 0, | 31 memset(recvfrom_addr_.data, 0, |
32 arraysize(recvfrom_addr_.data) * sizeof(*recvfrom_addr_.data)); | 32 arraysize(recvfrom_addr_.data) * sizeof(*recvfrom_addr_.data)); |
33 bound_addr_.size = 0; | 33 bound_addr_.size = 0; |
34 memset(bound_addr_.data, 0, | 34 memset(bound_addr_.data, 0, |
35 arraysize(bound_addr_.data) * sizeof(*bound_addr_.data)); | 35 arraysize(bound_addr_.data) * sizeof(*bound_addr_.data)); |
36 | 36 |
37 SendCreate(BROWSER, PpapiHostMsg_UDPSocketPrivate_Create()); | 37 SendCreate(BROWSER, PpapiHostMsg_UDPSocketPrivate_Create()); |
38 } | 38 } |
39 | 39 |
40 UDPSocketPrivateResource::~UDPSocketPrivateResource() { | 40 UDPSocketShared::~UDPSocketShared() { |
41 } | 41 } |
42 | 42 |
43 thunk::PPB_UDPSocket_Private_API* | 43 int32_t UDPSocketShared::SetSocketFeatureImpl(PP_UDPSocketFeature_Private name, |
44 UDPSocketPrivateResource::AsPPB_UDPSocket_Private_API() { | 44 const PP_Var& value) { |
45 return this; | |
46 } | |
47 | |
48 int32_t UDPSocketPrivateResource::SetSocketFeature( | |
49 PP_UDPSocketFeature_Private name, | |
50 PP_Var value) { | |
51 if (bound_ || closed_) | 45 if (bound_ || closed_) |
52 return PP_ERROR_FAILED; | 46 return PP_ERROR_FAILED; |
53 | 47 |
54 switch (name) { | 48 switch (name) { |
55 case PP_UDPSOCKETFEATURE_ADDRESS_REUSE: | 49 case PP_UDPSOCKETFEATURE_ADDRESS_REUSE: |
56 case PP_UDPSOCKETFEATURE_BROADCAST: | 50 case PP_UDPSOCKETFEATURE_BROADCAST: { |
57 if (value.type != PP_VARTYPE_BOOL) | 51 if (value.type != PP_VARTYPE_BOOL) |
58 return PP_ERROR_BADARGUMENT; | 52 return PP_ERROR_BADARGUMENT; |
59 SendBoolSocketFeature(static_cast<int32_t>(name), | 53 PpapiHostMsg_UDPSocketPrivate_SetBoolSocketFeature msg( |
60 PP_ToBool(value.value.as_bool)); | 54 static_cast<int32_t>(name), PP_ToBool(value.value.as_bool)); |
55 Post(BROWSER, msg); | |
61 break; | 56 break; |
62 default: | 57 } |
58 default: { | |
63 return PP_ERROR_BADARGUMENT; | 59 return PP_ERROR_BADARGUMENT; |
60 } | |
64 } | 61 } |
65 return PP_OK; | 62 return PP_OK; |
66 } | 63 } |
67 | 64 |
68 int32_t UDPSocketPrivateResource::Bind( | 65 int32_t UDPSocketShared::BindImpl(const PP_NetAddress_Private* addr, |
69 const PP_NetAddress_Private* addr, | 66 scoped_refptr<TrackedCallback> callback) { |
70 scoped_refptr<TrackedCallback> callback) { | |
71 if (!addr) | 67 if (!addr) |
72 return PP_ERROR_BADARGUMENT; | 68 return PP_ERROR_BADARGUMENT; |
73 if (bound_ || closed_) | 69 if (bound_ || closed_) |
74 return PP_ERROR_FAILED; | 70 return PP_ERROR_FAILED; |
75 if (TrackedCallback::IsPending(bind_callback_)) | 71 if (TrackedCallback::IsPending(bind_callback_)) |
76 return PP_ERROR_INPROGRESS; | 72 return PP_ERROR_INPROGRESS; |
77 | 73 |
78 bind_callback_ = callback; | 74 bind_callback_ = callback; |
79 | 75 |
80 // Send the request, the browser will call us back via BindReply. | 76 // Send the request, the browser will call us back via BindReply. |
81 SendBind(*addr); | 77 PpapiHostMsg_UDPSocketPrivate_Bind msg(*addr); |
78 Call<PpapiPluginMsg_UDPSocketPrivate_BindReply>( | |
79 BROWSER, | |
80 msg, | |
81 base::Bind(&UDPSocketShared::OnPluginMsgBindReply, | |
82 base::Unretained(this))); | |
82 return PP_OK_COMPLETIONPENDING; | 83 return PP_OK_COMPLETIONPENDING; |
83 } | 84 } |
84 | 85 |
85 PP_Bool UDPSocketPrivateResource::GetBoundAddress(PP_NetAddress_Private* addr) { | 86 PP_Bool UDPSocketShared::GetBoundAddressImpl(PP_NetAddress_Private* addr) { |
86 if (!addr || !bound_ || closed_) | 87 if (!addr || !bound_ || closed_) |
87 return PP_FALSE; | 88 return PP_FALSE; |
88 | 89 |
89 *addr = bound_addr_; | 90 *addr = bound_addr_; |
90 return PP_TRUE; | 91 return PP_TRUE; |
91 } | 92 } |
92 | 93 |
93 int32_t UDPSocketPrivateResource::RecvFrom( | 94 int32_t UDPSocketShared::RecvFromImpl(char* buffer, |
94 char* buffer, | 95 int32_t num_bytes, |
95 int32_t num_bytes, | 96 PP_Resource* addr, |
96 scoped_refptr<TrackedCallback> callback) { | 97 scoped_refptr<TrackedCallback> callback) { |
97 if (!buffer || num_bytes <= 0) | 98 if (!buffer || num_bytes <= 0) |
98 return PP_ERROR_BADARGUMENT; | 99 return PP_ERROR_BADARGUMENT; |
99 if (!bound_) | 100 if (!bound_) |
100 return PP_ERROR_FAILED; | 101 return PP_ERROR_FAILED; |
101 if (TrackedCallback::IsPending(recvfrom_callback_)) | 102 if (TrackedCallback::IsPending(recvfrom_callback_)) |
102 return PP_ERROR_INPROGRESS; | 103 return PP_ERROR_INPROGRESS; |
103 | 104 |
104 read_buffer_ = buffer; | 105 read_buffer_ = buffer; |
105 bytes_to_read_ = std::min(num_bytes, kMaxReadSize); | 106 bytes_to_read_ = std::min(num_bytes, kMaxReadSize); |
106 recvfrom_callback_ = callback; | 107 recvfrom_callback_ = callback; |
107 | 108 |
108 // Send the request, the browser will call us back via RecvFromReply. | 109 // Send the request, the browser will call us back via RecvFromReply. |
109 SendRecvFrom(bytes_to_read_); | 110 PpapiHostMsg_UDPSocketPrivate_RecvFrom msg(bytes_to_read_); |
111 Call<PpapiPluginMsg_UDPSocketPrivate_RecvFromReply>( | |
112 BROWSER, | |
113 msg, | |
114 base::Bind(&UDPSocketShared::OnPluginMsgRecvFromReply, | |
115 base::Unretained(this), addr)); | |
110 return PP_OK_COMPLETIONPENDING; | 116 return PP_OK_COMPLETIONPENDING; |
111 } | 117 } |
112 | 118 |
113 PP_Bool UDPSocketPrivateResource::GetRecvFromAddress( | 119 PP_Bool UDPSocketShared::GetRecvFromAddressImpl(PP_NetAddress_Private* addr) { |
114 PP_NetAddress_Private* addr) { | |
115 if (!addr) | 120 if (!addr) |
116 return PP_FALSE; | 121 return PP_FALSE; |
117 *addr = recvfrom_addr_; | 122 *addr = recvfrom_addr_; |
118 return PP_TRUE; | 123 return PP_TRUE; |
119 } | 124 } |
120 | 125 |
121 void UDPSocketPrivateResource::PostAbortIfNecessary( | 126 int32_t UDPSocketShared::SendToImpl(const char* buffer, |
122 scoped_refptr<TrackedCallback>* callback) { | 127 int32_t num_bytes, |
123 if (TrackedCallback::IsPending(*callback)) | 128 const PP_NetAddress_Private* addr, |
124 (*callback)->PostAbort(); | 129 scoped_refptr<TrackedCallback> callback) { |
125 } | |
126 | |
127 int32_t UDPSocketPrivateResource::SendTo( | |
128 const char* buffer, | |
129 int32_t num_bytes, | |
130 const PP_NetAddress_Private* addr, | |
131 scoped_refptr<TrackedCallback> callback) { | |
132 if (!buffer || num_bytes <= 0 || !addr) | 130 if (!buffer || num_bytes <= 0 || !addr) |
133 return PP_ERROR_BADARGUMENT; | 131 return PP_ERROR_BADARGUMENT; |
134 if (!bound_) | 132 if (!bound_) |
135 return PP_ERROR_FAILED; | 133 return PP_ERROR_FAILED; |
136 if (TrackedCallback::IsPending(sendto_callback_)) | 134 if (TrackedCallback::IsPending(sendto_callback_)) |
137 return PP_ERROR_INPROGRESS; | 135 return PP_ERROR_INPROGRESS; |
138 | 136 |
139 if (num_bytes > kMaxWriteSize) | 137 if (num_bytes > kMaxWriteSize) |
140 num_bytes = kMaxWriteSize; | 138 num_bytes = kMaxWriteSize; |
141 | 139 |
142 sendto_callback_ = callback; | 140 sendto_callback_ = callback; |
143 | 141 |
144 // Send the request, the browser will call us back via SendToReply. | 142 // Send the request, the browser will call us back via SendToReply. |
145 SendSendTo(std::string(buffer, num_bytes), *addr); | 143 PpapiHostMsg_UDPSocketPrivate_SendTo msg(std::string(buffer, num_bytes), |
144 *addr); | |
145 Call<PpapiPluginMsg_UDPSocketPrivate_SendToReply>( | |
146 BROWSER, | |
147 msg, | |
148 base::Bind(&UDPSocketShared::OnPluginMsgSendToReply, | |
149 base::Unretained(this))); | |
146 return PP_OK_COMPLETIONPENDING; | 150 return PP_OK_COMPLETIONPENDING; |
147 } | 151 } |
148 | 152 |
149 void UDPSocketPrivateResource::Close() { | 153 void UDPSocketShared::CloseImpl() { |
150 if(closed_) | 154 if(closed_) |
151 return; | 155 return; |
152 | 156 |
153 bound_ = false; | 157 bound_ = false; |
154 closed_ = true; | 158 closed_ = true; |
155 | 159 |
156 SendClose(); | 160 PpapiHostMsg_UDPSocketPrivate_Close msg; |
161 Post(BROWSER, msg); | |
157 | 162 |
158 PostAbortIfNecessary(&bind_callback_); | 163 PostAbortIfNecessary(&bind_callback_); |
159 PostAbortIfNecessary(&recvfrom_callback_); | 164 PostAbortIfNecessary(&recvfrom_callback_); |
160 PostAbortIfNecessary(&sendto_callback_); | 165 PostAbortIfNecessary(&sendto_callback_); |
161 } | 166 } |
162 | 167 |
163 void UDPSocketPrivateResource::SendBoolSocketFeature(int32_t name, bool value) { | 168 void UDPSocketShared::PostAbortIfNecessary( |
164 PpapiHostMsg_UDPSocketPrivate_SetBoolSocketFeature msg(name, value); | 169 scoped_refptr<TrackedCallback>* callback) { |
165 Post(BROWSER, msg); | 170 if (TrackedCallback::IsPending(*callback)) |
171 (*callback)->PostAbort(); | |
166 } | 172 } |
167 | 173 |
168 void UDPSocketPrivateResource::SendBind(const PP_NetAddress_Private& addr) { | 174 void UDPSocketShared::OnPluginMsgBindReply( |
169 PpapiHostMsg_UDPSocketPrivate_Bind msg(addr); | |
170 Call<PpapiPluginMsg_UDPSocketPrivate_BindReply>( | |
171 BROWSER, | |
172 msg, | |
173 base::Bind(&UDPSocketPrivateResource::OnPluginMsgBindReply, | |
174 base::Unretained(this))); | |
175 } | |
176 | |
177 void UDPSocketPrivateResource::SendRecvFrom(int32_t num_bytes) { | |
178 PpapiHostMsg_UDPSocketPrivate_RecvFrom msg(num_bytes); | |
179 Call<PpapiPluginMsg_UDPSocketPrivate_RecvFromReply>( | |
180 BROWSER, | |
181 msg, | |
182 base::Bind(&UDPSocketPrivateResource::OnPluginMsgRecvFromReply, | |
183 base::Unretained(this))); | |
184 } | |
185 | |
186 void UDPSocketPrivateResource::SendSendTo(const std::string& buffer, | |
187 const PP_NetAddress_Private& addr) { | |
188 PpapiHostMsg_UDPSocketPrivate_SendTo msg(buffer, addr); | |
189 Call<PpapiPluginMsg_UDPSocketPrivate_SendToReply>( | |
190 BROWSER, | |
191 msg, | |
192 base::Bind(&UDPSocketPrivateResource::OnPluginMsgSendToReply, | |
193 base::Unretained(this))); | |
194 } | |
195 | |
196 void UDPSocketPrivateResource::SendClose() { | |
197 PpapiHostMsg_UDPSocketPrivate_Close msg; | |
198 Post(BROWSER, msg); | |
199 } | |
200 | |
201 void UDPSocketPrivateResource::OnPluginMsgBindReply( | |
202 const ResourceMessageReplyParams& params, | 175 const ResourceMessageReplyParams& params, |
203 const PP_NetAddress_Private& bound_addr) { | 176 const PP_NetAddress_Private& bound_addr) { |
204 if (!TrackedCallback::IsPending(bind_callback_)) { | 177 if (!TrackedCallback::IsPending(bind_callback_)) { |
205 NOTREACHED(); | 178 NOTREACHED(); |
206 return; | 179 return; |
207 } | 180 } |
208 if (params.result() == PP_OK) | 181 if (params.result() == PP_OK) |
209 bound_ = true; | 182 bound_ = true; |
210 bound_addr_ = bound_addr; | 183 bound_addr_ = bound_addr; |
211 bind_callback_->Run(params.result()); | 184 bind_callback_->Run(params.result()); |
212 } | 185 } |
213 | 186 |
214 void UDPSocketPrivateResource::OnPluginMsgRecvFromReply( | 187 void UDPSocketShared::OnPluginMsgRecvFromReply( |
188 PP_Resource* output_addr, | |
215 const ResourceMessageReplyParams& params, | 189 const ResourceMessageReplyParams& params, |
216 const std::string& data, | 190 const std::string& data, |
217 const PP_NetAddress_Private& addr) { | 191 const PP_NetAddress_Private& addr) { |
218 if (!TrackedCallback::IsPending(recvfrom_callback_) || !read_buffer_) { | 192 if (!TrackedCallback::IsPending(recvfrom_callback_) || !read_buffer_) { |
bbudge
2013/06/10 12:59:56
Is the '!read_buffer' clause really needed? You ch
yzshen1
2013/06/10 22:10:35
Done.
| |
219 NOTREACHED(); | 193 NOTREACHED(); |
220 return; | 194 return; |
221 } | 195 } |
222 bool succeeded = (params.result() == PP_OK); | 196 bool succeeded = (params.result() == PP_OK); |
197 if (succeeded && output_addr) { | |
198 thunk::EnterResourceCreationNoLock enter(pp_instance()); | |
199 if (enter.succeeded()) { | |
200 *output_addr = enter.functions()->CreateNetAddressFromNetAddressPrivate( | |
201 pp_instance(), addr); | |
202 } else { | |
203 succeeded = false; | |
204 } | |
205 } | |
206 | |
223 if (succeeded) { | 207 if (succeeded) { |
224 CHECK_LE(static_cast<int32_t>(data.size()), bytes_to_read_); | 208 CHECK_LE(static_cast<int32_t>(data.size()), bytes_to_read_); |
225 if (!data.empty()) | 209 if (!data.empty()) |
226 memcpy(read_buffer_, data.c_str(), data.size()); | 210 memcpy(read_buffer_, data.c_str(), data.size()); |
227 } | 211 } |
212 | |
228 read_buffer_ = NULL; | 213 read_buffer_ = NULL; |
229 bytes_to_read_ = -1; | 214 bytes_to_read_ = -1; |
230 recvfrom_addr_ = addr; | 215 recvfrom_addr_ = addr; |
231 | 216 |
232 if (succeeded) | 217 if (succeeded) |
233 recvfrom_callback_->Run(static_cast<int32_t>(data.size())); | 218 recvfrom_callback_->Run(static_cast<int32_t>(data.size())); |
234 else | 219 else |
235 recvfrom_callback_->Run(params.result()); | 220 recvfrom_callback_->Run(params.result()); |
236 } | 221 } |
237 | 222 |
238 void UDPSocketPrivateResource::OnPluginMsgSendToReply( | 223 void UDPSocketShared::OnPluginMsgSendToReply( |
239 const ResourceMessageReplyParams& params, | 224 const ResourceMessageReplyParams& params, |
240 int32_t bytes_written) { | 225 int32_t bytes_written) { |
241 if (!TrackedCallback::IsPending(sendto_callback_)) { | 226 if (!TrackedCallback::IsPending(sendto_callback_)) { |
242 NOTREACHED(); | 227 NOTREACHED(); |
243 return; | 228 return; |
244 } | 229 } |
245 if (params.result() == PP_OK) | 230 if (params.result() == PP_OK) |
246 sendto_callback_->Run(bytes_written); | 231 sendto_callback_->Run(bytes_written); |
247 else | 232 else |
248 sendto_callback_->Run(params.result()); | 233 sendto_callback_->Run(params.result()); |
249 } | 234 } |
250 | 235 |
251 } // namespace proxy | 236 } // namespace proxy |
252 } // namespace ppapi | 237 } // namespace ppapi |
OLD | NEW |