blob: 5c993ea233ea1c9e9eaa33a89d3bfea027ceca43 [file] [log] [blame]
henrike@webrtc.orgf0488722014-05-13 18:00:26 +00001/*
2 * Copyright 2010 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Steve Anton10542f22019-01-11 09:11:00 -080011#include "rtc_base/socket_stream.h"
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000012
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020013#include "rtc_base/checks.h"
Yves Gerey988cc082018-10-23 12:03:01 +020014#include "rtc_base/socket.h"
nisseede5da42017-01-12 05:15:36 -080015
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000016namespace rtc {
17
Niels Möllerd0b88792021-08-12 10:32:30 +020018SocketStream::SocketStream(Socket* socket) : socket_(nullptr) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000019 Attach(socket);
20}
21
22SocketStream::~SocketStream() {
23 delete socket_;
24}
25
Niels Möllerd0b88792021-08-12 10:32:30 +020026void SocketStream::Attach(Socket* socket) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000027 if (socket_)
28 delete socket_;
29 socket_ = socket;
30 if (socket_) {
31 socket_->SignalConnectEvent.connect(this, &SocketStream::OnConnectEvent);
Yves Gerey665174f2018-06-19 15:03:05 +020032 socket_->SignalReadEvent.connect(this, &SocketStream::OnReadEvent);
33 socket_->SignalWriteEvent.connect(this, &SocketStream::OnWriteEvent);
34 socket_->SignalCloseEvent.connect(this, &SocketStream::OnCloseEvent);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000035 }
36}
37
Niels Möllerd0b88792021-08-12 10:32:30 +020038Socket* SocketStream::Detach() {
39 Socket* socket = socket_;
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000040 if (socket_) {
41 socket_->SignalConnectEvent.disconnect(this);
42 socket_->SignalReadEvent.disconnect(this);
43 socket_->SignalWriteEvent.disconnect(this);
44 socket_->SignalCloseEvent.disconnect(this);
deadbeef37f5ecf2017-02-27 14:06:41 -080045 socket_ = nullptr;
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000046 }
47 return socket;
48}
49
50StreamState SocketStream::GetState() const {
deadbeef37f5ecf2017-02-27 14:06:41 -080051 RTC_DCHECK(socket_ != nullptr);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000052 switch (socket_->GetState()) {
53 case Socket::CS_CONNECTED:
54 return SS_OPEN;
55 case Socket::CS_CONNECTING:
56 return SS_OPENING;
57 case Socket::CS_CLOSED:
58 default:
59 return SS_CLOSED;
60 }
61}
62
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000063StreamResult SocketStream::Read(rtc::ArrayView<uint8_t> buffer,
64 size_t& read,
65 int& error) {
deadbeef37f5ecf2017-02-27 14:06:41 -080066 RTC_DCHECK(socket_ != nullptr);
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000067 int result = socket_->Recv(buffer.data(), buffer.size(), nullptr);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000068 if (result < 0) {
69 if (socket_->IsBlocking())
70 return SR_BLOCK;
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000071 error = socket_->GetError();
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000072 return SR_ERROR;
73 }
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000074 if ((result > 0) || (buffer.size() == 0)) {
75 read = result;
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000076 return SR_SUCCESS;
77 }
78 return SR_EOS;
79}
80
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000081StreamResult SocketStream::Write(rtc::ArrayView<const uint8_t> data,
82 size_t& written,
83 int& error) {
deadbeef37f5ecf2017-02-27 14:06:41 -080084 RTC_DCHECK(socket_ != nullptr);
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000085 int result = socket_->Send(data.data(), data.size());
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000086 if (result < 0) {
87 if (socket_->IsBlocking())
88 return SR_BLOCK;
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000089 error = socket_->GetError();
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000090 return SR_ERROR;
91 }
Harald Alvestrand7ccd88f2022-11-16 08:59:58 +000092 written = result;
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000093 return SR_SUCCESS;
94}
95
96void SocketStream::Close() {
deadbeef37f5ecf2017-02-27 14:06:41 -080097 RTC_DCHECK(socket_ != nullptr);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000098 socket_->Close();
99}
100
Niels Möllerd0b88792021-08-12 10:32:30 +0200101void SocketStream::OnConnectEvent(Socket* socket) {
nisseede5da42017-01-12 05:15:36 -0800102 RTC_DCHECK(socket == socket_);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000103 SignalEvent(this, SE_OPEN | SE_READ | SE_WRITE, 0);
104}
105
Niels Möllerd0b88792021-08-12 10:32:30 +0200106void SocketStream::OnReadEvent(Socket* socket) {
nisseede5da42017-01-12 05:15:36 -0800107 RTC_DCHECK(socket == socket_);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000108 SignalEvent(this, SE_READ, 0);
109}
110
Niels Möllerd0b88792021-08-12 10:32:30 +0200111void SocketStream::OnWriteEvent(Socket* socket) {
nisseede5da42017-01-12 05:15:36 -0800112 RTC_DCHECK(socket == socket_);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000113 SignalEvent(this, SE_WRITE, 0);
114}
115
Niels Möllerd0b88792021-08-12 10:32:30 +0200116void SocketStream::OnCloseEvent(Socket* socket, int err) {
nisseede5da42017-01-12 05:15:36 -0800117 RTC_DCHECK(socket == socket_);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000118 SignalEvent(this, SE_CLOSE, err);
119}
120
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000121} // namespace rtc