| /* | 
 |  *  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 |