blob: f07145765235e363b1474956069c875d678123a2 [file] [log] [blame]
// Copyright 2018 The Fuchsia 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 "rfb_server.h"
#include <arpa/inet.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <vector>
namespace {
struct pixel_format {
uint8_t bits_per_pixel;
uint8_t depth;
uint8_t big_endian;
uint8_t true_color;
uint16_t red_max;
uint16_t green_max;
uint16_t blue_max;
uint8_t red_shift;
uint8_t green_shift;
uint8_t blue_shift;
uint8_t padding[3];
};
} // namespace
// See https://tools.ietf.org/html/rfc6143 for protocol documentation.
bool RFBServer::Initialize(uint32_t width, uint32_t height, uint32_t port) {
if (initialization_attempted_) {
return initialization_succeeded_;
}
initialization_attempted_ = true;
width_ = width;
height_ = height;
constexpr uint32_t kHeaderLength = 12;
const char kHeader[kHeaderLength + 1] = "RFB 003.008\n";
int listen_fd = socket(AF_INET6, SOCK_STREAM, 0);
if (listen_fd < 0) {
perror("socket failed\n");
return false;
}
int enabled = 1;
if (setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &enabled,
sizeof(enabled))) {
perror("so_reuseaddr failed\n");
return false;
}
struct sockaddr_in6 serveraddr;
memset(&serveraddr, 0, sizeof(serveraddr));
serveraddr.sin6_family = AF_INET6;
serveraddr.sin6_port = htons(port);
serveraddr.sin6_addr = in6addr_any;
int bind_result =
::bind(listen_fd, (struct sockaddr*)&serveraddr, sizeof(serveraddr));
if (bind_result < 0) {
perror("bind failed\n");
return false;
}
constexpr int kListenBacklog = 10;
if (listen(listen_fd, kListenBacklog)) {
perror("listen failed\n");
return false;
}
fprintf(stderr, "RFB layer waiting on port %u\n", port);
fd_ = fxl::UniqueFD(accept(listen_fd, nullptr, nullptr));
if (!fd_.is_valid()) {
perror("accept failed\n");
return false;
}
if (!SendBytes(kHeader, kHeaderLength)) return false;
char read_header[kHeaderLength + 1];
if (ReadEntireMessage(read_header, kHeaderLength) < 0) return false;
if (memcmp(read_header, kHeader, kHeaderLength) != 0) {
read_header[kHeaderLength] = 0;
fprintf(stderr, "Received unsupported header value %s\n", read_header);
return false;
}
constexpr uint8_t kSecurityTypeNone = 1;
uint8_t security_header[] = {1, kSecurityTypeNone};
if (!SendBytes(security_header, sizeof(security_header))) return false;
uint8_t security_type;
if (ReadEntireMessage(&security_type, sizeof(security_type)) < 0)
return false;
if (security_type != kSecurityTypeNone) return false;
uint32_t security_status = 0;
if (!SendBytes(&security_status, sizeof(security_status))) return false;
uint8_t shared_flag; // ignored.
if (ReadEntireMessage(&shared_flag, sizeof(shared_flag)) < 0) return false;
struct server_init {
uint16_t width;
uint16_t height;
struct pixel_format pixel_format;
uint32_t name_length;
uint8_t name_string[5];
} __attribute__((packed)) server_init;
server_init.width = htons(width);
server_init.height = htons(height);
server_init.pixel_format.bits_per_pixel = 32;
server_init.pixel_format.depth = 1;
server_init.pixel_format.big_endian = 0;
server_init.pixel_format.true_color = 1;
server_init.pixel_format.red_max = htons(255);
server_init.pixel_format.green_max = htons(255);
server_init.pixel_format.blue_max = htons(255);
server_init.pixel_format.red_shift = 0;
server_init.pixel_format.green_shift = 8;
server_init.pixel_format.blue_shift = 16;
server_init.name_length = htonl(5);
server_init.name_string[0] = 'm';
server_init.name_string[1] = 'a';
server_init.name_string[2] = 'g';
server_init.name_string[3] = 'm';
server_init.name_string[4] = 'a';
if (!SendBytes(&server_init, sizeof(server_init))) return false;
initialization_succeeded_ = true;
return true;
}
void RFBServer::WaitForFramebufferUpdate() {
if (!initialization_succeeded_) return;
while (true) {
uint8_t message_type;
int return_code = ReadEntireMessage(&message_type, sizeof(message_type));
if (return_code < 0) {
perror("ReadEntireMessage failure");
return;
}
switch (message_type) {
case 0: {
// In theory the server should transmit messages using this pixel format
// afterwards, but we just ignore it and use the one that was initially
// sent to the client.
struct set_pixel_format {
uint8_t padding[3];
pixel_format format;
} __attribute__((packed)) set_pixel_format;
if (ReadEntireMessage(&set_pixel_format, sizeof(set_pixel_format)) < 0)
return;
break;
}
case 2: {
struct set_encodings {
uint8_t padding;
uint16_t encoding_count;
} __attribute__((packed)) set_encodings;
if (ReadEntireMessage(&set_encodings, sizeof(set_encodings)) < 0)
return;
uint16_t encoding_count = ntohs(set_encodings.encoding_count);
std::vector<int32_t> encodings(encoding_count);
if (ReadEntireMessage(encodings.data(), encoding_count * 4) < 0) return;
break;
}
case 3: {
// The server should only send updates for the requested region, but
// instead we give updates for the entire frame and hope the client can
// understand.
struct framebuffer_update_request {
uint8_t incremental;
uint16_t x_position;
uint16_t y_position;
uint16_t width;
uint16_t height;
} __attribute__((packed)) update_request;
ReadEntireMessage(&update_request, sizeof(update_request));
return;
}
case 4: {
struct key_event {
uint8_t down_flag;
uint8_t padding[2];
uint32_t key;
} __attribute__((packed)) key_event;
if (ReadEntireMessage(&key_event, sizeof(key_event)) < 0) return;
break;
}
case 5: {
struct pointer_event {
uint8_t button_mask;
uint16_t x_position;
uint16_t y_position;
} __attribute__((packed)) pointer_event;
if (ReadEntireMessage(&pointer_event, sizeof(pointer_event)) < 0)
return;
break;
}
case 6: {
struct cut_text {
uint8_t padding[3];
uint32_t length;
} __attribute__((packed)) cut_text;
if (ReadEntireMessage(&cut_text, sizeof(cut_text)) < 0) return;
cut_text.length = ntohl(cut_text.length);
std::vector<uint8_t> text(cut_text.length);
if (ReadEntireMessage(text.data(), cut_text.length) < 0) return;
break;
}
}
}
}
void RFBServer::StartUpdate() {
if (!initialization_succeeded_) return;
struct update_header {
uint8_t type;
uint8_t padding;
uint16_t number_of_rectangles;
} update_header;
update_header.type = 0;
update_header.padding = 0;
update_header.number_of_rectangles = htons(1);
if (!SendBytes(&update_header, sizeof(update_header))) return;
struct rectangle_header {
uint16_t x_position;
uint16_t y_position;
uint16_t width;
uint16_t height;
int32_t encoding_type;
} rectangle_header;
rectangle_header.x_position = htons(0);
rectangle_header.y_position = htons(0);
rectangle_header.width = htons(width_);
rectangle_header.height = htons(height_);
rectangle_header.encoding_type = htonl(0);
SendBytes(&rectangle_header, sizeof(rectangle_header));
}
bool RFBServer::SendBytes(const void* data, uint32_t length) {
if (!fd_.is_valid()) return false;
const char* current_data = static_cast<const char*>(data);
uint32_t written = 0;
while (written < length) {
int current_write = send(fd_.get(), current_data, length - written, 0);
if (current_write < 0) {
if (errno == EINTR) continue;
return false;
}
current_data += current_write;
written += current_write;
}
return true;
}
int RFBServer::ReadEntireMessage(void* data, uint32_t size) {
uint32_t read = 0;
char* current_data = static_cast<char*>(data);
while (read < size) {
int current_read = recv(fd_.get(), current_data, size - read, 0);
if (current_read < 0) {
if (errno == EINTR) continue;
return current_read;
}
read += current_read;
current_data += current_read;
}
return read;
}