blob: dad3548d593d7454009bce4d60aa89b1b61c6d08 [file] [log] [blame]
// 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.
package rdpclient.rdp;
import streamer.BaseElement;
import streamer.ByteBuffer;
import streamer.Element;
import streamer.Link;
import streamer.Pipeline;
import streamer.PipelineImpl;
import streamer.debug.MockSink;
import streamer.debug.MockSource;
public class ServerIOChannelRouter extends BaseElement {
/**
* Demand Active PDU.
*/
public static final int PDUTYPE_DEMANDACTIVEPDU = 0x1;
/**
* Confirm Active PDU.
*/
public static final int PDUTYPE_CONFIRMACTIVEPDU = 0x3;
/**
* Deactivate All PDU.
*/
public static final int PDUTYPE_DEACTIVATEALLPDU = 0x6;
/**
* Data PDU (actual type is revealed by the pduType2 field in the Share Data
* Header).
*/
public static final int PDUTYPE_DATAPDU = 0x7;
/**
* Enhanced Security Server Redirection PDU.
*/
public static final int PDUTYPE_SERVER_REDIR_PKT = 0xA;
protected RdpState state;
public ServerIOChannelRouter(String id, RdpState state) {
super(id);
this.state = state;
}
/**
* @see http://msdn.microsoft.com/en-us/library/cc240576.aspx
*/
@Override
public void handleData(ByteBuffer buf, Link link) {
if (verbose)
System.out.println("[" + this + "] INFO: Data received: " + buf + ".");
int length = buf.readUnsignedShortLE();
if (buf.length != length)
{
// It is ServerErrorAlert-ValidClient
// Ignore it
//throw new RuntimeException("[" + this + "] ERROR: Incorrect PDU length: " + length + ", data: " + buf + ".");
}
int type = buf.readUnsignedShortLE() & 0xf;
// int sourceId = buf.readUnsignedShortLE();
buf.skipBytes(2);
switch (type) {
case PDUTYPE_DEMANDACTIVEPDU:
pushDataToPad("demand_active", buf);
break;
case PDUTYPE_CONFIRMACTIVEPDU:
throw new RuntimeException("Unexpected client CONFIRM ACTIVE PDU. Data: " + buf + ".");
case PDUTYPE_DEACTIVATEALLPDU:
// pushDataToPad("deactivate_all", buf);
/* ignore */buf.unref();
break;
case PDUTYPE_DATAPDU:
handleDataPdu(buf);
break;
case PDUTYPE_SERVER_REDIR_PKT:
// pushDataToPad("server_redir", buf);
/* ignore */buf.unref();
break;
default:
throw new RuntimeException("[" + this + "] ERROR: Unknown PDU type: " + type + ", data: " + buf + ".");
}
}
/**
* Graphics Update PDU.
*/
public static final int PDUTYPE2_UPDATE = 0x02;
/**
* Control PDU.
*/
public static final int PDUTYPE2_CONTROL = 0x14;
/**
* Pointer Update PDU.
*/
public static final int PDUTYPE2_POINTER = 0x1B;
/**
* Input Event PDU.
*/
public static final int PDUTYPE2_INPUT = 0x1C;
/**
* Synchronize PDU.
*/
public static final int PDUTYPE2_SYNCHRONIZE = 0x1F;
/**
* Refresh Rect PDU.
*/
public static final int PDUTYPE2_REFRESH_RECT = 0x21;
/**
* Play Sound PDU.
*/
public static final int PDUTYPE2_PLAY_SOUND = 0x22;
/**
* Suppress Output PDU.
*/
public static final int PDUTYPE2_SUPPRESS_OUTPUT = 0x23;
/**
* Shutdown Request PDU.
*/
public static final int PDUTYPE2_SHUTDOWN_REQUEST = 0x24;
/**
* Shutdown Request Denied PDU.
*/
public static final int PDUTYPE2_SHUTDOWN_DENIED = 0x25;
/**
* Save Session Info PDU.
*/
public static final int PDUTYPE2_SAVE_SESSION_INFO = 0x26;
/**
* Font List PDU.
*/
public static final int PDUTYPE2_FONTLIST = 0x27;
/**
* Font Map PDU.
*/
public static final int PDUTYPE2_FONTMAP = 0x28;
/**
* Set Keyboard Indicators PDU.
*/
public static final int PDUTYPE2_SET_KEYBOARD_INDICATORS = 0x29;
/**
* Persistent Key List PDU.
*/
public static final int PDUTYPE2_BITMAPCACHE_PERSISTENT_LIST = 0x2B;
/**
* Bitmap Cache Error PDU.
*/
public static final int PDUTYPE2_BITMAPCACHE_ERROR_PDU = 0x2C;
/**
* Set Keyboard IME Status PDU.
*/
public static final int PDUTYPE2_SET_KEYBOARD_IME_STATUS = 0x2D;
/**
* Offscreen Bitmap Cache Error PDU.
*/
public static final int PDUTYPE2_OFFSCRCACHE_ERROR_PDU = 0x2E;
/**
* Set Error Info PDU.
*/
public static final int PDUTYPE2_SET_ERROR_INFO_PDU = 0x2F;
/**
* DrawNineGrid Cache Error PDU.
*/
public static final int PDUTYPE2_DRAWNINEGRID_ERROR_PDU = 0x30;
/**
* GDI+ Error PDU.
*/
public static final int PDUTYPE2_DRAWGDIPLUS_ERROR_PDU = 0x31;
/**
* Auto-Reconnect Status PDU.
*/
public static final int PDUTYPE2_ARC_STATUS_PDU = 0x32;
/**
* Status Info PDU.
*/
public static final int PDUTYPE2_STATUS_INFO_PDU = 0x36;
/**
* Monitor Layout PDU.
*/
public static final int PDUTYPE2_MONITOR_LAYOUT_PDU = 0x37;
/**
* Indicates an Orders Update.
*/
public static final int UPDATETYPE_ORDERS = 0x0000;
/**
* Indicates a Bitmap Graphics Update.
*/
public static final int UPDATETYPE_BITMAP = 0x0001;
/**
* Indicates a Palette Update.
*/
public static final int UPDATETYPE_PALETTE = 0x0002;
/**
* Indicates a Synchronize Update.
*/
public static final int UPDATETYPE_SYNCHRONIZE = 0x0003;
/**
* @see http://msdn.microsoft.com/en-us/library/cc240577.aspx
*/
protected void handleDataPdu(ByteBuffer buf) {
// (4 bytes): A 32-bit, unsigned integer. Share identifier for the packet.
long shareId = buf.readUnsignedIntLE();
if (shareId != state.serverShareId)
throw new RuntimeException("Unexpected share ID: " + shareId + ".");
// buf.skipBytes(4);
// Padding.
buf.skipBytes(1);
// (1 byte): An 8-bit, unsigned integer. The stream identifier for the
// packet.
// int streamId = buf.readUnsignedByte();
buf.skipBytes(1);
// (2 bytes): A 16-bit, unsigned integer. The uncompressed length of the
// packet in bytes.
int uncompressedLength = buf.readUnsignedShortLE();
// (1 byte): An 8-bit, unsigned integer. The type of Data PDU.
int type2 = buf.readUnsignedByte();
// (1 byte): An 8-bit, unsigned integer. The compression type and flags
// specifying the data following the Share Data Header
int compressedType = buf.readUnsignedByte();
if (compressedType != 0)
throw new RuntimeException("Compression of protocol packets is not supported. Data: " + buf + ".");
// (2 bytes): A 16-bit, unsigned integer. The compressed length of the
// packet in bytes.
int compressedLength = buf.readUnsignedShortLE();
if (compressedLength != 0)
throw new RuntimeException("Compression of protocol packets is not supported. Data: " + buf + ".");
ByteBuffer data = buf.readBytes(uncompressedLength - 18);
buf.unref();
switch (type2) {
case PDUTYPE2_UPDATE: {
// (2 bytes): A 16-bit, unsigned integer. Type of the graphics update.
int updateType = data.readUnsignedShortLE();
ByteBuffer payload = data.readBytes(data.length - data.cursor);
data.unref();
switch (updateType) {
case UPDATETYPE_ORDERS:
pushDataToPad("orders", payload);
break;
case UPDATETYPE_BITMAP:
pushDataToPad("bitmap", payload);
break;
case UPDATETYPE_PALETTE:
pushDataToPad("palette", payload);
break;
case UPDATETYPE_SYNCHRONIZE:
// Ignore
payload.unref();
break;
}
break;
}
case PDUTYPE2_CONTROL:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_CONTROL ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_POINTER:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_POINTER ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_INPUT:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_INPUT ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SYNCHRONIZE:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SYNCHRONIZE ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_REFRESH_RECT:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_REFRESH_RECT ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_PLAY_SOUND:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_PLAY_SOUND ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SUPPRESS_OUTPUT:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SUPPRESS_OUTPUT ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SHUTDOWN_REQUEST:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SHUTDOWN_REQUEST ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SHUTDOWN_DENIED:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SHUTDOWN_DENIED ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SAVE_SESSION_INFO:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SAVE_SESSION_INFO ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_FONTLIST:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_FONTLIST ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_FONTMAP:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_FONTMAP ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SET_KEYBOARD_INDICATORS:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SET_KEYBOARD_INDICATORS ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_BITMAPCACHE_PERSISTENT_LIST:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_BITMAPCACHE_PERSISTENT_LIST ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_BITMAPCACHE_ERROR_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_BITMAPCACHE_ERROR_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SET_KEYBOARD_IME_STATUS:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SET_KEYBOARD_IME_STATUS ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_OFFSCRCACHE_ERROR_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_OFFSCRCACHE_ERROR_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_SET_ERROR_INFO_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_SET_ERROR_INFO_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_DRAWNINEGRID_ERROR_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_DRAWNINEGRID_ERROR_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_DRAWGDIPLUS_ERROR_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_DRAWGDIPLUS_ERROR_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_ARC_STATUS_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_ARC_STATUS_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_STATUS_INFO_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_STATUS_INFO_PDU ignored.");
// Ignore
data.unref();
break;
case PDUTYPE2_MONITOR_LAYOUT_PDU:
if (verbose)
System.out.println("[" + this + "] INFO: Packet PDUTYPE2_MONITOR_LAYOUT_PDU ignored.");
// Ignore
data.unref();
break;
default:
throw new RuntimeException("Unknow data PDU type: " + type2 + ", data: " + buf + ".");
}
}
/**
* Example.
*
*/
public static void main(String args[]) {
// System.setProperty("streamer.Link.debug", "true");
System.setProperty("streamer.Element.debug", "true");
// System.setProperty("streamer.Pipeline.debug", "true");
byte[] packet = new byte[] {
// TPKT
(byte)0x03, (byte)0x00, // TPKT Header: TPKT version = 3
(byte)0x00, (byte)0x1B, // TPKT length: 27 bytes
// X224
(byte)0x02, // X224 Length: 2 bytes
(byte)0xF0, // X224 Type: Data
(byte)0x80, // X224 EOT
// MCS
// Type: send data indication: 26 (0x1a, top 6 bits)
(byte)0x68, // ??
(byte)0x00, (byte)0x01, // User ID: 1002 (1001+1)
(byte)0x03, (byte)0xEB, // Channel ID: 1003
(byte)0x70, // Data priority: high, segmentation: begin|end
(byte)0x0D, // Payload length: 13 bytes
// Deactivate all PDU
(byte)0x0D, (byte)0x00, // Length: 13 bytes (LE)
// - PDUType: (0x16, LE)
// Type: (............0110) TS_PDUTYPE_DEACTIVATEALLPDU
// ProtocolVersion: (000000000001....) 1
(byte)0x16, (byte)0x00,
(byte)0xEA, (byte)0x03, // PDU source: 1002 (LE)
(byte)0xEA, (byte)0x03, (byte)0x01, (byte)0x00, // ShareID = 66538
(byte)0x01, (byte)0x00, // Length if source descriptor: 1 (LE)
(byte)0x00, // Source descriptor (should be set to 0): 0
};
MockSource source = new MockSource("source", ByteBuffer.convertByteArraysToByteBuffers(packet));
RdpState rdpState = new RdpState() {
{
serverShareId = 66538;
}
};
Element channel1003 = new ServerIOChannelRouter("channel_1003", rdpState);
Element mcs = new ServerMCSPDU("mcs");
Element tpkt = new ServerTpkt("tpkt");
Element x224 = new ServerX224DataPdu("x224");
Element sink = new MockSink("sink", ByteBuffer.convertByteArraysToByteBuffers(new byte[] {
// Deactivate all PDU
(byte)0x0D, (byte)0x00, // Length: 13 bytes (LE)
// - PDUType: 22 (0x16, LE)
// Type: (............0110) TS_PDUTYPE_DEACTIVATEALLPDU
// ProtocolVersion: (000000000001....) 1
(byte)0x16, (byte)0x00,
(byte)0xEA, (byte)0x03, // PDU source: 1002 (LE)
(byte)0xEA, (byte)0x03, (byte)0x01, (byte)0x00, // ShareID = 66538
(byte)0x01, (byte)0x00, // Length if source descriptor: 1 (LE)
(byte)0x00, // Source descriptor (should be set to 0): 0
}));
Pipeline pipeline = new PipelineImpl("test");
pipeline.add(source, tpkt, x224, mcs, channel1003, sink);
pipeline.link("source", "tpkt", "x224", "mcs >channel_1003", "channel_1003 >deactivate_all", "sink");
pipeline.runMainLoop("source", STDOUT, false, false);
}
}