| Index: content/renderer/pepper/pepper_websocket_host.cc | 
| diff --git a/content/renderer/pepper/pepper_websocket_host.cc b/content/renderer/pepper/pepper_websocket_host.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..ec6341a327169e0a66836c88b458db643f38b9fc | 
| --- /dev/null | 
| +++ b/content/renderer/pepper/pepper_websocket_host.cc | 
| @@ -0,0 +1,290 @@ | 
| +// Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +#include "content/renderer/pepper/pepper_websocket_host.h" | 
| + | 
| +#include <string> | 
| + | 
| +#include "content/public/renderer/renderer_ppapi_host.h" | 
| +#include "net/base/net_util.h" | 
| +#include "ppapi/c/pp_errors.h" | 
| +#include "ppapi/c/ppb_websocket.h" | 
| +#include "ppapi/host/dispatch_host_message.h" | 
| +#include "ppapi/host/host_message_context.h" | 
| +#include "ppapi/host/ppapi_host.h" | 
| +#include "ppapi/proxy/ppapi_messages.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURL.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebArrayBuffer.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebElement.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebPluginContainer.h" | 
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebSocket.h" | 
| + | 
| +using WebKit::WebArrayBuffer; | 
| +using WebKit::WebDocument; | 
| +using WebKit::WebString; | 
| +using WebKit::WebSocket; | 
| +using WebKit::WebURL; | 
| + | 
| +namespace content { | 
| + | 
| +PepperWebSocketHost::PepperWebSocketHost( | 
| +    RendererPpapiHost* host, | 
| +    PP_Instance instance, | 
| +    PP_Resource resource) | 
| +    : ResourceHost(host->GetPpapiHost(), instance, resource), | 
| +      renderer_ppapi_host_(host), | 
| +      connecting_(false), | 
| +      initiating_close_(false), | 
| +      accepting_close_(false), | 
| +      error_was_received_(false) { | 
| +} | 
| + | 
| +PepperWebSocketHost::~PepperWebSocketHost() { | 
| +  if (websocket_.get()) | 
| +    websocket_->disconnect(); | 
| +} | 
| + | 
| +int32_t PepperWebSocketHost::OnResourceMessageReceived( | 
| +    const IPC::Message& msg, | 
| +    ppapi::host::HostMessageContext* context) { | 
| +  IPC_BEGIN_MESSAGE_MAP(PepperWebSocketHost, msg) | 
| +    PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Connect, | 
| +                                      OnHostMsgConnect) | 
| +    PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Close, | 
| +                                      OnHostMsgClose) | 
| +    PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_SendText, | 
| +                                      OnHostMsgSendText) | 
| +    PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_SendBinary, | 
| +                                      OnHostMsgSendBinary) | 
| +    PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Fail, | 
| +                                      OnHostMsgFail) | 
| +  IPC_END_MESSAGE_MAP() | 
| +  return PP_ERROR_FAILED; | 
| +} | 
| + | 
| +void PepperWebSocketHost::didConnect() { | 
| +  std::string protocol; | 
| +  if (websocket_.get()) | 
| +    protocol = websocket_->subprotocol().utf8(); | 
| +  connecting_ = false; | 
| +  connect_reply_.params.set_result(PP_OK); | 
| +  host()->SendReply(connect_reply_, | 
| +                    PpapiPluginMsg_WebSocket_ConnectReply( | 
| +                        url_, | 
| +                        protocol)); | 
| +} | 
| + | 
| +void PepperWebSocketHost::didReceiveMessage(const WebKit::WebString& message) { | 
| +  // Dispose packets after receiving an error. | 
| +  if (error_was_received_) | 
| +    return; | 
| + | 
| +  // Send an IPC to transport received data. | 
| +  std::string string_message = message.utf8(); | 
| +  host()->SendUnsolicitedReply(pp_resource(), | 
| +                               PpapiPluginMsg_WebSocket_ReceiveTextReply( | 
| +                                  string_message)); | 
| +} | 
| + | 
| +void PepperWebSocketHost::didReceiveArrayBuffer( | 
| +    const WebKit::WebArrayBuffer& binaryData) { | 
| +  // Dispose packets after receiving an error. | 
| +  if (error_was_received_) | 
| +    return; | 
| + | 
| +  // Send an IPC to transport received data. | 
| +  uint8_t* data = static_cast<uint8_t*>(binaryData.data()); | 
| +  std::vector<uint8_t> array_message(data, data + binaryData.byteLength()); | 
| +  host()->SendUnsolicitedReply(pp_resource(), | 
| +                               PpapiPluginMsg_WebSocket_ReceiveBinaryReply( | 
| +                                  array_message)); | 
| +} | 
| + | 
| +void PepperWebSocketHost::didReceiveMessageError() { | 
| +  // Records the error, then stops receiving any frames after this error. | 
| +  // The error must be notified after all queued messages are read. | 
| +  error_was_received_ = true; | 
| + | 
| +  // Send an IPC to report the error. After this IPC, ReceiveTextReply and | 
| +  // ReceiveBinaryReply IPC are not sent anymore because |error_was_received_| | 
| +  // blocks. | 
| +  host()->SendUnsolicitedReply(pp_resource(), | 
| +                               PpapiPluginMsg_WebSocket_ErrorReply()); | 
| +} | 
| + | 
| +void PepperWebSocketHost::didUpdateBufferedAmount( | 
| +    unsigned long buffered_amount) { | 
| +  // Send an IPC to update buffered amount. | 
| +  host()->SendUnsolicitedReply(pp_resource(), | 
| +                               PpapiPluginMsg_WebSocket_BufferedAmountReply( | 
| +                                  buffered_amount)); | 
| +} | 
| + | 
| +void PepperWebSocketHost::didStartClosingHandshake() { | 
| +  accepting_close_ = true; | 
| + | 
| +  // Send an IPC to notice that server starts closing handshake. | 
| +  host()->SendUnsolicitedReply(pp_resource(), | 
| +                               PpapiPluginMsg_WebSocket_StateReply( | 
| +                                  PP_WEBSOCKETREADYSTATE_CLOSING)); | 
| +} | 
| + | 
| +void PepperWebSocketHost::didClose(unsigned long unhandled_buffered_amount, | 
| +                                   ClosingHandshakeCompletionStatus status, | 
| +                                   unsigned short code, | 
| +                                   const WebKit::WebString& reason) { | 
| +  if (connecting_) { | 
| +    connecting_ = false; | 
| +    connect_reply_.params.set_result(PP_ERROR_FAILED); | 
| +    host()->SendReply( | 
| +        connect_reply_, | 
| +        PpapiPluginMsg_WebSocket_ConnectReply(url_, std::string())); | 
| +  } | 
| + | 
| +  // Set close_was_clean_. | 
| +  bool was_clean = | 
| +      (initiating_close_ || accepting_close_) && | 
| +      !unhandled_buffered_amount && | 
| +      status == WebSocketClient::ClosingHandshakeComplete; | 
| + | 
| +  if (initiating_close_) { | 
| +    initiating_close_ = false; | 
| +    close_reply_.params.set_result(PP_OK); | 
| +    host()->SendReply(close_reply_, PpapiPluginMsg_WebSocket_CloseReply( | 
| +          unhandled_buffered_amount, | 
| +          was_clean, | 
| +          code, | 
| +          reason.utf8())); | 
| +  } else { | 
| +    accepting_close_ = false; | 
| +    host()->SendUnsolicitedReply(pp_resource(), | 
| +                                 PpapiPluginMsg_WebSocket_ClosedReply( | 
| +                                     unhandled_buffered_amount, | 
| +                                     was_clean, | 
| +                                     code, | 
| +                                     reason.utf8())); | 
| +  } | 
| + | 
| +  // Disconnect. | 
| +  if (websocket_.get()) | 
| +    websocket_->disconnect(); | 
| +} | 
| + | 
| +int32_t PepperWebSocketHost::OnHostMsgConnect( | 
| +    ppapi::host::HostMessageContext* context, | 
| +    const std::string& url, | 
| +    const std::vector<std::string>& protocols) { | 
| +  // Validate url and convert it to WebURL. | 
| +  GURL gurl(url); | 
| +  url_ = gurl.spec(); | 
| +  if (!gurl.is_valid()) | 
| +    return PP_ERROR_BADARGUMENT; | 
| +  if (!gurl.SchemeIs("ws") && !gurl.SchemeIs("wss")) | 
| +    return PP_ERROR_BADARGUMENT; | 
| +  if (gurl.has_ref()) | 
| +    return PP_ERROR_BADARGUMENT; | 
| +  if (!net::IsPortAllowedByDefault(gurl.IntPort())) | 
| +    return PP_ERROR_BADARGUMENT; | 
| +  WebURL web_url(gurl); | 
| + | 
| +  // Validate protocols. | 
| +  std::string protocol_string; | 
| +  for (std::vector<std::string>::const_iterator vector_it = protocols.begin(); | 
| +       vector_it != protocols.end(); | 
| +       ++vector_it) { | 
| + | 
| +    // Check containing characters. | 
| +    for (std::string::const_iterator string_it = vector_it->begin(); | 
| +        string_it != vector_it->end(); | 
| +        ++string_it) { | 
| +      uint8_t character = *string_it; | 
| +      // WebSocket specification says "(Subprotocol string must consist of) | 
| +      // characters in the range U+0021 to U+007E not including separator | 
| +      // characters as defined in [RFC2616]." | 
| +      const uint8_t minimumProtocolCharacter = '!';  // U+0021. | 
| +      const uint8_t maximumProtocolCharacter = '~';  // U+007E. | 
| +      if (character < minimumProtocolCharacter || | 
| +          character > maximumProtocolCharacter || | 
| +          character == '"' || character == '(' || character == ')' || | 
| +          character == ',' || character == '/' || | 
| +          (character >= ':' && character <= '@') ||  // U+003A - U+0040 | 
| +          (character >= '[' && character <= ']') ||  // U+005B - u+005D | 
| +          character == '{' || character == '}') | 
| +        return PP_ERROR_BADARGUMENT; | 
| +    } | 
| +    // Join protocols with the comma separator. | 
| +    if (vector_it != protocols.begin()) | 
| +      protocol_string.append(","); | 
| +    protocol_string.append(*vector_it); | 
| +  } | 
| + | 
| +  // Convert protocols to WebString. | 
| +  WebString web_protocols = WebString::fromUTF8(protocol_string); | 
| + | 
| +  // Create WebKit::WebSocket object and connect. | 
| +  WebKit::WebPluginContainer* container = | 
| +      renderer_ppapi_host_->GetContainerForInstance(pp_instance()); | 
| +  if (!container) | 
| +    return PP_ERROR_BADARGUMENT; | 
| +  // TODO(toyoshim) Remove following WebDocument object copy. | 
| +  WebDocument document = container->element().document(); | 
| +  websocket_.reset(WebSocket::create(document, this)); | 
| +  DCHECK(websocket_.get()); | 
| +  if (!websocket_.get()) | 
| +    return PP_ERROR_NOTSUPPORTED; | 
| + | 
| +  // Set receiving binary object type. | 
| +  websocket_->setBinaryType(WebSocket::BinaryTypeArrayBuffer); | 
| +  websocket_->connect(web_url, web_protocols); | 
| + | 
| +  connect_reply_ = context->MakeReplyMessageContext(); | 
| +  connecting_ = true; | 
| +  return PP_OK_COMPLETIONPENDING; | 
| +} | 
| + | 
| +int32_t PepperWebSocketHost::OnHostMsgClose( | 
| +    ppapi::host::HostMessageContext* context, | 
| +    int32_t code, | 
| +    const std::string& reason) { | 
| +  if (!websocket_.get()) | 
| +    return PP_ERROR_FAILED; | 
| +  close_reply_ = context->MakeReplyMessageContext(); | 
| +  initiating_close_ = true; | 
| +  WebString web_reason = WebString::fromUTF8(reason); | 
| +  websocket_->close(code, web_reason); | 
| +  return PP_OK_COMPLETIONPENDING; | 
| +} | 
| + | 
| +int32_t PepperWebSocketHost::OnHostMsgSendText( | 
| +    ppapi::host::HostMessageContext* context, | 
| +    const std::string& message) { | 
| +  if (websocket_.get()) { | 
| +    WebString web_message = WebString::fromUTF8(message); | 
| +    websocket_->sendText(web_message); | 
| +  } | 
| +  return PP_OK; | 
| +} | 
| + | 
| +int32_t PepperWebSocketHost::OnHostMsgSendBinary( | 
| +    ppapi::host::HostMessageContext* context, | 
| +    const std::vector<uint8_t>& message) { | 
| +  if (websocket_.get()) { | 
| +    WebArrayBuffer web_message = WebArrayBuffer::create(message.size(), 1); | 
| +    memcpy(web_message.data(), &message.front(), message.size()); | 
| +    websocket_->sendArrayBuffer(web_message); | 
| +  } | 
| +  return PP_OK; | 
| +} | 
| + | 
| +int32_t PepperWebSocketHost::OnHostMsgFail( | 
| +    ppapi::host::HostMessageContext* context, | 
| +    const std::string& message) { | 
| +  if (websocket_.get()) | 
| +    websocket_->fail(WebString::fromUTF8(message)); | 
| +  return PP_OK; | 
| +} | 
| + | 
| +}  // namespace content | 
|  |