use crate::rdp::error::RdpError;
use crate::rdp::util::{le_slice_to_string, parse_per_length_determinant, utf7_slice_to_string};
use crate::rdp::windows;
use nom::bytes::streaming::take;
use nom::combinator::{map_opt, map_res, opt};
use nom::number::streaming::{be_u16, be_u8, le_u16, le_u32, le_u8};
use nom::IResult;
fn millimeters_to_opt(x: u32) -> Option<u32> {
if x >= 10 && x <= 10_000 {
Some(x)
} else {
None
}
}
fn desktop_scale_to_opt(x: u32) -> Option<u32> {
if x >= 100 && x <= 500 {
Some(x)
} else {
None
}
}
fn device_scale_to_opt(x: u32) -> Option<u32> {
if x == 100 || x == 140 || x == 180 {
Some(x)
} else {
None
}
}
#[derive(Clone, Debug, PartialEq)]
pub enum TpktVersion {
T123 = 0x3,
}
#[derive(Clone, Debug, PartialEq)]
pub struct T123Tpkt {
pub child: T123TpktChild,
}
#[derive(Clone, Debug, PartialEq)]
pub enum T123TpktChild {
X224ConnectionRequest(X224ConnectionRequest),
X224ConnectionConfirm(X224ConnectionConfirm),
Data(X223Data),
Raw(Vec<u8>),
}
#[derive(Clone, Debug, PartialEq)]
pub enum X224Type {
ConnectionConfirm = 0xd,
ConnectionRequest = 0xe,
Data = 0xf,
}
#[derive(Clone, Debug, PartialEq)]
pub struct X224ConnectionRequest {
pub cdt: u8,
pub dst_ref: u16,
pub src_ref: u16,
pub class: u8,
pub options: u8,
pub cookie: Option<RdpCookie>,
pub negotiation_request: Option<NegotiationRequest>,
pub data: Vec<u8>,
}
#[derive(Clone, Debug, PartialEq)]
pub struct RdpCookie {
pub mstshash: String,
}
#[derive(Clone, Debug, PartialEq)]
pub enum X224ConnectionRequestType {
NegotiationRequest = 0x1,
NegotiationResponse = 0x2,
NegotiationFailure = 0x3,
}
#[derive(Clone, Debug, PartialEq)]
pub struct NegotiationRequest {
pub flags: NegotiationRequestFlags,
pub protocols: ProtocolFlags,
}
bitflags! {
#[derive(Default)]
pub struct NegotiationRequestFlags: u8 {
const RESTRICTED_ADMIN_MODE_REQUIRED = 0x1;
const REDIRECTED_AUTHENTICATION_MODE_REQUIRED = 0x2;
const CORRELATION_INFO_PRESENT = 0x8;
}
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum Protocol {
ProtocolRdp = 0x0,
ProtocolSsl = 0x1,
ProtocolHybrid = 0x2,
ProtocolRdsTls = 0x4,
ProtocolHybridEx = 0x8,
}
bitflags! {
pub struct ProtocolFlags: u32 {
const PROTOCOL_RDP = Protocol::ProtocolRdp as u32;
const PROTOCOL_SSL = Protocol::ProtocolSsl as u32;
const PROTOCOL_HYBRID = Protocol::ProtocolHybrid as u32;
const PROTOCOL_RDSTLS = Protocol::ProtocolRdsTls as u32;
const PROTOCOL_HYBRID_EX = Protocol::ProtocolHybridEx as u32;
}
}
#[derive(Clone, Debug, PartialEq)]
pub struct X224ConnectionConfirm {
pub cdt: u8,
pub dst_ref: u16,
pub src_ref: u16,
pub class: u8,
pub options: u8,
pub negotiation_from_server: Option<NegotiationFromServer>,
}
#[derive(Clone, Debug, PartialEq)]
pub enum NegotiationFromServer {
Response(NegotiationResponse),
Failure(NegotiationFailure),
}
#[derive(Clone, Debug, PartialEq)]
pub struct NegotiationResponse {
pub flags: NegotiationResponseFlags,
pub protocol: Protocol,
}
bitflags! {
#[derive(Default)]
pub struct NegotiationResponseFlags: u8 {
const EXTENDED_CLIENT_DATA_SUPPORTED = 0x1;
const DYNVC_GFX_PROTOCOL_SUPPORTED = 0x2;
const NEGRSP_FLAG_RESERVED = 0x4;
const RESTRICTED_ADMIN_MODE_SUPPORTED = 0x8;
const REDIRECTED_AUTHENTICATION_MODE_SUPPORTED = 0x10;
}
}
#[derive(Clone, Debug, PartialEq)]
pub struct NegotiationFailure {
pub code: NegotiationFailureCode,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum NegotiationFailureCode {
SslRequiredByServer = 0x1,
SslNotAllowedByServer = 0x2,
SslCertNotOnServer = 0x3,
InconsistentFlags = 0x4,
HybridRequiredByServer = 0x5,
SslWithUserAuthRequiredByServer = 0x6,
}
#[derive(Clone, Debug, PartialEq)]
pub struct X223Data {
pub child: X223DataChild,
}
#[derive(Clone, Debug, PartialEq)]
pub enum X223DataChild {
McsConnectRequest(McsConnectRequest),
McsConnectResponse(McsConnectResponse),
Raw(Vec<u8>),
}
#[derive(Clone, Debug, PartialEq)]
pub enum T125Type {
T125TypeMcsConnectRequest = 0x65, T125TypeMcsConnectResponse = 0x66, }
#[derive(Clone, Debug, PartialEq)]
pub struct McsConnectRequest {
pub children: Vec<McsConnectRequestChild>,
}
#[derive(Clone, Debug, PartialEq)]
pub enum McsConnectRequestChild {
CsClientCore(CsClientCoreData),
CsNet(CsNet),
CsUnknown(CsUnknown),
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum CsType {
Core = 0xc001,
Net = 0xc003,
}
#[derive(Clone, Debug, PartialEq)]
pub struct CsClientCoreData {
pub version: Option<RdpClientVersion>,
pub desktop_width: u16,
pub desktop_height: u16,
pub color_depth: Option<ColorDepth>,
pub sas_sequence: Option<SasSequence>,
pub keyboard_layout: u32, pub client_build: windows::OperatingSystem,
pub client_name: String,
pub keyboard_type: Option<KeyboardType>,
pub keyboard_subtype: u32,
pub keyboard_function_key: u32,
pub ime_file_name: String,
pub post_beta2_color_depth: Option<PostBeta2ColorDepth>,
pub client_product_id: Option<u16>,
pub serial_number: Option<u32>,
pub high_color_depth: Option<HighColorDepth>,
pub supported_color_depth: Option<SupportedColorDepth>,
pub early_capability_flags: Option<EarlyCapabilityFlags>,
pub client_dig_product_id: Option<String>,
pub connection_hint: Option<ConnectionHint>,
pub server_selected_protocol: Option<ProtocolFlags>,
pub desktop_physical_width: Option<u32>,
pub desktop_physical_height: Option<u32>,
pub desktop_orientation: Option<DesktopOrientation>,
pub desktop_scale_factor: Option<u32>,
pub device_scale_factor: Option<u32>,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
#[allow(non_camel_case_types)]
pub enum RdpClientVersion {
V4 = 0x80001,
V5_V8_1 = 0x80004,
V10_0 = 0x80005,
V10_1 = 0x80006,
V10_2 = 0x80007,
V10_3 = 0x80008,
V10_4 = 0x80009,
V10_5 = 0x8000a,
V10_6 = 0x8000b,
V10_7 = 0x8000c,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum ColorDepth {
RnsUdColor4Bpp = 0xca00,
RnsUdColor8Bpp = 0xca01,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum SasSequence {
RnsUdSasDel = 0xaa03,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum KeyboardType {
KbXt = 0x1,
KbIco = 0x2,
KbAt = 0x3,
KbEnhanced = 0x4,
Kb1050 = 0x5,
Kb9140 = 0x6,
KbJapanese = 0x7,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum PostBeta2ColorDepth {
RnsUdColorNotProvided = 0x0,
RnsUdColor4Bpp = 0xca00,
RnsUdColor8Bpp = 0xca01,
RnsUdColor16Bpp555 = 0xca02,
RnsUdColor16Bpp565 = 0xca03,
RnsUdColor24Bpp = 0xca04,
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum HighColorDepth {
HighColorNotProvided = 0x0,
HighColor4Bpp = 0x4,
HighColor8Bpp = 0x8,
HighColor15Bpp = 0xf,
HighColor16Bpp = 0x10,
HighColor24Bpp = 0x18,
}
bitflags! {
#[derive(Default)]
pub struct SupportedColorDepth: u16 {
const RNS_UD_24_BPP_SUPPORT = 0x1;
const RNS_UD_16_BPP_SUPPORT = 0x2;
const RNS_UD_15_BPP_SUPPORT = 0x4;
const RNS_UD_32_BPP_SUPPORT = 0x8;
}
}
bitflags! {
#[derive(Default)]
pub struct EarlyCapabilityFlags: u16 {
const RNS_UD_CS_SUPPORT_ERRINFO_PDF = 0x1;
const RNS_UD_CS_WANT_32BPP_SESSION = 0x2;
const RNS_UD_CS_SUPPORT_STATUSINFO_PDU = 0x4;
const RNS_UD_CS_STRONG_ASYMMETRIC_KEYS = 0x8;
const RNS_UD_CS_UNUSED = 0x10;
const RNS_UD_CS_VALID_CONNECTION_TYPE = 0x20;
const RNS_UD_CS_SUPPORT_MONITOR_LAYOUT_PDU = 0x40;
const RNS_UD_CS_SUPPORT_NETCHAR_AUTODETECT = 0x80;
const RNS_UD_CS_SUPPORT_DYNVC_GFX_PROTOCOL = 0x100;
const RNS_UD_CS_SUPPORT_DYNAMIC_TIME_ZONE = 0x200;
const RNS_UD_CS_SUPPORT_HEARTBEAT_PDU = 0x400;
}
}
#[derive(Clone, Debug, FromPrimitive, PartialEq)]
pub enum ConnectionHint {
ConnectionHintNotProvided = 0x0,
ConnectionHintModem = 0x1,
ConnectionHintBroadbandLow = 0x2,
ConnectionHintSatellite = 0x3,
ConnectionHintBroadbandHigh = 0x4,
ConnectionHintWan = 0x5,
ConnectionHintLan = 0x6,
ConnectionHintAutoDetect = 0x7,
}
#[derive(Clone, Copy, Debug, FromPrimitive, PartialEq)]
pub enum DesktopOrientation {
OrientationLandscape = 0,
OrientationPortrait = 90, OrientationLandscapeFlipped = 180, OrientationPortraitFlipped = 270, }
#[derive(Clone, Debug, PartialEq)]
pub struct CsNet {
pub channels: Vec<String>,
}
#[derive(Clone, Debug, PartialEq)]
pub struct CsUnknown {
pub typ: u16,
pub data: Vec<u8>,
}
#[derive(Clone, Debug, PartialEq)]
pub struct McsConnectResponse {}
pub fn parse_t123_tpkt(input: &[u8]) -> IResult<&[u8], T123Tpkt, RdpError> {
let (i1, _version) = verify!(input, be_u8, |&x| x == TpktVersion::T123 as u8)?;
let (i2, _reserved) = try_parse!(i1, be_u8);
let (i3, sz) = map_opt!(i2, be_u16, |x: u16| x.checked_sub(4))?;
let (i4, data) = take!(i3, sz)?;
let opt1: Option<T123TpktChild> = {
match opt!(data, parse_x224_connection_request_class_0) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(T123TpktChild::X224ConnectionRequest(x)),
None => None,
},
Err(e) => return Err(e),
}
};
let opt2: Option<T123TpktChild> = match opt1 {
Some(x) => Some(x),
None => match opt!(data, parse_x224_connection_confirm_class_0) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(T123TpktChild::X224ConnectionConfirm(x)),
None => None,
},
Err(e) => return Err(e),
},
};
let opt3: Option<T123TpktChild> = match opt2 {
Some(x) => Some(x),
None => match opt!(data, parse_x223_data_class_0) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(T123TpktChild::Data(x)),
None => None,
},
Err(e) => return Err(e),
},
};
let child: T123TpktChild = match opt3 {
Some(x) => x,
None => T123TpktChild::Raw(data.to_vec()),
};
return Ok((i4, T123Tpkt { child }));
}
fn take_4_4_bits(input: &[u8]) -> IResult<&[u8], (u8, u8), RdpError> {
map!(input, be_u8, |b| (b >> 4, b & 0xf))
}
fn parse_x224_connection_request(input: &[u8]) -> IResult<&[u8], X224ConnectionRequest, RdpError> {
let (i1, length) = verify!(input, be_u8, |&x| x != 0xff)?; let (i2, cr_cdt) = take_4_4_bits(i1)?;
let _ = verify!(i1, value!(cr_cdt.0), |&x| x
== X224Type::ConnectionRequest as u8)?;
let _ = verify!(i1, value!(cr_cdt.1), |&x| x == 0 || x == 1)?;
let (i3, dst_ref) = verify!(i2, be_u16, |&x| x == 0)?;
let (i4, src_ref) = try_parse!(i3, be_u16);
let (i5, class_options) = bits!(
i4,
tuple!(
verify!(take_bits!(4u8), |&x| x <= 4),
verify!(take_bits!(4u8), |&x| x <= 3)
)
)?;
let _ = verify!(i1, value!(length), |&x| x >= 6)?;
let i6 = i5;
let sz = length - 6;
let (i7, data) = {
if sz > 0 {
take!(i6, sz)?
} else {
(i6, &[][..])
}
};
let (j1, cookie) = {
if data.len() > 0 {
match opt!(data, parse_rdp_cookie) {
Ok((remainder, opt)) => (remainder, opt),
Err(e) => return Err(e),
}
} else {
(&[][..], None)
}
};
let (j2, negotiation_request) = {
if j1.len() > 0 {
match opt!(j1, parse_negotiation_request) {
Ok((remainder, opt)) => (remainder, opt),
Err(e) => return Err(e),
}
} else {
(&[][..], None)
}
};
return Ok((
i7,
X224ConnectionRequest {
cdt: cr_cdt.1,
dst_ref,
src_ref,
class: class_options.0,
options: class_options.1,
cookie,
negotiation_request,
data: j2.to_vec(),
},
));
}
fn parse_x224_connection_request_class_0(
input: &[u8],
) -> IResult<&[u8], X224ConnectionRequest, RdpError> {
let (i1, x224) = try_parse!(input, parse_x224_connection_request);
if x224.class == 0 && x224.options == 0 {
Ok((i1, x224))
} else {
Err(nom::Err::Error(RdpError::NotX224Class0Error))
}
}
fn parse_rdp_cookie(input: &[u8]) -> IResult<&[u8], RdpCookie, RdpError> {
do_parse! {
input,
_key: tag!(b"Cookie: ")
>> _name: tag!(b"mstshash=")
>> bytes: take_until_and_consume!("\r\n")
>> s: map_res!(value!(bytes), std::str::from_utf8)
>> (RdpCookie{ mstshash: String::from(s) })
}
}
fn parse_negotiation_request(input: &[u8]) -> IResult<&[u8], NegotiationRequest, RdpError> {
do_parse! {
input,
_typ: verify!(
le_u8,
|&x| x == X224ConnectionRequestType::NegotiationRequest as u8)
>> flags: map_opt!(
le_u8,
NegotiationRequestFlags::from_bits)
>> _length: verify!(
le_u16,
|&x| x == 8)
>> protocols: map_opt!(
le_u32,
ProtocolFlags::from_bits)
>> (NegotiationRequest { flags, protocols })
}
}
fn parse_x224_connection_confirm(input: &[u8]) -> IResult<&[u8], X224ConnectionConfirm, RdpError> {
let (i1, length) = verify!(input, be_u8, |&x| x != 0xff)?; let (i2, cr_cdt) = take_4_4_bits(i1)?;
let _ = verify!(i1, value!(cr_cdt.0), |&x| x
== X224Type::ConnectionConfirm as u8)?;
let _ = verify!(i1, value!(cr_cdt.1), |&x| x == 0 || x == 1)?;
let (i3, dst_ref) = verify!(i2, be_u16, |&x| x == 0)?;
let (i4, src_ref) = try_parse!(i3, be_u16);
let (i5, class_options) = bits!(
i4,
tuple!(
verify!(take_bits!(4u8), |&x| x <= 4),
verify!(take_bits!(4u8), |&x| x <= 3)
)
)?;
let _ = verify!(i1, value!(length), |&x| x >= 6)?;
let i6 = i5;
let sz = length - 6;
let (i7, negotiation_from_server) = {
if sz > 0 {
let (i7, data) = take!(i6, sz)?;
let opt1: Option<NegotiationFromServer> = match opt!(data, parse_negotiation_response) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(NegotiationFromServer::Response(x)),
None => None,
},
Err(e) => return Err(e),
};
let opt2: Option<NegotiationFromServer> = match opt1 {
Some(x) => Some(x),
None => match opt!(data, parse_negotiation_failure) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(NegotiationFromServer::Failure(x)),
None => None,
},
Err(e) => return Err(e),
},
};
(i7, opt2)
} else {
(i6, None)
}
};
return Ok((
i7,
X224ConnectionConfirm {
cdt: cr_cdt.1,
dst_ref,
src_ref,
class: class_options.0,
options: class_options.1,
negotiation_from_server,
},
));
}
fn parse_x224_connection_confirm_class_0(
input: &[u8],
) -> IResult<&[u8], X224ConnectionConfirm, RdpError> {
let (i1, x224) = try_parse!(input, parse_x224_connection_confirm);
if x224.class == 0 && x224.options == 0 {
Ok((i1, x224))
} else {
Err(nom::Err::Error(RdpError::NotX224Class0Error))
}
}
fn parse_negotiation_response(input: &[u8]) -> IResult<&[u8], NegotiationResponse, RdpError> {
do_parse! {
input,
_typ: verify!(
le_u8,
|&x| x == X224ConnectionRequestType::NegotiationResponse as u8)
>> flags: map_opt!(
le_u8,
NegotiationResponseFlags::from_bits)
>> _length: verify!(
le_u16,
|&x| x == 8)
>> protocol: map_opt!(
le_u32,
num::FromPrimitive::from_u32)
>> (NegotiationResponse { flags, protocol })
}
}
fn parse_negotiation_failure(input: &[u8]) -> IResult<&[u8], NegotiationFailure, RdpError> {
do_parse! {
input,
_typ: verify!(
le_u8,
|&x| x == X224ConnectionRequestType::NegotiationFailure as u8)
>> _flags: le_u8
>> _length: verify!(
le_u16,
|&x| x == 8)
>> code: map_opt!(
le_u32,
num::FromPrimitive::from_u32)
>> (NegotiationFailure { code })
}
}
fn parse_x223_data_class_0(input: &[u8]) -> IResult<&[u8], X223Data, RdpError> {
fn parser(input: &[u8]) -> IResult<&[u8], (u8, u8, u8), RdpError> {
bits!(
input,
tuple!(
verify!(take_bits!(4u8), |&x| x == 0xf),
verify!(take_bits!(3u8), |&x| x == 0),
verify!(take_bits!(1u8), |&x| x == 0)
)
)
}
let (i1, _length) = verify!(input, be_u8, |&x| x == 2)?;
let (i2, _dt_x_roa) = parser(i1)?;
let (i3, _eot) = verify!(i2, be_u8, |&x| x == 0x80)?;
let opt1: Option<X223DataChild> = match opt!(i3, parse_mcs_connect) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(X223DataChild::McsConnectRequest(x)),
None => None,
},
Err(e) => return Err(e),
};
let opt2: Option<X223DataChild> = match opt1 {
Some(x) => Some(x),
None => match opt!(i3, parse_mcs_connect_response) {
Ok((_remainder, opt)) => match opt {
Some(x) => Some(X223DataChild::McsConnectResponse(x)),
None => None,
},
Err(e) => return Err(e),
},
};
let child: X223DataChild = match opt2 {
Some(x) => x,
None => X223DataChild::Raw(i3.to_vec()),
};
return Ok((&[], X223Data { child }));
}
fn parse_mcs_connect(input: &[u8]) -> IResult<&[u8], McsConnectRequest, RdpError> {
let (i1, _ber_type) = verify!(
input,
le_u8,
|&x| x == 0x7f
)?;
let (i2, _t125_type) = verify!(i1, le_u8, |&x| x
== T125Type::T125TypeMcsConnectRequest as u8)?;
let (i3, _skipped) = take_until_and_consume!(i2, "Duca")?;
let (i4, data) = length_data!(i3, parse_per_length_determinant)?;
let mut remainder: &[u8] = data;
let mut children = Vec::new();
loop {
remainder = match opt!(remainder, parse_cs_client_core_data) {
Ok((rem, opt)) => match opt {
Some(core_data) => {
children.push(McsConnectRequestChild::CsClientCore(core_data));
rem
}
None => match opt!(remainder, parse_cs_net) {
Ok((rem, opt)) => match opt {
Some(net) => {
children.push(McsConnectRequestChild::CsNet(net));
rem
}
None => {
match opt!(remainder, parse_cs_unknown) {
Ok((rem, opt)) => match opt {
Some(unknown) => {
children.push(McsConnectRequestChild::CsUnknown(unknown));
rem
}
None => {
break;
}
},
Err(nom::Err::Incomplete(i)) => {
return Err(nom::Err::Incomplete(i))
}
Err(nom::Err::Failure(_)) | Err(nom::Err::Error(_)) => break,
}
}
},
Err(nom::Err::Incomplete(i)) => return Err(nom::Err::Incomplete(i)),
Err(nom::Err::Failure(_)) | Err(nom::Err::Error(_)) => break,
},
},
Err(nom::Err::Incomplete(i)) => return Err(nom::Err::Incomplete(i)),
Err(nom::Err::Failure(_)) | Err(nom::Err::Error(_)) => break,
};
if remainder.len() == 0 {
break;
}
}
return Ok((i4, McsConnectRequest { children }));
}
fn parse_cs_client_core_data(input: &[u8]) -> IResult<&[u8], CsClientCoreData> {
let (i1, _typ) = verify!(input, le_u16, |&x| x == CsType::Core as u16)?;
let (i2, sz) = map_opt!(i1, le_u16, |x: u16| x.checked_sub(4))?;
let (i3, data) = take!(i2, sz)?;
let (j1, version) = map!(data, le_u32, num::FromPrimitive::from_u32)?;
let (j2, desktop_width) = try_parse!(j1, le_u16);
let (j3, desktop_height) = try_parse!(j2, le_u16);
let (j4, color_depth) = map!(j3, le_u16, num::FromPrimitive::from_u16)?;
let (j5, sas_sequence) = map!(j4, le_u16, num::FromPrimitive::from_u16)?;
let (j6, keyboard_layout) = try_parse!(j5, le_u32);
let (j7, client_build) = map!(j6, le_u32, windows::build_number_to_os)?;
let (j8, client_name) = map_res!(j7, take!(32), le_slice_to_string)?;
let (j9, keyboard_type) = map!(j8, le_u32, num::FromPrimitive::from_u32)?;
let (j10, keyboard_subtype) = try_parse!(j9, le_u32);
let (j11, keyboard_function_key) = try_parse!(j10, le_u32);
let (j12, ime_file_name) = map_res!(j11, take!(64), le_slice_to_string)?;
let (j13, post_beta2_color_depth) =
match opt!(j12, map_opt!(le_u16, num::FromPrimitive::from_u16)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j12, None),
};
let (j14, client_product_id) = match post_beta2_color_depth {
None => (j13, None),
Some(_) => match opt!(j13, le_u16) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j13, None),
},
};
let (j15, serial_number) = match client_product_id {
None => (j14, None),
Some(_) => match opt!(j14, le_u32) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j14, None),
},
};
let (j16, high_color_depth) = match serial_number {
None => (j15, None),
Some(_) => {
match opt!(j15, map_opt!(le_u16, num::FromPrimitive::from_u16)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j15, None),
}
}
};
let (j17, supported_color_depth) = match high_color_depth {
None => (j16, None),
Some(_) => {
match opt!(j16, map_opt!(le_u16, SupportedColorDepth::from_bits)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j16, None),
}
}
};
let (j18, early_capability_flags) = match supported_color_depth {
None => (j17, None),
Some(_) => {
match opt!(j17, map_opt!(le_u16, EarlyCapabilityFlags::from_bits)) as IResult<&[u8], _>
{
Ok((rem, obj)) => (rem, obj),
_ => (j17, None),
}
}
};
let (j19, client_dig_product_id) = match early_capability_flags {
None => (j18, None),
Some(_) => {
match opt(map_res(take(64usize), le_slice_to_string))(j18) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j18, None),
}
}
};
let (j20, connection_hint) = match client_dig_product_id {
None => (j19, None),
Some(_) => {
match opt(map_opt(le_u8, num::FromPrimitive::from_u8))(j19) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j19, None),
}
}
};
let (j21, pad) = match connection_hint {
None => (j20, None),
Some(_) => match opt(take(1usize))(j20) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j20, None),
},
};
let (j22, server_selected_protocol) = match pad {
None => (j21, None),
Some(_) => {
match opt!(j21, map_opt!(le_u32, ProtocolFlags::from_bits)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j21, None),
}
}
};
let (j23, desktop_physical_width) = match server_selected_protocol {
None => (j22, None),
Some(_) => match opt!(j22, map_opt!(le_u32, millimeters_to_opt)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j22, None),
},
};
let (j24, desktop_physical_height) = match desktop_physical_width {
None => (j23, None),
Some(_) => match opt!(j23, map_opt!(le_u32, millimeters_to_opt)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j23, None),
},
};
let (j25, desktop_orientation) = match desktop_physical_height {
None => (j24, None),
Some(_) => {
match opt!(j24, map_opt!(le_u16, num::FromPrimitive::from_u16)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j24, None),
}
}
};
let (j26, desktop_scale_factor) = match desktop_orientation {
None => (j25, None),
Some(_) => match opt!(j25, map_opt!(le_u32, desktop_scale_to_opt)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j25, None),
},
};
let (_j27, device_scale_factor) = match desktop_scale_factor {
None => (j26, None),
Some(_) => match opt!(j26, map_opt!(le_u32, device_scale_to_opt)) as IResult<&[u8], _> {
Ok((rem, obj)) => (rem, obj),
_ => (j26, None),
},
};
return Ok((
i3,
CsClientCoreData {
version,
desktop_width,
desktop_height,
color_depth,
sas_sequence,
keyboard_layout,
client_build,
client_name,
keyboard_type,
keyboard_subtype,
keyboard_function_key,
ime_file_name,
post_beta2_color_depth,
client_product_id,
serial_number,
high_color_depth,
supported_color_depth,
early_capability_flags,
client_dig_product_id,
connection_hint,
server_selected_protocol,
desktop_physical_width,
desktop_physical_height,
desktop_orientation,
desktop_scale_factor,
device_scale_factor,
},
));
}
fn parse_cs_net(input: &[u8]) -> IResult<&[u8], CsNet> {
let (i1, _typ) = verify!(input, le_u16, |&x| x == CsType::Net as u16)?;
let (i2, sz) = map_opt!(i1, le_u16, |x: u16| x.checked_sub(8))?;
let (i3, count) = try_parse!(i2, le_u32);
let (i4, data) = take!(i3, sz)?;
let mut remainder: &[u8] = data;
let mut channels = Vec::new();
for _index in 0..count {
let (j1, name) = map_res!(remainder, take!(8), utf7_slice_to_string)?;
channels.push(name);
let (j2, _options) = try_parse!(j1, le_u32);
remainder = j2;
}
return Ok((i4, CsNet { channels }));
}
fn parse_cs_unknown(input: &[u8]) -> IResult<&[u8], CsUnknown> {
do_parse! {
input,
typ: map_opt!(
le_u16,
|x| {
let opt: Option<CsType> = num::FromPrimitive::from_u16(x);
match opt {
Some(_) => None,
None => Some(x),
}
})
>> sz: map_opt!(le_u16, |x: u16| x.checked_sub(4))
>> data: take!(sz)
>> (CsUnknown { typ, data: data.to_vec() })
}
}
fn parse_mcs_connect_response(input: &[u8]) -> IResult<&[u8], McsConnectResponse, RdpError> {
do_parse! {
input,
_ber_type: verify!(
le_u8,
|&x| x == 0x7f)
>> _t125_type: verify!(
le_u8,
|&x| x == T125Type::T125TypeMcsConnectResponse as u8)
>> (McsConnectResponse {})
}
}
#[cfg(test)]
mod tests_cookie_21182 {
use crate::rdp::parser::*;
static BYTES: [u8; 37] = [
0x03, 0x00, 0x00, 0x25, 0x20, 0xe0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x43, 0x6f, 0x6f, 0x6b,
0x69, 0x65, 0x3a, 0x20, 0x6d, 0x73, 0x74, 0x73, 0x68, 0x61, 0x73, 0x68, 0x3d, 0x75, 0x73,
0x65, 0x72, 0x31, 0x32, 0x33, 0x0d, 0x0a,
];
#[test]
fn test_t123_x224_cookie() {
let t123_bytes = &BYTES[..];
let t123_tpkt: T123Tpkt = T123Tpkt {
child: T123TpktChild::X224ConnectionRequest(X224ConnectionRequest {
cdt: 0,
dst_ref: 0,
src_ref: 0,
class: 0,
options: 0,
cookie: Some(RdpCookie {
mstshash: String::from("user123"),
}),
negotiation_request: None,
data: Vec::new(),
}),
};
assert_eq!(Ok((&[][..], t123_tpkt)), parse_t123_tpkt(t123_bytes));
}
}
#[cfg(test)]
mod tests_negotiate_49350 {
use crate::rdp::parser::*;
static BYTES: [u8; 20] = [
0x03, 0x00, 0x00, 0x13, 0x0e, 0xe0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x08, 0x00,
0x00, 0x00, 0x00, 0x00, 0xff,
];
static TPKT_BEGIN: usize = 0;
static X224_BEGIN: usize = TPKT_BEGIN + 4;
static NEG_REQ_BEGIN: usize = X224_BEGIN + 7;
static NEG_REQ_END: usize = NEG_REQ_BEGIN + 8;
static X224_END: usize = NEG_REQ_END;
static TPKT_END: usize = X224_END;
static PADDING_BEGIN: usize = TPKT_END;
#[test]
fn test_t123_x224_negotiate() {
let t123_bytes = &BYTES[TPKT_BEGIN..];
let t123_tpkt: T123Tpkt = T123Tpkt {
child: T123TpktChild::X224ConnectionRequest(X224ConnectionRequest {
cdt: 0,
dst_ref: 0,
src_ref: 0,
class: 0,
options: 0,
cookie: None,
negotiation_request: Some(NegotiationRequest {
flags: NegotiationRequestFlags::empty(),
protocols: ProtocolFlags::PROTOCOL_RDP,
}),
data: Vec::new(),
}),
};
assert_eq!(
Ok((&BYTES[PADDING_BEGIN..][..], t123_tpkt)),
parse_t123_tpkt(t123_bytes)
)
}
}
#[cfg(test)]
mod tests_core_49350 {
use crate::rdp::parser::*;
static BYTES: [u8; 429] = [
0x03, 0x00, 0x01, 0xac, 0x02, 0xf0, 0x80, 0x7f, 0x65, 0x82, 0x01, 0xa0, 0x04, 0x01, 0x01,
0x04, 0x01, 0x01, 0x01, 0x01, 0xff, 0x30, 0x19, 0x02, 0x01, 0x22, 0x02, 0x01, 0x02, 0x02,
0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x02, 0xff, 0xff,
0x02, 0x01, 0x02, 0x30, 0x19, 0x02, 0x01, 0x01, 0x02, 0x01, 0x01, 0x02, 0x01, 0x01, 0x02,
0x01, 0x01, 0x02, 0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x02, 0x04, 0x20, 0x02, 0x01, 0x02,
0x30, 0x1c, 0x02, 0x02, 0xff, 0xff, 0x02, 0x02, 0xfc, 0x17, 0x02, 0x02, 0xff, 0xff, 0x02,
0x01, 0x01, 0x02, 0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x02, 0xff, 0xff, 0x02, 0x01, 0x02,
0x04, 0x82, 0x01, 0x3f, 0x00, 0x05, 0x00, 0x14, 0x7c, 0x00, 0x01, 0x81, 0x36, 0x00, 0x08,
0x00, 0x10, 0x00, 0x01, 0xc0, 0x00, 0x44, 0x75, 0x63, 0x61, 0x81, 0x28, 0x01, 0xc0, 0xd8,
0x00, 0x04, 0x00, 0x08, 0x00, 0x00, 0x05, 0x00, 0x03, 0x01, 0xca, 0x03, 0xaa, 0x09, 0x04,
0x00, 0x00, 0x71, 0x17, 0x00, 0x00, 0x53, 0x00, 0x45, 0x00, 0x52, 0x00, 0x56, 0x00, 0x45,
0x00, 0x52, 0x00, 0x2d, 0x00, 0x58, 0x00, 0x59, 0x00, 0x5a, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x0c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xca, 0x01, 0x00, 0x00, 0x00,
0x00, 0x00, 0x08, 0x00, 0x0f, 0x00, 0x09, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x04, 0xc0, 0x0c, 0x00, 0x0d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x02, 0xc0, 0x0c, 0x00, 0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xc0, 0x38,
0x00, 0x04, 0x00, 0x00, 0x00, 0x72, 0x64, 0x70, 0x64, 0x72, 0x00, 0x00, 0x00, 0x00, 0x00,
0x80, 0x80, 0x72, 0x64, 0x70, 0x73, 0x6e, 0x64, 0x00, 0x00, 0x00, 0x00, 0x00, 0xc0, 0x64,
0x72, 0x64, 0x79, 0x6e, 0x76, 0x63, 0x00, 0x00, 0x00, 0x80, 0xc0, 0x63, 0x6c, 0x69, 0x70,
0x72, 0x64, 0x72, 0x00, 0x00, 0x00, 0xa0, 0xc0, 0xff,
];
static TPKT_BEGIN: usize = 0;
static X223_BEGIN: usize = TPKT_BEGIN + 4;
static MCS_CONNECT_BEGIN: usize = X223_BEGIN + 3;
static MCS_CONNECT_END: usize = MCS_CONNECT_BEGIN + 421;
static X223_END: usize = MCS_CONNECT_END;
static TPKT_END: usize = X223_END;
static PADDING_BEGIN: usize = TPKT_END;
#[test]
fn test_t123_x223_connect_core() {
let t123_bytes = &BYTES[TPKT_BEGIN..];
let core_data = CsClientCoreData {
version: Some(RdpClientVersion::V5_V8_1),
desktop_width: 1280,
desktop_height: 768,
color_depth: Some(ColorDepth::RnsUdColor8Bpp),
sas_sequence: Some(SasSequence::RnsUdSasDel),
keyboard_layout: 0x409,
client_build: windows::OperatingSystem {
build: windows::Build::Vista_6001,
suffix: windows::Suffix::Sp1,
},
client_name: String::from("SERVER-XYZ"),
keyboard_type: Some(KeyboardType::KbEnhanced),
keyboard_subtype: 0,
keyboard_function_key: 12,
ime_file_name: String::from(""),
post_beta2_color_depth: Some(PostBeta2ColorDepth::RnsUdColor8Bpp),
client_product_id: Some(1),
serial_number: Some(0),
high_color_depth: Some(HighColorDepth::HighColor8Bpp),
supported_color_depth: Some(
SupportedColorDepth::RNS_UD_15_BPP_SUPPORT
| SupportedColorDepth::RNS_UD_16_BPP_SUPPORT
| SupportedColorDepth::RNS_UD_24_BPP_SUPPORT
| SupportedColorDepth::RNS_UD_32_BPP_SUPPORT,
),
early_capability_flags: Some(
EarlyCapabilityFlags::RNS_UD_CS_SUPPORT_ERRINFO_PDF
| EarlyCapabilityFlags::RNS_UD_CS_STRONG_ASYMMETRIC_KEYS,
),
client_dig_product_id: Some(String::from("")),
connection_hint: Some(ConnectionHint::ConnectionHintNotProvided),
server_selected_protocol: Some(ProtocolFlags::PROTOCOL_RDP),
desktop_physical_width: None,
desktop_physical_height: None,
desktop_orientation: None,
desktop_scale_factor: None,
device_scale_factor: None,
};
let mut children = Vec::new();
children.push(McsConnectRequestChild::CsClientCore(core_data));
children.push(McsConnectRequestChild::CsUnknown(CsUnknown {
typ: 0xc004,
data: BYTES[0x160..0x160 + 0x8].to_vec(),
}));
children.push(McsConnectRequestChild::CsUnknown(CsUnknown {
typ: 0xc002,
data: BYTES[0x16c..0x16c + 0x8].to_vec(),
}));
let mut channels = Vec::new();
channels.push(String::from("rdpdr"));
channels.push(String::from("rdpsnd"));
channels.push(String::from("drdynvc"));
channels.push(String::from("cliprdr"));
children.push(McsConnectRequestChild::CsNet(CsNet { channels }));
let t123_tpkt: T123Tpkt = T123Tpkt {
child: T123TpktChild::Data(X223Data {
child: X223DataChild::McsConnectRequest(McsConnectRequest { children }),
}),
};
assert_eq!(
Ok((&BYTES[PADDING_BEGIN..][..], t123_tpkt)),
parse_t123_tpkt(t123_bytes)
);
}
}
#[cfg(test)]
mod tests_x223_response_49350 {
use crate::rdp::parser::*;
static BYTES: [u8; 9] = [0x03, 0x00, 0x00, 0x09, 0x02, 0xf0, 0x80, 0x7f, 0x66];
#[test]
fn test_x223_response() {
let t123_bytes = &BYTES[..];
assert_eq!(
Ok((
&[][..],
T123Tpkt {
child: T123TpktChild::Data(X223Data {
child: X223DataChild::McsConnectResponse(McsConnectResponse {}),
})
}
)),
parse_t123_tpkt(t123_bytes)
)
}
}
#[cfg(test)]
mod tests_t123_raw_49350 {
use crate::rdp::parser::*;
static BYTES: [u8; 9] = [0x03, 0x00, 0x00, 0x09, 0x03, 0xf0, 0x80, 0x7f, 0x65];
#[test]
fn test_t123_raw() {
let t123_bytes = &BYTES[..];
assert_eq!(
Ok((
&[][..],
T123Tpkt {
child: T123TpktChild::Raw(BYTES[4..].to_vec())
}
)),
parse_t123_tpkt(t123_bytes)
)
}
}
#[cfg(test)]
mod tests_x224_raw_49350 {
use crate::rdp::parser::*;
static BYTES: [u8; 19] = [
0x03, 0x00, 0x00, 0x13, 0x0e, 0xe0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x08, 0x00,
0x00, 0x00, 0x00, 0x00,
];
#[test]
fn test_x224_raw() {
let t123_bytes = &BYTES[..];
assert_eq!(
Ok((
&[][..],
T123Tpkt {
child: T123TpktChild::X224ConnectionRequest(X224ConnectionRequest {
cdt: 0,
dst_ref: 0,
src_ref: 0,
class: 0,
options: 0,
cookie: None,
negotiation_request: None,
data: BYTES[11..].to_vec(),
})
}
)),
parse_t123_tpkt(t123_bytes)
)
}
}
#[cfg(test)]
mod tests_x223_raw_49350 {
use crate::rdp::parser::*;
static BYTES: [u8; 9] = [0x03, 0x00, 0x00, 0x09, 0x02, 0xf0, 0x80, 0x7f, 0xff];
#[test]
fn test_x223_raw() {
let t123_bytes = &BYTES[..];
assert_eq!(
Ok((
&[][..],
T123Tpkt {
child: T123TpktChild::Data(X223Data {
child: X223DataChild::Raw(BYTES[7..].to_vec()),
})
}
)),
parse_t123_tpkt(t123_bytes)
)
}
}
#[cfg(test)]
mod tests_negotiate_incomplete_49350 {
use crate::rdp::parser::*;
use nom;
static BYTES: [u8; 19] = [
0x03, 0x00, 0x00, 0x13, 0x0e, 0xe0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x08, 0x00,
0x00, 0x00, 0x00, 0x00,
];
static TPKT_BEGIN: usize = 0;
static X224_BEGIN: usize = TPKT_BEGIN + 4;
static NEG_REQ_BEGIN: usize = X224_BEGIN + 7;
static NEG_REQ_END: usize = NEG_REQ_BEGIN + 8;
static X224_END: usize = NEG_REQ_END;
static TPKT_END: usize = X224_END;
#[test]
fn test_t123_incomplete() {
let t123_bytes = &BYTES[TPKT_BEGIN..TPKT_END - 1];
assert_eq!(
Err(nom::Err::Incomplete(nom::Needed::Size(
TPKT_END - TPKT_BEGIN - 4
))),
parse_t123_tpkt(t123_bytes)
)
}
#[test]
fn test_x224_incomplete() {
let x224_bytes = &BYTES[X224_BEGIN..X224_END - 1];
assert_eq!(
Err(nom::Err::Incomplete(nom::Needed::Size(
X224_END - X224_BEGIN - (6 + 1)
))),
parse_x224_connection_request_class_0(x224_bytes)
)
}
#[test]
fn test_negotiate_incomplete() {
let neg_req_bytes = &BYTES[NEG_REQ_BEGIN..NEG_REQ_END - 1];
assert_eq!(
Err(nom::Err::Incomplete(nom::Needed::Size(
NEG_REQ_END - NEG_REQ_BEGIN - (1 + 1 + 2)
))),
parse_negotiation_request(neg_req_bytes)
)
}
}
#[cfg(test)]
mod tests_core_incomplete_49350 {
use crate::rdp::parser::*;
use nom;
static BYTES: [u8; 428] = [
0x03, 0x00, 0x01, 0xac, 0x02, 0xf0, 0x80, 0x7f, 0x65, 0x82, 0x01, 0xa0, 0x04, 0x01, 0x01,
0x04, 0x01, 0x01, 0x01, 0x01, 0xff, 0x30, 0x19, 0x02, 0x01, 0x22, 0x02, 0x01, 0x02, 0x02,
0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x02, 0xff, 0xff,
0x02, 0x01, 0x02, 0x30, 0x19, 0x02, 0x01, 0x01, 0x02, 0x01, 0x01, 0x02, 0x01, 0x01, 0x02,
0x01, 0x01, 0x02, 0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x02, 0x04, 0x20, 0x02, 0x01, 0x02,
0x30, 0x1c, 0x02, 0x02, 0xff, 0xff, 0x02, 0x02, 0xfc, 0x17, 0x02, 0x02, 0xff, 0xff, 0x02,
0x01, 0x01, 0x02, 0x01, 0x00, 0x02, 0x01, 0x01, 0x02, 0x02, 0xff, 0xff, 0x02, 0x01, 0x02,
0x04, 0x82, 0x01, 0x3f, 0x00, 0x05, 0x00, 0x14, 0x7c, 0x00, 0x01, 0x81, 0x36, 0x00, 0x08,
0x00, 0x10, 0x00, 0x01, 0xc0, 0x00, 0x44, 0x75, 0x63, 0x61, 0x81, 0x28, 0x01, 0xc0, 0xd8,
0x00, 0x04, 0x00, 0x08, 0x00, 0x00, 0x05, 0x00, 0x03, 0x01, 0xca, 0x03, 0xaa, 0x09, 0x04,
0x00, 0x00, 0x71, 0x17, 0x00, 0x00, 0x53, 0x00, 0x45, 0x00, 0x52, 0x00, 0x56, 0x00, 0x45,
0x00, 0x52, 0x00, 0x2d, 0x00, 0x58, 0x00, 0x59, 0x00, 0x5a, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x0c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xca, 0x01, 0x00, 0x00, 0x00,
0x00, 0x00, 0x08, 0x00, 0x0f, 0x00, 0x09, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x04, 0xc0, 0x0c, 0x00, 0x0d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x02, 0xc0, 0x0c, 0x00, 0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xc0, 0x38,
0x00, 0x04, 0x00, 0x00, 0x00, 0x72, 0x64, 0x70, 0x64, 0x72, 0x00, 0x00, 0x00, 0x00, 0x00,
0x80, 0x80, 0x72, 0x64, 0x70, 0x73, 0x6e, 0x64, 0x00, 0x00, 0x00, 0x00, 0x00, 0xc0, 0x64,
0x72, 0x64, 0x79, 0x6e, 0x76, 0x63, 0x00, 0x00, 0x00, 0x80, 0xc0, 0x63, 0x6c, 0x69, 0x70,
0x72, 0x64, 0x72, 0x00, 0x00, 0x00, 0xa0, 0xc0,
];
static X223_BEGIN: usize = 4;
static MCS_CONNECT_BEGIN: usize = X223_BEGIN + 3;
static MCS_CONNECT_END: usize = MCS_CONNECT_BEGIN + 421;
static _X223_END: usize = MCS_CONNECT_END;
#[test]
fn test_x223_incomplete() {
let x223_bytes = &BYTES[X223_BEGIN..X223_BEGIN + 2];
assert_eq!(
Err(nom::Err::Incomplete(nom::Needed::Size(1))),
parse_x223_data_class_0(x223_bytes)
)
}
#[test]
fn test_connect_incomplete() {
let connect_bytes = &BYTES[MCS_CONNECT_BEGIN..MCS_CONNECT_END - 1];
assert_eq!(
Err(nom::Err::Incomplete(nom::Needed::Size(0x128))),
parse_mcs_connect(connect_bytes)
)
}
}