blob: 53cb38932349c1fdfedc17a9558b0ecc209e0e36 [file] [log] [blame]
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -08001/*
2 * Copyright (C) 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "host/frontend/vnc_server/screen_connector.h"
18
19#include <atomic>
20#include <condition_variable>
21
22#include <gflags/gflags.h>
23
Cody Schuffelen134ff032019-11-22 00:25:32 -080024#include <common/vsoc/lib/screen_region_view.h>
25#include <host/libs/config/cuttlefish_config.h>
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080026#include "host/frontend/vnc_server/vnc_utils.h"
27
28DEFINE_int32(frame_server_fd, -1, "");
29
30namespace cvd {
31namespace vnc {
32
33namespace {
Cody Schuffelen134ff032019-11-22 00:25:32 -080034class VSoCScreenConnector : public ScreenConnector {
35 public:
36 int WaitForNewFrameSince(std::uint32_t* seq_num) override {
37 if (!screen_view_) return -1;
38 return screen_view_->WaitForNewFrameSince(seq_num);
39 }
40
41 void* GetBuffer(int buffer_idx) override {
42 if (!screen_view_) return nullptr;
43 return screen_view_->GetBuffer(buffer_idx);
44 }
45
46 private:
47 vsoc::screen::ScreenRegionView* screen_view_ =
48 vsoc::screen::ScreenRegionView::GetInstance(vsoc::GetDomain().c_str());
49};
50
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080051// TODO(b/128852363): Substitute with one based on memory shared with the
52// wayland mock
53class SocketBasedScreenConnector : public ScreenConnector {
54 public:
Jorge E. Moreira37e91b42019-06-20 17:53:35 -070055 SocketBasedScreenConnector() {
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080056 screen_server_thread_ = std::thread([this]() { ServerLoop(); });
57 }
58
59 int WaitForNewFrameSince(std::uint32_t* seq_num) override {
60 std::unique_lock<std::mutex> lock(new_frame_mtx_);
61 while (seq_num_ == *seq_num) {
62 new_frame_cond_var_.wait(lock);
63 }
Jorge E. Moreirab15851c2019-05-07 18:09:59 -070064 *seq_num = seq_num_;
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080065 return newest_buffer_;
66 }
67
68 void* GetBuffer(int buffer_idx) override {
69 if (buffer_idx < 0) return nullptr;
70 buffer_idx %= NUM_BUFFERS_;
71 return &buffer_[buffer_idx * ScreenSizeInBytes()];
72 }
73
74 private:
75 static constexpr int NUM_BUFFERS_ = 4;
76
77 void ServerLoop() {
78 if (FLAGS_frame_server_fd < 0) {
79 LOG(FATAL) << "Invalid file descriptor: " << FLAGS_frame_server_fd;
80 return;
81 }
82 auto server = SharedFD::Dup(FLAGS_frame_server_fd);
83 close(FLAGS_frame_server_fd);
84 if (!server->IsOpen()) {
85 LOG(FATAL) << "Unable to dup screen server: " << server->StrError();
86 return;
87 }
88
89 int current_buffer = 0;
90
91 while (1) {
Jorge E. Moreira52b8fca2019-06-12 16:26:41 -070092 LOG(INFO) << "Screen Connector accepting connections...";
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080093 auto conn = SharedFD::Accept(*server);
Jorge E. Moreira52b8fca2019-06-12 16:26:41 -070094 if (!conn->IsOpen()) {
95 LOG(ERROR) << "Disconnected fd returned from accept";
96 continue;
97 }
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080098 while (conn->IsOpen()) {
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -080099 int32_t size = 0;
Cody Schuffelen0b82c632019-10-23 14:00:24 -0700100 if (conn->Read(&size, sizeof(size)) < 0) {
101 LOG(ERROR) << "Failed to read from hwcomposer: "
102 << conn->StrError();
103 break;
104 }
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -0800105 auto buff = reinterpret_cast<uint8_t*>(GetBuffer(current_buffer));
106 while (size > 0) {
107 auto read = conn->Read(buff, size);
108 if (read < 0) {
109 LOG(ERROR) << "Failed to read from hwcomposer: "
110 << conn->StrError();
Cody Schuffelen0b82c632019-10-23 14:00:24 -0700111 conn->Close();
112 break;
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -0800113 }
114 size -= read;
115 buff += read;
116 }
117 BroadcastNewFrame(current_buffer);
118 current_buffer = (current_buffer + 1) % NUM_BUFFERS_;
119 }
120 }
121 }
122
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -0800123 void BroadcastNewFrame(int buffer_idx) {
124 {
125 std::lock_guard<std::mutex> lock(new_frame_mtx_);
126 seq_num_++;
127 newest_buffer_ = buffer_idx;
128 }
129 new_frame_cond_var_.notify_all();
130 }
131
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -0800132 std::vector<std::uint8_t> buffer_ =
133 std::vector<std::uint8_t>(NUM_BUFFERS_ * ScreenSizeInBytes());
134 std::uint32_t seq_num_{0};
135 int newest_buffer_ = 0;
136 std::condition_variable new_frame_cond_var_;
137 std::mutex new_frame_mtx_;
138 std::thread screen_server_thread_;
139};
140} // namespace
141
142ScreenConnector* ScreenConnector::Get() {
Cody Schuffelen134ff032019-11-22 00:25:32 -0800143 auto config = vsoc::CuttlefishConfig::Get();
144 if (config->enable_ivserver()) {
145 return new VSoCScreenConnector();
146 } else {
147 return new SocketBasedScreenConnector();
148 }
Jorge E. Moreirac87c2c72019-03-06 16:12:23 -0800149}
150
151} // namespace vnc
Jorge E. Moreira37e91b42019-06-20 17:53:35 -0700152} // namespace cvd