| OLD | NEW |
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 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_resource_base.h" | 5 #include "ppapi/proxy/udp_socket_resource_base.h" |
| 6 | 6 |
| 7 #include <algorithm> | 7 #include <algorithm> |
| 8 #include <cstring> | 8 #include <cstring> |
| 9 | 9 |
| 10 #include "base/logging.h" | 10 #include "base/logging.h" |
| (...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 61 } | 61 } |
| 62 | 62 |
| 63 int32_t UDPSocketResourceBase::SetOptionImpl( | 63 int32_t UDPSocketResourceBase::SetOptionImpl( |
| 64 PP_UDPSocket_Option name, | 64 PP_UDPSocket_Option name, |
| 65 const PP_Var& value, | 65 const PP_Var& value, |
| 66 bool check_bind_state, | 66 bool check_bind_state, |
| 67 scoped_refptr<TrackedCallback> callback) { | 67 scoped_refptr<TrackedCallback> callback) { |
| 68 if (closed_) | 68 if (closed_) |
| 69 return PP_ERROR_FAILED; | 69 return PP_ERROR_FAILED; |
| 70 | 70 |
| 71 SocketOptionData option_data; | 71 // Check if socket is expected to be bound or not according to the option. |
| 72 switch (name) { | 72 switch (name) { |
| 73 case PP_UDPSOCKET_OPTION_ADDRESS_REUSE: | 73 case PP_UDPSOCKET_OPTION_ADDRESS_REUSE: |
| 74 case PP_UDPSOCKET_OPTION_BROADCAST: { | 74 case PP_UDPSOCKET_OPTION_BROADCAST: |
| 75 case PP_UDPSOCKET_OPTION_MULTICAST_LOOP: |
| 76 case PP_UDPSOCKET_OPTION_MULTICAST_TTL: { |
| 75 if ((check_bind_state || name == PP_UDPSOCKET_OPTION_ADDRESS_REUSE) && | 77 if ((check_bind_state || name == PP_UDPSOCKET_OPTION_ADDRESS_REUSE) && |
| 76 bind_called_) { | 78 bind_called_) { |
| 77 // SetOption should fail in this case in order to give predictable | 79 // SetOption should fail in this case in order to give predictable |
| 78 // behavior while binding. Note that we use |bind_called_| rather | 80 // behavior while binding. Note that we use |bind_called_| rather |
| 79 // than |bound_| since the latter is only set on successful completion | 81 // than |bound_| since the latter is only set on successful completion |
| 80 // of Bind(). | 82 // of Bind(). |
| 81 return PP_ERROR_FAILED; | 83 return PP_ERROR_FAILED; |
| 82 } | 84 } |
| 85 break; |
| 86 } |
| 87 case PP_UDPSOCKET_OPTION_SEND_BUFFER_SIZE: |
| 88 case PP_UDPSOCKET_OPTION_RECV_BUFFER_SIZE: { |
| 89 if (check_bind_state && !bound_) |
| 90 return PP_ERROR_FAILED; |
| 91 break; |
| 92 } |
| 93 } |
| 94 |
| 95 SocketOptionData option_data; |
| 96 switch (name) { |
| 97 case PP_UDPSOCKET_OPTION_ADDRESS_REUSE: |
| 98 case PP_UDPSOCKET_OPTION_BROADCAST: |
| 99 case PP_UDPSOCKET_OPTION_MULTICAST_LOOP: { |
| 83 if (value.type != PP_VARTYPE_BOOL) | 100 if (value.type != PP_VARTYPE_BOOL) |
| 84 return PP_ERROR_BADARGUMENT; | 101 return PP_ERROR_BADARGUMENT; |
| 85 option_data.SetBool(PP_ToBool(value.value.as_bool)); | 102 option_data.SetBool(PP_ToBool(value.value.as_bool)); |
| 86 break; | 103 break; |
| 87 } | 104 } |
| 88 case PP_UDPSOCKET_OPTION_SEND_BUFFER_SIZE: | 105 case PP_UDPSOCKET_OPTION_SEND_BUFFER_SIZE: |
| 89 case PP_UDPSOCKET_OPTION_RECV_BUFFER_SIZE: { | 106 case PP_UDPSOCKET_OPTION_RECV_BUFFER_SIZE: { |
| 90 if (check_bind_state && !bound_) | |
| 91 return PP_ERROR_FAILED; | |
| 92 if (value.type != PP_VARTYPE_INT32) | 107 if (value.type != PP_VARTYPE_INT32) |
| 93 return PP_ERROR_BADARGUMENT; | 108 return PP_ERROR_BADARGUMENT; |
| 94 option_data.SetInt32(value.value.as_int); | 109 option_data.SetInt32(value.value.as_int); |
| 95 break; | 110 break; |
| 96 } | 111 } |
| 112 case PP_UDPSOCKET_OPTION_MULTICAST_TTL: { |
| 113 int32_t ival = value.value.as_int; |
| 114 if (value.type != PP_VARTYPE_INT32 && (ival < 0 || ival > 255)) |
| 115 return PP_ERROR_BADARGUMENT; |
| 116 option_data.SetInt32(ival); |
| 117 break; |
| 118 } |
| 97 default: { | 119 default: { |
| 98 NOTREACHED(); | 120 NOTREACHED(); |
| 99 return PP_ERROR_BADARGUMENT; | 121 return PP_ERROR_BADARGUMENT; |
| 100 } | 122 } |
| 101 } | 123 } |
| 102 | 124 |
| 103 Call<PpapiPluginMsg_UDPSocket_SetOptionReply>( | 125 Call<PpapiPluginMsg_UDPSocket_SetOptionReply>( |
| 104 BROWSER, | 126 BROWSER, |
| 105 PpapiHostMsg_UDPSocket_SetOption(name, option_data), | 127 PpapiHostMsg_UDPSocket_SetOption(name, option_data), |
| 106 base::Bind(&UDPSocketResourceBase::OnPluginMsgSetOptionReply, | 128 base::Bind(&UDPSocketResourceBase::OnPluginMsgGeneralReply, |
| 107 base::Unretained(this), | 129 base::Unretained(this), |
| 108 callback), | 130 callback), |
| 109 callback); | 131 callback); |
| 110 return PP_OK_COMPLETIONPENDING; | 132 return PP_OK_COMPLETIONPENDING; |
| 111 } | 133 } |
| 112 | 134 |
| 113 int32_t UDPSocketResourceBase::BindImpl( | 135 int32_t UDPSocketResourceBase::BindImpl( |
| 114 const PP_NetAddress_Private* addr, | 136 const PP_NetAddress_Private* addr, |
| 115 scoped_refptr<TrackedCallback> callback) { | 137 scoped_refptr<TrackedCallback> callback) { |
| 116 if (!addr) | 138 if (!addr) |
| (...skipping 109 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 226 while (!sendto_callbacks_.empty()) { | 248 while (!sendto_callbacks_.empty()) { |
| 227 scoped_refptr<TrackedCallback> callback = sendto_callbacks_.front(); | 249 scoped_refptr<TrackedCallback> callback = sendto_callbacks_.front(); |
| 228 sendto_callbacks_.pop(); | 250 sendto_callbacks_.pop(); |
| 229 PostAbortIfNecessary(&callback); | 251 PostAbortIfNecessary(&callback); |
| 230 } | 252 } |
| 231 | 253 |
| 232 read_buffer_ = NULL; | 254 read_buffer_ = NULL; |
| 233 bytes_to_read_ = -1; | 255 bytes_to_read_ = -1; |
| 234 } | 256 } |
| 235 | 257 |
| 258 int32_t UDPSocketResourceBase::JoinGroupImpl( |
| 259 const PP_NetAddress_Private *group, |
| 260 scoped_refptr<TrackedCallback> callback) { |
| 261 DCHECK(group); |
| 262 |
| 263 Call<PpapiPluginMsg_UDPSocket_JoinGroupReply>( |
| 264 BROWSER, |
| 265 PpapiHostMsg_UDPSocket_JoinGroup(*group), |
| 266 base::Bind(&UDPSocketResourceBase::OnPluginMsgGeneralReply, |
| 267 base::Unretained(this), |
| 268 callback), |
| 269 callback); |
| 270 return PP_OK_COMPLETIONPENDING; |
| 271 } |
| 272 |
| 273 int32_t UDPSocketResourceBase::LeaveGroupImpl( |
| 274 const PP_NetAddress_Private *group, |
| 275 scoped_refptr<TrackedCallback> callback) { |
| 276 DCHECK(group); |
| 277 |
| 278 Call<PpapiPluginMsg_UDPSocket_LeaveGroupReply>( |
| 279 BROWSER, |
| 280 PpapiHostMsg_UDPSocket_LeaveGroup(*group), |
| 281 base::Bind(&UDPSocketResourceBase::OnPluginMsgGeneralReply, |
| 282 base::Unretained(this), |
| 283 callback), |
| 284 callback); |
| 285 return PP_OK_COMPLETIONPENDING; |
| 286 } |
| 287 |
| 236 void UDPSocketResourceBase::OnReplyReceived( | 288 void UDPSocketResourceBase::OnReplyReceived( |
| 237 const ResourceMessageReplyParams& params, | 289 const ResourceMessageReplyParams& params, |
| 238 const IPC::Message& msg) { | 290 const IPC::Message& msg) { |
| 239 PPAPI_BEGIN_MESSAGE_MAP(UDPSocketResourceBase, msg) | 291 PPAPI_BEGIN_MESSAGE_MAP(UDPSocketResourceBase, msg) |
| 240 PPAPI_DISPATCH_PLUGIN_RESOURCE_CALL( | 292 PPAPI_DISPATCH_PLUGIN_RESOURCE_CALL( |
| 241 PpapiPluginMsg_UDPSocket_PushRecvResult, | 293 PpapiPluginMsg_UDPSocket_PushRecvResult, |
| 242 OnPluginMsgPushRecvResult) | 294 OnPluginMsgPushRecvResult) |
| 243 PPAPI_DISPATCH_PLUGIN_RESOURCE_CALL_UNHANDLED( | 295 PPAPI_DISPATCH_PLUGIN_RESOURCE_CALL_UNHANDLED( |
| 244 PluginResource::OnReplyReceived(params, msg)) | 296 PluginResource::OnReplyReceived(params, msg)) |
| 245 PPAPI_END_MESSAGE_MAP() | 297 PPAPI_END_MESSAGE_MAP() |
| 246 } | 298 } |
| 247 | 299 |
| 248 void UDPSocketResourceBase::PostAbortIfNecessary( | 300 void UDPSocketResourceBase::PostAbortIfNecessary( |
| 249 scoped_refptr<TrackedCallback>* callback) { | 301 scoped_refptr<TrackedCallback>* callback) { |
| 250 if (TrackedCallback::IsPending(*callback)) | 302 if (TrackedCallback::IsPending(*callback)) |
| 251 (*callback)->PostAbort(); | 303 (*callback)->PostAbort(); |
| 252 } | 304 } |
| 253 | 305 |
| 254 void UDPSocketResourceBase::OnPluginMsgSetOptionReply( | 306 void UDPSocketResourceBase::OnPluginMsgGeneralReply( |
| 255 scoped_refptr<TrackedCallback> callback, | 307 scoped_refptr<TrackedCallback> callback, |
| 256 const ResourceMessageReplyParams& params) { | 308 const ResourceMessageReplyParams& params) { |
| 257 if (TrackedCallback::IsPending(callback)) | 309 if (TrackedCallback::IsPending(callback)) |
| 258 RunCallback(callback, params.result()); | 310 RunCallback(callback, params.result()); |
| 259 } | 311 } |
| 260 | 312 |
| 261 void UDPSocketResourceBase::OnPluginMsgBindReply( | 313 void UDPSocketResourceBase::OnPluginMsgBindReply( |
| 262 const ResourceMessageReplyParams& params, | 314 const ResourceMessageReplyParams& params, |
| 263 const PP_NetAddress_Private& bound_addr) { | 315 const PP_NetAddress_Private& bound_addr) { |
| 264 // It is possible that |bind_callback_| is pending while |closed_| is true: | 316 // It is possible that |bind_callback_| is pending while |closed_| is true: |
| (...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 365 if (result == PP_OK && !data.empty()) | 417 if (result == PP_OK && !data.empty()) |
| 366 memcpy(output_buffer, data.c_str(), data.size()); | 418 memcpy(output_buffer, data.c_str(), data.size()); |
| 367 | 419 |
| 368 recvfrom_addr_ = addr; | 420 recvfrom_addr_ = addr; |
| 369 | 421 |
| 370 return result == PP_OK ? static_cast<int32_t>(data.size()) : result; | 422 return result == PP_OK ? static_cast<int32_t>(data.size()) : result; |
| 371 } | 423 } |
| 372 | 424 |
| 373 } // namespace proxy | 425 } // namespace proxy |
| 374 } // namespace ppapi | 426 } // namespace ppapi |
| OLD | NEW |