/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ #include "channels/rdpdr/rdpdr-messages.h" #include "channels/rdpdr/rdpdr.h" #include "rdp.h" #include "settings.h" #include #include #include #include #include /** * Sends a Client Announce Reply message. The Client Announce Reply message is * required to be sent in response to the Server Announce Request message. See: * * https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/d6fe6d1b-c145-4a6f-99aa-4fe3cdcea398 * * @param svc * The guac_rdp_common_svc representing the static virtual channel being * used for RDPDR. * * @param major * The major version of the RDPDR protocol in use. This value must always * be 1. * * @param minor * The minor version of the RDPDR protocol in use. This value must be * either 2, 5, 10, 12, or 13. * * @param client_id * The client ID received in the Server Announce Request, or a randomly * generated ID. */ static void guac_rdpdr_send_client_announce_reply(guac_rdp_common_svc* svc, unsigned int major, unsigned int minor, unsigned int client_id) { wStream* output_stream = Stream_New(NULL, 12); /* Write header */ Stream_Write_UINT16(output_stream, RDPDR_CTYP_CORE); Stream_Write_UINT16(output_stream, PAKID_CORE_CLIENTID_CONFIRM); /* Write content */ Stream_Write_UINT16(output_stream, major); Stream_Write_UINT16(output_stream, minor); Stream_Write_UINT32(output_stream, client_id); guac_rdp_common_svc_write(svc, output_stream); } /** * Sends a Client Name Request message. The Client Name Request message is used * by the client to announce its own name. See: * * https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/902497f1-3b1c-4aee-95f8-1668f9b7b7d2 * * @param svc * The guac_rdp_common_svc representing the static virtual channel being * used for RDPDR. * * @param name * The name that should be used for the client. */ static void guac_rdpdr_send_client_name_request(guac_rdp_common_svc* svc, const char* name) { int name_bytes = strlen(name) + 1; wStream* output_stream = Stream_New(NULL, 16 + name_bytes); /* Write header */ Stream_Write_UINT16(output_stream, RDPDR_CTYP_CORE); Stream_Write_UINT16(output_stream, PAKID_CORE_CLIENT_NAME); /* Write content */ Stream_Write_UINT32(output_stream, 0); /* ASCII */ Stream_Write_UINT32(output_stream, 0); /* 0 required by RDPDR spec */ Stream_Write_UINT32(output_stream, name_bytes); Stream_Write(output_stream, name, name_bytes); guac_rdp_common_svc_write(svc, output_stream); } /** * Sends a Client Core Capability Response message. The Client Core Capability * Response message is used to announce the client's capabilities, in response * to receiving the server's capabilities via a Server Core Capability Request. * See: * * https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/f513bf87-cca0-488a-ac5c-18cf18f4a7e1 * * @param svc * The guac_rdp_common_svc representing the static virtual channel being * used for RDPDR. */ static void guac_rdpdr_send_client_capability(guac_rdp_common_svc* svc) { wStream* output_stream = Stream_New(NULL, 256); guac_client_log(svc->client, GUAC_LOG_DEBUG, "Sending capabilities..."); /* Write header */ Stream_Write_UINT16(output_stream, RDPDR_CTYP_CORE); Stream_Write_UINT16(output_stream, PAKID_CORE_CLIENT_CAPABILITY); /* Capability count + padding */ Stream_Write_UINT16(output_stream, 3); Stream_Write_UINT16(output_stream, 0); /* Padding */ /* General capability header */ Stream_Write_UINT16(output_stream, CAP_GENERAL_TYPE); Stream_Write_UINT16(output_stream, 44); Stream_Write_UINT32(output_stream, GENERAL_CAPABILITY_VERSION_02); /* General capability data */ Stream_Write_UINT32(output_stream, GUAC_OS_TYPE); /* osType - required to be ignored */ Stream_Write_UINT32(output_stream, 0); /* osVersion */ Stream_Write_UINT16(output_stream, 1); /* protocolMajor - must be set to 1 */ Stream_Write_UINT16(output_stream, RDPDR_MINOR_RDP_VERSION_5_2); /* protocolMinor */ Stream_Write_UINT32(output_stream, 0xFFFF); /* ioCode1 */ Stream_Write_UINT32(output_stream, 0); /* ioCode2 */ Stream_Write_UINT32(output_stream, RDPDR_DEVICE_REMOVE_PDUS | RDPDR_CLIENT_DISPLAY_NAME_PDU | RDPDR_USER_LOGGEDON_PDU); /* extendedPDU */ Stream_Write_UINT32(output_stream, 0); /* extraFlags1 */ Stream_Write_UINT32(output_stream, 0); /* extraFlags2 */ Stream_Write_UINT32(output_stream, 0); /* SpecialTypeDeviceCap */ /* Printer support header */ Stream_Write_UINT16(output_stream, CAP_PRINTER_TYPE); Stream_Write_UINT16(output_stream, 8); Stream_Write_UINT32(output_stream, PRINT_CAPABILITY_VERSION_01); /* Drive support header */ Stream_Write_UINT16(output_stream, CAP_DRIVE_TYPE); Stream_Write_UINT16(output_stream, 8); Stream_Write_UINT32(output_stream, DRIVE_CAPABILITY_VERSION_02); guac_rdp_common_svc_write(svc, output_stream); guac_client_log(svc->client, GUAC_LOG_DEBUG, "Capabilities sent."); } /** * Sends a Client Device List Announce Request message. The Client Device List * Announce Request message is used by the client to enumerate all devices * which should be made available within the RDP session via RDPDR. See: * * https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/10ef9ada-cba2-4384-ab60-7b6290ed4a9a * * @param svc * The guac_rdp_common_svc representing the static virtual channel being * used for RDPDR. */ static void guac_rdpdr_send_client_device_list_announce_request(guac_rdp_common_svc* svc) { guac_rdpdr* rdpdr = (guac_rdpdr*) svc->data; /* Calculate number of bytes needed for the stream */ int streamBytes = 16; for (int i=0; i < rdpdr->devices_registered; i++) streamBytes += rdpdr->devices[i].device_announce_len; /* Allocate the stream */ wStream* output_stream = Stream_New(NULL, streamBytes); /* Write header */ Stream_Write_UINT16(output_stream, RDPDR_CTYP_CORE); Stream_Write_UINT16(output_stream, PAKID_CORE_DEVICELIST_ANNOUNCE); /* Get the stream for each of the devices. */ Stream_Write_UINT32(output_stream, rdpdr->devices_registered); for (int i=0; idevices_registered; i++) { Stream_Write(output_stream, Stream_Buffer(rdpdr->devices[i].device_announce), rdpdr->devices[i].device_announce_len); guac_client_log(svc->client, GUAC_LOG_DEBUG, "Registered device %i (%s)", rdpdr->devices[i].device_id, rdpdr->devices[i].device_name); } guac_rdp_common_svc_write(svc, output_stream); guac_client_log(svc->client, GUAC_LOG_DEBUG, "All supported devices sent."); } void guac_rdpdr_process_server_announce(guac_rdp_common_svc* svc, wStream* input_stream) { unsigned int major, minor, client_id; if (Stream_GetRemainingLength(input_stream) < 8) return; Stream_Read_UINT16(input_stream, major); Stream_Read_UINT16(input_stream, minor); Stream_Read_UINT32(input_stream, client_id); /* Must choose own client ID if minor not >= 12 */ if (minor < 12) client_id = random() & 0xFFFF; guac_client_log(svc->client, GUAC_LOG_INFO, "Connected to RDPDR %u.%u as client 0x%04x", major, minor, client_id); /* Respond to announce */ guac_rdpdr_send_client_announce_reply(svc, major, minor, client_id); /* Name request */ guac_rdpdr_send_client_name_request(svc, ((guac_rdp_client*) svc->client->data)->settings->client_name); } void guac_rdpdr_process_clientid_confirm(guac_rdp_common_svc* svc, wStream* input_stream) { guac_client_log(svc->client, GUAC_LOG_DEBUG, "Client ID confirmed"); } void guac_rdpdr_process_device_reply(guac_rdp_common_svc* svc, wStream* input_stream) { guac_rdpdr* rdpdr = (guac_rdpdr*) svc->data; unsigned int device_id, ntstatus; int severity, c, n, facility, code; if (Stream_GetRemainingLength(input_stream) < 8) return; Stream_Read_UINT32(input_stream, device_id); Stream_Read_UINT32(input_stream, ntstatus); severity = (ntstatus & 0xC0000000) >> 30; c = (ntstatus & 0x20000000) >> 29; n = (ntstatus & 0x10000000) >> 28; facility = (ntstatus & 0x0FFF0000) >> 16; code = ntstatus & 0x0000FFFF; /* Log error / information */ if (device_id < rdpdr->devices_registered) { if (severity == 0x0) guac_client_log(svc->client, GUAC_LOG_DEBUG, "Device %i (%s) connected successfully", device_id, rdpdr->devices[device_id].device_name); else guac_client_log(svc->client, GUAC_LOG_ERROR, "Problem connecting device %i (%s): " "severity=0x%x, c=0x%x, n=0x%x, facility=0x%x, code=0x%x", device_id, rdpdr->devices[device_id].device_name, severity, c, n, facility, code); } else guac_client_log(svc->client, GUAC_LOG_ERROR, "Unknown device ID: 0x%08x", device_id); } void guac_rdpdr_process_device_iorequest(guac_rdp_common_svc* svc, wStream* input_stream) { guac_rdpdr* rdpdr = (guac_rdpdr*) svc->data; guac_rdpdr_iorequest iorequest; if (Stream_GetRemainingLength(input_stream) < 20) return; /* Read header */ Stream_Read_UINT32(input_stream, iorequest.device_id); Stream_Read_UINT32(input_stream, iorequest.file_id); Stream_Read_UINT32(input_stream, iorequest.completion_id); Stream_Read_UINT32(input_stream, iorequest.major_func); Stream_Read_UINT32(input_stream, iorequest.minor_func); /* If printer, run printer handlers */ if (iorequest.device_id >= 0 && iorequest.device_id < rdpdr->devices_registered) { /* Call handler on device */ guac_rdpdr_device* device = &(rdpdr->devices[iorequest.device_id]); device->iorequest_handler(svc, device, &iorequest, input_stream); } else guac_client_log(svc->client, GUAC_LOG_ERROR, "Unknown device ID: " "0x%08x", iorequest.device_id); } void guac_rdpdr_process_server_capability(guac_rdp_common_svc* svc, wStream* input_stream) { int count; int i; if (Stream_GetRemainingLength(input_stream) < 4) return; /* Read header */ Stream_Read_UINT16(input_stream, count); Stream_Seek(input_stream, 2); /* Parse capabilities */ for (i=0; iclient, GUAC_LOG_DEBUG, "Ignoring server capability set type=0x%04x, length=%i", type, length); Stream_Seek(input_stream, length - 4); } /* Send own capabilities */ guac_rdpdr_send_client_capability(svc); } void guac_rdpdr_process_user_loggedon(guac_rdp_common_svc* svc, wStream* input_stream) { guac_client_log(svc->client, GUAC_LOG_INFO, "RDPDR user logged on"); guac_rdpdr_send_client_device_list_announce_request(svc); } void guac_rdpdr_process_prn_cache_data(guac_rdp_common_svc* svc, wStream* input_stream) { guac_client_log(svc->client, GUAC_LOG_DEBUG, "Ignoring printer cached configuration data"); } void guac_rdpdr_process_prn_using_xps(guac_rdp_common_svc* svc, wStream* input_stream) { guac_client_log(svc->client, GUAC_LOG_WARNING, "Printer unexpectedly switched to XPS mode"); }