|  | /* | 
|  | *  Copyright 2010 The WebRTC Project Authors. All rights reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "rtc_base/socket_stream.h" | 
|  |  | 
|  | #include "rtc_base/checks.h" | 
|  | #include "rtc_base/socket.h" | 
|  |  | 
|  | namespace rtc { | 
|  |  | 
|  | SocketStream::SocketStream(Socket* socket) : socket_(nullptr) { | 
|  | Attach(socket); | 
|  | } | 
|  |  | 
|  | SocketStream::~SocketStream() { | 
|  | delete socket_; | 
|  | } | 
|  |  | 
|  | void SocketStream::Attach(Socket* socket) { | 
|  | if (socket_) | 
|  | delete socket_; | 
|  | socket_ = socket; | 
|  | if (socket_) { | 
|  | socket_->SignalConnectEvent.connect(this, &SocketStream::OnConnectEvent); | 
|  | socket_->SignalReadEvent.connect(this, &SocketStream::OnReadEvent); | 
|  | socket_->SignalWriteEvent.connect(this, &SocketStream::OnWriteEvent); | 
|  | socket_->SignalCloseEvent.connect(this, &SocketStream::OnCloseEvent); | 
|  | } | 
|  | } | 
|  |  | 
|  | Socket* SocketStream::Detach() { | 
|  | Socket* socket = socket_; | 
|  | if (socket_) { | 
|  | socket_->SignalConnectEvent.disconnect(this); | 
|  | socket_->SignalReadEvent.disconnect(this); | 
|  | socket_->SignalWriteEvent.disconnect(this); | 
|  | socket_->SignalCloseEvent.disconnect(this); | 
|  | socket_ = nullptr; | 
|  | } | 
|  | return socket; | 
|  | } | 
|  |  | 
|  | StreamState SocketStream::GetState() const { | 
|  | RTC_DCHECK(socket_ != nullptr); | 
|  | switch (socket_->GetState()) { | 
|  | case Socket::CS_CONNECTED: | 
|  | return SS_OPEN; | 
|  | case Socket::CS_CONNECTING: | 
|  | return SS_OPENING; | 
|  | case Socket::CS_CLOSED: | 
|  | default: | 
|  | return SS_CLOSED; | 
|  | } | 
|  | } | 
|  |  | 
|  | StreamResult SocketStream::Read(rtc::ArrayView<uint8_t> buffer, | 
|  | size_t& read, | 
|  | int& error) { | 
|  | RTC_DCHECK(socket_ != nullptr); | 
|  | int result = socket_->Recv(buffer.data(), buffer.size(), nullptr); | 
|  | if (result < 0) { | 
|  | if (socket_->IsBlocking()) | 
|  | return SR_BLOCK; | 
|  | error = socket_->GetError(); | 
|  | return SR_ERROR; | 
|  | } | 
|  | if ((result > 0) || (buffer.size() == 0)) { | 
|  | read = result; | 
|  | return SR_SUCCESS; | 
|  | } | 
|  | return SR_EOS; | 
|  | } | 
|  |  | 
|  | StreamResult SocketStream::Write(rtc::ArrayView<const uint8_t> data, | 
|  | size_t& written, | 
|  | int& error) { | 
|  | RTC_DCHECK(socket_ != nullptr); | 
|  | int result = socket_->Send(data.data(), data.size()); | 
|  | if (result < 0) { | 
|  | if (socket_->IsBlocking()) | 
|  | return SR_BLOCK; | 
|  | error = socket_->GetError(); | 
|  | return SR_ERROR; | 
|  | } | 
|  | written = result; | 
|  | return SR_SUCCESS; | 
|  | } | 
|  |  | 
|  | void SocketStream::Close() { | 
|  | RTC_DCHECK(socket_ != nullptr); | 
|  | socket_->Close(); | 
|  | } | 
|  |  | 
|  | void SocketStream::OnConnectEvent(Socket* socket) { | 
|  | RTC_DCHECK(socket == socket_); | 
|  | SignalEvent(this, SE_OPEN | SE_READ | SE_WRITE, 0); | 
|  | } | 
|  |  | 
|  | void SocketStream::OnReadEvent(Socket* socket) { | 
|  | RTC_DCHECK(socket == socket_); | 
|  | SignalEvent(this, SE_READ, 0); | 
|  | } | 
|  |  | 
|  | void SocketStream::OnWriteEvent(Socket* socket) { | 
|  | RTC_DCHECK(socket == socket_); | 
|  | SignalEvent(this, SE_WRITE, 0); | 
|  | } | 
|  |  | 
|  | void SocketStream::OnCloseEvent(Socket* socket, int err) { | 
|  | RTC_DCHECK(socket == socket_); | 
|  | SignalEvent(this, SE_CLOSE, err); | 
|  | } | 
|  |  | 
|  | }  // namespace rtc |