From 2eb1ff1e3edbb1f84514a2793eebd8cc444cb8b0 Mon Sep 17 00:00:00 2001 From: Isaiah Becker-Mayer Date: Mon, 11 Jul 2022 16:39:02 -0400 Subject: [PATCH] [v9] Backport #12530, #12531, #12405, #12615, and #12684 (#14012) * Restructure `rdpdr.rs` into a multi file module (#12530) * Adds go build flags for directory sharing and some basic scaffolding for handling them (#12531) * TDP Shared Directory Announce and Acknowledge (#12405) * RDP <--> TDP Translation Architecture (#12615) * RBAC for directory sharing (#12684) --- Cargo.lock | 14 +- api/types/role.go | 3 + api/types/types.pb.go | 1712 +++++++++-------- api/types/types.proto | 9 + lib/services/role.go | 16 + lib/services/role_test.go | 210 +- lib/srv/desktop/dir_sharing_disabled.go | 24 + lib/srv/desktop/dir_sharing_enabled.go | 24 + lib/srv/desktop/rdp/rdpclient/Cargo.toml | 6 +- lib/srv/desktop/rdp/rdpclient/client.go | 93 +- .../desktop/rdp/rdpclient/client_common.go | 4 + lib/srv/desktop/rdp/rdpclient/librdprs.h | 85 +- lib/srv/desktop/rdp/rdpclient/src/cliprdr.rs | 6 +- lib/srv/desktop/rdp/rdpclient/src/errors.rs | 8 + lib/srv/desktop/rdp/rdpclient/src/lib.rs | 283 ++- lib/srv/desktop/rdp/rdpclient/src/rdpdr.rs | 725 ------- .../desktop/rdp/rdpclient/src/rdpdr/consts.rs | 167 ++ .../desktop/rdp/rdpclient/src/rdpdr/flags.rs | 200 ++ .../desktop/rdp/rdpclient/src/rdpdr/mod.rs | 1655 ++++++++++++++++ .../rdp/rdpclient/src/{ => rdpdr}/scard.rs | 0 lib/srv/desktop/rdp/rdpclient/src/util.rs | 51 +- lib/srv/desktop/tdp/proto.go | 244 ++- lib/srv/desktop/windows_server.go | 2 + lib/web/desktop/playback.go | 2 - lib/web/resources_test.go | 1 + 25 files changed, 3855 insertions(+), 1689 deletions(-) create mode 100644 lib/srv/desktop/dir_sharing_disabled.go create mode 100644 lib/srv/desktop/dir_sharing_enabled.go delete mode 100644 lib/srv/desktop/rdp/rdpclient/src/rdpdr.rs create mode 100644 lib/srv/desktop/rdp/rdpclient/src/rdpdr/consts.rs create mode 100644 lib/srv/desktop/rdp/rdpclient/src/rdpdr/flags.rs create mode 100644 lib/srv/desktop/rdp/rdpclient/src/rdpdr/mod.rs rename lib/srv/desktop/rdp/rdpclient/src/{ => rdpdr}/scard.rs (100%) diff --git a/Cargo.lock b/Cargo.lock index 1d113e65e1064..622c48b24c29e 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -972,6 +972,7 @@ dependencies = [ "rand_chacha 0.3.1", "rdp-rs", "rsa", + "utf16string", "uuid", ] @@ -1322,11 +1323,20 @@ version = "0.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" +[[package]] +name = "utf16string" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b62a1e85e12d5d712bf47a85f426b73d303e2d00a90de5f3004df3596e9d216" +dependencies = [ + "byteorder", +] + [[package]] name = "uuid" -version = "0.8.2" +version = "1.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7" +checksum = "dd6469f4314d5f1ffec476e05f17cc9a78bc7a27a6a857842170bdf8d6f98d2f" dependencies = [ "getrandom 0.2.6", ] diff --git a/api/types/role.go b/api/types/role.go index 51a9c0561fc3c..6bf96a4f3682e 100644 --- a/api/types/role.go +++ b/api/types/role.go @@ -643,6 +643,9 @@ func (r *RoleV5) CheckAndSetDefaults() error { if r.Spec.Options.DesktopClipboard == nil { r.Spec.Options.DesktopClipboard = NewBoolOption(true) } + if r.Spec.Options.DesktopDirectorySharing == nil { + r.Spec.Options.DesktopDirectorySharing = NewBoolOption(true) + } switch r.Version { case V3: diff --git a/api/types/types.pb.go b/api/types/types.pb.go index 47541bf0e3483..0fde30bc51e08 100644 --- a/api/types/types.pb.go +++ b/api/types/types.pb.go @@ -4487,10 +4487,14 @@ type RoleOptions struct { CertExtensions []*CertExtension `protobuf:"bytes,17,rep,name=CertExtensions,proto3" json:"cert_extensions,omitempty"` // MaxKubernetesConnections defines the maximum number of concurrent // Kubernetes sessions a user may hold. - MaxKubernetesConnections int64 `protobuf:"varint,18,opt,name=MaxKubernetesConnections,proto3" json:"max_kubernetes_connections,omitempty"` - XXX_NoUnkeyedLiteral struct{} `json:"-"` - XXX_unrecognized []byte `json:"-"` - XXX_sizecache int32 `json:"-"` + MaxKubernetesConnections int64 `protobuf:"varint,18,opt,name=MaxKubernetesConnections,proto3" json:"max_kubernetes_connections,omitempty"` + // DesktopDirectorySharing indicates whether directory sharing is allowed between the user's + // workstation and the remote desktop. It defaults to false unless explicitly set to + // true. + DesktopDirectorySharing *BoolOption `protobuf:"bytes,19,opt,name=DesktopDirectorySharing,proto3,customtype=BoolOption" json:"desktop_directory_sharing"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` } func (m *RoleOptions) Reset() { *m = RoleOptions{} } @@ -9163,7 +9167,7 @@ func init() { func init() { proto.RegisterFile("types.proto", fileDescriptor_d938547f84707355) } var fileDescriptor_d938547f84707355 = []byte{ - // 11979 bytes of a gzipped FileDescriptorProto + // 12012 bytes of a gzipped FileDescriptorProto 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xdc, 0x7d, 0x6d, 0x6c, 0x1c, 0x49, 0x76, 0x98, 0x7a, 0x86, 0x1f, 0x33, 0x8f, 0x43, 0x72, 0x58, 0xd4, 0x07, 0xa5, 0xd5, 0xee, 0xe8, 0x7a, 0x77, 0xb5, 0x92, 0x6e, 0x57, 0x3c, 0x51, 0xb7, 0x3a, 0xef, 0xed, 0xd7, 0xcd, 0x70, 0x28, @@ -9171,748 +9175,750 @@ var fileDescriptor_d938547f84707355 = []byte{ 0x66, 0x7a, 0xae, 0xbb, 0x47, 0x12, 0xfd, 0x01, 0x3b, 0x1f, 0x17, 0xc3, 0x30, 0x7c, 0x1f, 0xc1, 0x19, 0xb6, 0x03, 0x27, 0x76, 0x8c, 0x18, 0x89, 0x63, 0x38, 0x08, 0x9c, 0x04, 0x49, 0xfe, 0x04, 0x30, 0x10, 0x18, 0xf7, 0x23, 0x41, 0xfc, 0x2f, 0xb0, 0x13, 0x30, 0xf1, 0x5d, 0xfe, 0x98, 0x40, - 0x7e, 0x05, 0x08, 0xe0, 0x4b, 0x8c, 0x04, 0xf5, 0xaa, 0xaa, 0xbb, 0xaa, 0xa7, 0x87, 0x1c, 0xad, - 0xb4, 0xc0, 0x69, 0x7f, 0x91, 0xf3, 0xea, 0xbd, 0xd7, 0xf5, 0xf9, 0xea, 0xbd, 0xaa, 0xf7, 0x5e, - 0xc1, 0x44, 0x74, 0xd0, 0xa5, 0xe1, 0xf5, 0x6e, 0xe0, 0x47, 0x3e, 0x19, 0xc5, 0x1f, 0x17, 0x4e, - 0xef, 0xfa, 0xbb, 0x3e, 0x42, 0xe6, 0xd9, 0x7f, 0xbc, 0xf0, 0x42, 0x65, 0xd7, 0xf7, 0x77, 0x5b, - 0x74, 0x1e, 0x7f, 0xed, 0xf4, 0x1e, 0xcc, 0x47, 0x5e, 0x9b, 0x86, 0x91, 0xd3, 0xee, 0x0a, 0x84, - 0xc5, 0x5d, 0x2f, 0xda, 0xeb, 0xed, 0x5c, 0x6f, 0xfa, 0xed, 0xf9, 0xdd, 0xc0, 0x79, 0xe8, 0x45, - 0x4e, 0xe4, 0xf9, 0x1d, 0xa7, 0x35, 0x1f, 0xd1, 0x16, 0xed, 0xfa, 0x41, 0x34, 0xef, 0x74, 0xbd, - 0x79, 0xfc, 0xc6, 0xfc, 0xa3, 0xc0, 0xe9, 0x76, 0x69, 0x90, 0xfc, 0xc3, 0x99, 0x98, 0xff, 0x30, - 0x0f, 0xc5, 0x7b, 0x94, 0x76, 0xab, 0x2d, 0xef, 0x21, 0x25, 0x2f, 0xc3, 0xc8, 0xaa, 0xd3, 0xa6, - 0x73, 0xc6, 0x25, 0xe3, 0x4a, 0xb1, 0x36, 0x7d, 0x74, 0x58, 0x99, 0x08, 0x69, 0xf0, 0x90, 0x06, - 0x76, 0xc7, 0x69, 0x53, 0x0b, 0x0b, 0xc9, 0x67, 0xa1, 0xc8, 0xfe, 0x86, 0x5d, 0xa7, 0x49, 0xe7, - 0x72, 0x88, 0x39, 0x79, 0x74, 0x58, 0x29, 0x76, 0x24, 0xd0, 0x4a, 0xca, 0xc9, 0x65, 0x18, 0x5f, - 0xa1, 0x4e, 0x48, 0x97, 0xeb, 0x73, 0xf9, 0x4b, 0xc6, 0x95, 0x7c, 0xad, 0x74, 0x74, 0x58, 0x29, - 0xb4, 0x18, 0xc8, 0xf6, 0x5c, 0x4b, 0x16, 0x92, 0x65, 0x18, 0x5f, 0x7a, 0xdc, 0xf5, 0x02, 0x1a, - 0xce, 0x8d, 0x5c, 0x32, 0xae, 0x4c, 0x2c, 0x5c, 0xb8, 0xce, 0xdb, 0x7f, 0x5d, 0xb6, 0xff, 0xfa, - 0x86, 0x6c, 0x7f, 0x6d, 0xf6, 0xfb, 0x87, 0x95, 0x53, 0x47, 0x87, 0x95, 0x71, 0xca, 0x49, 0xbe, - 0xf3, 0xdf, 0x2a, 0x86, 0x25, 0xe9, 0xc9, 0x3b, 0x30, 0xb2, 0x71, 0xd0, 0xa5, 0x73, 0xc5, 0x4b, - 0xc6, 0x95, 0xa9, 0x85, 0x97, 0xae, 0xf3, 0x1e, 0x8f, 0x1b, 0x99, 0xfc, 0xc7, 0xb0, 0x6a, 0x85, - 0xa3, 0xc3, 0xca, 0x08, 0x43, 0xb1, 0x90, 0x8a, 0xbc, 0x01, 0x63, 0x77, 0xfd, 0x30, 0x5a, 0xae, - 0xcf, 0x01, 0x36, 0xed, 0xcc, 0xd1, 0x61, 0x65, 0x66, 0xcf, 0x0f, 0x23, 0xdb, 0x73, 0x5f, 0xf7, - 0xdb, 0x5e, 0x44, 0xdb, 0xdd, 0xe8, 0xc0, 0x12, 0x48, 0xe6, 0x0e, 0x4c, 0x6a, 0xfc, 0xc8, 0x04, - 0x8c, 0x6f, 0xae, 0xde, 0x5b, 0x5d, 0xdb, 0x5e, 0x2d, 0x9f, 0x22, 0x05, 0x18, 0x59, 0x5d, 0xab, - 0x2f, 0x95, 0x0d, 0x32, 0x0e, 0xf9, 0xea, 0xfa, 0x7a, 0x39, 0x47, 0x4a, 0x50, 0xa8, 0x57, 0x37, - 0xaa, 0xb5, 0x6a, 0x63, 0xa9, 0x9c, 0x27, 0xb3, 0x30, 0xbd, 0xbd, 0xbc, 0x5a, 0x5f, 0xdb, 0x6e, - 0xd8, 0xf5, 0xa5, 0xc6, 0xbd, 0x8d, 0xb5, 0xf5, 0xf2, 0x08, 0x99, 0x02, 0xb8, 0xb7, 0x59, 0x5b, - 0xb2, 0x56, 0x97, 0x36, 0x96, 0x1a, 0xe5, 0x51, 0xf3, 0x97, 0xf2, 0x50, 0xb8, 0x4f, 0x23, 0xc7, - 0x75, 0x22, 0x87, 0x5c, 0xd4, 0x86, 0x08, 0x6b, 0xaf, 0x8c, 0xcd, 0xcb, 0xfd, 0x63, 0x33, 0x7a, - 0x74, 0x58, 0x31, 0xde, 0x50, 0xc7, 0xe4, 0x6d, 0x98, 0xa8, 0xd3, 0xb0, 0x19, 0x78, 0x5d, 0x36, - 0x5f, 0x70, 0x5c, 0x8a, 0xb5, 0xf3, 0x47, 0x87, 0x95, 0x33, 0x6e, 0x02, 0x56, 0xda, 0xaa, 0x62, - 0x93, 0x65, 0x18, 0x5b, 0x71, 0x76, 0x68, 0x2b, 0x9c, 0x1b, 0xbd, 0x94, 0xbf, 0x32, 0xb1, 0xf0, - 0x82, 0xe8, 0x5f, 0x59, 0xc1, 0xeb, 0xbc, 0x74, 0xa9, 0x13, 0x05, 0x07, 0xb5, 0xd3, 0x47, 0x87, - 0x95, 0x72, 0x0b, 0x01, 0x6a, 0xdf, 0x71, 0x14, 0xd2, 0x48, 0xc6, 0x7c, 0xec, 0xc4, 0x31, 0x7f, - 0xf1, 0xfb, 0x87, 0x15, 0x83, 0x8d, 0x85, 0x18, 0xf3, 0x84, 0x9f, 0x3e, 0xfa, 0x97, 0x20, 0xb7, - 0x5c, 0x9f, 0x1b, 0xc7, 0xb9, 0x56, 0x3e, 0x3a, 0xac, 0x94, 0xb4, 0x61, 0xcb, 0x2d, 0xd7, 0x2f, + 0x7e, 0xf9, 0x97, 0x2f, 0x31, 0x12, 0xd4, 0xab, 0xaa, 0xee, 0xaa, 0x9e, 0x1e, 0x72, 0xb4, 0xd2, + 0x02, 0xa7, 0xfd, 0x45, 0xce, 0xab, 0xf7, 0x5e, 0xd7, 0xc7, 0xab, 0x57, 0xaf, 0xaa, 0xde, 0x7b, + 0x05, 0x13, 0xd1, 0x41, 0x97, 0x86, 0xd7, 0xbb, 0x81, 0x1f, 0xf9, 0x64, 0x14, 0x7f, 0x5c, 0x38, + 0xbd, 0xeb, 0xef, 0xfa, 0x08, 0x99, 0x67, 0xff, 0xf1, 0xc2, 0x0b, 0x95, 0x5d, 0xdf, 0xdf, 0x6d, + 0xd1, 0x79, 0xfc, 0xb5, 0xd3, 0x7b, 0x30, 0x1f, 0x79, 0x6d, 0x1a, 0x46, 0x4e, 0xbb, 0x2b, 0x10, + 0x16, 0x77, 0xbd, 0x68, 0xaf, 0xb7, 0x73, 0xbd, 0xe9, 0xb7, 0xe7, 0x77, 0x03, 0xe7, 0xa1, 0x17, + 0x39, 0x91, 0xe7, 0x77, 0x9c, 0xd6, 0x7c, 0x44, 0x5b, 0xb4, 0xeb, 0x07, 0xd1, 0xbc, 0xd3, 0xf5, + 0xe6, 0xf1, 0x1b, 0xf3, 0x8f, 0x02, 0xa7, 0xdb, 0xa5, 0x41, 0xf2, 0x0f, 0x67, 0x62, 0xfe, 0xe3, + 0x3c, 0x14, 0xef, 0x51, 0xda, 0xad, 0xb6, 0xbc, 0x87, 0x94, 0xbc, 0x0c, 0x23, 0xab, 0x4e, 0x9b, + 0xce, 0x19, 0x97, 0x8c, 0x2b, 0xc5, 0xda, 0xf4, 0xd1, 0x61, 0x65, 0x22, 0xa4, 0xc1, 0x43, 0x1a, + 0xd8, 0x1d, 0xa7, 0x4d, 0x2d, 0x2c, 0x24, 0x9f, 0x85, 0x22, 0xfb, 0x1b, 0x76, 0x9d, 0x26, 0x9d, + 0xcb, 0x21, 0xe6, 0xe4, 0xd1, 0x61, 0xa5, 0xd8, 0x91, 0x40, 0x2b, 0x29, 0x27, 0x97, 0x61, 0x7c, + 0x85, 0x3a, 0x21, 0x5d, 0xae, 0xcf, 0xe5, 0x2f, 0x19, 0x57, 0xf2, 0xb5, 0xd2, 0xd1, 0x61, 0xa5, + 0xd0, 0x62, 0x20, 0xdb, 0x73, 0x2d, 0x59, 0x48, 0x96, 0x61, 0x7c, 0xe9, 0x71, 0xd7, 0x0b, 0x68, + 0x38, 0x37, 0x72, 0xc9, 0xb8, 0x32, 0xb1, 0x70, 0xe1, 0x3a, 0x6f, 0xff, 0x75, 0xd9, 0xfe, 0xeb, + 0x1b, 0xb2, 0xfd, 0xb5, 0xd9, 0xef, 0x1f, 0x56, 0x4e, 0x1d, 0x1d, 0x56, 0xc6, 0x29, 0x27, 0xf9, + 0xce, 0xff, 0xa8, 0x18, 0x96, 0xa4, 0x27, 0xef, 0xc0, 0xc8, 0xc6, 0x41, 0x97, 0xce, 0x15, 0x2f, + 0x19, 0x57, 0xa6, 0x16, 0x5e, 0xba, 0xce, 0x7b, 0x3c, 0x6e, 0x64, 0xf2, 0x1f, 0xc3, 0xaa, 0x15, + 0x8e, 0x0e, 0x2b, 0x23, 0x0c, 0xc5, 0x42, 0x2a, 0xf2, 0x06, 0x8c, 0xdd, 0xf5, 0xc3, 0x68, 0xb9, + 0x3e, 0x07, 0xd8, 0xb4, 0x33, 0x47, 0x87, 0x95, 0x99, 0x3d, 0x3f, 0x8c, 0x6c, 0xcf, 0x7d, 0xdd, + 0x6f, 0x7b, 0x11, 0x6d, 0x77, 0xa3, 0x03, 0x4b, 0x20, 0x99, 0x3b, 0x30, 0xa9, 0xf1, 0x23, 0x13, + 0x30, 0xbe, 0xb9, 0x7a, 0x6f, 0x75, 0x6d, 0x7b, 0xb5, 0x7c, 0x8a, 0x14, 0x60, 0x64, 0x75, 0xad, + 0xbe, 0x54, 0x36, 0xc8, 0x38, 0xe4, 0xab, 0xeb, 0xeb, 0xe5, 0x1c, 0x29, 0x41, 0xa1, 0x5e, 0xdd, + 0xa8, 0xd6, 0xaa, 0x8d, 0xa5, 0x72, 0x9e, 0xcc, 0xc2, 0xf4, 0xf6, 0xf2, 0x6a, 0x7d, 0x6d, 0xbb, + 0x61, 0xd7, 0x97, 0x1a, 0xf7, 0x36, 0xd6, 0xd6, 0xcb, 0x23, 0x64, 0x0a, 0xe0, 0xde, 0x66, 0x6d, + 0xc9, 0x5a, 0x5d, 0xda, 0x58, 0x6a, 0x94, 0x47, 0xcd, 0x5f, 0xca, 0x43, 0xe1, 0x3e, 0x8d, 0x1c, + 0xd7, 0x89, 0x1c, 0x72, 0x51, 0x1b, 0x22, 0xac, 0xbd, 0x32, 0x36, 0x2f, 0xf7, 0x8f, 0xcd, 0xe8, + 0xd1, 0x61, 0xc5, 0x78, 0x43, 0x1d, 0x93, 0xb7, 0x61, 0xa2, 0x4e, 0xc3, 0x66, 0xe0, 0x75, 0x99, + 0xbc, 0xe0, 0xb8, 0x14, 0x6b, 0xe7, 0x8f, 0x0e, 0x2b, 0x67, 0xdc, 0x04, 0xac, 0xb4, 0x55, 0xc5, + 0x26, 0xcb, 0x30, 0xb6, 0xe2, 0xec, 0xd0, 0x56, 0x38, 0x37, 0x7a, 0x29, 0x7f, 0x65, 0x62, 0xe1, + 0x05, 0xd1, 0xbf, 0xb2, 0x82, 0xd7, 0x79, 0xe9, 0x52, 0x27, 0x0a, 0x0e, 0x6a, 0xa7, 0x8f, 0x0e, + 0x2b, 0xe5, 0x16, 0x02, 0xd4, 0xbe, 0xe3, 0x28, 0xa4, 0x91, 0x8c, 0xf9, 0xd8, 0x89, 0x63, 0xfe, + 0xe2, 0xf7, 0x0f, 0x2b, 0x06, 0x1b, 0x0b, 0x31, 0xe6, 0x09, 0x3f, 0x7d, 0xf4, 0x2f, 0x41, 0x6e, + 0xb9, 0x3e, 0x37, 0x8e, 0xb2, 0x56, 0x3e, 0x3a, 0xac, 0x94, 0xb4, 0x61, 0xcb, 0x2d, 0xd7, 0x2f, 0xbc, 0x05, 0x13, 0x4a, 0x1d, 0x49, 0x19, 0xf2, 0xfb, 0xf4, 0x80, 0xf7, 0xa7, 0xc5, 0xfe, 0x25, 0xa7, 0x61, 0xf4, 0xa1, 0xd3, 0xea, 0x89, 0x0e, 0xb4, 0xf8, 0x8f, 0x2f, 0xe6, 0x7e, 0xc2, 0x30, - 0xff, 0xee, 0x08, 0x14, 0x2c, 0x9f, 0xaf, 0x33, 0x72, 0x15, 0x46, 0x1b, 0x91, 0x13, 0xc9, 0xa1, + 0xff, 0xfe, 0x08, 0x14, 0x2c, 0x9f, 0xcf, 0x33, 0x72, 0x15, 0x46, 0x1b, 0x91, 0x13, 0xc9, 0xa1, 0x98, 0x3d, 0x3a, 0xac, 0x4c, 0x87, 0x0c, 0xa0, 0x7c, 0x8f, 0x63, 0x30, 0xd4, 0xf5, 0x3d, 0x27, 0x94, 0x43, 0x82, 0xa8, 0x5d, 0x06, 0x50, 0x51, 0x11, 0x83, 0x5c, 0x86, 0x91, 0xfb, 0xbe, 0x4b, 0xc5, 0xa8, 0x90, 0xa3, 0xc3, 0xca, 0x54, 0xdb, 0x77, 0x55, 0x44, 0x2c, 0x27, 0xaf, 0x43, 0x71, - 0xb1, 0x17, 0x04, 0xb4, 0xc3, 0xa6, 0xea, 0x08, 0x22, 0x4f, 0x1d, 0x1d, 0x56, 0xa0, 0xc9, 0x81, - 0x6c, 0x71, 0x25, 0x08, 0xac, 0xab, 0x1b, 0x91, 0x13, 0x44, 0xd4, 0x9d, 0x1b, 0x1d, 0xaa, 0xab, - 0xd9, 0xf2, 0x9a, 0x09, 0x39, 0x49, 0xba, 0xab, 0x05, 0x27, 0x72, 0x17, 0x26, 0xee, 0x04, 0x4e, - 0x93, 0xae, 0xd3, 0xc0, 0xf3, 0x5d, 0x1c, 0xc3, 0x7c, 0xed, 0xf2, 0xd1, 0x61, 0xe5, 0xec, 0x2e, - 0x03, 0xdb, 0x5d, 0x84, 0x27, 0xd4, 0x3f, 0x3a, 0xac, 0x14, 0xea, 0xbd, 0x00, 0x7b, 0xcf, 0x52, - 0x49, 0xc9, 0x4f, 0xb3, 0x21, 0x09, 0x23, 0xec, 0x5a, 0xea, 0xe2, 0xe8, 0x1d, 0x5f, 0x45, 0x53, - 0x54, 0xf1, 0x6c, 0xcb, 0x09, 0x23, 0x3b, 0xe0, 0x74, 0xa9, 0x7a, 0xaa, 0x2c, 0xc9, 0x1a, 0x14, - 0x1a, 0xcd, 0x3d, 0xea, 0xf6, 0x5a, 0x74, 0xae, 0x80, 0xec, 0xcf, 0x89, 0x89, 0x2b, 0xc7, 0x53, - 0x16, 0xd7, 0x2e, 0x08, 0xde, 0x24, 0x14, 0x10, 0xa5, 0xef, 0x63, 0x26, 0x5f, 0x2c, 0xfc, 0xc6, - 0xef, 0x54, 0x4e, 0xfd, 0xe2, 0x7f, 0xbd, 0x74, 0xca, 0xfc, 0xd7, 0x39, 0x28, 0xa7, 0x99, 0x90, - 0x07, 0x30, 0xb9, 0xd9, 0x75, 0x9d, 0x88, 0x2e, 0xb6, 0x3c, 0xda, 0x89, 0x42, 0x9c, 0x24, 0xc7, - 0xb7, 0xe9, 0x15, 0xf1, 0xdd, 0xb9, 0x1e, 0x12, 0xda, 0x4d, 0x4e, 0x99, 0x6a, 0x95, 0xce, 0x36, - 0xf9, 0x4e, 0x03, 0xe5, 0x74, 0x88, 0x33, 0xec, 0xc9, 0xbe, 0xc3, 0x25, 0xfc, 0x80, 0xef, 0x08, - 0xb6, 0x62, 0x02, 0x75, 0xdc, 0x9d, 0x03, 0x9c, 0x99, 0xc3, 0x4f, 0x20, 0x46, 0x92, 0x31, 0x81, - 0x18, 0xd8, 0xfc, 0x1f, 0x06, 0x4c, 0x59, 0x34, 0xf4, 0x7b, 0x41, 0x93, 0xde, 0xa5, 0x8e, 0x4b, - 0x03, 0x36, 0xfd, 0xef, 0x79, 0x1d, 0x57, 0xac, 0x29, 0x9c, 0xfe, 0xfb, 0x5e, 0x47, 0x5d, 0xc2, - 0x58, 0x4e, 0x3e, 0x07, 0xe3, 0x8d, 0xde, 0x0e, 0xa2, 0xf2, 0x35, 0x75, 0x16, 0x47, 0xac, 0xb7, - 0x63, 0xa7, 0xd0, 0x25, 0x1a, 0x99, 0x87, 0xf1, 0x2d, 0x1a, 0x84, 0x89, 0xc4, 0x43, 0xc9, 0xfe, - 0x90, 0x83, 0x54, 0x02, 0x81, 0x45, 0xee, 0x24, 0x52, 0x57, 0xec, 0x49, 0xd3, 0x29, 0x59, 0x97, - 0x4c, 0x95, 0xb6, 0x80, 0xa8, 0x53, 0x45, 0x62, 0x99, 0xdf, 0xcd, 0x41, 0xb9, 0xee, 0x44, 0xce, - 0x8e, 0x13, 0x8a, 0xfe, 0xdc, 0xba, 0xc9, 0xe4, 0xb8, 0xd2, 0x50, 0x94, 0xe3, 0xac, 0xe6, 0x1f, - 0xbb, 0x79, 0xaf, 0xa6, 0x9b, 0x37, 0xc1, 0x36, 0x48, 0xd1, 0xbc, 0xa4, 0x51, 0xef, 0x9e, 0xdc, - 0xa8, 0xb2, 0x68, 0x54, 0x41, 0x36, 0x2a, 0x69, 0x0a, 0x79, 0x17, 0x46, 0x1a, 0x5d, 0xda, 0x14, - 0x42, 0x44, 0xca, 0x7e, 0xbd, 0x71, 0x0c, 0x61, 0xeb, 0x66, 0xad, 0x24, 0xd8, 0x8c, 0x84, 0x5d, - 0xda, 0xb4, 0x90, 0x4c, 0x59, 0x34, 0xdf, 0x1b, 0x83, 0xd3, 0x59, 0x64, 0xe4, 0x5d, 0x7d, 0x73, - 0xe2, 0xdd, 0xf3, 0xc2, 0xc0, 0xcd, 0x69, 0xce, 0xd0, 0xb7, 0xa7, 0x6b, 0x50, 0x58, 0x67, 0x13, - 0xb2, 0xe9, 0xb7, 0x44, 0xcf, 0x31, 0xa9, 0x58, 0xe8, 0x4a, 0x98, 0x61, 0xc5, 0xe5, 0xe4, 0x05, - 0xc8, 0x6f, 0x5a, 0xcb, 0xa2, 0xbb, 0x8a, 0x47, 0x87, 0x95, 0x7c, 0x2f, 0xf0, 0xe6, 0x0c, 0x8b, - 0x41, 0xc9, 0x3c, 0x8c, 0x2d, 0x56, 0x17, 0x69, 0x10, 0x61, 0x37, 0x95, 0x6a, 0xe7, 0xd8, 0x6c, - 0x69, 0x3a, 0x76, 0x93, 0x06, 0x91, 0xf6, 0x79, 0x81, 0x46, 0x3e, 0x0b, 0xf9, 0xea, 0x76, 0x43, - 0xf4, 0x0c, 0x88, 0x9e, 0xa9, 0x6e, 0x37, 0x6a, 0x93, 0xa2, 0x23, 0xf2, 0xce, 0xa3, 0x90, 0x71, - 0xaf, 0x6e, 0x37, 0xd4, 0xd1, 0x1a, 0x3b, 0x66, 0xb4, 0xae, 0x40, 0x81, 0xe9, 0x19, 0x6c, 0x83, - 0x47, 0xa1, 0x58, 0xe4, 0xea, 0xd3, 0x9e, 0x80, 0x59, 0x71, 0x29, 0x79, 0x39, 0x56, 0x5b, 0x0a, - 0x09, 0x3f, 0xa1, 0xb6, 0x48, 0x65, 0x85, 0x3c, 0x86, 0xc9, 0xfa, 0x41, 0xc7, 0x69, 0x7b, 0x4d, - 0xb1, 0x85, 0x17, 0x71, 0x0b, 0xbf, 0x7e, 0xcc, 0x30, 0x5e, 0xd7, 0x08, 0xf8, 0xae, 0x2e, 0x85, - 0xef, 0x9c, 0xcb, 0xcb, 0xec, 0xf4, 0x0e, 0x3f, 0x67, 0x58, 0xfa, 0x87, 0xd8, 0x5a, 0x92, 0x22, - 0x12, 0xf5, 0xaa, 0x64, 0xda, 0x49, 0x70, 0xb2, 0x96, 0x02, 0x01, 0x51, 0xd7, 0x52, 0xbc, 0xe9, - 0xbe, 0x0b, 0xf9, 0x3b, 0x8b, 0xeb, 0x73, 0x13, 0xc8, 0x83, 0x08, 0x1e, 0x77, 0x16, 0xd7, 0x17, - 0x5b, 0x7e, 0xcf, 0x6d, 0x7c, 0xb8, 0x52, 0x3b, 0x27, 0xd8, 0x4c, 0xee, 0x36, 0xbb, 0x5a, 0x8d, - 0x18, 0x1d, 0x59, 0x82, 0x82, 0x6c, 0xe5, 0x5c, 0x09, 0x79, 0xcc, 0xa4, 0x1a, 0xbf, 0x75, 0x93, - 0xaf, 0x35, 0x57, 0xfc, 0x56, 0x6b, 0x21, 0x71, 0x2e, 0x6c, 0x03, 0xe9, 0xef, 0x97, 0x0c, 0x4d, - 0xe2, 0xb3, 0xaa, 0x26, 0x31, 0xb1, 0x70, 0x46, 0x7c, 0x6b, 0xd1, 0x6f, 0xb7, 0x9d, 0x8e, 0x8b, - 0xb4, 0x5b, 0x0b, 0xaa, 0x82, 0x51, 0x85, 0xa9, 0xa4, 0x22, 0x2b, 0x5e, 0x18, 0x91, 0x79, 0x28, - 0x4a, 0x08, 0xdb, 0x44, 0xf2, 0x99, 0x55, 0xb6, 0x12, 0x1c, 0xf3, 0x4f, 0x72, 0x00, 0x49, 0xc9, - 0x73, 0x2a, 0x67, 0xbe, 0xa0, 0xc9, 0x99, 0x33, 0xe9, 0x09, 0x3a, 0x50, 0xc2, 0x90, 0xf7, 0x61, - 0x8c, 0xa9, 0x5c, 0x3d, 0xa9, 0x52, 0x9e, 0x4b, 0x93, 0x62, 0xe1, 0xd6, 0xcd, 0xda, 0x94, 0x20, - 0x1e, 0x0b, 0x11, 0x62, 0x09, 0x32, 0x45, 0x44, 0xfd, 0xd1, 0x68, 0x32, 0x18, 0x42, 0x38, 0x5d, - 0x51, 0xa4, 0x8b, 0x91, 0xac, 0x47, 0x29, 0x5d, 0x14, 0xd9, 0x72, 0x9e, 0xcb, 0x16, 0xde, 0xa9, - 0xe3, 0x42, 0xb6, 0xa4, 0x25, 0x0b, 0xef, 0xc0, 0x13, 0x25, 0x4b, 0x37, 0xbd, 0x6c, 0x47, 0x70, - 0x1a, 0x5c, 0xc9, 0xec, 0x95, 0xac, 0x05, 0x7b, 0xe9, 0xa4, 0x05, 0x9b, 0x5e, 0xae, 0x37, 0x07, - 0xc9, 0xb2, 0x33, 0x72, 0x75, 0x39, 0x8f, 0x54, 0x72, 0x94, 0x69, 0x6f, 0xf3, 0xa5, 0x39, 0x36, - 0x70, 0x69, 0x9e, 0xc9, 0x5c, 0x9a, 0x7c, 0x61, 0xbe, 0x0d, 0xa3, 0xd5, 0x9f, 0xe9, 0x05, 0x54, - 0xe8, 0x7e, 0x25, 0xf9, 0x4d, 0x06, 0x8b, 0xd7, 0xf4, 0xb4, 0xc3, 0x7e, 0xaa, 0x3a, 0x33, 0x96, - 0xb3, 0x2f, 0x6f, 0xac, 0x34, 0x84, 0x5e, 0x47, 0x52, 0xdd, 0xb2, 0xb1, 0xa2, 0x54, 0x3b, 0xd2, - 0x5a, 0xcd, 0xa8, 0xc8, 0x3c, 0xe4, 0xaa, 0x75, 0x34, 0x16, 0x27, 0x16, 0x8a, 0xf2, 0xb3, 0xf5, - 0xda, 0x69, 0x41, 0x52, 0x72, 0x34, 0xfb, 0xa1, 0x5a, 0x27, 0x35, 0x18, 0xbd, 0x7f, 0xd0, 0xf8, - 0x70, 0x45, 0x08, 0xb2, 0x59, 0x39, 0xaf, 0x19, 0x6c, 0x0d, 0x77, 0xa1, 0x30, 0xa9, 0x71, 0xfb, - 0x20, 0xfc, 0x46, 0x4b, 0xad, 0x31, 0xa2, 0x7d, 0x72, 0x02, 0xe4, 0x9f, 0x1b, 0x8a, 0xae, 0x21, - 0xe6, 0x3a, 0xb3, 0x69, 0xc5, 0x8c, 0x33, 0x12, 0xcd, 0xa7, 0x6f, 0xc6, 0xc5, 0xf3, 0xed, 0x2a, - 0x1f, 0xfd, 0x5c, 0xdf, 0xe8, 0x4f, 0x28, 0x3b, 0x19, 0x1f, 0xf3, 0xb8, 0x2f, 0xf2, 0x1f, 0xbb, - 0x2f, 0xcc, 0xbf, 0x34, 0xf0, 0x7b, 0xe4, 0x75, 0x18, 0xb3, 0xe8, 0x6e, 0xb2, 0xe9, 0xa3, 0xf1, - 0x18, 0x20, 0x44, 0xad, 0x24, 0xc7, 0xc1, 0x1d, 0x85, 0xba, 0xe1, 0x9e, 0xf7, 0x20, 0x12, 0x35, - 0x8d, 0x77, 0x14, 0x01, 0x56, 0x76, 0x14, 0x01, 0xd1, 0x76, 0x14, 0x01, 0x63, 0x73, 0xdd, 0xaa, - 0x37, 0x44, 0x03, 0x64, 0x6b, 0xad, 0xba, 0x32, 0x69, 0x02, 0x57, 0x9b, 0x34, 0x56, 0xbd, 0x41, - 0x6e, 0x41, 0xb1, 0xda, 0x6c, 0xfa, 0x3d, 0xc5, 0xfa, 0x9a, 0x3b, 0x3a, 0xac, 0x9c, 0x76, 0x38, - 0x50, 0x3f, 0x2b, 0x48, 0x50, 0xcd, 0x5a, 0x52, 0x6b, 0xc6, 0x63, 0xb1, 0xd5, 0x0b, 0x23, 0x1a, - 0x2c, 0xd7, 0x45, 0x93, 0x91, 0x47, 0x93, 0x03, 0x53, 0x3c, 0x62, 0x54, 0xf3, 0xbf, 0x18, 0x58, - 0x63, 0xf2, 0x16, 0xc0, 0x72, 0x87, 0x69, 0xd8, 0x4d, 0x1a, 0x33, 0x40, 0x2b, 0xde, 0x13, 0x50, - 0x9d, 0x83, 0x82, 0xac, 0x7f, 0x3a, 0x37, 0xf4, 0xa7, 0xd9, 0x27, 0xa5, 0xbe, 0x2e, 0x0e, 0x74, - 0xc4, 0x27, 0x03, 0x01, 0x4d, 0x7d, 0x32, 0x41, 0x26, 0x97, 0x61, 0x7c, 0xb9, 0x7a, 0xbf, 0xda, - 0x8b, 0xf6, 0xb0, 0xbf, 0x0a, 0x5c, 0x72, 0x7a, 0x4e, 0xdb, 0x76, 0x7a, 0xd1, 0x9e, 0x25, 0x0b, - 0xcd, 0x5f, 0x34, 0x60, 0x42, 0x11, 0x1a, 0xac, 0xaa, 0xeb, 0x81, 0xff, 0x11, 0x6d, 0x46, 0x7a, - 0x2f, 0x75, 0x39, 0x30, 0x55, 0xd5, 0x18, 0x35, 0xd5, 0x3b, 0xb9, 0x27, 0xe8, 0x1d, 0x73, 0x5e, - 0xc8, 0x22, 0x66, 0x8c, 0x28, 0x67, 0x2d, 0x68, 0x8c, 0x30, 0x65, 0x4b, 0x35, 0x46, 0x58, 0xb9, - 0xf9, 0xfb, 0x06, 0x93, 0x21, 0x64, 0x1e, 0xe0, 0x1e, 0x3d, 0x88, 0x9c, 0x9d, 0xdb, 0x5e, 0x4b, - 0x3b, 0x43, 0xdb, 0x47, 0xa8, 0xfd, 0xc0, 0x6b, 0x51, 0x4b, 0x41, 0x21, 0x37, 0xa1, 0x70, 0x2f, - 0xd8, 0x79, 0x13, 0xd1, 0x73, 0xf1, 0x5e, 0x30, 0xbb, 0x1f, 0xec, 0xbc, 0x89, 0xc8, 0xea, 0x7c, - 0x95, 0x88, 0xc4, 0x84, 0xb1, 0xba, 0xdf, 0x76, 0x3c, 0xb9, 0xff, 0x02, 0xdb, 0xc4, 0x5c, 0x84, - 0x58, 0xa2, 0x84, 0xed, 0x3e, 0x8d, 0xf5, 0x55, 0x31, 0x31, 0x71, 0xf7, 0x09, 0xbb, 0x1d, 0x8b, - 0xc1, 0xcc, 0x3f, 0x34, 0x60, 0x42, 0x11, 0x8d, 0xe4, 0xf3, 0xe2, 0xbc, 0xc1, 0xc0, 0xd3, 0xb2, - 0xb3, 0xfd, 0xc2, 0x93, 0x95, 0x72, 0xbd, 0xa1, 0xed, 0xbb, 0x54, 0x9c, 0x3e, 0x24, 0x12, 0x25, - 0x37, 0x8c, 0x44, 0x79, 0x0b, 0x80, 0x2b, 0x95, 0xd8, 0x9d, 0xca, 0xbc, 0x51, 0x4e, 0x17, 0xd5, - 0xc1, 0x48, 0x90, 0x4d, 0x0b, 0x4a, 0xaa, 0x34, 0x21, 0x35, 0x98, 0x14, 0x36, 0x94, 0xd0, 0x42, - 0x78, 0x3f, 0x5f, 0x64, 0xdb, 0x9b, 0xe0, 0xd6, 0x6f, 0xd3, 0xe9, 0x24, 0xe6, 0xdf, 0xc8, 0x41, - 0x41, 0x40, 0x16, 0x9e, 0x53, 0x05, 0xe9, 0x4d, 0x4d, 0x41, 0x92, 0x72, 0x57, 0xd1, 0xdc, 0x17, - 0x4e, 0x30, 0xc0, 0xde, 0x82, 0x92, 0xec, 0x02, 0xd4, 0x33, 0xaf, 0xc2, 0xb8, 0x3c, 0x42, 0xe0, - 0x5a, 0xe6, 0xb4, 0xc6, 0x73, 0x6b, 0xc1, 0x92, 0xe5, 0xe6, 0x77, 0x47, 0x25, 0x2d, 0xff, 0x12, - 0xeb, 0xc2, 0xaa, 0xeb, 0x06, 0x6a, 0x17, 0x3a, 0xae, 0x1b, 0x58, 0x08, 0x65, 0x83, 0xbf, 0xde, - 0xdb, 0x69, 0x79, 0x4d, 0xc4, 0x51, 0x56, 0x62, 0x17, 0xa1, 0x36, 0x43, 0x55, 0x07, 0x3f, 0x41, - 0xd6, 0xec, 0x9f, 0xfc, 0xb1, 0xf6, 0xcf, 0x4f, 0x41, 0x71, 0xb1, 0xed, 0x6a, 0xfa, 0x91, 0x99, - 0xd1, 0x29, 0xd7, 0x63, 0x24, 0xae, 0x19, 0x5d, 0x14, 0x7d, 0x74, 0xba, 0xd9, 0x76, 0xfb, 0xb5, - 0xa2, 0x84, 0xa5, 0x66, 0xc0, 0x8c, 0x3e, 0x8d, 0x01, 0x73, 0x0b, 0x8a, 0x9b, 0x21, 0xdd, 0xe8, - 0x75, 0x3a, 0xb4, 0x85, 0xba, 0x52, 0x81, 0xcb, 0xb3, 0x5e, 0x48, 0xed, 0x08, 0xa1, 0x6a, 0x05, - 0x62, 0x54, 0x75, 0x5a, 0x8d, 0x1f, 0x33, 0xad, 0x3e, 0x0f, 0x23, 0xd5, 0x6e, 0x57, 0x5a, 0x76, - 0xf1, 0xe6, 0xdd, 0xed, 0xe2, 0x76, 0x3a, 0xe5, 0x74, 0xbb, 0xba, 0x9d, 0x86, 0xd8, 0x84, 0x02, - 0xb9, 0xd7, 0xdb, 0xa1, 0x41, 0x87, 0x46, 0x34, 0x14, 0xe2, 0x3e, 0x9c, 0x03, 0xe4, 0x31, 0x27, - 0x0f, 0xd0, 0xd3, 0x08, 0x68, 0x95, 0x9f, 0xdb, 0xef, 0xed, 0x50, 0x5b, 0xec, 0x1b, 0x6a, 0xdf, - 0x65, 0x30, 0xbc, 0xd0, 0x80, 0x29, 0xbd, 0xff, 0x9f, 0x81, 0xc6, 0xf3, 0xc1, 0x48, 0xa1, 0x50, - 0x2e, 0x9a, 0xbf, 0x94, 0x83, 0x89, 0x6a, 0xb7, 0xfb, 0x9c, 0x1f, 0xaf, 0xfc, 0x84, 0xb6, 0xaa, - 0xcf, 0x26, 0xa3, 0xf7, 0x04, 0x27, 0x2b, 0x7f, 0x65, 0xc0, 0x74, 0x8a, 0x42, 0xad, 0xbd, 0x31, - 0xe4, 0x71, 0x43, 0x6e, 0xc8, 0xe3, 0x86, 0xfc, 0xe0, 0xe3, 0x06, 0x75, 0xcd, 0x8c, 0x3c, 0xcd, - 0x9a, 0x79, 0x0d, 0xf2, 0xd5, 0x6e, 0x57, 0xf4, 0x4a, 0x29, 0xe9, 0x95, 0xad, 0x9b, 0x7c, 0x73, - 0x73, 0xba, 0x5d, 0x8b, 0x61, 0x98, 0x6f, 0x40, 0x11, 0xc1, 0x28, 0xd1, 0x2e, 0x89, 0xa5, 0xc0, - 0xc5, 0x99, 0x46, 0xc6, 0xa7, 0xbd, 0xf9, 0x7f, 0x0c, 0x18, 0xc5, 0xdf, 0xcf, 0xe9, 0x74, 0x59, - 0xd0, 0xa6, 0x4b, 0x59, 0x99, 0x2e, 0xc3, 0x4c, 0x94, 0x3f, 0xca, 0x63, 0x6f, 0x89, 0x29, 0x22, - 0x0c, 0x56, 0x23, 0xc3, 0x60, 0x7d, 0x0a, 0x01, 0xbe, 0x9f, 0x36, 0x5d, 0xf3, 0x38, 0x18, 0x2f, - 0xa7, 0xab, 0xfa, 0x4c, 0xac, 0xd6, 0xbb, 0x40, 0x96, 0x3b, 0x21, 0x6d, 0xf6, 0x02, 0xda, 0xd8, - 0xf7, 0xba, 0x5b, 0x34, 0xf0, 0x1e, 0x1c, 0x08, 0x6d, 0x13, 0x65, 0xac, 0x27, 0x4a, 0xed, 0x70, - 0xdf, 0xeb, 0x32, 0x35, 0xc1, 0x7b, 0x70, 0x60, 0x65, 0xd0, 0x90, 0xf7, 0x61, 0xdc, 0xa2, 0x8f, - 0x02, 0x2f, 0xa2, 0xa2, 0x6f, 0xa7, 0x62, 0xdb, 0x02, 0xa1, 0x5c, 0xdf, 0x09, 0xf8, 0x0f, 0x75, - 0xfc, 0x45, 0xf9, 0x27, 0x67, 0xdf, 0x7d, 0x6f, 0x14, 0xd7, 0xc2, 0x09, 0xd7, 0x80, 0xc7, 0x9c, - 0x3e, 0xe8, 0x83, 0x99, 0x7f, 0x92, 0xc1, 0xdc, 0x82, 0x12, 0x33, 0x29, 0x53, 0xc7, 0x10, 0x17, - 0x93, 0xb1, 0xbc, 0xae, 0x16, 0x1f, 0x77, 0x03, 0xa8, 0xf1, 0x21, 0x76, 0x7a, 0x92, 0xf0, 0x9b, - 0xc5, 0x17, 0x15, 0xc6, 0x19, 0xd3, 0x23, 0x16, 0x1d, 0x4d, 0xde, 0x59, 0x4f, 0x3c, 0x31, 0xc6, - 0x9e, 0x6e, 0x62, 0x8c, 0x7f, 0x9c, 0x89, 0x91, 0xbe, 0x7b, 0x2d, 0x3c, 0xc9, 0xdd, 0xeb, 0x85, - 0xf7, 0x61, 0xa6, 0xaf, 0x87, 0x9f, 0xe4, 0xfe, 0xf2, 0x93, 0x9b, 0x96, 0x3f, 0x1f, 0xf7, 0x0b, - 0x59, 0x40, 0x13, 0xd7, 0x0b, 0x68, 0x33, 0x42, 0xd1, 0x2b, 0xa4, 0x65, 0x20, 0x60, 0x29, 0x1b, - 0x1c, 0x61, 0xe4, 0x3d, 0x18, 0xe7, 0xf7, 0x3f, 0xe1, 0x5c, 0x0e, 0xc7, 0x7e, 0x52, 0x7c, 0x91, - 0x43, 0xc5, 0x25, 0x3c, 0xc7, 0x50, 0x7b, 0x55, 0x10, 0x99, 0x77, 0x60, 0x4c, 0xdc, 0x1f, 0x1d, - 0xbf, 0x2e, 0x2a, 0x30, 0xba, 0x95, 0xf4, 0x0c, 0x9e, 0xf9, 0xf3, 0x46, 0x58, 0x1c, 0x6e, 0xfe, - 0x8a, 0x01, 0x53, 0x7a, 0x2b, 0xc9, 0x75, 0x18, 0x13, 0x17, 0x9c, 0x06, 0x5e, 0x70, 0xb2, 0xd6, - 0x8c, 0xf1, 0xab, 0x4d, 0xed, 0x42, 0x53, 0x60, 0x31, 0xd1, 0x2f, 0x38, 0x60, 0x5b, 0x84, 0xe8, - 0x17, 0x93, 0xd4, 0x92, 0x65, 0xcc, 0x8c, 0xb3, 0x68, 0xd8, 0x6b, 0x45, 0xaa, 0x19, 0x17, 0x20, - 0xc4, 0x12, 0x25, 0xe6, 0xa1, 0x01, 0xd0, 0x68, 0xdc, 0xbd, 0x47, 0x0f, 0xd6, 0x1d, 0x2f, 0x40, - 0x53, 0x18, 0x57, 0xe3, 0x3d, 0x31, 0x5a, 0x25, 0x61, 0x0a, 0xf3, 0x95, 0xbb, 0x4f, 0x0f, 0x34, - 0x53, 0x58, 0xa2, 0xe2, 0x92, 0x0f, 0xbc, 0x87, 0x4e, 0x44, 0x19, 0x61, 0x0e, 0x09, 0xf9, 0x92, - 0xe7, 0xd0, 0x14, 0xa5, 0x82, 0x4c, 0xbe, 0x0e, 0x53, 0xc9, 0x2f, 0xf4, 0xaa, 0xc8, 0xa3, 0x9d, - 0x28, 0x67, 0x84, 0x5e, 0x58, 0x7b, 0xe9, 0xe8, 0xb0, 0x72, 0x41, 0xe1, 0x6a, 0x33, 0x2c, 0x85, - 0x75, 0x8a, 0x99, 0xf9, 0xbb, 0x06, 0xc0, 0xc6, 0x4a, 0x43, 0x36, 0xf0, 0x32, 0x8c, 0xc4, 0xa7, - 0x54, 0x25, 0x6e, 0x6f, 0xa7, 0x0c, 0x4a, 0x2c, 0x27, 0x2f, 0x43, 0x3e, 0x69, 0xc9, 0xcc, 0xd1, - 0x61, 0x65, 0x52, 0x6f, 0x01, 0x2b, 0x25, 0x77, 0x60, 0x7c, 0xa8, 0x3a, 0xe3, 0xec, 0xcc, 0xa8, - 0xab, 0xa4, 0xc6, 0x51, 0xf8, 0x60, 0x7b, 0xe3, 0xd3, 0x3b, 0x0a, 0xdf, 0xce, 0xc1, 0x34, 0xeb, - 0xd7, 0x6a, 0x2f, 0xda, 0xf3, 0x03, 0x2f, 0x3a, 0x78, 0x6e, 0xad, 0xe2, 0x77, 0x34, 0x85, 0xe8, - 0x82, 0x14, 0x5b, 0x6a, 0xdb, 0x86, 0x32, 0x8e, 0xff, 0x62, 0x1c, 0x66, 0x33, 0xa8, 0xc8, 0xeb, - 0xc2, 0xb5, 0x28, 0x39, 0x87, 0x42, 0xd7, 0xa1, 0x1f, 0x1d, 0x56, 0x4a, 0x12, 0x7d, 0x23, 0x71, - 0x25, 0x5a, 0x80, 0x09, 0x61, 0xfa, 0xac, 0x26, 0x1a, 0x35, 0xfa, 0xa4, 0xc8, 0x73, 0x36, 0x14, - 0x4d, 0x2a, 0x12, 0xa9, 0x42, 0x69, 0x71, 0x8f, 0x36, 0xf7, 0xbd, 0xce, 0xee, 0x3d, 0x7a, 0xc0, - 0xf5, 0xa5, 0x52, 0xed, 0x45, 0x66, 0x69, 0x35, 0x05, 0x9c, 0x0d, 0xa9, 0x6e, 0xc4, 0x69, 0x24, - 0xe4, 0x3d, 0x98, 0x68, 0x78, 0xbb, 0x1d, 0xc9, 0x61, 0x04, 0x39, 0x5c, 0x3c, 0x3a, 0xac, 0x9c, - 0x0d, 0x39, 0xb8, 0x9f, 0x81, 0x4a, 0x40, 0xae, 0xc2, 0xa8, 0xe5, 0xb7, 0x28, 0xdf, 0x86, 0x85, - 0xb3, 0x4a, 0xc0, 0x00, 0xea, 0xd1, 0x2d, 0x62, 0x90, 0xbb, 0x30, 0xce, 0xfe, 0xb9, 0xef, 0x74, - 0xe7, 0xc6, 0x50, 0x6e, 0x93, 0x58, 0xc1, 0x47, 0x68, 0xd7, 0xeb, 0xec, 0xaa, 0x3a, 0x7e, 0x8b, - 0xda, 0x6d, 0xa7, 0xab, 0xed, 0x8b, 0x1c, 0x91, 0x6c, 0xc1, 0x44, 0x22, 0x08, 0xc2, 0xb9, 0x71, - 0xed, 0xa2, 0x2b, 0x29, 0xa9, 0x7d, 0x46, 0x30, 0x3b, 0x17, 0xb5, 0x42, 0x9c, 0xdb, 0x5d, 0x86, - 0xaf, 0x37, 0x46, 0x61, 0xa4, 0xd9, 0x20, 0x85, 0xc1, 0x36, 0x88, 0x71, 0xa2, 0x0d, 0xe2, 0x02, - 0x88, 0x4e, 0xaa, 0xb6, 0x76, 0x85, 0x6f, 0xd9, 0xd5, 0xc1, 0x13, 0xec, 0x7a, 0x82, 0x8c, 0x6b, - 0x92, 0x9f, 0x76, 0x89, 0xfe, 0x77, 0x5a, 0xbb, 0xda, 0x69, 0x57, 0x8c, 0xca, 0xba, 0x21, 0x11, - 0x35, 0xd2, 0x02, 0x97, 0xdd, 0x90, 0x94, 0x24, 0xdd, 0xf0, 0xd1, 0xa3, 0x68, 0x50, 0x37, 0x28, - 0x8c, 0xc8, 0x2a, 0x40, 0xb5, 0x19, 0x79, 0x0f, 0x29, 0x4e, 0x89, 0x09, 0xad, 0x23, 0x16, 0xab, - 0xf7, 0xe8, 0x41, 0x83, 0x46, 0xb1, 0xdb, 0xc6, 0x19, 0x07, 0x51, 0x53, 0xd3, 0xc4, 0x52, 0x38, - 0x90, 0x2e, 0x9c, 0xa9, 0xba, 0xae, 0xc7, 0xfd, 0x0d, 0x37, 0x02, 0x36, 0x7f, 0x5d, 0x64, 0x5d, - 0xca, 0x66, 0x7d, 0x55, 0xb0, 0xfe, 0x8c, 0x13, 0x53, 0xd9, 0x11, 0x27, 0x4b, 0x7f, 0x26, 0x9b, - 0xb1, 0xb9, 0x06, 0x53, 0x7a, 0x97, 0xea, 0x9e, 0x76, 0x25, 0x28, 0x58, 0x8d, 0xaa, 0xdd, 0xb8, - 0x5b, 0xbd, 0x51, 0x36, 0x48, 0x19, 0x4a, 0xe2, 0xd7, 0x82, 0xbd, 0xf0, 0xe6, 0xad, 0x72, 0x4e, - 0x83, 0xbc, 0x79, 0x63, 0xa1, 0x9c, 0x37, 0xff, 0xc8, 0x80, 0x82, 0xac, 0x1f, 0xb9, 0x05, 0xf9, - 0x46, 0xe3, 0x6e, 0xea, 0x7e, 0x35, 0xd9, 0x7a, 0xf9, 0x26, 0x13, 0x86, 0x7b, 0xea, 0x26, 0xd3, - 0x68, 0xdc, 0x65, 0x74, 0x1b, 0x2b, 0x0d, 0xa1, 0xb4, 0x64, 0x4c, 0xd7, 0x99, 0x01, 0x97, 0x4e, - 0xb7, 0x20, 0xff, 0xc1, 0xf6, 0x86, 0xb0, 0x86, 0x32, 0xc6, 0x17, 0xe9, 0x3e, 0x7a, 0xa4, 0x6e, - 0x7d, 0x8c, 0xc0, 0xb4, 0x60, 0x42, 0x59, 0x5a, 0x5c, 0x89, 0x68, 0xfb, 0xb1, 0x0f, 0x9a, 0x50, - 0x22, 0x18, 0xc4, 0x12, 0x25, 0x4c, 0xe7, 0x59, 0xf1, 0x9b, 0x4e, 0x4b, 0x68, 0x23, 0xa8, 0xf3, - 0xb4, 0x18, 0xc0, 0xe2, 0x70, 0xf3, 0x8f, 0x0d, 0x28, 0xaf, 0x07, 0xfe, 0x43, 0x8f, 0x49, 0xe0, - 0x0d, 0x7f, 0x9f, 0x76, 0xb6, 0x6e, 0x90, 0x37, 0xa4, 0x10, 0xe0, 0x2a, 0xdc, 0x39, 0x46, 0x85, - 0x42, 0xe0, 0x47, 0x87, 0x15, 0x68, 0x1c, 0x84, 0x11, 0x6d, 0xb3, 0x72, 0x29, 0x08, 0x14, 0x57, - 0xbe, 0xdc, 0xf0, 0xee, 0x41, 0x27, 0xb8, 0xf2, 0x55, 0x60, 0x14, 0xab, 0xa3, 0x78, 0x68, 0x8c, - 0x46, 0x0c, 0x60, 0x71, 0xb8, 0x22, 0xb0, 0xbf, 0x9b, 0xeb, 0x6b, 0xc3, 0xc2, 0xa7, 0xca, 0xc5, - 0x46, 0x6f, 0xdc, 0x50, 0x9b, 0xd8, 0x57, 0xe0, 0x74, 0xba, 0x4b, 0xf0, 0x5c, 0xa4, 0x0a, 0xd3, - 0x3a, 0x5c, 0x1e, 0x91, 0x9c, 0xcb, 0xfc, 0xd6, 0xd6, 0x82, 0x95, 0xc6, 0x37, 0x7f, 0x60, 0x40, - 0x11, 0xff, 0xb5, 0x7a, 0x2d, 0xca, 0x34, 0x9b, 0xea, 0x76, 0x43, 0x5c, 0x72, 0xa9, 0x17, 0x51, - 0xce, 0xa3, 0xd0, 0x16, 0x37, 0x62, 0x9a, 0x1c, 0x89, 0x91, 0x05, 0x29, 0xbf, 0xd2, 0x0b, 0xc5, - 0x0c, 0x8d, 0x49, 0xf9, 0xdd, 0x5f, 0x98, 0x22, 0x15, 0xc8, 0x6c, 0xfc, 0xd8, 0x2f, 0xbf, 0x25, - 0x8f, 0x86, 0x71, 0xfc, 0x90, 0xce, 0xd7, 0xae, 0x4e, 0x24, 0x1a, 0x79, 0x03, 0xc6, 0xd8, 0xa7, - 0x2d, 0x79, 0x31, 0x82, 0x56, 0x05, 0xd6, 0x31, 0xd0, 0x6e, 0x18, 0x39, 0x92, 0xf9, 0x6f, 0x72, - 0xe9, 0x0e, 0x14, 0x5a, 0xc0, 0x13, 0xae, 0x8d, 0xb7, 0x61, 0xb4, 0xda, 0x6a, 0xf9, 0x8f, 0x84, - 0x94, 0x90, 0xc7, 0x34, 0x71, 0xff, 0xf1, 0x1d, 0xd6, 0x61, 0x28, 0xda, 0xd5, 0x36, 0x03, 0x90, - 0x45, 0x28, 0x56, 0xb7, 0x1b, 0xcb, 0xcb, 0xf5, 0x8d, 0x8d, 0x15, 0xe1, 0x41, 0xfd, 0xaa, 0xec, - 0x1f, 0xcf, 0x73, 0xed, 0x28, 0x6a, 0x0d, 0x70, 0xb0, 0x4c, 0xe8, 0xc8, 0xbb, 0x00, 0x1f, 0xf8, - 0x5e, 0xe7, 0x3e, 0x8d, 0xf6, 0x7c, 0x57, 0x34, 0x9e, 0xa9, 0x14, 0x13, 0x1f, 0xf9, 0x5e, 0xc7, - 0x6e, 0x23, 0x98, 0xd5, 0x3d, 0x41, 0xb2, 0x94, 0xff, 0x59, 0x4f, 0xd7, 0xfc, 0x08, 0x75, 0x98, - 0xd1, 0xa4, 0xa7, 0x77, 0xfc, 0x28, 0x7d, 0x6f, 0x23, 0xd1, 0xcc, 0x5f, 0xcd, 0xc1, 0x14, 0xb7, - 0x54, 0xf9, 0x84, 0x79, 0x6e, 0x17, 0xe3, 0xdb, 0xda, 0x62, 0x3c, 0x2f, 0x37, 0x06, 0xa5, 0x69, - 0x43, 0x2d, 0xc5, 0x3d, 0x20, 0xfd, 0x34, 0xc4, 0x92, 0xe7, 0x29, 0xc3, 0xac, 0xc2, 0x1b, 0xc9, - 0x7d, 0x74, 0x88, 0x44, 0x36, 0x8a, 0xc2, 0xd0, 0xd2, 0x78, 0x98, 0xbf, 0x92, 0x83, 0x49, 0x45, - 0x9f, 0x7c, 0x6e, 0x3b, 0xfe, 0x8b, 0x5a, 0xc7, 0xcb, 0x3b, 0x08, 0xa5, 0x65, 0x43, 0xf5, 0x7b, - 0x0f, 0x66, 0xfa, 0x48, 0xd2, 0x6a, 0xb9, 0x31, 0x8c, 0x5a, 0xfe, 0x7a, 0xff, 0x85, 0x39, 0xf7, - 0xb6, 0x8e, 0x2f, 0xcc, 0xd5, 0x1b, 0xfa, 0x6f, 0xe7, 0xe0, 0xb4, 0xf8, 0x55, 0xed, 0xb9, 0x5e, - 0xb4, 0xe8, 0x77, 0x1e, 0x78, 0xbb, 0xcf, 0xed, 0x58, 0x54, 0xb5, 0xb1, 0xa8, 0xe8, 0x63, 0xa1, - 0x34, 0x70, 0xf0, 0x90, 0x98, 0x7f, 0x1b, 0x60, 0x6e, 0x10, 0x01, 0x33, 0xfb, 0x15, 0xab, 0x0a, - 0xcd, 0xfe, 0x94, 0xc5, 0xca, 0xed, 0xa9, 0xc4, 0x41, 0x24, 0x37, 0x84, 0x83, 0xc8, 0x0a, 0x94, + 0xb1, 0x17, 0x04, 0xb4, 0xc3, 0x44, 0x75, 0x04, 0x91, 0xa7, 0x8e, 0x0e, 0x2b, 0xd0, 0xe4, 0x40, + 0x36, 0xb9, 0x12, 0x04, 0xd6, 0xd5, 0x8d, 0xc8, 0x09, 0x22, 0xea, 0xce, 0x8d, 0x0e, 0xd5, 0xd5, + 0x6c, 0x7a, 0xcd, 0x84, 0x9c, 0x24, 0xdd, 0xd5, 0x82, 0x13, 0xb9, 0x0b, 0x13, 0x77, 0x02, 0xa7, + 0x49, 0xd7, 0x69, 0xe0, 0xf9, 0x2e, 0x8e, 0x61, 0xbe, 0x76, 0xf9, 0xe8, 0xb0, 0x72, 0x76, 0x97, + 0x81, 0xed, 0x2e, 0xc2, 0x13, 0xea, 0x1f, 0x1d, 0x56, 0x0a, 0xf5, 0x5e, 0x80, 0xbd, 0x67, 0xa9, + 0xa4, 0xe4, 0xa7, 0xd9, 0x90, 0x84, 0x11, 0x76, 0x2d, 0x75, 0x71, 0xf4, 0x8e, 0xaf, 0xa2, 0x29, + 0xaa, 0x78, 0xb6, 0xe5, 0x84, 0x91, 0x1d, 0x70, 0xba, 0x54, 0x3d, 0x55, 0x96, 0x64, 0x0d, 0x0a, + 0x8d, 0xe6, 0x1e, 0x75, 0x7b, 0x2d, 0x3a, 0x57, 0x40, 0xf6, 0xe7, 0x84, 0xe0, 0xca, 0xf1, 0x94, + 0xc5, 0xb5, 0x0b, 0x82, 0x37, 0x09, 0x05, 0x44, 0xe9, 0xfb, 0x98, 0xc9, 0x17, 0x0b, 0xbf, 0xf1, + 0x3b, 0x95, 0x53, 0xbf, 0xf8, 0xdf, 0x2f, 0x9d, 0x32, 0xff, 0x6d, 0x0e, 0xca, 0x69, 0x26, 0xe4, + 0x01, 0x4c, 0x6e, 0x76, 0x5d, 0x27, 0xa2, 0x8b, 0x2d, 0x8f, 0x76, 0xa2, 0x10, 0x85, 0xe4, 0xf8, + 0x36, 0xbd, 0x22, 0xbe, 0x3b, 0xd7, 0x43, 0x42, 0xbb, 0xc9, 0x29, 0x53, 0xad, 0xd2, 0xd9, 0x26, + 0xdf, 0x69, 0xa0, 0x9e, 0x0e, 0x51, 0xc2, 0x9e, 0xec, 0x3b, 0x5c, 0xc3, 0x0f, 0xf8, 0x8e, 0x60, + 0x2b, 0x04, 0xa8, 0xe3, 0xee, 0x1c, 0xa0, 0x64, 0x0e, 0x2f, 0x40, 0x8c, 0x24, 0x43, 0x80, 0x18, + 0xd8, 0xfc, 0x5f, 0x06, 0x4c, 0x59, 0x34, 0xf4, 0x7b, 0x41, 0x93, 0xde, 0xa5, 0x8e, 0x4b, 0x03, + 0x26, 0xfe, 0xf7, 0xbc, 0x8e, 0x2b, 0xe6, 0x14, 0x8a, 0xff, 0xbe, 0xd7, 0x51, 0xa7, 0x30, 0x96, + 0x93, 0xcf, 0xc1, 0x78, 0xa3, 0xb7, 0x83, 0xa8, 0x7c, 0x4e, 0x9d, 0xc5, 0x11, 0xeb, 0xed, 0xd8, + 0x29, 0x74, 0x89, 0x46, 0xe6, 0x61, 0x7c, 0x8b, 0x06, 0x61, 0xa2, 0xf1, 0x50, 0xb3, 0x3f, 0xe4, + 0x20, 0x95, 0x40, 0x60, 0x91, 0x3b, 0x89, 0xd6, 0x15, 0x6b, 0xd2, 0x74, 0x4a, 0xd7, 0x25, 0xa2, + 0xd2, 0x16, 0x10, 0x55, 0x54, 0x24, 0x96, 0xf9, 0xdd, 0x1c, 0x94, 0xeb, 0x4e, 0xe4, 0xec, 0x38, + 0xa1, 0xe8, 0xcf, 0xad, 0x9b, 0x4c, 0x8f, 0x2b, 0x0d, 0x45, 0x3d, 0xce, 0x6a, 0xfe, 0xb1, 0x9b, + 0xf7, 0x6a, 0xba, 0x79, 0x13, 0x6c, 0x81, 0x14, 0xcd, 0x4b, 0x1a, 0xf5, 0xee, 0xc9, 0x8d, 0x2a, + 0x8b, 0x46, 0x15, 0x64, 0xa3, 0x92, 0xa6, 0x90, 0x77, 0x61, 0xa4, 0xd1, 0xa5, 0x4d, 0xa1, 0x44, + 0xa4, 0xee, 0xd7, 0x1b, 0xc7, 0x10, 0xb6, 0x6e, 0xd6, 0x4a, 0x82, 0xcd, 0x48, 0xd8, 0xa5, 0x4d, + 0x0b, 0xc9, 0x94, 0x49, 0xf3, 0xbd, 0x31, 0x38, 0x9d, 0x45, 0x46, 0xde, 0xd5, 0x17, 0x27, 0xde, + 0x3d, 0x2f, 0x0c, 0x5c, 0x9c, 0xe6, 0x0c, 0x7d, 0x79, 0xba, 0x06, 0x85, 0x75, 0x26, 0x90, 0x4d, + 0xbf, 0x25, 0x7a, 0x8e, 0x69, 0xc5, 0x42, 0x57, 0xc2, 0x0c, 0x2b, 0x2e, 0x27, 0x2f, 0x40, 0x7e, + 0xd3, 0x5a, 0x16, 0xdd, 0x55, 0x3c, 0x3a, 0xac, 0xe4, 0x7b, 0x81, 0x37, 0x67, 0x58, 0x0c, 0x4a, + 0xe6, 0x61, 0x6c, 0xb1, 0xba, 0x48, 0x83, 0x08, 0xbb, 0xa9, 0x54, 0x3b, 0xc7, 0xa4, 0xa5, 0xe9, + 0xd8, 0x4d, 0x1a, 0x44, 0xda, 0xe7, 0x05, 0x1a, 0xf9, 0x2c, 0xe4, 0xab, 0xdb, 0x0d, 0xd1, 0x33, + 0x20, 0x7a, 0xa6, 0xba, 0xdd, 0xa8, 0x4d, 0x8a, 0x8e, 0xc8, 0x3b, 0x8f, 0x42, 0xc6, 0xbd, 0xba, + 0xdd, 0x50, 0x47, 0x6b, 0xec, 0x98, 0xd1, 0xba, 0x02, 0x05, 0x66, 0x67, 0xb0, 0x05, 0x1e, 0x95, + 0x62, 0x91, 0x9b, 0x4f, 0x7b, 0x02, 0x66, 0xc5, 0xa5, 0xe4, 0xe5, 0xd8, 0x6c, 0x29, 0x24, 0xfc, + 0x84, 0xd9, 0x22, 0x8d, 0x15, 0xf2, 0x18, 0x26, 0xeb, 0x07, 0x1d, 0xa7, 0xed, 0x35, 0xc5, 0x12, + 0x5e, 0xc4, 0x25, 0xfc, 0xfa, 0x31, 0xc3, 0x78, 0x5d, 0x23, 0xe0, 0xab, 0xba, 0x54, 0xbe, 0x73, + 0x2e, 0x2f, 0xb3, 0xd3, 0x2b, 0xfc, 0x9c, 0x61, 0xe9, 0x1f, 0x62, 0x73, 0x49, 0xaa, 0x48, 0xb4, + 0xab, 0x12, 0xb1, 0x93, 0xe0, 0x64, 0x2e, 0x05, 0x02, 0xa2, 0xce, 0xa5, 0x78, 0xd1, 0x7d, 0x17, + 0xf2, 0x77, 0x16, 0xd7, 0xe7, 0x26, 0x90, 0x07, 0x11, 0x3c, 0xee, 0x2c, 0xae, 0x2f, 0xb6, 0xfc, + 0x9e, 0xdb, 0xf8, 0x70, 0xa5, 0x76, 0x4e, 0xb0, 0x99, 0xdc, 0x6d, 0x76, 0xb5, 0x1a, 0x31, 0x3a, + 0xb2, 0x04, 0x05, 0xd9, 0xca, 0xb9, 0x12, 0xf2, 0x98, 0x49, 0x35, 0x7e, 0xeb, 0x26, 0x9f, 0x6b, + 0xae, 0xf8, 0xad, 0xd6, 0x42, 0xe2, 0x5c, 0xd8, 0x06, 0xd2, 0xdf, 0x2f, 0x19, 0x96, 0xc4, 0x67, + 0x55, 0x4b, 0x62, 0x62, 0xe1, 0x8c, 0xf8, 0xd6, 0xa2, 0xdf, 0x6e, 0x3b, 0x1d, 0x17, 0x69, 0xb7, + 0x16, 0x54, 0x03, 0xa3, 0x0a, 0x53, 0x49, 0x45, 0x56, 0xbc, 0x30, 0x22, 0xf3, 0x50, 0x94, 0x10, + 0xb6, 0x88, 0xe4, 0x33, 0xab, 0x6c, 0x25, 0x38, 0xe6, 0x9f, 0xe4, 0x00, 0x92, 0x92, 0xe7, 0x54, + 0xcf, 0x7c, 0x41, 0xd3, 0x33, 0x67, 0xd2, 0x02, 0x3a, 0x50, 0xc3, 0x90, 0xf7, 0x61, 0x8c, 0x99, + 0x5c, 0x3d, 0x69, 0x52, 0x9e, 0x4b, 0x93, 0x62, 0xe1, 0xd6, 0xcd, 0xda, 0x94, 0x20, 0x1e, 0x0b, + 0x11, 0x62, 0x09, 0x32, 0x45, 0x45, 0xfd, 0xd1, 0x68, 0x32, 0x18, 0x42, 0x39, 0x5d, 0x51, 0xb4, + 0x8b, 0x91, 0xcc, 0x47, 0xa9, 0x5d, 0x14, 0xdd, 0x72, 0x9e, 0xeb, 0x16, 0xde, 0xa9, 0xe3, 0x42, + 0xb7, 0xa4, 0x35, 0x0b, 0xef, 0xc0, 0x13, 0x35, 0x4b, 0x37, 0x3d, 0x6d, 0x47, 0x50, 0x0c, 0xae, + 0x64, 0xf6, 0x4a, 0xd6, 0x84, 0xbd, 0x74, 0xd2, 0x84, 0x4d, 0x4f, 0xd7, 0x9b, 0x83, 0x74, 0xd9, + 0x19, 0x39, 0xbb, 0x9c, 0x47, 0x2a, 0x39, 0xea, 0xb4, 0xb7, 0xf9, 0xd4, 0x1c, 0x1b, 0x38, 0x35, + 0xcf, 0x64, 0x4e, 0x4d, 0x3e, 0x31, 0xdf, 0x86, 0xd1, 0xea, 0xcf, 0xf4, 0x02, 0x2a, 0x6c, 0xbf, + 0x92, 0xfc, 0x26, 0x83, 0xc5, 0x73, 0x7a, 0xda, 0x61, 0x3f, 0x55, 0x9b, 0x19, 0xcb, 0xd9, 0x97, + 0x37, 0x56, 0x1a, 0xc2, 0xae, 0x23, 0xa9, 0x6e, 0xd9, 0x58, 0x51, 0xaa, 0x1d, 0x69, 0xad, 0x66, + 0x54, 0x64, 0x1e, 0x72, 0xd5, 0x3a, 0x6e, 0x16, 0x27, 0x16, 0x8a, 0xf2, 0xb3, 0xf5, 0xda, 0x69, + 0x41, 0x52, 0x72, 0xb4, 0xfd, 0x43, 0xb5, 0x4e, 0x6a, 0x30, 0x7a, 0xff, 0xa0, 0xf1, 0xe1, 0x8a, + 0x50, 0x64, 0xb3, 0x52, 0xae, 0x19, 0x6c, 0x0d, 0x57, 0xa1, 0x30, 0xa9, 0x71, 0xfb, 0x20, 0xfc, + 0x46, 0x4b, 0xad, 0x31, 0xa2, 0x7d, 0x72, 0x0a, 0xe4, 0x5f, 0x1a, 0x8a, 0xad, 0x21, 0x64, 0x9d, + 0xed, 0x69, 0x85, 0xc4, 0x19, 0x89, 0xe5, 0xd3, 0x27, 0x71, 0xb1, 0xbc, 0x5d, 0xe5, 0xa3, 0x9f, + 0xeb, 0x1b, 0xfd, 0x09, 0x65, 0x25, 0xe3, 0x63, 0x1e, 0xf7, 0x45, 0xfe, 0x63, 0xf7, 0x85, 0xf9, + 0x97, 0x06, 0x7e, 0x8f, 0xbc, 0x0e, 0x63, 0x16, 0xdd, 0x4d, 0x16, 0x7d, 0xdc, 0x3c, 0x06, 0x08, + 0x51, 0x2b, 0xc9, 0x71, 0x70, 0x45, 0xa1, 0x6e, 0xb8, 0xe7, 0x3d, 0x88, 0x44, 0x4d, 0xe3, 0x15, + 0x45, 0x80, 0x95, 0x15, 0x45, 0x40, 0xb4, 0x15, 0x45, 0xc0, 0x98, 0xac, 0x5b, 0xf5, 0x86, 0x68, + 0x80, 0x6c, 0xad, 0x55, 0x57, 0x84, 0x26, 0x70, 0x35, 0xa1, 0xb1, 0xea, 0x0d, 0x72, 0x0b, 0x8a, + 0xd5, 0x66, 0xd3, 0xef, 0x29, 0xbb, 0xaf, 0xb9, 0xa3, 0xc3, 0xca, 0x69, 0x87, 0x03, 0xf5, 0xb3, + 0x82, 0x04, 0xd5, 0xac, 0x25, 0xb5, 0x66, 0x3c, 0x16, 0x5b, 0xbd, 0x30, 0xa2, 0xc1, 0x72, 0x5d, + 0x34, 0x19, 0x79, 0x34, 0x39, 0x30, 0xc5, 0x23, 0x46, 0x35, 0xff, 0x9b, 0x81, 0x35, 0x26, 0x6f, + 0x01, 0x2c, 0x77, 0x98, 0x85, 0xdd, 0xa4, 0x31, 0x03, 0xdc, 0xc5, 0x7b, 0x02, 0xaa, 0x73, 0x50, + 0x90, 0xf5, 0x4f, 0xe7, 0x86, 0xfe, 0x34, 0xfb, 0xa4, 0xb4, 0xd7, 0xc5, 0x81, 0x8e, 0xf8, 0x64, + 0x20, 0xa0, 0xa9, 0x4f, 0x26, 0xc8, 0xe4, 0x32, 0x8c, 0x2f, 0x57, 0xef, 0x57, 0x7b, 0xd1, 0x1e, + 0xf6, 0x57, 0x81, 0x6b, 0x4e, 0xcf, 0x69, 0xdb, 0x4e, 0x2f, 0xda, 0xb3, 0x64, 0xa1, 0xf9, 0x8b, + 0x06, 0x4c, 0x28, 0x4a, 0x83, 0x55, 0x75, 0x3d, 0xf0, 0x3f, 0xa2, 0xcd, 0x48, 0xef, 0xa5, 0x2e, + 0x07, 0xa6, 0xaa, 0x1a, 0xa3, 0xa6, 0x7a, 0x27, 0xf7, 0x04, 0xbd, 0x63, 0xce, 0x0b, 0x5d, 0xc4, + 0x36, 0x23, 0xca, 0x59, 0x0b, 0x6e, 0x46, 0x98, 0xb1, 0xa5, 0x6e, 0x46, 0x58, 0xb9, 0xf9, 0xfb, + 0x06, 0xd3, 0x21, 0x64, 0x1e, 0xe0, 0x1e, 0x3d, 0x88, 0x9c, 0x9d, 0xdb, 0x5e, 0x4b, 0x3b, 0x43, + 0xdb, 0x47, 0xa8, 0xfd, 0xc0, 0x6b, 0x51, 0x4b, 0x41, 0x21, 0x37, 0xa1, 0x70, 0x2f, 0xd8, 0x79, + 0x13, 0xd1, 0x73, 0xf1, 0x5a, 0x30, 0xbb, 0x1f, 0xec, 0xbc, 0x89, 0xc8, 0xaa, 0xbc, 0x4a, 0x44, + 0x62, 0xc2, 0x58, 0xdd, 0x6f, 0x3b, 0x9e, 0x5c, 0x7f, 0x81, 0x2d, 0x62, 0x2e, 0x42, 0x2c, 0x51, + 0xc2, 0x56, 0x9f, 0xc6, 0xfa, 0xaa, 0x10, 0x4c, 0x5c, 0x7d, 0xc2, 0x6e, 0xc7, 0x62, 0x30, 0xf3, + 0x0f, 0x0d, 0x98, 0x50, 0x54, 0x23, 0xf9, 0xbc, 0x38, 0x6f, 0x30, 0xf0, 0xb4, 0xec, 0x6c, 0xbf, + 0xf2, 0x64, 0xa5, 0xdc, 0x6e, 0x68, 0xfb, 0x2e, 0x15, 0xa7, 0x0f, 0x89, 0x46, 0xc9, 0x0d, 0xa3, + 0x51, 0xde, 0x02, 0xe0, 0x46, 0x25, 0x76, 0xa7, 0x22, 0x37, 0xca, 0xe9, 0xa2, 0x3a, 0x18, 0x09, + 0xb2, 0x69, 0x41, 0x49, 0xd5, 0x26, 0xa4, 0x06, 0x93, 0x62, 0x0f, 0x25, 0xac, 0x10, 0xde, 0xcf, + 0x17, 0xd9, 0xf2, 0x26, 0xb8, 0xf5, 0xef, 0xe9, 0x74, 0x12, 0xf3, 0x6f, 0xe5, 0xa0, 0x20, 0x20, + 0x0b, 0xcf, 0xa9, 0x81, 0xf4, 0xa6, 0x66, 0x20, 0x49, 0xbd, 0xab, 0x58, 0xee, 0x0b, 0x27, 0x6c, + 0xc0, 0xde, 0x82, 0x92, 0xec, 0x02, 0xb4, 0x33, 0xaf, 0xc2, 0xb8, 0x3c, 0x42, 0xe0, 0x56, 0xe6, + 0xb4, 0xc6, 0x73, 0x6b, 0xc1, 0x92, 0xe5, 0xe6, 0x77, 0x47, 0x25, 0x2d, 0xff, 0x12, 0xeb, 0xc2, + 0xaa, 0xeb, 0x06, 0x6a, 0x17, 0x3a, 0xae, 0x1b, 0x58, 0x08, 0x65, 0x83, 0xbf, 0xde, 0xdb, 0x69, + 0x79, 0x4d, 0xc4, 0x51, 0x66, 0x62, 0x17, 0xa1, 0x36, 0x43, 0x55, 0x07, 0x3f, 0x41, 0xd6, 0xf6, + 0x3f, 0xf9, 0x63, 0xf7, 0x3f, 0x3f, 0x05, 0xc5, 0xc5, 0xb6, 0xab, 0xd9, 0x47, 0x66, 0x46, 0xa7, + 0x5c, 0x8f, 0x91, 0xb8, 0x65, 0x74, 0x51, 0xf4, 0xd1, 0xe9, 0x66, 0xdb, 0xed, 0xb7, 0x8a, 0x12, + 0x96, 0xda, 0x06, 0x66, 0xf4, 0x69, 0x36, 0x30, 0xb7, 0xa0, 0xb8, 0x19, 0xd2, 0x8d, 0x5e, 0xa7, + 0x43, 0x5b, 0x68, 0x2b, 0x15, 0xb8, 0x3e, 0xeb, 0x85, 0xd4, 0x8e, 0x10, 0xaa, 0x56, 0x20, 0x46, + 0x55, 0xc5, 0x6a, 0xfc, 0x18, 0xb1, 0xfa, 0x3c, 0x8c, 0x54, 0xbb, 0x5d, 0xb9, 0xb3, 0x8b, 0x17, + 0xef, 0x6e, 0x17, 0x97, 0xd3, 0x29, 0xa7, 0xdb, 0xd5, 0xf7, 0x69, 0x88, 0x4d, 0x28, 0x90, 0x7b, + 0xbd, 0x1d, 0x1a, 0x74, 0x68, 0x44, 0x43, 0xa1, 0xee, 0xc3, 0x39, 0x40, 0x1e, 0x73, 0xf2, 0x00, + 0x3d, 0x8d, 0x80, 0xbb, 0xf2, 0x73, 0xfb, 0xbd, 0x1d, 0x6a, 0x8b, 0x75, 0x43, 0xed, 0xbb, 0x0c, + 0x86, 0x17, 0x1a, 0x30, 0xa5, 0xf7, 0xff, 0x33, 0xb0, 0x78, 0x3e, 0x18, 0x29, 0x14, 0xca, 0x45, + 0xf3, 0x97, 0x72, 0x30, 0x51, 0xed, 0x76, 0x9f, 0xf3, 0xe3, 0x95, 0x9f, 0xd0, 0x66, 0xf5, 0xd9, + 0x64, 0xf4, 0x9e, 0xe0, 0x64, 0xe5, 0xaf, 0x0d, 0x98, 0x4e, 0x51, 0xa8, 0xb5, 0x37, 0x86, 0x3c, + 0x6e, 0xc8, 0x0d, 0x79, 0xdc, 0x90, 0x1f, 0x7c, 0xdc, 0xa0, 0xce, 0x99, 0x91, 0xa7, 0x99, 0x33, + 0xaf, 0x41, 0xbe, 0xda, 0xed, 0x8a, 0x5e, 0x29, 0x25, 0xbd, 0xb2, 0x75, 0x93, 0x2f, 0x6e, 0x4e, + 0xb7, 0x6b, 0x31, 0x0c, 0xf3, 0x0d, 0x28, 0x22, 0x18, 0x35, 0xda, 0x25, 0x31, 0x15, 0xb8, 0x3a, + 0xd3, 0xc8, 0xb8, 0xd8, 0x9b, 0xff, 0xc7, 0x80, 0x51, 0xfc, 0xfd, 0x9c, 0x8a, 0xcb, 0x82, 0x26, + 0x2e, 0x65, 0x45, 0x5c, 0x86, 0x11, 0x94, 0x3f, 0xca, 0x63, 0x6f, 0x09, 0x11, 0x11, 0x1b, 0x56, + 0x23, 0x63, 0xc3, 0xfa, 0x14, 0x0a, 0x7c, 0x3f, 0xbd, 0x75, 0xcd, 0xe3, 0x60, 0xbc, 0x9c, 0xae, + 0xea, 0x33, 0xd9, 0xb5, 0xde, 0x05, 0xb2, 0xdc, 0x09, 0x69, 0xb3, 0x17, 0xd0, 0xc6, 0xbe, 0xd7, + 0xdd, 0xa2, 0x81, 0xf7, 0xe0, 0x40, 0x58, 0x9b, 0xa8, 0x63, 0x3d, 0x51, 0x6a, 0x87, 0xfb, 0x5e, + 0x97, 0x99, 0x09, 0xde, 0x83, 0x03, 0x2b, 0x83, 0x86, 0xbc, 0x0f, 0xe3, 0x16, 0x7d, 0x14, 0x78, + 0x11, 0x15, 0x7d, 0x3b, 0x15, 0xef, 0x2d, 0x10, 0xca, 0xed, 0x9d, 0x80, 0xff, 0x50, 0xc7, 0x5f, + 0x94, 0x7f, 0x72, 0xfb, 0xbb, 0xef, 0x8d, 0xe2, 0x5c, 0x38, 0xe1, 0x1a, 0xf0, 0x98, 0xd3, 0x07, + 0x7d, 0x30, 0xf3, 0x4f, 0x32, 0x98, 0x5b, 0x50, 0x62, 0x5b, 0xca, 0xd4, 0x31, 0xc4, 0xc5, 0x64, + 0x2c, 0xaf, 0xab, 0xc5, 0xc7, 0xdd, 0x00, 0x6a, 0x7c, 0x88, 0x9d, 0x16, 0x12, 0x7e, 0xb3, 0xf8, + 0xa2, 0xc2, 0x38, 0x43, 0x3c, 0x62, 0xd5, 0xd1, 0xe4, 0x9d, 0xf5, 0xc4, 0x82, 0x31, 0xf6, 0x74, + 0x82, 0x31, 0xfe, 0x71, 0x04, 0x23, 0x7d, 0xf7, 0x5a, 0x78, 0x92, 0xbb, 0xd7, 0x0b, 0xef, 0xc3, + 0x4c, 0x5f, 0x0f, 0x3f, 0xc9, 0xfd, 0xe5, 0x27, 0x27, 0x96, 0x3f, 0x1f, 0xf7, 0x0b, 0x59, 0xc0, + 0x2d, 0xae, 0x17, 0xd0, 0x66, 0x84, 0xaa, 0x57, 0x68, 0xcb, 0x40, 0xc0, 0x52, 0x7b, 0x70, 0x84, + 0x91, 0xf7, 0x60, 0x9c, 0xdf, 0xff, 0x84, 0x73, 0x39, 0x1c, 0xfb, 0x49, 0xf1, 0x45, 0x0e, 0x15, + 0x97, 0xf0, 0x1c, 0x43, 0xed, 0x55, 0x41, 0x64, 0xde, 0x81, 0x31, 0x71, 0x7f, 0x74, 0xfc, 0xbc, + 0xa8, 0xc0, 0xe8, 0x56, 0xd2, 0x33, 0x78, 0xe6, 0xcf, 0x1b, 0x61, 0x71, 0xb8, 0xf9, 0x2b, 0x06, + 0x4c, 0xe9, 0xad, 0x24, 0xd7, 0x61, 0x4c, 0x5c, 0x70, 0x1a, 0x78, 0xc1, 0xc9, 0x5a, 0x33, 0xc6, + 0xaf, 0x36, 0xb5, 0x0b, 0x4d, 0x81, 0xc5, 0x54, 0xbf, 0xe0, 0x80, 0x6d, 0x11, 0xaa, 0x5f, 0x08, + 0xa9, 0x25, 0xcb, 0xd8, 0x36, 0xce, 0xa2, 0x61, 0xaf, 0x15, 0xa9, 0xdb, 0xb8, 0x00, 0x21, 0x96, + 0x28, 0x31, 0x0f, 0x0d, 0x80, 0x46, 0xe3, 0xee, 0x3d, 0x7a, 0xb0, 0xee, 0x78, 0x01, 0x6e, 0x85, + 0x71, 0x36, 0xde, 0x13, 0xa3, 0x55, 0x12, 0x5b, 0x61, 0x3e, 0x73, 0xf7, 0xe9, 0x81, 0xb6, 0x15, + 0x96, 0xa8, 0x38, 0xe5, 0x03, 0xef, 0xa1, 0x13, 0x51, 0x46, 0x98, 0x43, 0x42, 0x3e, 0xe5, 0x39, + 0x34, 0x45, 0xa9, 0x20, 0x93, 0xaf, 0xc3, 0x54, 0xf2, 0x0b, 0xbd, 0x2a, 0xf2, 0xb8, 0x4f, 0x94, + 0x12, 0xa1, 0x17, 0xd6, 0x5e, 0x3a, 0x3a, 0xac, 0x5c, 0x50, 0xb8, 0xda, 0x0c, 0x4b, 0x61, 0x9d, + 0x62, 0x66, 0xfe, 0xae, 0x01, 0xb0, 0xb1, 0xd2, 0x90, 0x0d, 0xbc, 0x0c, 0x23, 0xf1, 0x29, 0x55, + 0x89, 0xef, 0xb7, 0x53, 0x1b, 0x4a, 0x2c, 0x27, 0x2f, 0x43, 0x3e, 0x69, 0xc9, 0xcc, 0xd1, 0x61, + 0x65, 0x52, 0x6f, 0x01, 0x2b, 0x25, 0x77, 0x60, 0x7c, 0xa8, 0x3a, 0xa3, 0x74, 0x66, 0xd4, 0x55, + 0x52, 0xe3, 0x28, 0x7c, 0xb0, 0xbd, 0xf1, 0xe9, 0x1d, 0x85, 0x6f, 0xe7, 0x60, 0x9a, 0xf5, 0x6b, + 0xb5, 0x17, 0xed, 0xf9, 0x81, 0x17, 0x1d, 0x3c, 0xb7, 0xbb, 0xe2, 0x77, 0x34, 0x83, 0xe8, 0x82, + 0x54, 0x5b, 0x6a, 0xdb, 0x86, 0xda, 0x1c, 0xff, 0xc5, 0x38, 0xcc, 0x66, 0x50, 0x91, 0xd7, 0x85, + 0x6b, 0x51, 0x72, 0x0e, 0x85, 0xae, 0x43, 0x3f, 0x3a, 0xac, 0x94, 0x24, 0xfa, 0x46, 0xe2, 0x4a, + 0xb4, 0x00, 0x13, 0x62, 0xeb, 0xb3, 0x9a, 0x58, 0xd4, 0xe8, 0x93, 0x22, 0xcf, 0xd9, 0x50, 0x35, + 0xa9, 0x48, 0xa4, 0x0a, 0xa5, 0xc5, 0x3d, 0xda, 0xdc, 0xf7, 0x3a, 0xbb, 0xf7, 0xe8, 0x01, 0xb7, + 0x97, 0x4a, 0xb5, 0x17, 0xd9, 0x4e, 0xab, 0x29, 0xe0, 0x6c, 0x48, 0xf5, 0x4d, 0x9c, 0x46, 0x42, + 0xde, 0x83, 0x89, 0x86, 0xb7, 0xdb, 0x91, 0x1c, 0x46, 0x90, 0xc3, 0xc5, 0xa3, 0xc3, 0xca, 0xd9, + 0x90, 0x83, 0xfb, 0x19, 0xa8, 0x04, 0xe4, 0x2a, 0x8c, 0x5a, 0x7e, 0x8b, 0xf2, 0x65, 0x58, 0x38, + 0xab, 0x04, 0x0c, 0xa0, 0x1e, 0xdd, 0x22, 0x06, 0xb9, 0x0b, 0xe3, 0xec, 0x9f, 0xfb, 0x4e, 0x77, + 0x6e, 0x0c, 0xf5, 0x36, 0x89, 0x0d, 0x7c, 0x84, 0x76, 0xbd, 0xce, 0xae, 0x6a, 0xe3, 0xb7, 0xa8, + 0xdd, 0x76, 0xba, 0xda, 0xba, 0xc8, 0x11, 0xc9, 0x16, 0x4c, 0x24, 0x8a, 0x20, 0x9c, 0x1b, 0xd7, + 0x2e, 0xba, 0x92, 0x92, 0xda, 0x67, 0x04, 0xb3, 0x73, 0x51, 0x2b, 0x44, 0xd9, 0xee, 0x32, 0x7c, + 0xbd, 0x31, 0x0a, 0x23, 0x6d, 0x0f, 0x52, 0x18, 0xbc, 0x07, 0x31, 0x4e, 0xdc, 0x83, 0xb8, 0x00, + 0xa2, 0x93, 0xaa, 0xad, 0x5d, 0xe1, 0x5b, 0x76, 0x75, 0xb0, 0x80, 0x5d, 0x4f, 0x90, 0x71, 0x4e, + 0xf2, 0xd3, 0x2e, 0xd1, 0xff, 0x4e, 0x6b, 0x57, 0x3b, 0xed, 0x8a, 0x51, 0x59, 0x37, 0x24, 0xaa, + 0x46, 0xee, 0xc0, 0x65, 0x37, 0x24, 0x25, 0x49, 0x37, 0x7c, 0xf4, 0x28, 0x1a, 0xd4, 0x0d, 0x0a, + 0x23, 0xb2, 0x0a, 0x50, 0x6d, 0x46, 0xde, 0x43, 0x8a, 0x22, 0x31, 0xa1, 0x75, 0xc4, 0x62, 0xf5, + 0x1e, 0x3d, 0x68, 0xd0, 0x28, 0x76, 0xdb, 0x38, 0xe3, 0x20, 0x6a, 0x4a, 0x4c, 0x2c, 0x85, 0x03, + 0xe9, 0xc2, 0x99, 0xaa, 0xeb, 0x7a, 0xdc, 0xdf, 0x70, 0x23, 0x60, 0xf2, 0xeb, 0x22, 0xeb, 0x52, + 0x36, 0xeb, 0xab, 0x82, 0xf5, 0x67, 0x9c, 0x98, 0xca, 0x8e, 0x38, 0x59, 0xfa, 0x33, 0xd9, 0x8c, + 0xcd, 0x35, 0x98, 0xd2, 0xbb, 0x54, 0xf7, 0xb4, 0x2b, 0x41, 0xc1, 0x6a, 0x54, 0xed, 0xc6, 0xdd, + 0xea, 0x8d, 0xb2, 0x41, 0xca, 0x50, 0x12, 0xbf, 0x16, 0xec, 0x85, 0x37, 0x6f, 0x95, 0x73, 0x1a, + 0xe4, 0xcd, 0x1b, 0x0b, 0xe5, 0xbc, 0xf9, 0x47, 0x06, 0x14, 0x64, 0xfd, 0xc8, 0x2d, 0xc8, 0x37, + 0x1a, 0x77, 0x53, 0xf7, 0xab, 0xc9, 0xd2, 0xcb, 0x17, 0x99, 0x30, 0xdc, 0x53, 0x17, 0x99, 0x46, + 0xe3, 0x2e, 0xa3, 0xdb, 0x58, 0x69, 0x08, 0xa3, 0x25, 0x43, 0x5c, 0x67, 0x06, 0x5c, 0x3a, 0xdd, + 0x82, 0xfc, 0x07, 0xdb, 0x1b, 0x62, 0x37, 0x94, 0x31, 0xbe, 0x48, 0xf7, 0xd1, 0x23, 0x75, 0xe9, + 0x63, 0x04, 0xa6, 0x05, 0x13, 0xca, 0xd4, 0xe2, 0x46, 0x44, 0xdb, 0x8f, 0x7d, 0xd0, 0x84, 0x11, + 0xc1, 0x20, 0x96, 0x28, 0x61, 0x36, 0xcf, 0x8a, 0xdf, 0x74, 0x5a, 0xc2, 0x1a, 0x41, 0x9b, 0xa7, + 0xc5, 0x00, 0x16, 0x87, 0x9b, 0x7f, 0x6c, 0x40, 0x79, 0x3d, 0xf0, 0x1f, 0x7a, 0x4c, 0x03, 0x6f, + 0xf8, 0xfb, 0xb4, 0xb3, 0x75, 0x83, 0xbc, 0x21, 0x95, 0x00, 0x37, 0xe1, 0xce, 0x31, 0x2a, 0x54, + 0x02, 0x3f, 0x3a, 0xac, 0x40, 0xe3, 0x20, 0x8c, 0x68, 0x9b, 0x95, 0x4b, 0x45, 0xa0, 0xb8, 0xf2, + 0xe5, 0x86, 0x77, 0x0f, 0x3a, 0xc1, 0x95, 0xaf, 0x02, 0xa3, 0x58, 0x1d, 0xc5, 0x43, 0x63, 0x34, + 0x62, 0x00, 0x8b, 0xc3, 0x15, 0x85, 0xfd, 0xdd, 0x5c, 0x5f, 0x1b, 0x16, 0x3e, 0x55, 0x2e, 0x36, + 0x7a, 0xe3, 0x86, 0x5a, 0xc4, 0xbe, 0x02, 0xa7, 0xd3, 0x5d, 0x82, 0xe7, 0x22, 0x55, 0x98, 0xd6, + 0xe1, 0xf2, 0x88, 0xe4, 0x5c, 0xe6, 0xb7, 0xb6, 0x16, 0xac, 0x34, 0xbe, 0xf9, 0x03, 0x03, 0x8a, + 0xf8, 0xaf, 0xd5, 0x6b, 0x51, 0x66, 0xd9, 0x54, 0xb7, 0x1b, 0xe2, 0x92, 0x4b, 0xbd, 0x88, 0x72, + 0x1e, 0x85, 0xb6, 0xb8, 0x11, 0xd3, 0xf4, 0x48, 0x8c, 0x2c, 0x48, 0xf9, 0x95, 0x5e, 0x28, 0x24, + 0x34, 0x26, 0xe5, 0x77, 0x7f, 0x61, 0x8a, 0x54, 0x20, 0xb3, 0xf1, 0x63, 0xbf, 0xfc, 0x96, 0x3c, + 0x1a, 0xc6, 0xf1, 0x43, 0x3a, 0x5f, 0xbb, 0x3a, 0x91, 0x68, 0xe4, 0x0d, 0x18, 0x63, 0x9f, 0xb6, + 0xe4, 0xc5, 0x08, 0xee, 0x2a, 0xb0, 0x8e, 0x81, 0x76, 0xc3, 0xc8, 0x91, 0xcc, 0x7f, 0x97, 0x4b, + 0x77, 0xa0, 0xb0, 0x02, 0x9e, 0x70, 0x6e, 0xbc, 0x0d, 0xa3, 0xd5, 0x56, 0xcb, 0x7f, 0x24, 0xb4, + 0x84, 0x3c, 0xa6, 0x89, 0xfb, 0x8f, 0xaf, 0xb0, 0x0e, 0x43, 0xd1, 0xae, 0xb6, 0x19, 0x80, 0x2c, + 0x42, 0xb1, 0xba, 0xdd, 0x58, 0x5e, 0xae, 0x6f, 0x6c, 0xac, 0x08, 0x0f, 0xea, 0x57, 0x65, 0xff, + 0x78, 0x9e, 0x6b, 0x47, 0x51, 0x6b, 0x80, 0x83, 0x65, 0x42, 0x47, 0xde, 0x05, 0xf8, 0xc0, 0xf7, + 0x3a, 0xf7, 0x69, 0xb4, 0xe7, 0xbb, 0xa2, 0xf1, 0xcc, 0xa4, 0x98, 0xf8, 0xc8, 0xf7, 0x3a, 0x76, + 0x1b, 0xc1, 0xac, 0xee, 0x09, 0x92, 0xa5, 0xfc, 0xcf, 0x7a, 0xba, 0xe6, 0x47, 0x68, 0xc3, 0x8c, + 0x26, 0x3d, 0xbd, 0xe3, 0x47, 0xe9, 0x7b, 0x1b, 0x89, 0x66, 0xfe, 0x6a, 0x0e, 0xa6, 0xf8, 0x4e, + 0x95, 0x0b, 0xcc, 0x73, 0x3b, 0x19, 0xdf, 0xd6, 0x26, 0xe3, 0x79, 0xb9, 0x30, 0x28, 0x4d, 0x1b, + 0x6a, 0x2a, 0xee, 0x01, 0xe9, 0xa7, 0x21, 0x96, 0x3c, 0x4f, 0x19, 0x66, 0x16, 0xde, 0x48, 0xee, + 0xa3, 0x43, 0x24, 0xb2, 0x51, 0x15, 0x86, 0x96, 0xc6, 0xc3, 0xfc, 0x95, 0x1c, 0x4c, 0x2a, 0xf6, + 0xe4, 0x73, 0xdb, 0xf1, 0x5f, 0xd4, 0x3a, 0x5e, 0xde, 0x41, 0x28, 0x2d, 0x1b, 0xaa, 0xdf, 0x7b, + 0x30, 0xd3, 0x47, 0x92, 0x36, 0xcb, 0x8d, 0x61, 0xcc, 0xf2, 0xd7, 0xfb, 0x2f, 0xcc, 0xb9, 0xb7, + 0x75, 0x7c, 0x61, 0xae, 0xde, 0xd0, 0x7f, 0x3b, 0x07, 0xa7, 0xc5, 0xaf, 0x6a, 0xcf, 0xf5, 0xa2, + 0x45, 0xbf, 0xf3, 0xc0, 0xdb, 0x7d, 0x6e, 0xc7, 0xa2, 0xaa, 0x8d, 0x45, 0x45, 0x1f, 0x0b, 0xa5, + 0x81, 0x83, 0x87, 0xc4, 0xfc, 0xbb, 0x00, 0x73, 0x83, 0x08, 0xd8, 0xb6, 0x5f, 0xd9, 0x55, 0xe1, + 0xb6, 0x3f, 0xb5, 0x63, 0xe5, 0xfb, 0xa9, 0xc4, 0x41, 0x24, 0x37, 0x84, 0x83, 0xc8, 0x0a, 0x94, 0xf1, 0x53, 0x0d, 0x1a, 0xb2, 0x4e, 0x08, 0x13, 0x57, 0xcf, 0x4b, 0x47, 0x87, 0x95, 0x8b, 0x0e, 0x2b, 0xb3, 0x43, 0x51, 0x68, 0xf7, 0x02, 0x4f, 0xe1, 0xd1, 0x47, 0x49, 0x7e, 0xd7, 0x80, 0x29, 0x04, 0x2e, 0x3d, 0xa4, 0x9d, 0x08, 0x99, 0x8d, 0x88, 0x4b, 0x9a, 0x38, 0xa2, 0xa6, 0x11, 0x05, 0x5e, 0x67, 0x17, 0x0f, 0x92, 0xc2, 0xda, 0x0e, 0xeb, 0x85, 0x3f, 0x3f, 0xac, 0xbc, 0xf3, 0x71, - 0xa2, 0x74, 0x04, 0xab, 0x90, 0x19, 0xf2, 0xbc, 0xa2, 0x14, 0x3f, 0x9b, 0xaa, 0x66, 0xaa, 0x46, - 0xe4, 0x27, 0xe1, 0xdc, 0x52, 0xc7, 0xd9, 0x69, 0xd1, 0x45, 0xbf, 0x13, 0x79, 0x9d, 0x9e, 0xdf, - 0x0b, 0x6b, 0x4e, 0x73, 0xbf, 0xd7, 0x0d, 0xc5, 0x61, 0x27, 0xb6, 0xbc, 0x19, 0x17, 0xda, 0x3b, - 0xbc, 0x54, 0x61, 0x39, 0x88, 0x01, 0xb9, 0x0b, 0x33, 0xbc, 0xa8, 0xda, 0x8b, 0xfc, 0x46, 0xd3, - 0x69, 0x79, 0x9d, 0x5d, 0x3c, 0x03, 0x2d, 0xd4, 0x2e, 0x30, 0xdb, 0xd2, 0xe9, 0x45, 0xbe, 0x1d, - 0x72, 0xb8, 0xc2, 0xaf, 0x9f, 0x88, 0x2c, 0xc3, 0xb4, 0x45, 0x1d, 0xf7, 0xbe, 0xf3, 0x78, 0xd1, - 0xe9, 0x3a, 0x4d, 0x2f, 0x3a, 0x40, 0xcb, 0x2c, 0x5f, 0xab, 0x1c, 0x1d, 0x56, 0x5e, 0x08, 0xa8, - 0xe3, 0xda, 0x6d, 0xe7, 0xb1, 0xdd, 0x14, 0x85, 0x0a, 0xb3, 0x34, 0x5d, 0xcc, 0xca, 0xeb, 0xc4, - 0xac, 0x8a, 0x69, 0x56, 0x5e, 0x67, 0x30, 0xab, 0x84, 0x4e, 0xb2, 0xda, 0x70, 0x82, 0x5d, 0x1a, - 0xf1, 0x43, 0x42, 0xb8, 0x64, 0x5c, 0x31, 0x14, 0x56, 0x11, 0x96, 0xd9, 0x78, 0x60, 0x98, 0x66, - 0xa5, 0xd0, 0xb1, 0x99, 0xb7, 0x1d, 0x78, 0x11, 0x55, 0x5b, 0x38, 0x81, 0xd5, 0xc2, 0xfe, 0xc7, - 0x63, 0xd2, 0x41, 0x4d, 0xec, 0xa3, 0x4c, 0xb8, 0x29, 0x8d, 0x2c, 0xf5, 0x71, 0xcb, 0x6e, 0x65, - 0x1f, 0x65, 0xcc, 0x4d, 0x6d, 0xe7, 0x24, 0xb6, 0x53, 0xe1, 0x36, 0xa0, 0xa1, 0x7d, 0x94, 0x64, - 0x95, 0x75, 0x5a, 0x44, 0x3b, 0x6c, 0x46, 0x8b, 0x43, 0xd2, 0x29, 0xac, 0xda, 0x2b, 0xc2, 0xa6, - 0x2e, 0x07, 0xb2, 0xd8, 0xce, 0x38, 0x32, 0x4d, 0x13, 0x93, 0x9f, 0x83, 0xe9, 0xcd, 0x90, 0xde, - 0x5e, 0x5e, 0x6f, 0x2c, 0x75, 0xdc, 0xae, 0xef, 0x75, 0xa2, 0xb9, 0x69, 0xb4, 0xb4, 0x6f, 0x9c, - 0x20, 0x74, 0xae, 0xab, 0x34, 0x18, 0x73, 0xc3, 0xc7, 0xad, 0x17, 0x52, 0xfb, 0x81, 0xd7, 0x0d, - 0x6d, 0x2a, 0xca, 0xd4, 0x71, 0x4b, 0x7d, 0xca, 0xbc, 0x0b, 0x33, 0x7d, 0x6c, 0xc8, 0x14, 0x00, - 0x03, 0xda, 0x9b, 0xab, 0x8d, 0xa5, 0x8d, 0xf2, 0x29, 0x66, 0x48, 0xe2, 0xef, 0xa5, 0xd5, 0x6a, - 0x6d, 0x65, 0xa9, 0x5e, 0x36, 0xc8, 0x0c, 0x4c, 0x22, 0xa4, 0xbe, 0xdc, 0xe0, 0xa0, 0xdc, 0x07, - 0x23, 0x85, 0xd1, 0xf2, 0x98, 0x55, 0xe6, 0x4b, 0x37, 0x62, 0x0b, 0x00, 0xf7, 0x14, 0xf3, 0x37, - 0x73, 0x70, 0x5e, 0x6e, 0x2b, 0x34, 0x7a, 0xe4, 0x07, 0xfb, 0x5e, 0x67, 0xf7, 0x39, 0xdf, 0x1d, - 0x6e, 0x6b, 0xbb, 0xc3, 0x2b, 0xa9, 0x9d, 0x3a, 0xd5, 0xca, 0x63, 0xb6, 0x88, 0x3f, 0x1b, 0x85, - 0x17, 0x8f, 0xa5, 0x22, 0x1f, 0xb2, 0xdd, 0xdc, 0xa3, 0x9d, 0x68, 0xd9, 0x6d, 0x51, 0x66, 0x4e, - 0xfa, 0xbd, 0x48, 0x1c, 0xca, 0xbf, 0x7c, 0x74, 0x58, 0x99, 0xe5, 0x01, 0x33, 0xb6, 0xe7, 0xb6, - 0xa8, 0x1d, 0xf1, 0x62, 0x6d, 0xba, 0xf5, 0x53, 0x33, 0x96, 0x71, 0xf8, 0xde, 0x72, 0x27, 0xa2, - 0xc1, 0x43, 0x87, 0xc7, 0x0d, 0x08, 0x96, 0xfb, 0x94, 0x76, 0x6d, 0x87, 0x95, 0xda, 0x9e, 0x28, - 0xd6, 0x59, 0xf6, 0x51, 0x93, 0xdb, 0x0a, 0xcb, 0x45, 0x66, 0xe4, 0xdc, 0x77, 0x1e, 0x0b, 0xcd, - 0x1d, 0xcf, 0x89, 0x15, 0x96, 0xdc, 0x57, 0xb0, 0xed, 0x3c, 0xb6, 0xfa, 0x49, 0xc8, 0xd7, 0xe1, - 0x8c, 0xd8, 0x80, 0x98, 0x30, 0x0e, 0xfc, 0x96, 0x6c, 0xf1, 0x08, 0xf2, 0x7a, 0xed, 0xe8, 0xb0, - 0x72, 0x4e, 0x6c, 0x5f, 0x76, 0x93, 0x63, 0x64, 0xb6, 0x3a, 0x9b, 0x0b, 0xd9, 0x60, 0x1b, 0x72, - 0xaa, 0x3b, 0xee, 0xd3, 0x30, 0x74, 0x76, 0xa5, 0x96, 0xcf, 0x6f, 0xc6, 0x94, 0xce, 0xb4, 0xdb, - 0xbc, 0xdc, 0x1a, 0x48, 0x49, 0xee, 0xc2, 0xd4, 0x36, 0xdd, 0x51, 0xc7, 0x67, 0x2c, 0x16, 0x55, - 0xe5, 0x47, 0x74, 0x67, 0xf0, 0xe0, 0xa4, 0xe8, 0x88, 0x07, 0x33, 0xeb, 0x81, 0xff, 0xf8, 0x80, - 0x99, 0xac, 0xb4, 0x43, 0x03, 0x74, 0x52, 0x1b, 0x47, 0x61, 0x30, 0x97, 0x68, 0xc8, 0x7a, 0x79, - 0xed, 0x33, 0x47, 0x87, 0x95, 0x17, 0xbb, 0x0c, 0x6c, 0xb7, 0x04, 0xdc, 0x4e, 0x45, 0xcf, 0xf5, - 0x73, 0x25, 0x3f, 0x0d, 0xd3, 0x96, 0xdf, 0x8b, 0xbc, 0xce, 0x6e, 0x23, 0x0a, 0x9c, 0x88, 0xee, - 0xf2, 0x0d, 0x29, 0xf1, 0x86, 0x4b, 0x95, 0xf2, 0x03, 0xf6, 0x80, 0x03, 0xed, 0x50, 0x40, 0xb5, - 0x1d, 0x41, 0x27, 0x30, 0x7f, 0x3d, 0x07, 0x73, 0x62, 0x18, 0x2c, 0xda, 0xf4, 0x03, 0xf7, 0xf9, - 0x5f, 0xf6, 0x4b, 0xda, 0xb2, 0x7f, 0x39, 0xf6, 0xb5, 0xca, 0x6a, 0xe4, 0x31, 0xab, 0xfe, 0x9f, - 0x19, 0x70, 0xf1, 0x38, 0x22, 0xd6, 0x3b, 0xb1, 0x7f, 0x62, 0xb1, 0xcf, 0x0f, 0xb1, 0x0b, 0xb3, - 0x38, 0x9e, 0x78, 0x00, 0x1e, 0xde, 0xf5, 0xc3, 0x08, 0x4f, 0x21, 0x73, 0x9a, 0x43, 0x44, 0xcd, - 0xf7, 0x5b, 0xb8, 0x5f, 0xd5, 0x5e, 0x67, 0xdb, 0xd2, 0x9f, 0x1f, 0x56, 0x80, 0x81, 0xb8, 0x47, - 0x21, 0xd3, 0x5d, 0xf8, 0x8c, 0xc1, 0xf3, 0xf5, 0xd0, 0x46, 0x2f, 0x96, 0x7d, 0x7a, 0x10, 0x5a, - 0x59, 0xac, 0xf1, 0xa4, 0xa9, 0xda, 0x8b, 0xf6, 0xd6, 0x03, 0xfa, 0x80, 0x06, 0xb4, 0xd3, 0xa4, - 0x9f, 0xb2, 0x93, 0x26, 0xbd, 0x71, 0x43, 0x99, 0x59, 0xff, 0x6f, 0x0c, 0x4e, 0x67, 0x91, 0xb1, - 0x7e, 0x51, 0x34, 0xfb, 0x74, 0xa8, 0xf5, 0xdf, 0x32, 0xa0, 0xd4, 0xa0, 0x4d, 0xbf, 0xe3, 0xde, - 0x76, 0x9a, 0x91, 0x2f, 0x5d, 0x4b, 0x6c, 0x2e, 0xd9, 0x18, 0xdc, 0x7e, 0x80, 0x05, 0xda, 0x09, - 0xc7, 0x97, 0x86, 0x53, 0xa8, 0x9b, 0x3e, 0x3a, 0xf4, 0x46, 0x6c, 0x4e, 0x26, 0x9f, 0xc0, 0xdb, - 0x19, 0xed, 0xa3, 0xa4, 0x06, 0x93, 0x8b, 0x7e, 0xa7, 0x43, 0xd9, 0x0f, 0xc5, 0x3d, 0x15, 0x1d, - 0x4a, 0x9b, 0xb2, 0x20, 0x7d, 0xd2, 0xa1, 0x93, 0x90, 0x9b, 0x90, 0xdf, 0x5c, 0xb8, 0x2d, 0xc6, - 0x40, 0x3a, 0xdd, 0x6d, 0x2e, 0xdc, 0x46, 0x9b, 0x9d, 0xe9, 0x41, 0x93, 0xbd, 0x85, 0x07, 0xea, - 0x59, 0xee, 0xe6, 0xc2, 0x6d, 0xb2, 0x06, 0x33, 0x16, 0xfd, 0x46, 0xcf, 0x0b, 0xa8, 0x58, 0x00, - 0xf7, 0x6f, 0x57, 0x71, 0x2c, 0x0a, 0x5c, 0x8e, 0x05, 0xbc, 0x50, 0xda, 0x28, 0x76, 0xfb, 0x81, - 0x1a, 0x5e, 0xd8, 0x4f, 0x4b, 0x7e, 0x01, 0xce, 0xd4, 0xbd, 0x50, 0xd4, 0x99, 0x1f, 0xa2, 0xba, - 0x78, 0x9f, 0x3a, 0x36, 0x60, 0x39, 0x7c, 0x21, 0x73, 0x39, 0x7c, 0xc6, 0x8d, 0x99, 0xd8, 0xfc, - 0x84, 0xd6, 0x4d, 0xfb, 0xf5, 0x66, 0x7f, 0x87, 0x7c, 0x04, 0x53, 0x78, 0x6a, 0x85, 0xe7, 0xca, - 0xe8, 0xea, 0x3d, 0x3e, 0xe0, 0xcb, 0x9f, 0xcb, 0xfc, 0xf2, 0x05, 0x3c, 0x04, 0xb3, 0xf1, 0x74, - 0x1a, 0xdd, 0xc2, 0x35, 0x5b, 0x47, 0xe3, 0x4c, 0x3e, 0x80, 0x69, 0xb1, 0xe9, 0xac, 0x3d, 0xd8, - 0xd8, 0xa3, 0x75, 0xe7, 0x40, 0x38, 0x53, 0xa0, 0x1e, 0x2b, 0x76, 0x2a, 0xdb, 0x7f, 0x60, 0x47, - 0x7b, 0xd4, 0x76, 0x1d, 0x4d, 0x3c, 0xa7, 0x08, 0xc9, 0xcf, 0xc2, 0xc4, 0x8a, 0x8f, 0x17, 0x68, - 0x28, 0x6a, 0x8a, 0xc8, 0xe7, 0x2b, 0x18, 0x5e, 0xcc, 0xc1, 0xa9, 0x4d, 0xe4, 0x47, 0x87, 0x95, - 0xb7, 0x9f, 0x74, 0x16, 0x2a, 0x1f, 0xb0, 0xd4, 0xaf, 0x91, 0x45, 0x28, 0x6c, 0xd3, 0x1d, 0xd6, - 0xda, 0x74, 0x68, 0x9c, 0x04, 0x73, 0x79, 0xf1, 0x48, 0xfc, 0x52, 0x6f, 0xa7, 0x24, 0x86, 0xf9, - 0x6f, 0x0d, 0x9c, 0x81, 0xe4, 0x1a, 0x3a, 0xb4, 0xc5, 0x9e, 0xf2, 0x68, 0x21, 0x3b, 0xdd, 0xae, - 0xee, 0xeb, 0xce, 0x51, 0x98, 0x39, 0x7d, 0xdb, 0x69, 0xd2, 0x48, 0x9e, 0xbb, 0x22, 0xf2, 0x03, - 0x84, 0xa8, 0xe6, 0x34, 0xc7, 0x21, 0x5f, 0x86, 0xd3, 0x75, 0xfa, 0xd0, 0x6b, 0xd2, 0x6a, 0x14, - 0xd1, 0x90, 0xb7, 0x76, 0xb1, 0xca, 0x2f, 0x28, 0x8b, 0xb5, 0x57, 0x8e, 0x0e, 0x2b, 0x97, 0x5c, - 0x2c, 0xb7, 0x9d, 0x04, 0xc1, 0x6e, 0x3a, 0x2a, 0xaf, 0x4c, 0x0e, 0xe6, 0x3f, 0xc8, 0x25, 0x3d, - 0x40, 0x5e, 0x83, 0x11, 0x6b, 0x3d, 0xae, 0x3f, 0xbf, 0x7b, 0x4c, 0x55, 0x1f, 0x11, 0xc8, 0x57, - 0xe1, 0x8c, 0xc2, 0x07, 0x27, 0x07, 0x75, 0x59, 0x85, 0x78, 0x63, 0x5e, 0xc5, 0xcb, 0x26, 0xa5, - 0x26, 0x0e, 0xc7, 0x48, 0xd5, 0x28, 0x9b, 0x07, 0x6b, 0xac, 0x52, 0x50, 0xa7, 0x1d, 0x8f, 0xf3, - 0x56, 0x1a, 0xab, 0xf2, 0x76, 0x11, 0x21, 0xdd, 0xd8, 0x2c, 0x0e, 0x64, 0x01, 0x0a, 0x75, 0x2f, - 0x64, 0x16, 0x81, 0x2b, 0x3c, 0xd3, 0x78, 0xb4, 0xa1, 0x80, 0x69, 0xd1, 0x86, 0x02, 0x66, 0xfe, - 0x95, 0xa1, 0x64, 0x75, 0x78, 0x4e, 0xf7, 0x9a, 0x5b, 0xda, 0x5e, 0x73, 0x5a, 0x90, 0xc6, 0xad, - 0x62, 0x65, 0x99, 0xfa, 0xc1, 0x34, 0x4c, 0x6a, 0x48, 0xe8, 0xe8, 0xbb, 0x19, 0xd2, 0x80, 0x9f, - 0x8a, 0x7e, 0xba, 0x1c, 0x7d, 0xe3, 0x76, 0x0d, 0xe5, 0xbf, 0xf9, 0x17, 0x06, 0x5a, 0xcb, 0x2a, - 0x05, 0xeb, 0x0d, 0x06, 0x52, 0x7b, 0xa3, 0x17, 0xd2, 0xc0, 0x42, 0x28, 0x77, 0x0b, 0x5c, 0xd1, - 0xdd, 0x02, 0x5b, 0x16, 0x83, 0x91, 0x2f, 0xc1, 0xe8, 0x26, 0xea, 0xfe, 0xba, 0x67, 0x49, 0xcc, - 0x1f, 0x0b, 0xf9, 0x7a, 0xec, 0xb1, 0x7f, 0x55, 0x71, 0x82, 0x65, 0xa4, 0x01, 0xe3, 0x8b, 0x01, - 0xc5, 0xfc, 0x0d, 0x23, 0xc3, 0x5f, 0x01, 0x36, 0x39, 0x49, 0xfa, 0x0a, 0x50, 0x70, 0x32, 0x7f, - 0x2d, 0x07, 0x24, 0x69, 0x23, 0x6d, 0x06, 0x34, 0x0a, 0x9f, 0xdb, 0x41, 0x7f, 0x5f, 0x1b, 0xf4, - 0x17, 0xfb, 0x06, 0x9d, 0x37, 0x6f, 0xa8, 0xb1, 0xff, 0x63, 0x03, 0xce, 0x66, 0x13, 0x92, 0x97, - 0x61, 0x6c, 0x6d, 0x63, 0x5d, 0x3a, 0x27, 0x89, 0xa6, 0xf8, 0x5d, 0xd4, 0x69, 0x2d, 0x51, 0x44, - 0xde, 0x80, 0xb1, 0x0f, 0xad, 0x45, 0xb6, 0xd9, 0x29, 0xf1, 0x3b, 0xdf, 0x08, 0xec, 0xa6, 0x6e, - 0x2c, 0x09, 0x24, 0x75, 0x6c, 0xf3, 0xcf, 0x6c, 0x6c, 0xbf, 0x9d, 0x83, 0xe9, 0x6a, 0xb3, 0x49, - 0xc3, 0x90, 0xa9, 0x32, 0x34, 0x8c, 0x9e, 0xdb, 0x81, 0xcd, 0x76, 0x3b, 0xd2, 0xda, 0x36, 0xd4, - 0xa8, 0xfe, 0x89, 0x01, 0x67, 0x24, 0xd5, 0x43, 0x8f, 0x3e, 0xda, 0xd8, 0x0b, 0x68, 0xb8, 0xe7, - 0xb7, 0xdc, 0x61, 0x23, 0xd1, 0x70, 0x4f, 0xf7, 0x5a, 0x11, 0x0d, 0xd4, 0x23, 0xf2, 0x07, 0x08, - 0xd1, 0xf6, 0x74, 0x84, 0x90, 0x79, 0x18, 0xaf, 0x76, 0xbb, 0x81, 0xff, 0x90, 0x2f, 0xfb, 0x49, - 0x71, 0x23, 0xca, 0x41, 0xda, 0x0d, 0x2a, 0x07, 0xb1, 0x6a, 0xd4, 0x69, 0x87, 0xfb, 0x54, 0x4f, - 0xf2, 0x6a, 0xb8, 0xb4, 0xa3, 0xea, 0x56, 0x58, 0x6e, 0x7e, 0x6b, 0x04, 0x4a, 0x6a, 0x43, 0x88, - 0x09, 0x63, 0xdc, 0x41, 0x46, 0x75, 0x54, 0x70, 0x10, 0x62, 0x89, 0x92, 0xc4, 0xef, 0x28, 0x77, - 0xa2, 0xdf, 0xd1, 0x36, 0x4c, 0xae, 0x07, 0x7e, 0xd7, 0x0f, 0xa9, 0xcb, 0x53, 0xf0, 0x70, 0xa9, - 0x35, 0x1b, 0x3b, 0xe3, 0xf2, 0x3e, 0xc7, 0x73, 0x40, 0x54, 0xe4, 0xbb, 0x02, 0xdb, 0x4e, 0x27, - 0xe8, 0xd1, 0xf9, 0xf0, 0x2b, 0x06, 0x27, 0x14, 0x01, 0x0b, 0xf1, 0x15, 0x03, 0x83, 0xe8, 0x57, - 0x0c, 0x0c, 0xa2, 0x2e, 0x8b, 0xd1, 0x67, 0xb5, 0x2c, 0xc8, 0xaf, 0x19, 0x30, 0x51, 0xed, 0x74, - 0x84, 0xdf, 0x91, 0x8c, 0x63, 0x3f, 0x93, 0x5c, 0x33, 0x70, 0xc7, 0x54, 0x7e, 0xcb, 0xf0, 0x35, - 0x71, 0xcb, 0xf0, 0xf6, 0xc7, 0xba, 0x65, 0xd8, 0x08, 0x1c, 0x2f, 0x0a, 0xf1, 0x3a, 0x39, 0xf9, - 0xa0, 0xea, 0x7c, 0xac, 0xd4, 0x83, 0xbc, 0x0d, 0xe5, 0x78, 0x3e, 0x2e, 0x77, 0x5c, 0xfa, 0x98, - 0x72, 0x37, 0xad, 0x49, 0x1e, 0xe3, 0xa8, 0x5d, 0x9f, 0xa4, 0x11, 0xcd, 0x6f, 0x1b, 0x70, 0x56, - 0x9d, 0x10, 0x8d, 0xde, 0x4e, 0xdb, 0x43, 0xc3, 0x85, 0x5c, 0x87, 0xa2, 0x18, 0xaf, 0x58, 0xed, - 0xeb, 0xcf, 0xdb, 0x94, 0xa0, 0x90, 0x25, 0x36, 0x44, 0x8c, 0x87, 0xb0, 0xf2, 0x67, 0x53, 0xcb, - 0x8d, 0x15, 0xd5, 0xe6, 0x44, 0x67, 0x97, 0x03, 0xfc, 0xad, 0x8f, 0x1d, 0x83, 0x98, 0xef, 0xc1, - 0x8c, 0x5e, 0xcb, 0x06, 0xc5, 0x20, 0x38, 0xd9, 0x34, 0x23, 0xbb, 0x69, 0xb2, 0xdc, 0xdc, 0x06, - 0xd2, 0x47, 0x1f, 0xe2, 0x55, 0x19, 0x8d, 0xe4, 0x55, 0xae, 0x3c, 0xa8, 0xea, 0x43, 0x8c, 0x33, - 0x98, 0x4d, 0xa8, 0xdd, 0x8d, 0xa4, 0xe6, 0x5f, 0x17, 0x61, 0x36, 0x43, 0x74, 0x9c, 0xb0, 0xb5, - 0x57, 0xf4, 0xc5, 0x53, 0x8c, 0x7d, 0x12, 0xe4, 0x92, 0x79, 0x4f, 0x66, 0xab, 0x3a, 0x66, 0xa9, - 0x1c, 0x97, 0xc2, 0xea, 0x93, 0xd8, 0xde, 0x55, 0xb7, 0xa1, 0xd1, 0x67, 0xe6, 0x36, 0x54, 0x83, - 0x49, 0xd1, 0x2a, 0xb1, 0x94, 0xc7, 0x12, 0x83, 0x3e, 0xe0, 0x05, 0x76, 0xdf, 0x92, 0xd6, 0x49, - 0x38, 0x8f, 0xd0, 0x6f, 0x3d, 0xa4, 0x82, 0xc7, 0xb8, 0xca, 0x03, 0x0b, 0x32, 0x79, 0x28, 0x24, - 0xe4, 0x0f, 0x0c, 0x20, 0x02, 0xa2, 0xae, 0xe7, 0xc2, 0x71, 0xeb, 0xd9, 0x7d, 0x36, 0xeb, 0xf9, - 0x45, 0x59, 0xc7, 0xec, 0x75, 0x9d, 0x51, 0x2d, 0xf2, 0x4f, 0x0c, 0x98, 0xe1, 0xbe, 0x2b, 0x6a, - 0x65, 0x8b, 0xc7, 0x55, 0xb6, 0xf9, 0x6c, 0x2a, 0x7b, 0x31, 0xc4, 0xcf, 0x0e, 0xa8, 0x6b, 0x7f, - 0xa5, 0xc8, 0x4f, 0x02, 0xc4, 0x2b, 0x4a, 0xfa, 0x48, 0x5e, 0xcc, 0x90, 0x02, 0x31, 0x52, 0x12, - 0xe6, 0x19, 0xc5, 0x74, 0xaa, 0x57, 0x51, 0xc2, 0x8d, 0xfc, 0x02, 0x9c, 0x66, 0xeb, 0x25, 0x86, - 0x08, 0x4f, 0xbb, 0xb9, 0x09, 0xfc, 0xca, 0xe7, 0x07, 0x6f, 0xed, 0xd7, 0xb3, 0xc8, 0x78, 0xa4, - 0x4a, 0x92, 0x48, 0x20, 0x6a, 0xab, 0x06, 0x62, 0x16, 0x05, 0xba, 0xd4, 0x62, 0xed, 0xc3, 0xb9, - 0x12, 0x7e, 0x33, 0x53, 0xbe, 0x9d, 0x97, 0x6b, 0x81, 0xcb, 0xb7, 0x50, 0x0f, 0x35, 0x41, 0x10, - 0xf9, 0x10, 0x48, 0xa3, 0xb7, 0xbb, 0x4b, 0xc3, 0x88, 0xba, 0x1c, 0x46, 0x83, 0x70, 0x6e, 0x12, - 0xe5, 0x03, 0x1e, 0x30, 0x85, 0xb2, 0xd4, 0x0e, 0x64, 0xb1, 0x3a, 0x49, 0xfa, 0x89, 0x2f, 0xec, - 0xc0, 0xf9, 0x81, 0xcd, 0xcc, 0x08, 0x23, 0x99, 0xd7, 0xc3, 0x48, 0xce, 0x0f, 0x12, 0x87, 0xa1, - 0x1a, 0x4a, 0xf2, 0xdb, 0x46, 0x4a, 0xfe, 0x09, 0x65, 0x85, 0x27, 0xf6, 0x1b, 0xb4, 0x41, 0xe4, - 0x30, 0xc5, 0x00, 0x97, 0x90, 0xb9, 0x44, 0x49, 0x62, 0x12, 0x52, 0x95, 0xb0, 0x28, 0x2b, 0x9f, - 0x52, 0x14, 0x9a, 0xff, 0xd2, 0x00, 0xc2, 0x6b, 0xb8, 0xe8, 0x74, 0x9d, 0x1d, 0xaf, 0xe5, 0x45, - 0x1e, 0x0d, 0xc9, 0x3d, 0x28, 0x0b, 0x16, 0xcc, 0x6c, 0x57, 0x3d, 0xc4, 0xc4, 0x15, 0x72, 0x5c, - 0x66, 0xa7, 0xd5, 0x9a, 0x3e, 0xc2, 0x01, 0x83, 0x97, 0x7b, 0x8a, 0xc1, 0x33, 0x7f, 0x68, 0xc0, - 0xf9, 0xfe, 0x6a, 0x8b, 0x2f, 0xc7, 0x9d, 0x67, 0x9c, 0xd0, 0x79, 0x59, 0xad, 0xcc, 0xe1, 0x41, - 0xc6, 0x33, 0x6b, 0x65, 0x3e, 0x39, 0x03, 0x7d, 0xf2, 0x56, 0xfe, 0x72, 0x0e, 0x4a, 0xeb, 0xad, - 0xde, 0xae, 0xd7, 0xa9, 0x3b, 0x91, 0xf3, 0xdc, 0x9a, 0x14, 0x6f, 0x69, 0x26, 0x45, 0xec, 0x13, - 0x16, 0x37, 0x6c, 0xb8, 0x24, 0x6b, 0x06, 0x4c, 0x27, 0x24, 0x7c, 0x95, 0xde, 0x85, 0x11, 0xf6, - 0x43, 0x68, 0x28, 0x97, 0xfa, 0x18, 0x23, 0xd6, 0xf5, 0xf8, 0x3f, 0xa1, 0xe4, 0xeb, 0xa9, 0xed, - 0x90, 0xc3, 0x85, 0x2f, 0xf0, 0xcc, 0x54, 0x4f, 0x9e, 0x45, 0xf3, 0x5f, 0x18, 0x50, 0x4e, 0xb7, - 0x84, 0xdc, 0x83, 0x71, 0xc6, 0xc9, 0x8b, 0xb3, 0x5c, 0xbd, 0x32, 0xa0, 0xcd, 0xd7, 0x05, 0x1a, - 0xaf, 0x1e, 0x76, 0x3e, 0xe5, 0x10, 0x4b, 0x72, 0xb8, 0x60, 0x41, 0x49, 0xc5, 0xca, 0xa8, 0xdd, - 0xeb, 0xba, 0x68, 0x3a, 0x9b, 0xdd, 0x0f, 0x6a, 0xad, 0x7f, 0x4b, 0xab, 0xb5, 0x10, 0x4a, 0xc3, - 0xa6, 0x2b, 0xc4, 0xa0, 0x38, 0x9e, 0x0b, 0x45, 0x9d, 0x67, 0x32, 0x6d, 0x8a, 0x1e, 0x14, 0xc7, - 0x61, 0xcc, 0x16, 0xe1, 0xdf, 0x13, 0xf3, 0x0c, 0x6d, 0x91, 0x2e, 0x42, 0x54, 0x7d, 0x96, 0xe3, - 0x98, 0x7f, 0x3f, 0x0f, 0x67, 0x93, 0xea, 0xf1, 0xe4, 0x8d, 0xeb, 0x4e, 0xe0, 0xb4, 0xc3, 0x13, - 0x56, 0xc0, 0x95, 0xbe, 0xaa, 0x61, 0xd0, 0xb7, 0xac, 0x9a, 0x52, 0x21, 0x33, 0x55, 0x21, 0x34, - 0xe2, 0x78, 0x85, 0x64, 0x35, 0xc8, 0x3d, 0xc8, 0x37, 0x68, 0x24, 0x42, 0x43, 0x2f, 0xf7, 0xf5, - 0xaa, 0x5a, 0xaf, 0xeb, 0x0d, 0x1a, 0xf1, 0x41, 0xe4, 0xde, 0xf5, 0x54, 0xf3, 0x76, 0x67, 0xea, - 0xf8, 0x36, 0x8c, 0x2d, 0x3d, 0xee, 0xd2, 0x66, 0x24, 0x22, 0x42, 0xaf, 0x1e, 0xcf, 0x8f, 0xe3, - 0x2a, 0x71, 0xa7, 0x14, 0x01, 0x6a, 0x67, 0x71, 0x94, 0x0b, 0xb7, 0xa0, 0x20, 0x3f, 0xfe, 0x44, - 0xf1, 0x93, 0x6f, 0xc1, 0x84, 0xf2, 0x91, 0x27, 0x9a, 0xf4, 0x7f, 0x6d, 0xc0, 0x18, 0x13, 0x7a, - 0x5b, 0x6f, 0x3e, 0xa7, 0x12, 0xe9, 0xa6, 0x26, 0x91, 0x66, 0x94, 0x40, 0x1f, 0x5c, 0x97, 0x6f, - 0x9e, 0x20, 0x8b, 0x0e, 0x0d, 0x80, 0x04, 0x99, 0xdc, 0x81, 0x71, 0x91, 0xd1, 0x45, 0x64, 0x46, - 0x55, 0x23, 0x87, 0x64, 0xe6, 0xa8, 0x58, 0xcb, 0xf1, 0xbb, 0x69, 0xb5, 0x50, 0x52, 0x93, 0x7a, - 0xe2, 0x5d, 0xad, 0x86, 0xaa, 0x32, 0x36, 0x8b, 0x7e, 0x87, 0x47, 0x92, 0x28, 0x39, 0xa8, 0x06, - 0xb8, 0x59, 0x57, 0xc5, 0xc1, 0x46, 0xfe, 0x38, 0x26, 0x67, 0x05, 0x93, 0xec, 0x33, 0x8f, 0xff, - 0x3d, 0xc1, 0x63, 0x33, 0x64, 0xc5, 0xde, 0x85, 0xd2, 0x6d, 0x3f, 0x78, 0xe4, 0x04, 0x6e, 0x75, - 0x97, 0x0a, 0xbf, 0xf8, 0x02, 0x3a, 0xb7, 0x4f, 0x3e, 0xe0, 0x70, 0xdb, 0x61, 0x05, 0x3f, 0x3a, - 0xac, 0x8c, 0xd4, 0x7c, 0xbf, 0x65, 0x69, 0xe8, 0x64, 0x0d, 0x26, 0xef, 0x3b, 0x8f, 0xc5, 0xed, - 0xde, 0xc6, 0xc6, 0x8a, 0xf0, 0x4a, 0xb9, 0x7a, 0x74, 0x58, 0x39, 0xdf, 0x76, 0x1e, 0xc7, 0xb7, - 0x82, 0x83, 0x1d, 0xc0, 0x75, 0x7a, 0xe2, 0xc1, 0xd4, 0xba, 0x1f, 0x44, 0xe2, 0x23, 0x4c, 0xa7, - 0xcd, 0x0f, 0xb8, 0x9c, 0x9b, 0xcf, 0xbc, 0x9c, 0x3b, 0xcf, 0x14, 0x79, 0xfb, 0x41, 0x4c, 0xae, - 0x05, 0x14, 0x6a, 0x8c, 0xc9, 0xbb, 0x30, 0xb3, 0x48, 0x83, 0xc8, 0x7b, 0xe0, 0x35, 0x9d, 0x88, - 0xde, 0xf6, 0x83, 0xb6, 0x13, 0x89, 0x03, 0x15, 0x34, 0xa8, 0x9b, 0x94, 0x73, 0x6a, 0x3b, 0x91, - 0xd5, 0x8f, 0x49, 0xbe, 0x9a, 0xe5, 0xe7, 0x33, 0x8a, 0xcd, 0x7f, 0x83, 0x29, 0x05, 0x19, 0x7e, - 0x3e, 0x03, 0xba, 0x20, 0xc3, 0xe3, 0x67, 0xf7, 0xb8, 0x4b, 0xd2, 0x42, 0xed, 0x86, 0xb8, 0xb0, - 0x3d, 0xf9, 0x12, 0x34, 0x1e, 0xb7, 0x01, 0x97, 0xa1, 0x0b, 0x90, 0xaf, 0xad, 0xdf, 0xc6, 0x23, - 0x12, 0x71, 0x29, 0x49, 0x3b, 0x7b, 0x4e, 0xa7, 0x89, 0xba, 0x8c, 0xf0, 0x74, 0x50, 0x05, 0x5e, - 0x6d, 0xfd, 0x36, 0x71, 0x60, 0x76, 0x9d, 0x06, 0x6d, 0x2f, 0xfa, 0xf2, 0x8d, 0x1b, 0xca, 0x40, - 0x15, 0xb0, 0x6a, 0xf3, 0xa2, 0x6a, 0x95, 0x2e, 0xa2, 0xd8, 0x8f, 0x6f, 0xdc, 0xc8, 0x1c, 0x8e, - 0xb8, 0x62, 0x59, 0xbc, 0xc8, 0x12, 0x4c, 0xdd, 0x77, 0x1e, 0x8b, 0xeb, 0xeb, 0xd8, 0xc6, 0xcb, - 0x63, 0x3c, 0x00, 0x4e, 0xac, 0x66, 0x52, 0xa4, 0x0e, 0xb1, 0x4e, 0x44, 0xde, 0x81, 0x89, 0x64, - 0x7a, 0x85, 0x78, 0x71, 0x99, 0xe7, 0x8e, 0xa0, 0xca, 0xe4, 0xd4, 0xce, 0x92, 0x14, 0x74, 0xb2, - 0x19, 0x9b, 0xe8, 0x5c, 0x21, 0x45, 0xf7, 0xc8, 0x62, 0x6d, 0x5e, 0x35, 0xd1, 0x1d, 0x2c, 0xd1, - 0x9a, 0x35, 0x1d, 0xab, 0xe8, 0xdc, 0xaf, 0xc6, 0xd2, 0xb9, 0x28, 0x96, 0xff, 0x7a, 0xe0, 0xb7, - 0xbb, 0x11, 0xfa, 0x49, 0xa6, 0x2c, 0xff, 0x2e, 0x96, 0x64, 0x58, 0xfe, 0x9c, 0x24, 0xfb, 0x56, - 0x7e, 0xf2, 0x29, 0x6e, 0xe5, 0x29, 0x8c, 0xac, 0xf8, 0xcd, 0x7d, 0x74, 0x8c, 0x2c, 0xd6, 0x3e, - 0x64, 0xf2, 0xa3, 0xe5, 0x37, 0xf7, 0x9f, 0xdd, 0x6d, 0x32, 0xb2, 0x27, 0xab, 0xac, 0xed, 0x6c, - 0x5a, 0x89, 0x4f, 0xa3, 0xe3, 0x64, 0x72, 0xd3, 0xa6, 0x95, 0x71, 0x45, 0x85, 0xcf, 0x42, 0xd9, - 0x10, 0x4b, 0x27, 0x27, 0x14, 0xca, 0x75, 0x1a, 0xee, 0x47, 0x7e, 0x77, 0xb1, 0xe5, 0x75, 0x77, - 0x7c, 0x27, 0x70, 0xe7, 0xca, 0x03, 0x04, 0xc6, 0x6b, 0x99, 0x02, 0x63, 0xc6, 0xe5, 0xf4, 0x76, - 0x53, 0x32, 0xb0, 0xfa, 0x58, 0x92, 0xaf, 0xc2, 0x14, 0x5b, 0x2d, 0x4b, 0x8f, 0x23, 0xda, 0xe1, - 0x53, 0x69, 0x06, 0xb7, 0xfa, 0xd3, 0x4a, 0x68, 0x65, 0x5c, 0xc8, 0x27, 0x29, 0x4a, 0x0f, 0x1a, - 0x13, 0xa8, 0x93, 0x54, 0x67, 0x45, 0x5c, 0x98, 0xbb, 0xef, 0x3c, 0x56, 0x12, 0x11, 0x29, 0xb3, - 0x9e, 0xe0, 0x8c, 0xbd, 0x72, 0x74, 0x58, 0x79, 0x85, 0xcd, 0xd8, 0xfd, 0x18, 0x69, 0xc0, 0x02, - 0x18, 0xc8, 0xc9, 0xfc, 0xc9, 0x54, 0xcf, 0x93, 0x65, 0x18, 0x17, 0xed, 0x14, 0x7b, 0x5b, 0x7f, - 0x8f, 0xbd, 0x98, 0xd9, 0x63, 0xe3, 0xa2, 0xc7, 0x2c, 0x49, 0x6f, 0xfe, 0x3b, 0x03, 0x26, 0xb5, - 0x46, 0x91, 0x5b, 0x8a, 0x4b, 0x4d, 0xe2, 0x0a, 0xa7, 0xe1, 0x64, 0xbe, 0x6b, 0x70, 0x4b, 0xf8, - 0x51, 0xe5, 0x06, 0xd3, 0x65, 0x66, 0x7a, 0x93, 0x09, 0x15, 0xf2, 0xc7, 0x27, 0x54, 0x18, 0x19, - 0x90, 0x50, 0xe1, 0x3b, 0x93, 0x30, 0xa5, 0x6f, 0xa3, 0x4c, 0xaf, 0x5d, 0xf1, 0x77, 0xbd, 0x8e, - 0xb4, 0x8e, 0x79, 0x8a, 0x10, 0x84, 0x68, 0x8f, 0x04, 0x20, 0x84, 0xbc, 0x0a, 0x10, 0x5f, 0x00, - 0x4b, 0x03, 0x58, 0x3c, 0x69, 0xa0, 0x14, 0x90, 0x9f, 0x02, 0x58, 0xf5, 0x5d, 0x1a, 0x67, 0x99, - 0x39, 0xe6, 0xd8, 0xea, 0x35, 0x71, 0x6c, 0x25, 0x9e, 0x21, 0x38, 0x3a, 0xac, 0x9c, 0xe9, 0xf8, - 0x2e, 0xed, 0x4f, 0x2f, 0xa3, 0x70, 0x24, 0x5f, 0x84, 0x51, 0xab, 0xd7, 0xa2, 0x32, 0xe9, 0xc9, - 0x84, 0x5c, 0x56, 0xbd, 0x96, 0x92, 0x9e, 0x34, 0xe8, 0xa5, 0x6f, 0x2b, 0x18, 0x80, 0xbc, 0x0f, - 0xc0, 0x66, 0xce, 0x9d, 0xc0, 0xef, 0x75, 0x65, 0x54, 0x35, 0x1a, 0xcb, 0xca, 0xa4, 0xdb, 0xc5, - 0x42, 0xf5, 0xe3, 0x09, 0x09, 0x59, 0x83, 0x71, 0x21, 0xa4, 0xc4, 0x6d, 0xc0, 0x4b, 0x59, 0xe7, - 0x50, 0x8a, 0xa6, 0x22, 0xb2, 0x90, 0x20, 0x58, 0x3f, 0x1a, 0xe2, 0xc6, 0xfe, 0x3b, 0x50, 0x64, - 0xec, 0x99, 0x41, 0x1f, 0x8a, 0x1d, 0x0a, 0x7d, 0x1a, 0x95, 0x0a, 0x31, 0xe3, 0x5f, 0xcb, 0x85, - 0x16, 0x13, 0x90, 0xaf, 0x62, 0xde, 0x20, 0xd1, 0xd5, 0xc7, 0x1e, 0x67, 0x5e, 0xee, 0xeb, 0xea, - 0xd3, 0x4e, 0xb7, 0x9b, 0x91, 0x68, 0x2d, 0xe6, 0x47, 0x76, 0xe3, 0xf8, 0xa5, 0x38, 0x47, 0xf5, - 0x31, 0x1f, 0xb8, 0xd6, 0xf7, 0x81, 0x39, 0x19, 0x92, 0xd3, 0x9f, 0x2d, 0x48, 0xe3, 0x4b, 0xba, - 0x50, 0x4e, 0xd6, 0xb3, 0xf8, 0x16, 0x1c, 0xf7, 0xad, 0x37, 0xfa, 0xbe, 0xa5, 0x0e, 0x60, 0xdf, - 0xe7, 0xfa, 0xb8, 0x13, 0x37, 0xc9, 0x27, 0x2c, 0xbe, 0x37, 0x71, 0xdc, 0xf7, 0x5e, 0xed, 0xfb, - 0xde, 0xac, 0xbb, 0xd3, 0xff, 0x9d, 0x14, 0x4f, 0xf2, 0x0e, 0x4c, 0x4a, 0x08, 0xae, 0x0f, 0x3c, - 0x46, 0x14, 0x56, 0x84, 0xbb, 0x83, 0x8e, 0x6c, 0x7a, 0xaa, 0x1c, 0x15, 0x59, 0xa5, 0xe6, 0xb3, - 0x63, 0x52, 0xa3, 0x4e, 0xcf, 0x0a, 0x1d, 0x99, 0x7c, 0x05, 0x26, 0x96, 0xdb, 0xac, 0x21, 0x7e, - 0xc7, 0x89, 0x28, 0x6e, 0x79, 0xc9, 0xd1, 0xac, 0x52, 0xa2, 0x4c, 0x55, 0x9e, 0x93, 0x33, 0x29, - 0x52, 0x55, 0x06, 0x85, 0x82, 0x75, 0x1e, 0x3f, 0xe4, 0x11, 0x73, 0x38, 0x14, 0x1b, 0xdc, 0x8b, - 0x19, 0xc7, 0xa3, 0x0a, 0x7b, 0xdc, 0x31, 0xf8, 0xd9, 0x91, 0x2d, 0x16, 0x84, 0xd6, 0x79, 0x3a, - 0x4f, 0xf2, 0x2e, 0x4c, 0x88, 0x68, 0xd1, 0xaa, 0xb5, 0x1a, 0xce, 0x95, 0xb1, 0xf1, 0x98, 0xe7, - 0x4e, 0x06, 0x96, 0xda, 0x4e, 0x90, 0xba, 0x23, 0x4b, 0xf0, 0xc9, 0x97, 0xe1, 0xf4, 0xb6, 0xd7, - 0x71, 0xfd, 0x47, 0xa1, 0x10, 0xe0, 0x42, 0xd0, 0xcd, 0x24, 0x7e, 0x43, 0x8f, 0x78, 0xb9, 0x2d, - 0x37, 0xc7, 0x3e, 0xc1, 0x97, 0xc9, 0x81, 0xfc, 0x7c, 0x1f, 0x67, 0x3e, 0x83, 0xc8, 0x71, 0x33, - 0x68, 0xa1, 0x6f, 0x06, 0xf5, 0x7f, 0x3e, 0x3d, 0x9d, 0x32, 0x3f, 0x43, 0x7c, 0x20, 0xba, 0x66, - 0xf3, 0x81, 0xef, 0x75, 0xe6, 0x66, 0xb5, 0x17, 0x60, 0x62, 0x37, 0x5e, 0xc4, 0x5b, 0xf7, 0x5b, - 0x5e, 0xf3, 0xa0, 0x66, 0x1e, 0x1d, 0x56, 0x5e, 0x4a, 0xeb, 0x4c, 0x1f, 0xf9, 0xda, 0x11, 0x46, - 0x06, 0x6b, 0xf2, 0x15, 0x28, 0xb1, 0xbf, 0xb1, 0x82, 0x79, 0x5a, 0xbb, 0x50, 0x53, 0x30, 0xc5, - 0x77, 0x70, 0x8c, 0x30, 0x9c, 0x35, 0x43, 0xf7, 0xd4, 0x58, 0x99, 0x3f, 0x34, 0xe0, 0x74, 0x56, - 0x5d, 0x4f, 0xc8, 0x1d, 0x64, 0xa6, 0xae, 0xd6, 0xf1, 0xf4, 0x83, 0x5f, 0xad, 0xc7, 0x17, 0xea, - 0x15, 0x18, 0x65, 0x16, 0xb9, 0x74, 0x14, 0xc3, 0xed, 0x90, 0x59, 0xed, 0xa1, 0xc5, 0xe1, 0x0c, - 0x01, 0x1d, 0xfc, 0x71, 0xbf, 0x1c, 0xe5, 0x08, 0x18, 0x05, 0x60, 0x71, 0x38, 0x43, 0x60, 0xdb, - 0xae, 0xdc, 0x26, 0x10, 0x81, 0xed, 0xc6, 0xa1, 0xc5, 0xe1, 0xe4, 0x32, 0x8c, 0xaf, 0x75, 0x56, - 0xa8, 0xf3, 0x90, 0x8a, 0x7b, 0x2d, 0x3c, 0xad, 0xf1, 0x3b, 0x76, 0x8b, 0xc1, 0x2c, 0x59, 0x68, - 0x7e, 0xcf, 0x80, 0x99, 0xbe, 0x6e, 0x3a, 0x39, 0x3d, 0xd2, 0xf1, 0x97, 0x88, 0xc3, 0xb4, 0x8f, - 0x57, 0x7f, 0x24, 0xbb, 0xfa, 0xe6, 0x5f, 0xe6, 0xe1, 0xdc, 0x80, 0x5d, 0x2b, 0x71, 0x00, 0x30, - 0x4e, 0x74, 0x00, 0xf8, 0x1a, 0xdb, 0x25, 0x1c, 0xaf, 0x1d, 0x6e, 0xf8, 0x49, 0x8d, 0x93, 0xbb, - 0x12, 0x2c, 0x93, 0xf9, 0x47, 0x64, 0xae, 0x8c, 0xf3, 0x4d, 0xa4, 0xb0, 0x23, 0xbf, 0xef, 0x64, - 0x5a, 0x67, 0xd6, 0x77, 0x05, 0x9f, 0xff, 0x31, 0xb9, 0x82, 0xd7, 0x2f, 0xbe, 0x46, 0x9e, 0xe9, - 0xc5, 0x57, 0xf6, 0x51, 0xfc, 0xe8, 0xd3, 0x5c, 0x38, 0xfc, 0x87, 0xd4, 0xa5, 0xff, 0x8f, 0xe3, - 0x50, 0x5f, 0x85, 0xd1, 0xed, 0x3d, 0x1a, 0x48, 0xfd, 0x16, 0x2b, 0xf2, 0x88, 0x01, 0xd4, 0x8a, - 0x20, 0x86, 0xf9, 0xb3, 0x50, 0x52, 0x3f, 0x86, 0x6b, 0x99, 0xfd, 0x16, 0x8b, 0x89, 0xaf, 0x65, - 0x06, 0xb0, 0x38, 0xfc, 0xc4, 0x6c, 0x63, 0x49, 0x2f, 0xe4, 0x4f, 0xea, 0x05, 0xf3, 0xdf, 0x1b, - 0x30, 0x82, 0xc9, 0x16, 0xde, 0x84, 0xa2, 0x3c, 0x90, 0x55, 0x13, 0x10, 0xcc, 0xca, 0xf3, 0xda, - 0x50, 0xf7, 0x9a, 0x10, 0x40, 0xf6, 0xa9, 0x2d, 0x1a, 0xec, 0x68, 0xce, 0x35, 0x0f, 0x19, 0x40, - 0xfd, 0x14, 0x62, 0x3c, 0x41, 0x97, 0xa0, 0x03, 0x91, 0xb0, 0xa7, 0xf8, 0x82, 0xe7, 0x0e, 0x44, - 0x7d, 0xc6, 0x93, 0xc4, 0x32, 0x7f, 0xc3, 0x80, 0x33, 0x99, 0x7a, 0x00, 0xfb, 0x2a, 0x57, 0x38, - 0x94, 0x19, 0x91, 0xd6, 0x36, 0x38, 0xc6, 0x93, 0x38, 0x0a, 0x3d, 0xc1, 0xf0, 0x7e, 0x06, 0x8a, - 0xb1, 0x7d, 0x46, 0x4e, 0xcb, 0xa1, 0xc3, 0x53, 0x3b, 0x69, 0xcc, 0xfc, 0xb5, 0x01, 0x63, 0xac, - 0x0a, 0xcf, 0x6d, 0xc4, 0x47, 0xf6, 0x19, 0x2e, 0x6b, 0xd2, 0x50, 0x71, 0x1e, 0xbf, 0x3b, 0x06, - 0x90, 0x20, 0x93, 0x1d, 0x98, 0x5a, 0x5b, 0xae, 0x2f, 0x2e, 0xbb, 0xb4, 0x13, 0xe1, 0x5d, 0x62, - 0x2a, 0x83, 0x01, 0x33, 0x2c, 0x83, 0x8e, 0xd3, 0x12, 0x08, 0x07, 0xc9, 0xf2, 0xf4, 0x3d, 0xb7, - 0x69, 0x7b, 0x31, 0x9d, 0xaa, 0x90, 0xe9, 0x1c, 0xd9, 0x37, 0x1a, 0xd5, 0xfb, 0x2b, 0xca, 0x37, - 0x72, 0x43, 0x7e, 0x23, 0x74, 0xda, 0xad, 0x01, 0xdf, 0xd0, 0x39, 0x92, 0x3d, 0x28, 0xdf, 0x41, - 0xd9, 0xad, 0x7c, 0x25, 0x7f, 0xfc, 0x57, 0x5e, 0x16, 0x5f, 0x79, 0x81, 0x0b, 0xfd, 0xec, 0xef, - 0xf4, 0x71, 0x4d, 0x66, 0xee, 0xc8, 0x89, 0x33, 0xf7, 0xef, 0x18, 0x30, 0xc6, 0x37, 0x87, 0xf8, - 0x11, 0x94, 0xcc, 0xed, 0x67, 0xfb, 0xd9, 0x6c, 0x3f, 0xe5, 0x08, 0xff, 0x53, 0x0d, 0x70, 0x5e, - 0x46, 0xea, 0xa9, 0x17, 0x55, 0xe4, 0x41, 0x3d, 0x2a, 0xa6, 0xbc, 0x24, 0x71, 0xb7, 0xe2, 0x8f, - 0xa9, 0xa8, 0x5c, 0x38, 0x86, 0xfa, 0xbe, 0xe3, 0xf8, 0x53, 0xbe, 0xef, 0xb8, 0x02, 0x45, 0xe1, - 0x3f, 0x54, 0x3b, 0x10, 0xe6, 0xa7, 0x3c, 0x60, 0x89, 0xe1, 0x4a, 0x62, 0x6f, 0x0e, 0xb2, 0x77, - 0xb4, 0xb4, 0x7c, 0x31, 0x22, 0x59, 0x83, 0x62, 0x12, 0xae, 0x52, 0xd4, 0x6e, 0x5b, 0x63, 0xb8, - 0x70, 0xb0, 0xe5, 0x11, 0x91, 0x99, 0xd1, 0x29, 0x09, 0x0f, 0xf3, 0x5b, 0x06, 0x94, 0xd3, 0xf3, - 0x85, 0xbc, 0x03, 0x13, 0x71, 0xc4, 0x50, 0xec, 0xc5, 0x80, 0xc7, 0xa5, 0x49, 0x88, 0x91, 0xe6, - 0xcf, 0xa0, 0xa2, 0x93, 0x05, 0x28, 0xb0, 0x65, 0xa7, 0xe4, 0x65, 0x46, 0x79, 0xd2, 0x13, 0x30, - 0xf5, 0xf6, 0x50, 0xe2, 0x29, 0xab, 0xf6, 0x3f, 0xe5, 0x61, 0x42, 0x19, 0x2c, 0x72, 0x15, 0x0a, - 0xcb, 0xe1, 0x8a, 0xdf, 0xdc, 0xa7, 0xae, 0xb8, 0x94, 0xc0, 0xe7, 0x3b, 0xbd, 0xd0, 0x6e, 0x21, - 0xd0, 0x8a, 0x8b, 0x49, 0x0d, 0x26, 0xf9, 0x7f, 0x32, 0x32, 0x34, 0x97, 0x1c, 0xa8, 0x72, 0x64, - 0x19, 0x13, 0xaa, 0xee, 0xb0, 0x1a, 0x09, 0xf9, 0x3a, 0x00, 0x07, 0xb0, 0xf1, 0x1d, 0xc2, 0x7d, - 0x58, 0x2e, 0xe0, 0x33, 0xe2, 0x03, 0x91, 0xa7, 0xb6, 0x10, 0xa7, 0x82, 0xc2, 0x10, 0x9f, 0x0e, - 0xf4, 0x9b, 0xfb, 0xc3, 0x3f, 0x1e, 0x9a, 0x3c, 0x1d, 0xe8, 0x37, 0xf7, 0xed, 0x6c, 0x5f, 0x32, - 0x95, 0x25, 0xf9, 0xb6, 0x01, 0x17, 0x2c, 0xda, 0xf4, 0x1f, 0xd2, 0xe0, 0xa0, 0x1a, 0x21, 0x96, - 0xfa, 0xc5, 0x93, 0x1d, 0xd7, 0x6e, 0x8a, 0x2f, 0xbe, 0x16, 0x08, 0x2e, 0x18, 0x22, 0xd3, 0xee, - 0x46, 0xf6, 0x31, 0x55, 0x38, 0xe6, 0x93, 0xe6, 0x9f, 0x19, 0xca, 0x12, 0x20, 0xab, 0x50, 0x8c, - 0x27, 0x8b, 0x38, 0x70, 0x8c, 0x95, 0x23, 0x09, 0xb7, 0xe8, 0x83, 0xda, 0x0b, 0xe2, 0xfe, 0x60, - 0x36, 0x9e, 0x72, 0xda, 0x8a, 0x90, 0x40, 0xf2, 0x25, 0x18, 0xc1, 0xa1, 0x3a, 0x39, 0x91, 0x97, - 0xdc, 0x6a, 0x46, 0xd8, 0x18, 0x61, 0xad, 0x91, 0x92, 0x7c, 0x4e, 0xf8, 0x92, 0xe4, 0xb5, 0x14, - 0xb9, 0x0c, 0xc4, 0xea, 0x11, 0xef, 0x31, 0x89, 0xfb, 0xa2, 0x32, 0x5b, 0xff, 0xa6, 0x01, 0xb3, - 0x9b, 0x0b, 0xb7, 0x2d, 0xba, 0xeb, 0x61, 0x64, 0xad, 0xe7, 0xe3, 0x45, 0x2f, 0x39, 0x0f, 0x79, - 0xcb, 0x79, 0x24, 0x12, 0x6e, 0x62, 0xec, 0x42, 0xe0, 0x3c, 0xb2, 0x18, 0x8c, 0xbc, 0x0e, 0xc5, - 0x7b, 0xf4, 0xe0, 0xae, 0xd3, 0x71, 0xc5, 0x3b, 0x1a, 0x25, 0x9e, 0x9c, 0x65, 0x9f, 0x1e, 0xd8, - 0x7b, 0x08, 0xb5, 0x12, 0x04, 0xbc, 0xc5, 0xee, 0xed, 0xdc, 0xa3, 0xfc, 0xb2, 0xaf, 0x24, 0x6e, - 0xb1, 0x7b, 0x3b, 0xe8, 0x1e, 0xcf, 0x4b, 0xcc, 0x3f, 0xc8, 0x43, 0x39, 0xbd, 0xfa, 0xc9, 0xfb, - 0x50, 0x5a, 0x77, 0xc2, 0xf0, 0x91, 0x1f, 0xb8, 0x77, 0x9d, 0x70, 0x4f, 0x54, 0x05, 0x0d, 0xcf, - 0xae, 0x80, 0xdb, 0x7b, 0x8e, 0x96, 0x23, 0x4e, 0x23, 0x60, 0x5a, 0xc1, 0x86, 0x70, 0xcd, 0x57, - 0x56, 0x71, 0xe4, 0x47, 0xdd, 0x54, 0xee, 0x4f, 0x89, 0x46, 0x5c, 0x98, 0x4e, 0xf5, 0x45, 0xbc, - 0x80, 0xe2, 0x18, 0xc3, 0x74, 0x4f, 0xf1, 0x93, 0xbc, 0xde, 0xc2, 0x03, 0xcc, 0x9f, 0x25, 0x4b, - 0xb4, 0xbc, 0x07, 0x3a, 0x11, 0x79, 0x0b, 0x60, 0x73, 0xe1, 0x36, 0xda, 0x9f, 0x34, 0x10, 0xbe, - 0xdd, 0x78, 0x2c, 0xc3, 0x98, 0x34, 0x39, 0x58, 0xb5, 0x1a, 0x12, 0x64, 0xf2, 0x26, 0xe4, 0x79, - 0xd4, 0xa2, 0x9a, 0xd9, 0xea, 0xfe, 0xed, 0x2a, 0x0f, 0xf4, 0xe2, 0x17, 0xfb, 0xfa, 0x0d, 0x09, - 0xc3, 0x27, 0x2b, 0x4a, 0xcc, 0xdb, 0x98, 0x96, 0xe1, 0x47, 0x82, 0xe3, 0xde, 0x1f, 0x22, 0xf8, - 0xed, 0xf7, 0xf3, 0x50, 0x8c, 0xbf, 0x49, 0x08, 0xa0, 0x26, 0x26, 0x6e, 0xe5, 0xf1, 0x7f, 0x72, - 0x1e, 0x0a, 0x52, 0xf9, 0x12, 0x37, 0xf3, 0xe3, 0xa1, 0x50, 0xbc, 0xe6, 0x40, 0x6a, 0x59, 0x5c, - 0xf1, 0xb2, 0xe4, 0x4f, 0x72, 0x03, 0x62, 0x15, 0x6a, 0x90, 0xae, 0x35, 0xc2, 0xa6, 0xb2, 0x15, - 0xa3, 0x91, 0x29, 0xc8, 0x79, 0xdc, 0x17, 0xbc, 0x68, 0xe5, 0x3c, 0x97, 0xbc, 0x0f, 0x05, 0xc7, - 0x75, 0xa9, 0x6b, 0x3b, 0xd1, 0x10, 0x4f, 0xdc, 0x16, 0x18, 0x37, 0xbe, 0xd7, 0x21, 0x55, 0x35, - 0x22, 0x55, 0x28, 0xe2, 0x0b, 0xa7, 0xbd, 0x70, 0xa8, 0x67, 0x51, 0x13, 0x0e, 0x05, 0x46, 0xb6, - 0x19, 0x52, 0x97, 0xbc, 0x06, 0x23, 0x6c, 0x8a, 0x89, 0x9d, 0x32, 0xce, 0x51, 0xb8, 0xb6, 0xb1, - 0xce, 0x3b, 0xec, 0xee, 0x29, 0x0b, 0x11, 0xc8, 0x2b, 0x90, 0xef, 0x2d, 0x3c, 0x10, 0x7b, 0x60, - 0x39, 0x99, 0x60, 0x31, 0x1a, 0x2b, 0x26, 0x37, 0xa1, 0xf0, 0x48, 0x0f, 0x57, 0x3c, 0x93, 0x1a, - 0xba, 0x18, 0x3f, 0x46, 0xac, 0x15, 0x60, 0x8c, 0x07, 0x07, 0x9a, 0x2f, 0x01, 0x24, 0x9f, 0xee, - 0x77, 0xa0, 0x30, 0xbf, 0x0e, 0xc5, 0xf8, 0x93, 0xe4, 0x45, 0x50, 0xd6, 0x30, 0x5f, 0x6f, 0x56, - 0x71, 0x3f, 0x5e, 0xc9, 0xe7, 0x60, 0xbc, 0xcb, 0x46, 0x55, 0xa6, 0xd3, 0xb5, 0xd8, 0x32, 0x66, - 0xcb, 0x66, 0x0e, 0xc6, 0xc5, 0xb4, 0xe5, 0x71, 0x0d, 0x96, 0xfc, 0x69, 0xfe, 0x76, 0x0e, 0x13, - 0x14, 0x28, 0xf5, 0x24, 0x2f, 0xc3, 0x64, 0x33, 0xa0, 0xb8, 0x51, 0x3b, 0x4c, 0x61, 0x14, 0xdf, - 0x29, 0x25, 0xc0, 0x65, 0x97, 0x5c, 0x86, 0xe9, 0x24, 0xbf, 0xaf, 0xdd, 0xdc, 0x11, 0xc1, 0xca, - 0x25, 0x6b, 0xb2, 0x2b, 0x13, 0xfc, 0x2e, 0xee, 0x60, 0x0c, 0x43, 0x59, 0x0d, 0x0c, 0x8c, 0x64, - 0xae, 0xde, 0xa2, 0x35, 0xad, 0xc0, 0xf1, 0x42, 0xe6, 0x2c, 0x8c, 0x39, 0xce, 0x6e, 0xcf, 0xe3, - 0xfe, 0xd4, 0x25, 0x4b, 0xfc, 0x22, 0x9f, 0x85, 0x99, 0xd0, 0xdb, 0xed, 0x38, 0x51, 0x2f, 0xa0, - 0x72, 0xf5, 0xe1, 0x94, 0x9a, 0xb4, 0xca, 0x71, 0x81, 0x5c, 0x7f, 0x6f, 0x00, 0x51, 0xbf, 0xe7, - 0xef, 0x7c, 0x44, 0x9b, 0x7c, 0xaa, 0x95, 0xac, 0x19, 0xa5, 0x64, 0x0d, 0x0b, 0xc8, 0x67, 0xa0, - 0x14, 0xd0, 0x10, 0x95, 0x55, 0xec, 0x36, 0xcc, 0xdf, 0x63, 0x4d, 0x48, 0x18, 0x13, 0x7d, 0x35, - 0x98, 0xe9, 0x5b, 0x83, 0xe4, 0x0d, 0x6e, 0xf7, 0x08, 0xcd, 0xa5, 0xc4, 0xcd, 0x3c, 0x26, 0xbe, - 0x53, 0x8f, 0x62, 0x73, 0x24, 0xb3, 0x03, 0x25, 0x75, 0xe7, 0x39, 0x21, 0x0c, 0xfc, 0x2c, 0x3a, - 0x76, 0x72, 0x89, 0x38, 0x76, 0x74, 0x58, 0xc9, 0x79, 0x2e, 0xba, 0x73, 0x5e, 0x81, 0x82, 0xd4, - 0x9f, 0xd4, 0xc7, 0x5f, 0x84, 0xaa, 0x7d, 0x60, 0xc5, 0xa5, 0xe6, 0x6b, 0x30, 0x2e, 0x36, 0x97, - 0xe3, 0x0f, 0xb8, 0xcc, 0x6f, 0xe6, 0x60, 0xda, 0xa2, 0x6c, 0x81, 0x8b, 0x67, 0x55, 0x3e, 0x65, - 0x99, 0x8e, 0xb5, 0xb6, 0x1d, 0x93, 0x75, 0xe1, 0x0f, 0x0d, 0x98, 0xcd, 0xc0, 0xfd, 0x58, 0xa9, - 0xd1, 0x6e, 0x41, 0xb1, 0xee, 0x39, 0xad, 0xaa, 0xeb, 0xc6, 0x0e, 0xaa, 0xa8, 0x27, 0xbb, 0x6c, - 0x39, 0x39, 0x0c, 0xaa, 0xaa, 0x19, 0x31, 0x2a, 0xb9, 0x26, 0x26, 0x45, 0x92, 0xbc, 0x51, 0xe6, - 0x52, 0x06, 0x5e, 0xa7, 0x24, 0x93, 0x32, 0x86, 0x01, 0x72, 0x60, 0x72, 0xf1, 0xfa, 0xdc, 0x0e, - 0x5d, 0x76, 0x18, 0x60, 0xba, 0x79, 0x43, 0x19, 0xe4, 0xdf, 0xca, 0xc1, 0xd9, 0x6c, 0xc2, 0x8f, - 0x9b, 0xe5, 0x0e, 0x53, 0x5e, 0x28, 0xe9, 0xaa, 0x51, 0x91, 0xe2, 0xf9, 0x31, 0x10, 0x3f, 0x41, - 0x20, 0x0f, 0x60, 0x72, 0xc5, 0x09, 0xa3, 0xbb, 0xd4, 0x09, 0xa2, 0x1d, 0xea, 0x44, 0x43, 0xe8, - 0xf6, 0xf1, 0xd3, 0xd3, 0xb8, 0xa9, 0xed, 0x49, 0xca, 0xf4, 0xd3, 0xd3, 0x1a, 0xdb, 0x78, 0xa2, - 0x8c, 0x0c, 0x31, 0x51, 0xbe, 0x01, 0xd3, 0x0d, 0xda, 0x76, 0xba, 0x7b, 0x7e, 0x40, 0xc5, 0x99, - 0xfc, 0x75, 0x98, 0x8c, 0x41, 0x99, 0xb3, 0x45, 0x2f, 0xd6, 0xf0, 0x95, 0x8e, 0x48, 0x44, 0x89, + 0xa2, 0x74, 0x04, 0xab, 0x90, 0x6d, 0xe4, 0x79, 0x45, 0x29, 0x7e, 0x36, 0x55, 0xcd, 0x54, 0x8d, + 0xc8, 0x4f, 0xc2, 0xb9, 0xa5, 0x8e, 0xb3, 0xd3, 0xa2, 0x8b, 0x7e, 0x27, 0xf2, 0x3a, 0x3d, 0xbf, + 0x17, 0xd6, 0x9c, 0xe6, 0x7e, 0xaf, 0x1b, 0x8a, 0xc3, 0x4e, 0x6c, 0x79, 0x33, 0x2e, 0xb4, 0x77, + 0x78, 0xa9, 0xc2, 0x72, 0x10, 0x03, 0x72, 0x17, 0x66, 0x78, 0x51, 0xb5, 0x17, 0xf9, 0x8d, 0xa6, + 0xd3, 0xf2, 0x3a, 0xbb, 0x78, 0x06, 0x5a, 0xa8, 0x5d, 0x60, 0x7b, 0x4b, 0xa7, 0x17, 0xf9, 0x76, + 0xc8, 0xe1, 0x0a, 0xbf, 0x7e, 0x22, 0xb2, 0x0c, 0xd3, 0x16, 0x75, 0xdc, 0xfb, 0xce, 0xe3, 0x45, + 0xa7, 0xeb, 0x34, 0xbd, 0xe8, 0x00, 0x77, 0x66, 0xf9, 0x5a, 0xe5, 0xe8, 0xb0, 0xf2, 0x42, 0x40, + 0x1d, 0xd7, 0x6e, 0x3b, 0x8f, 0xed, 0xa6, 0x28, 0x54, 0x98, 0xa5, 0xe9, 0x62, 0x56, 0x5e, 0x27, + 0x66, 0x55, 0x4c, 0xb3, 0xf2, 0x3a, 0x83, 0x59, 0x25, 0x74, 0x92, 0xd5, 0x86, 0x13, 0xec, 0xd2, + 0x88, 0x1f, 0x12, 0xc2, 0x25, 0xe3, 0x8a, 0xa1, 0xb0, 0x8a, 0xb0, 0xcc, 0xc6, 0x03, 0xc3, 0x34, + 0x2b, 0x85, 0x8e, 0x49, 0xde, 0x76, 0xe0, 0x45, 0x54, 0x6d, 0xe1, 0x04, 0x56, 0x0b, 0xfb, 0x1f, + 0x8f, 0x49, 0x07, 0x35, 0xb1, 0x8f, 0x32, 0xe1, 0xa6, 0x34, 0xb2, 0xd4, 0xc7, 0x2d, 0xbb, 0x95, + 0x7d, 0x94, 0x31, 0x37, 0xb5, 0x9d, 0x93, 0xd8, 0x4e, 0x85, 0xdb, 0x80, 0x86, 0xf6, 0x51, 0x92, + 0x55, 0xd6, 0x69, 0x11, 0xed, 0x30, 0x89, 0x16, 0x87, 0xa4, 0x53, 0x58, 0xb5, 0x57, 0xc4, 0x9e, + 0xba, 0x1c, 0xc8, 0x62, 0x3b, 0xe3, 0xc8, 0x34, 0x4d, 0x4c, 0x7e, 0x0e, 0xa6, 0x37, 0x43, 0x7a, + 0x7b, 0x79, 0xbd, 0xb1, 0xd4, 0x71, 0xbb, 0xbe, 0xd7, 0x89, 0xe6, 0xa6, 0x71, 0xa7, 0x7d, 0xe3, + 0x04, 0xa5, 0x73, 0x5d, 0xa5, 0xc1, 0x98, 0x1b, 0x3e, 0x6e, 0xbd, 0x90, 0xda, 0x0f, 0xbc, 0x6e, + 0x68, 0x53, 0x51, 0xa6, 0x8e, 0x5b, 0xea, 0x53, 0xe6, 0x5d, 0x98, 0xe9, 0x63, 0x43, 0xa6, 0x00, + 0x18, 0xd0, 0xde, 0x5c, 0x6d, 0x2c, 0x6d, 0x94, 0x4f, 0xb1, 0x8d, 0x24, 0xfe, 0x5e, 0x5a, 0xad, + 0xd6, 0x56, 0x96, 0xea, 0x65, 0x83, 0xcc, 0xc0, 0x24, 0x42, 0xea, 0xcb, 0x0d, 0x0e, 0xca, 0x7d, + 0x30, 0x52, 0x18, 0x2d, 0x8f, 0x59, 0x65, 0x3e, 0x75, 0x23, 0x36, 0x01, 0x70, 0x4d, 0x31, 0x7f, + 0x33, 0x07, 0xe7, 0xe5, 0xb2, 0x42, 0xa3, 0x47, 0x7e, 0xb0, 0xef, 0x75, 0x76, 0x9f, 0xf3, 0xd5, + 0xe1, 0xb6, 0xb6, 0x3a, 0xbc, 0x92, 0x5a, 0xa9, 0x53, 0xad, 0x3c, 0x66, 0x89, 0xf8, 0xb3, 0x51, + 0x78, 0xf1, 0x58, 0x2a, 0xf2, 0x21, 0x5b, 0xcd, 0x3d, 0xda, 0x89, 0x96, 0xdd, 0x16, 0x65, 0xdb, + 0x49, 0xbf, 0x17, 0x89, 0x43, 0xf9, 0x97, 0x8f, 0x0e, 0x2b, 0xb3, 0x3c, 0x60, 0xc6, 0xf6, 0xdc, + 0x16, 0xb5, 0x23, 0x5e, 0xac, 0x89, 0x5b, 0x3f, 0x35, 0x63, 0x19, 0x87, 0xef, 0x2d, 0x77, 0x22, + 0x1a, 0x3c, 0x74, 0x78, 0xdc, 0x80, 0x60, 0xb9, 0x4f, 0x69, 0xd7, 0x76, 0x58, 0xa9, 0xed, 0x89, + 0x62, 0x9d, 0x65, 0x1f, 0x35, 0xb9, 0xad, 0xb0, 0x5c, 0x64, 0x9b, 0x9c, 0xfb, 0xce, 0x63, 0x61, + 0xb9, 0xe3, 0x39, 0xb1, 0xc2, 0x92, 0xfb, 0x0a, 0xb6, 0x9d, 0xc7, 0x56, 0x3f, 0x09, 0xf9, 0x3a, + 0x9c, 0x11, 0x0b, 0x10, 0x53, 0xc6, 0x81, 0xdf, 0x92, 0x2d, 0x1e, 0x41, 0x5e, 0xaf, 0x1d, 0x1d, + 0x56, 0xce, 0x89, 0xe5, 0xcb, 0x6e, 0x72, 0x8c, 0xcc, 0x56, 0x67, 0x73, 0x21, 0x1b, 0x6c, 0x41, + 0x4e, 0x75, 0xc7, 0x7d, 0x1a, 0x86, 0xce, 0xae, 0xb4, 0xf2, 0xf9, 0xcd, 0x98, 0xd2, 0x99, 0x76, + 0x9b, 0x97, 0x5b, 0x03, 0x29, 0xc9, 0x5d, 0x98, 0xda, 0xa6, 0x3b, 0xea, 0xf8, 0x8c, 0xc5, 0xaa, + 0xaa, 0xfc, 0x88, 0xee, 0x0c, 0x1e, 0x9c, 0x14, 0x1d, 0xf1, 0x60, 0x66, 0x3d, 0xf0, 0x1f, 0x1f, + 0xb0, 0x2d, 0x2b, 0xed, 0xd0, 0x00, 0x9d, 0xd4, 0xc6, 0x51, 0x19, 0xcc, 0x25, 0x16, 0xb2, 0x5e, + 0x5e, 0xfb, 0xcc, 0xd1, 0x61, 0xe5, 0xc5, 0x2e, 0x03, 0xdb, 0x2d, 0x01, 0xb7, 0x53, 0xd1, 0x73, + 0xfd, 0x5c, 0xc9, 0x4f, 0xc3, 0xb4, 0xe5, 0xf7, 0x22, 0xaf, 0xb3, 0xdb, 0x88, 0x02, 0x27, 0xa2, + 0xbb, 0x7c, 0x41, 0x4a, 0xbc, 0xe1, 0x52, 0xa5, 0xfc, 0x80, 0x3d, 0xe0, 0x40, 0x3b, 0x14, 0x50, + 0x6d, 0x45, 0xd0, 0x09, 0xcc, 0x5f, 0xcf, 0xc1, 0x9c, 0x18, 0x06, 0x8b, 0x36, 0xfd, 0xc0, 0x7d, + 0xfe, 0xa7, 0xfd, 0x92, 0x36, 0xed, 0x5f, 0x8e, 0x7d, 0xad, 0xb2, 0x1a, 0x79, 0xcc, 0xac, 0xff, + 0x17, 0x06, 0x5c, 0x3c, 0x8e, 0x88, 0xf5, 0x4e, 0xec, 0x9f, 0x58, 0xec, 0xf3, 0x43, 0xec, 0xc2, + 0x2c, 0x8e, 0x27, 0x1e, 0x80, 0x87, 0x77, 0xfd, 0x30, 0xc2, 0x53, 0xc8, 0x9c, 0xe6, 0x10, 0x51, + 0xf3, 0xfd, 0x16, 0xae, 0x57, 0xb5, 0xd7, 0xd9, 0xb2, 0xf4, 0xe7, 0x87, 0x15, 0x60, 0x20, 0xee, + 0x51, 0xc8, 0x6c, 0x17, 0x2e, 0x31, 0x78, 0xbe, 0x1e, 0xda, 0xe8, 0xc5, 0xb2, 0x4f, 0x0f, 0x42, + 0x2b, 0x8b, 0x35, 0x9e, 0x34, 0x55, 0x7b, 0xd1, 0xde, 0x7a, 0x40, 0x1f, 0xd0, 0x80, 0x76, 0x9a, + 0xf4, 0x53, 0x76, 0xd2, 0xa4, 0x37, 0x6e, 0xa8, 0x6d, 0xd6, 0xff, 0x1b, 0x83, 0xd3, 0x59, 0x64, + 0xac, 0x5f, 0x14, 0xcb, 0x3e, 0x1d, 0x6a, 0xfd, 0x77, 0x0c, 0x28, 0x35, 0x68, 0xd3, 0xef, 0xb8, + 0xb7, 0x9d, 0x66, 0xe4, 0x4b, 0xd7, 0x12, 0x9b, 0x6b, 0x36, 0x06, 0xb7, 0x1f, 0x60, 0x81, 0x76, + 0xc2, 0xf1, 0xa5, 0xe1, 0x0c, 0xea, 0xa6, 0x8f, 0x0e, 0xbd, 0x11, 0x93, 0xc9, 0xe4, 0x13, 0x78, + 0x3b, 0xa3, 0x7d, 0x94, 0xd4, 0x60, 0x72, 0xd1, 0xef, 0x74, 0x28, 0xfb, 0xa1, 0xb8, 0xa7, 0xa2, + 0x43, 0x69, 0x53, 0x16, 0xa4, 0x4f, 0x3a, 0x74, 0x12, 0x72, 0x13, 0xf2, 0x9b, 0x0b, 0xb7, 0xc5, + 0x18, 0x48, 0xa7, 0xbb, 0xcd, 0x85, 0xdb, 0xb8, 0x67, 0x67, 0x76, 0xd0, 0x64, 0x6f, 0xe1, 0x81, + 0x7a, 0x96, 0xbb, 0xb9, 0x70, 0x9b, 0xac, 0xc1, 0x8c, 0x45, 0xbf, 0xd1, 0xf3, 0x02, 0x2a, 0x26, + 0xc0, 0xfd, 0xdb, 0x55, 0x1c, 0x8b, 0x02, 0xd7, 0x63, 0x01, 0x2f, 0x94, 0x7b, 0x14, 0xbb, 0xfd, + 0x40, 0x0d, 0x2f, 0xec, 0xa7, 0x25, 0xbf, 0x00, 0x67, 0xea, 0x5e, 0x28, 0xea, 0xcc, 0x0f, 0x51, + 0x5d, 0xbc, 0x4f, 0x1d, 0x1b, 0x30, 0x1d, 0xbe, 0x90, 0x39, 0x1d, 0x3e, 0xe3, 0xc6, 0x4c, 0x6c, + 0x7e, 0x42, 0xeb, 0xa6, 0xfd, 0x7a, 0xb3, 0xbf, 0x43, 0x3e, 0x82, 0x29, 0x3c, 0xb5, 0xc2, 0x73, + 0x65, 0x74, 0xf5, 0x1e, 0x1f, 0xf0, 0xe5, 0xcf, 0x65, 0x7e, 0xf9, 0x02, 0x1e, 0x82, 0xd9, 0x78, + 0x3a, 0x8d, 0x6e, 0xe1, 0xda, 0x5e, 0x47, 0xe3, 0x4c, 0x3e, 0x80, 0x69, 0xb1, 0xe8, 0xac, 0x3d, + 0xd8, 0xd8, 0xa3, 0x75, 0xe7, 0x40, 0x38, 0x53, 0xa0, 0x1d, 0x2b, 0x56, 0x2a, 0xdb, 0x7f, 0x60, + 0x47, 0x7b, 0xd4, 0x76, 0x1d, 0x4d, 0x3d, 0xa7, 0x08, 0xc9, 0xcf, 0xc2, 0xc4, 0x8a, 0x8f, 0x17, + 0x68, 0xa8, 0x6a, 0x8a, 0xc8, 0xe7, 0x2b, 0x18, 0x5e, 0xcc, 0xc1, 0xa9, 0x45, 0xe4, 0x47, 0x87, + 0x95, 0xb7, 0x9f, 0x54, 0x0a, 0x95, 0x0f, 0x58, 0xea, 0xd7, 0xc8, 0x22, 0x14, 0xb6, 0xe9, 0x0e, + 0x6b, 0x6d, 0x3a, 0x34, 0x4e, 0x82, 0xb9, 0xbe, 0x78, 0x24, 0x7e, 0xa9, 0xb7, 0x53, 0x12, 0xc3, + 0xfc, 0xf7, 0x06, 0x4a, 0x20, 0xb9, 0x86, 0x0e, 0x6d, 0xb1, 0xa7, 0x3c, 0xee, 0x90, 0x9d, 0x6e, + 0x57, 0xf7, 0x75, 0xe7, 0x28, 0x6c, 0x3b, 0x7d, 0xdb, 0x69, 0xd2, 0x48, 0x9e, 0xbb, 0x22, 0xf2, + 0x03, 0x84, 0xa8, 0xdb, 0x69, 0x8e, 0x43, 0xbe, 0x0c, 0xa7, 0xeb, 0xf4, 0xa1, 0xd7, 0xa4, 0xd5, + 0x28, 0xa2, 0x21, 0x6f, 0xed, 0x62, 0x95, 0x5f, 0x50, 0x16, 0x6b, 0xaf, 0x1c, 0x1d, 0x56, 0x2e, + 0xb9, 0x58, 0x6e, 0x3b, 0x09, 0x82, 0xdd, 0x74, 0x54, 0x5e, 0x99, 0x1c, 0xcc, 0x7f, 0x94, 0x4b, + 0x7a, 0x80, 0xbc, 0x06, 0x23, 0xd6, 0x7a, 0x5c, 0x7f, 0x7e, 0xf7, 0x98, 0xaa, 0x3e, 0x22, 0x90, + 0xaf, 0xc2, 0x19, 0x85, 0x0f, 0x0a, 0x07, 0x75, 0x59, 0x85, 0x78, 0x63, 0x5e, 0xc5, 0xcb, 0x26, + 0xa5, 0x26, 0x0e, 0xc7, 0x48, 0xd5, 0x28, 0x9b, 0x07, 0x6b, 0xac, 0x52, 0x50, 0xa7, 0x1d, 0x8f, + 0xf3, 0x56, 0x1a, 0xab, 0xf2, 0x76, 0x11, 0x21, 0xdd, 0xd8, 0x2c, 0x0e, 0x64, 0x01, 0x0a, 0x75, + 0x2f, 0x64, 0x3b, 0x02, 0x57, 0x78, 0xa6, 0xf1, 0x68, 0x43, 0x01, 0xd3, 0xa2, 0x0d, 0x05, 0xcc, + 0xfc, 0x6b, 0x43, 0xc9, 0xea, 0xf0, 0x9c, 0xae, 0x35, 0xb7, 0xb4, 0xb5, 0xe6, 0xb4, 0x20, 0x8d, + 0x5b, 0xc5, 0xca, 0x32, 0xed, 0x83, 0x69, 0x98, 0xd4, 0x90, 0xd0, 0xd1, 0x77, 0x33, 0xa4, 0x01, + 0x3f, 0x15, 0xfd, 0x74, 0x39, 0xfa, 0xc6, 0xed, 0x1a, 0xca, 0x7f, 0xf3, 0x2f, 0x0c, 0xdc, 0x2d, + 0xab, 0x14, 0xac, 0x37, 0x18, 0x48, 0xed, 0x8d, 0x5e, 0x48, 0x03, 0x0b, 0xa1, 0xdc, 0x2d, 0x70, + 0x45, 0x77, 0x0b, 0x6c, 0x59, 0x0c, 0x46, 0xbe, 0x04, 0xa3, 0x9b, 0x68, 0xfb, 0xeb, 0x9e, 0x25, + 0x31, 0x7f, 0x2c, 0xe4, 0xf3, 0xb1, 0xc7, 0xfe, 0x55, 0xd5, 0x09, 0x96, 0x91, 0x06, 0x8c, 0x2f, + 0x06, 0x14, 0xf3, 0x37, 0x8c, 0x0c, 0x7f, 0x05, 0xd8, 0xe4, 0x24, 0xe9, 0x2b, 0x40, 0xc1, 0xc9, + 0xfc, 0xb5, 0x1c, 0x90, 0xa4, 0x8d, 0xb4, 0x19, 0xd0, 0x28, 0x7c, 0x6e, 0x07, 0xfd, 0x7d, 0x6d, + 0xd0, 0x5f, 0xec, 0x1b, 0x74, 0xde, 0xbc, 0xa1, 0xc6, 0xfe, 0x8f, 0x0d, 0x38, 0x9b, 0x4d, 0x48, + 0x5e, 0x86, 0xb1, 0xb5, 0x8d, 0x75, 0xe9, 0x9c, 0x24, 0x9a, 0xe2, 0x77, 0xd1, 0xa6, 0xb5, 0x44, + 0x11, 0x79, 0x03, 0xc6, 0x3e, 0xb4, 0x16, 0xd9, 0x62, 0xa7, 0xc4, 0xef, 0x7c, 0x23, 0xb0, 0x9b, + 0xfa, 0x66, 0x49, 0x20, 0xa9, 0x63, 0x9b, 0x7f, 0x66, 0x63, 0xfb, 0xed, 0x1c, 0x4c, 0x57, 0x9b, + 0x4d, 0x1a, 0x86, 0xcc, 0x94, 0xa1, 0x61, 0xf4, 0xdc, 0x0e, 0x6c, 0xb6, 0xdb, 0x91, 0xd6, 0xb6, + 0xa1, 0x46, 0xf5, 0x4f, 0x0c, 0x38, 0x23, 0xa9, 0x1e, 0x7a, 0xf4, 0xd1, 0xc6, 0x5e, 0x40, 0xc3, + 0x3d, 0xbf, 0xe5, 0x0e, 0x1b, 0x89, 0x86, 0x6b, 0xba, 0xd7, 0x8a, 0x68, 0xa0, 0x1e, 0x91, 0x3f, + 0x40, 0x88, 0xb6, 0xa6, 0x23, 0x84, 0xcc, 0xc3, 0x78, 0xb5, 0xdb, 0x0d, 0xfc, 0x87, 0x7c, 0xda, + 0x4f, 0x8a, 0x1b, 0x51, 0x0e, 0xd2, 0x6e, 0x50, 0x39, 0x88, 0x55, 0xa3, 0x4e, 0x3b, 0xdc, 0xa7, + 0x7a, 0x92, 0x57, 0xc3, 0xa5, 0x1d, 0xd5, 0xb6, 0xc2, 0x72, 0xf3, 0x5b, 0x23, 0x50, 0x52, 0x1b, + 0x42, 0x4c, 0x18, 0xe3, 0x0e, 0x32, 0xaa, 0xa3, 0x82, 0x83, 0x10, 0x4b, 0x94, 0x24, 0x7e, 0x47, + 0xb9, 0x13, 0xfd, 0x8e, 0xb6, 0x61, 0x72, 0x3d, 0xf0, 0xbb, 0x7e, 0x48, 0x5d, 0x9e, 0x82, 0x87, + 0x6b, 0xad, 0xd9, 0xd8, 0x19, 0x97, 0xf7, 0x39, 0x9e, 0x03, 0xa2, 0x21, 0xdf, 0x15, 0xd8, 0x76, + 0x3a, 0x41, 0x8f, 0xce, 0x87, 0x5f, 0x31, 0x38, 0xa1, 0x08, 0x58, 0x88, 0xaf, 0x18, 0x18, 0x44, + 0xbf, 0x62, 0x60, 0x10, 0x75, 0x5a, 0x8c, 0x3e, 0xab, 0x69, 0x41, 0x7e, 0xcd, 0x80, 0x89, 0x6a, + 0xa7, 0x23, 0xfc, 0x8e, 0x64, 0x1c, 0xfb, 0x99, 0xe4, 0x9a, 0x81, 0x3b, 0xa6, 0xf2, 0x5b, 0x86, + 0xaf, 0x89, 0x5b, 0x86, 0xb7, 0x3f, 0xd6, 0x2d, 0xc3, 0x46, 0xe0, 0x78, 0x51, 0x88, 0xd7, 0xc9, + 0xc9, 0x07, 0x55, 0xe7, 0x63, 0xa5, 0x1e, 0xe4, 0x6d, 0x28, 0xc7, 0xf2, 0xb8, 0xdc, 0x71, 0xe9, + 0x63, 0xca, 0xdd, 0xb4, 0x26, 0x79, 0x8c, 0xa3, 0x76, 0x7d, 0x92, 0x46, 0x34, 0xbf, 0x6d, 0xc0, + 0x59, 0x55, 0x20, 0x1a, 0xbd, 0x9d, 0xb6, 0x87, 0x1b, 0x17, 0x72, 0x1d, 0x8a, 0x62, 0xbc, 0x62, + 0xb3, 0xaf, 0x3f, 0x6f, 0x53, 0x82, 0x42, 0x96, 0xd8, 0x10, 0x31, 0x1e, 0x62, 0x97, 0x3f, 0x9b, + 0x9a, 0x6e, 0xac, 0xa8, 0x36, 0x27, 0x3a, 0xbb, 0x1c, 0xe0, 0x6f, 0x7d, 0xec, 0x18, 0xc4, 0x7c, + 0x0f, 0x66, 0xf4, 0x5a, 0x36, 0x28, 0x06, 0xc1, 0xc9, 0xa6, 0x19, 0xd9, 0x4d, 0x93, 0xe5, 0xe6, + 0x36, 0x90, 0x3e, 0xfa, 0x10, 0xaf, 0xca, 0x68, 0x24, 0xaf, 0x72, 0xe5, 0x41, 0x55, 0x1f, 0x62, + 0x9c, 0xc1, 0x6c, 0x42, 0xed, 0x6e, 0x24, 0x35, 0xff, 0xa6, 0x08, 0xb3, 0x19, 0xaa, 0xe3, 0x84, + 0xa5, 0xbd, 0xa2, 0x4f, 0x9e, 0x62, 0xec, 0x93, 0x20, 0xa7, 0xcc, 0x7b, 0x32, 0x5b, 0xd5, 0x31, + 0x53, 0xe5, 0xb8, 0x14, 0x56, 0x9f, 0xc4, 0xf2, 0xae, 0xba, 0x0d, 0x8d, 0x3e, 0x33, 0xb7, 0xa1, + 0x1a, 0x4c, 0x8a, 0x56, 0x89, 0xa9, 0x3c, 0x96, 0x6c, 0xe8, 0x03, 0x5e, 0x60, 0xf7, 0x4d, 0x69, + 0x9d, 0x84, 0xf3, 0x08, 0xfd, 0xd6, 0x43, 0x2a, 0x78, 0x8c, 0xab, 0x3c, 0xb0, 0x20, 0x93, 0x87, + 0x42, 0x42, 0xfe, 0xc0, 0x00, 0x22, 0x20, 0xea, 0x7c, 0x2e, 0x1c, 0x37, 0x9f, 0xdd, 0x67, 0x33, + 0x9f, 0x5f, 0x94, 0x75, 0xcc, 0x9e, 0xd7, 0x19, 0xd5, 0x22, 0xff, 0xcc, 0x80, 0x19, 0xee, 0xbb, + 0xa2, 0x56, 0xb6, 0x78, 0x5c, 0x65, 0x9b, 0xcf, 0xa6, 0xb2, 0x17, 0x43, 0xfc, 0xec, 0x80, 0xba, + 0xf6, 0x57, 0x8a, 0xfc, 0x24, 0x40, 0x3c, 0xa3, 0xa4, 0x8f, 0xe4, 0xc5, 0x0c, 0x2d, 0x10, 0x23, + 0x25, 0x61, 0x9e, 0x51, 0x4c, 0xa7, 0x7a, 0x15, 0x25, 0xdc, 0xc8, 0x2f, 0xc0, 0x69, 0x36, 0x5f, + 0x62, 0x88, 0xf0, 0xb4, 0x9b, 0x9b, 0xc0, 0xaf, 0x7c, 0x7e, 0xf0, 0xd2, 0x7e, 0x3d, 0x8b, 0x8c, + 0x47, 0xaa, 0x24, 0x89, 0x04, 0xa2, 0xb6, 0xba, 0x41, 0xcc, 0xa2, 0x40, 0x97, 0x5a, 0xac, 0x7d, + 0x38, 0x57, 0xc2, 0x6f, 0x66, 0xea, 0xb7, 0xf3, 0x72, 0x2e, 0x70, 0xfd, 0x16, 0xea, 0xa1, 0x26, + 0x08, 0x22, 0x1f, 0x02, 0x69, 0xf4, 0x76, 0x77, 0x69, 0x18, 0x51, 0x97, 0xc3, 0x68, 0x10, 0xce, + 0x4d, 0xa2, 0x7e, 0xc0, 0x03, 0xa6, 0x50, 0x96, 0xda, 0x81, 0x2c, 0x56, 0x85, 0xa4, 0x9f, 0xf8, + 0xc2, 0x0e, 0x9c, 0x1f, 0xd8, 0xcc, 0x8c, 0x30, 0x92, 0x79, 0x3d, 0x8c, 0xe4, 0xfc, 0x20, 0x75, + 0x18, 0xaa, 0xa1, 0x24, 0xbf, 0x6d, 0xa4, 0xf4, 0x9f, 0x30, 0x56, 0x78, 0x62, 0xbf, 0x41, 0x0b, + 0x44, 0x0e, 0x53, 0x0c, 0x70, 0x0d, 0x99, 0x4b, 0x8c, 0x24, 0xa6, 0x21, 0x55, 0x0d, 0x8b, 0xba, + 0xf2, 0x29, 0x55, 0xa1, 0xf9, 0xaf, 0x0d, 0x20, 0xbc, 0x86, 0x8b, 0x4e, 0xd7, 0xd9, 0xf1, 0x5a, + 0x5e, 0xe4, 0xd1, 0x90, 0xdc, 0x83, 0xb2, 0x60, 0xc1, 0xb6, 0xed, 0xaa, 0x87, 0x98, 0xb8, 0x42, + 0x8e, 0xcb, 0xec, 0xb4, 0x59, 0xd3, 0x47, 0x38, 0x60, 0xf0, 0x72, 0x4f, 0x31, 0x78, 0xe6, 0x0f, + 0x0d, 0x38, 0xdf, 0x5f, 0x6d, 0xf1, 0xe5, 0xb8, 0xf3, 0x8c, 0x13, 0x3a, 0x2f, 0xab, 0x95, 0x39, + 0x3c, 0xc8, 0x78, 0x66, 0xad, 0xcc, 0x27, 0x67, 0xa0, 0x4f, 0xde, 0xca, 0x5f, 0xce, 0x41, 0x69, + 0xbd, 0xd5, 0xdb, 0xf5, 0x3a, 0x75, 0x27, 0x72, 0x9e, 0xdb, 0x2d, 0xc5, 0x5b, 0xda, 0x96, 0x22, + 0xf6, 0x09, 0x8b, 0x1b, 0x36, 0x5c, 0x92, 0x35, 0x03, 0xa6, 0x13, 0x12, 0x3e, 0x4b, 0xef, 0xc2, + 0x08, 0xfb, 0x21, 0x2c, 0x94, 0x4b, 0x7d, 0x8c, 0x11, 0xeb, 0x7a, 0xfc, 0x9f, 0x30, 0xf2, 0xf5, + 0xd4, 0x76, 0xc8, 0xe1, 0xc2, 0x17, 0x78, 0x66, 0xaa, 0x27, 0xcf, 0xa2, 0xf9, 0xaf, 0x0c, 0x28, + 0xa7, 0x5b, 0x42, 0xee, 0xc1, 0x38, 0xe3, 0xe4, 0xc5, 0x59, 0xae, 0x5e, 0x19, 0xd0, 0xe6, 0xeb, + 0x02, 0x8d, 0x57, 0x0f, 0x3b, 0x9f, 0x72, 0x88, 0x25, 0x39, 0x5c, 0xb0, 0xa0, 0xa4, 0x62, 0x65, + 0xd4, 0xee, 0x75, 0x5d, 0x35, 0x9d, 0xcd, 0xee, 0x07, 0xb5, 0xd6, 0xbf, 0xa5, 0xd5, 0x5a, 0x28, + 0xa5, 0x61, 0xd3, 0x15, 0x62, 0x50, 0x1c, 0xcf, 0x85, 0xa2, 0xca, 0x99, 0x4c, 0x9b, 0xa2, 0x07, + 0xc5, 0x71, 0x18, 0xdb, 0x8b, 0xf0, 0xef, 0x09, 0x39, 0xc3, 0xbd, 0x48, 0x17, 0x21, 0xaa, 0x3d, + 0xcb, 0x71, 0xcc, 0x7f, 0x98, 0x87, 0xb3, 0x49, 0xf5, 0x78, 0xf2, 0xc6, 0x75, 0x27, 0x70, 0xda, + 0xe1, 0x09, 0x33, 0xe0, 0x4a, 0x5f, 0xd5, 0x30, 0xe8, 0x5b, 0x56, 0x4d, 0xa9, 0x90, 0x99, 0xaa, + 0x10, 0x6e, 0xe2, 0x78, 0x85, 0x64, 0x35, 0xc8, 0x3d, 0xc8, 0x37, 0x68, 0x24, 0x42, 0x43, 0x2f, + 0xf7, 0xf5, 0xaa, 0x5a, 0xaf, 0xeb, 0x0d, 0x1a, 0xf1, 0x41, 0xe4, 0xde, 0xf5, 0x54, 0xf3, 0x76, + 0x67, 0xe6, 0xf8, 0x36, 0x8c, 0x2d, 0x3d, 0xee, 0xd2, 0x66, 0x24, 0x22, 0x42, 0xaf, 0x1e, 0xcf, + 0x8f, 0xe3, 0x2a, 0x71, 0xa7, 0x14, 0x01, 0x6a, 0x67, 0x71, 0x94, 0x0b, 0xb7, 0xa0, 0x20, 0x3f, + 0xfe, 0x44, 0xf1, 0x93, 0x6f, 0xc1, 0x84, 0xf2, 0x91, 0x27, 0x12, 0xfa, 0xbf, 0x31, 0x60, 0x8c, + 0x29, 0xbd, 0xad, 0x37, 0x9f, 0x53, 0x8d, 0x74, 0x53, 0xd3, 0x48, 0x33, 0x4a, 0xa0, 0x0f, 0xce, + 0xcb, 0x37, 0x4f, 0xd0, 0x45, 0x87, 0x06, 0x40, 0x82, 0x4c, 0xee, 0xc0, 0xb8, 0xc8, 0xe8, 0x22, + 0x32, 0xa3, 0xaa, 0x91, 0x43, 0x32, 0x73, 0x54, 0x6c, 0xe5, 0xf8, 0xdd, 0xb4, 0x59, 0x28, 0xa9, + 0x49, 0x3d, 0xf1, 0xae, 0x56, 0x43, 0x55, 0x19, 0x9b, 0x45, 0xbf, 0xc3, 0x23, 0x49, 0x94, 0x1c, + 0x54, 0x03, 0xdc, 0xac, 0xab, 0xe2, 0x60, 0x23, 0x7f, 0x1c, 0x93, 0xb3, 0x82, 0x49, 0xf6, 0x99, + 0xc7, 0x5f, 0x95, 0x78, 0x6c, 0x86, 0xac, 0xd8, 0xbb, 0x50, 0xba, 0xed, 0x07, 0x8f, 0x9c, 0xc0, + 0xad, 0xee, 0x52, 0xe1, 0x17, 0x5f, 0x40, 0xe7, 0xf6, 0xc9, 0x07, 0x1c, 0x6e, 0x3b, 0xac, 0xe0, + 0x47, 0x87, 0x95, 0x91, 0x9a, 0xef, 0xb7, 0x2c, 0x0d, 0x9d, 0xac, 0xc1, 0xe4, 0x7d, 0xe7, 0xb1, + 0xb8, 0xdd, 0xdb, 0xd8, 0x58, 0x11, 0x5e, 0x29, 0x57, 0x8f, 0x0e, 0x2b, 0xe7, 0xdb, 0xce, 0xe3, + 0xf8, 0x56, 0x70, 0xb0, 0x03, 0xb8, 0x4e, 0x4f, 0x3c, 0x98, 0x5a, 0xf7, 0x83, 0x48, 0x7c, 0x84, + 0xd9, 0xb4, 0xf9, 0x01, 0x97, 0x73, 0xf3, 0x99, 0x97, 0x73, 0xe7, 0x99, 0x21, 0x6f, 0x3f, 0x88, + 0xc9, 0xb5, 0x80, 0x42, 0x8d, 0x31, 0x79, 0x17, 0x66, 0x16, 0x69, 0x10, 0x79, 0x0f, 0xbc, 0xa6, + 0x13, 0xd1, 0xdb, 0x7e, 0xd0, 0x76, 0x22, 0x71, 0xa0, 0x82, 0x1b, 0xea, 0x26, 0xe5, 0x9c, 0xda, + 0x4e, 0x64, 0xf5, 0x63, 0x92, 0xaf, 0x66, 0xf9, 0xf9, 0x8c, 0x62, 0xf3, 0xdf, 0x60, 0x46, 0x41, + 0x86, 0x9f, 0xcf, 0x80, 0x2e, 0xc8, 0xf0, 0xf8, 0xd9, 0x3d, 0xee, 0x92, 0xb4, 0x50, 0xbb, 0x21, + 0x2e, 0x6c, 0x4f, 0xbe, 0x04, 0x8d, 0xc7, 0x6d, 0xc0, 0x65, 0xe8, 0x02, 0xe4, 0x6b, 0xeb, 0xb7, + 0xf1, 0x88, 0x44, 0x5c, 0x4a, 0xd2, 0xce, 0x9e, 0xd3, 0x69, 0xa2, 0x2d, 0x23, 0x3c, 0x1d, 0x54, + 0x85, 0x57, 0x5b, 0xbf, 0x4d, 0x1c, 0x98, 0x5d, 0xa7, 0x41, 0xdb, 0x8b, 0xbe, 0x7c, 0xe3, 0x86, + 0x32, 0x50, 0x05, 0xac, 0xda, 0xbc, 0xa8, 0x5a, 0xa5, 0x8b, 0x28, 0xf6, 0xe3, 0x1b, 0x37, 0x32, + 0x87, 0x23, 0xae, 0x58, 0x16, 0x2f, 0xb2, 0x04, 0x53, 0xf7, 0x9d, 0xc7, 0xe2, 0xfa, 0x3a, 0xde, + 0xe3, 0xe5, 0x31, 0x1e, 0x00, 0x05, 0xab, 0x99, 0x14, 0xa9, 0x43, 0xac, 0x13, 0x91, 0x77, 0x60, + 0x22, 0x11, 0xaf, 0x10, 0x2f, 0x2e, 0xf3, 0xdc, 0x11, 0x54, 0x11, 0x4e, 0xed, 0x2c, 0x49, 0x41, + 0x27, 0x9b, 0xf1, 0x16, 0x9d, 0x1b, 0xa4, 0xe8, 0x1e, 0x59, 0xac, 0xcd, 0xab, 0x5b, 0x74, 0x07, + 0x4b, 0xb4, 0x66, 0x4d, 0xc7, 0x26, 0x3a, 0xf7, 0xab, 0xb1, 0x74, 0x2e, 0xca, 0xce, 0x7f, 0x3d, + 0xf0, 0xdb, 0xdd, 0x08, 0xfd, 0x24, 0x53, 0x3b, 0xff, 0x2e, 0x96, 0x64, 0xec, 0xfc, 0x39, 0x49, + 0xf6, 0xad, 0xfc, 0xe4, 0x53, 0xdc, 0xca, 0x53, 0x18, 0x59, 0xf1, 0x9b, 0xfb, 0xe8, 0x18, 0x59, + 0xac, 0x7d, 0xc8, 0xf4, 0x47, 0xcb, 0x6f, 0xee, 0x3f, 0xbb, 0xdb, 0x64, 0x64, 0x4f, 0x56, 0x59, + 0xdb, 0x99, 0x58, 0x89, 0x4f, 0xa3, 0xe3, 0x64, 0x72, 0xd3, 0xa6, 0x95, 0x71, 0x43, 0x85, 0x4b, + 0xa1, 0x6c, 0x88, 0xa5, 0x93, 0x13, 0x0a, 0xe5, 0x3a, 0x0d, 0xf7, 0x23, 0xbf, 0xbb, 0xd8, 0xf2, + 0xba, 0x3b, 0xbe, 0x13, 0xb8, 0x73, 0xe5, 0x01, 0x0a, 0xe3, 0xb5, 0x4c, 0x85, 0x31, 0xe3, 0x72, + 0x7a, 0xbb, 0x29, 0x19, 0x58, 0x7d, 0x2c, 0xc9, 0x57, 0x61, 0x8a, 0xcd, 0x96, 0xa5, 0xc7, 0x11, + 0xed, 0x70, 0x51, 0x9a, 0xc1, 0xa5, 0xfe, 0xb4, 0x12, 0x5a, 0x19, 0x17, 0x72, 0x21, 0x45, 0xed, + 0x41, 0x63, 0x02, 0x55, 0x48, 0x75, 0x56, 0xc4, 0x85, 0xb9, 0xfb, 0xce, 0x63, 0x25, 0x11, 0x91, + 0x22, 0xf5, 0x04, 0x25, 0xf6, 0xca, 0xd1, 0x61, 0xe5, 0x15, 0x26, 0xb1, 0xfb, 0x31, 0xd2, 0x80, + 0x09, 0x30, 0x90, 0x13, 0xf9, 0x59, 0x38, 0x27, 0x9a, 0x55, 0xc7, 0x6c, 0x06, 0x7e, 0x70, 0xd0, + 0xd8, 0x73, 0x02, 0x36, 0x71, 0x67, 0x9f, 0x4c, 0xc3, 0xca, 0x0e, 0x73, 0x25, 0x1f, 0x3b, 0xe4, + 0x8c, 0xac, 0x41, 0x5f, 0x30, 0x7f, 0x32, 0x35, 0xec, 0x64, 0x19, 0xc6, 0x05, 0xae, 0x58, 0x58, + 0xfb, 0xbf, 0xfe, 0x62, 0xe6, 0xd7, 0xc7, 0xc5, 0xd7, 0x2d, 0x49, 0x6f, 0xfe, 0x07, 0x03, 0x26, + 0xb5, 0x1e, 0x25, 0xb7, 0x14, 0x7f, 0x9e, 0xc4, 0x0f, 0x4f, 0xc3, 0xc9, 0x7c, 0x54, 0xe1, 0x96, + 0x70, 0xe2, 0xca, 0x0d, 0xa6, 0xcb, 0x4c, 0x33, 0x27, 0xb3, 0x39, 0xe4, 0x8f, 0xcf, 0xe6, 0x30, + 0x32, 0x20, 0x9b, 0xc3, 0x77, 0x26, 0x61, 0x4a, 0x5f, 0xc3, 0x99, 0x51, 0xbd, 0xe2, 0xef, 0x7a, + 0x1d, 0xb9, 0x35, 0xe7, 0xf9, 0x49, 0x10, 0xa2, 0xbd, 0x50, 0x80, 0x10, 0xf2, 0x2a, 0x40, 0x7c, + 0xfb, 0x2c, 0x77, 0xdf, 0xe2, 0x3d, 0x05, 0xa5, 0x80, 0xfc, 0x14, 0xc0, 0xaa, 0xef, 0xd2, 0x38, + 0xc5, 0xcd, 0x31, 0x67, 0x66, 0xaf, 0x89, 0x33, 0x33, 0xf1, 0x06, 0xc2, 0xd1, 0x61, 0xe5, 0x4c, + 0xc7, 0x77, 0x69, 0x7f, 0x6e, 0x1b, 0x85, 0x23, 0xf9, 0x22, 0x8c, 0x5a, 0xbd, 0x16, 0x95, 0x19, + 0x57, 0x26, 0xe4, 0x9c, 0xee, 0xb5, 0x94, 0xdc, 0xa8, 0x41, 0x2f, 0x7d, 0x55, 0xc2, 0x00, 0xe4, + 0x7d, 0x00, 0x26, 0xb6, 0x77, 0x02, 0xbf, 0xd7, 0x95, 0x21, 0xdd, 0xb8, 0x53, 0x57, 0x24, 0x7e, + 0x17, 0x0b, 0xd5, 0x8f, 0x27, 0x24, 0x64, 0x0d, 0xc6, 0x85, 0x86, 0x14, 0x57, 0x11, 0x2f, 0x65, + 0x1d, 0x82, 0x29, 0x66, 0x92, 0x48, 0x81, 0x82, 0x60, 0xfd, 0x5c, 0x8a, 0x9f, 0x34, 0xbc, 0x03, + 0x45, 0xc6, 0x7e, 0x33, 0xa4, 0x22, 0xd0, 0xbb, 0xc8, 0x1d, 0x2a, 0x95, 0x0a, 0xf5, 0x42, 0x7d, + 0xa3, 0x9f, 0x10, 0x90, 0xaf, 0x62, 0xd2, 0x22, 0xd1, 0xd5, 0xc7, 0x9e, 0xa5, 0x5e, 0xee, 0xeb, + 0xea, 0xd3, 0x4e, 0xb7, 0x9b, 0x91, 0xe5, 0x2d, 0xe6, 0x47, 0x76, 0xe3, 0xe0, 0xa9, 0x38, 0x41, + 0xf6, 0x31, 0x1f, 0xb8, 0xd6, 0xf7, 0x81, 0x39, 0x19, 0x0f, 0xd4, 0x9f, 0xaa, 0x48, 0xe3, 0x4b, + 0xba, 0x50, 0x4e, 0x94, 0x89, 0xf8, 0x16, 0x1c, 0xf7, 0xad, 0x37, 0xfa, 0xbe, 0xa5, 0x0e, 0x60, + 0xdf, 0xe7, 0xfa, 0xb8, 0x13, 0x37, 0x49, 0x66, 0x2c, 0xbe, 0x37, 0x71, 0xdc, 0xf7, 0x5e, 0xed, + 0xfb, 0xde, 0xac, 0xbb, 0xd3, 0xff, 0x9d, 0x14, 0x4f, 0xf2, 0x0e, 0x4c, 0x4a, 0x08, 0xce, 0x0f, + 0x3c, 0xc3, 0x14, 0x5b, 0x18, 0x77, 0x07, 0xbd, 0xe8, 0xf4, 0x3c, 0x3d, 0x2a, 0xb2, 0x4a, 0xcd, + 0xa5, 0x63, 0x52, 0xa3, 0x4e, 0x4b, 0x85, 0x8e, 0x4c, 0xbe, 0x02, 0x13, 0xcb, 0x6d, 0xd6, 0x10, + 0xbf, 0xe3, 0x44, 0x14, 0xd7, 0xdb, 0xe4, 0x5c, 0x58, 0x29, 0x51, 0x44, 0x95, 0x27, 0x04, 0x4d, + 0x8a, 0x54, 0x7b, 0x45, 0xa1, 0x60, 0x9d, 0xc7, 0x4f, 0x98, 0x84, 0x0c, 0x87, 0x62, 0x75, 0x7d, + 0x31, 0xe3, 0x6c, 0x56, 0x61, 0x8f, 0xcb, 0x15, 0x3f, 0xb8, 0xb2, 0xc5, 0x84, 0xd0, 0x3a, 0x4f, + 0xe7, 0x49, 0xde, 0x85, 0x09, 0x11, 0xaa, 0x5a, 0xb5, 0x56, 0xc3, 0xb9, 0x32, 0x36, 0x1e, 0x93, + 0xec, 0xc9, 0xa8, 0x56, 0xdb, 0x09, 0x52, 0x17, 0x74, 0x09, 0x3e, 0xf9, 0x32, 0x9c, 0xde, 0xf6, + 0x3a, 0xae, 0xff, 0x28, 0x14, 0x0a, 0x5c, 0x28, 0xba, 0x99, 0xc4, 0x69, 0xe9, 0x11, 0x2f, 0xb7, + 0xe5, 0x42, 0xd3, 0xa7, 0xf8, 0x32, 0x39, 0x90, 0x9f, 0xef, 0xe3, 0xcc, 0x25, 0x88, 0x1c, 0x27, + 0x41, 0x0b, 0x7d, 0x12, 0xd4, 0xff, 0xf9, 0xb4, 0x38, 0x65, 0x7e, 0x86, 0xf8, 0x40, 0x74, 0xb3, + 0xea, 0x03, 0xdf, 0xeb, 0xcc, 0xcd, 0x6a, 0xcf, 0xcf, 0xc4, 0x3e, 0xc4, 0x88, 0xb7, 0xee, 0xb7, + 0xbc, 0xe6, 0x41, 0xcd, 0x3c, 0x3a, 0xac, 0xbc, 0x94, 0x36, 0xd8, 0x3e, 0xf2, 0xb5, 0xf3, 0x93, + 0x0c, 0xd6, 0xe4, 0x2b, 0x50, 0x62, 0x7f, 0x63, 0xeb, 0xf6, 0xb4, 0x76, 0x9b, 0xa7, 0x60, 0x8a, + 0xef, 0xe0, 0x18, 0x61, 0x2c, 0x6d, 0x86, 0xe1, 0xab, 0xb1, 0x32, 0x7f, 0x68, 0xc0, 0xe9, 0xac, + 0xba, 0x9e, 0x90, 0xb8, 0xc8, 0x4c, 0xdd, 0xeb, 0xe3, 0xd1, 0x0b, 0xbf, 0xd7, 0x8f, 0x6f, 0xf3, + 0x2b, 0x30, 0x7a, 0xcf, 0xeb, 0xb8, 0xd2, 0x4b, 0x0d, 0x97, 0xc3, 0x7d, 0x06, 0xb0, 0x38, 0x9c, + 0x21, 0x60, 0x74, 0x01, 0xae, 0x97, 0xa3, 0x1c, 0x01, 0x43, 0x10, 0x2c, 0x0e, 0x67, 0x08, 0x6c, + 0xd9, 0x95, 0xcb, 0x04, 0x22, 0xb0, 0xd5, 0x38, 0xb4, 0x38, 0x9c, 0x5c, 0x86, 0xf1, 0xb5, 0xce, + 0x0a, 0x75, 0x1e, 0x52, 0x71, 0xa9, 0x86, 0x47, 0x45, 0x7e, 0xc7, 0x6e, 0x31, 0x98, 0x25, 0x0b, + 0xcd, 0xef, 0x19, 0x30, 0xd3, 0xd7, 0x4d, 0x27, 0xe7, 0x66, 0x3a, 0xfe, 0x06, 0x73, 0x98, 0xf6, + 0xf1, 0xea, 0x8f, 0x64, 0x57, 0xdf, 0xfc, 0xcb, 0x3c, 0x9c, 0x1b, 0xb0, 0x6a, 0x25, 0xde, 0x07, + 0xc6, 0x89, 0xde, 0x07, 0x5f, 0x63, 0xab, 0x84, 0xe3, 0xb5, 0xc3, 0x0d, 0x3f, 0xa9, 0x71, 0x72, + 0x51, 0x83, 0x65, 0x32, 0xf9, 0x89, 0x4c, 0xd4, 0x71, 0xbe, 0x89, 0x14, 0x76, 0xe4, 0xf7, 0x1d, + 0x8b, 0xeb, 0xcc, 0xfa, 0xee, 0xff, 0xf3, 0x3f, 0x26, 0xf7, 0xff, 0xfa, 0xad, 0xdb, 0xc8, 0x33, + 0xbd, 0x75, 0xcb, 0xbe, 0x07, 0x18, 0x7d, 0x9a, 0xdb, 0x8e, 0xff, 0x94, 0xf2, 0x38, 0xf8, 0x71, + 0x1c, 0xea, 0xab, 0x30, 0xba, 0xbd, 0x47, 0x03, 0x69, 0xdf, 0x62, 0x45, 0x1e, 0x31, 0x80, 0x5a, + 0x11, 0xc4, 0x30, 0x7f, 0x16, 0x4a, 0xea, 0xc7, 0x70, 0x2e, 0xb3, 0xdf, 0x62, 0x32, 0xf1, 0xb9, + 0xcc, 0x00, 0x16, 0x87, 0x9f, 0x98, 0xea, 0x2c, 0xe9, 0x85, 0xfc, 0x49, 0xbd, 0x60, 0xfe, 0x47, + 0x03, 0x46, 0x30, 0xd3, 0xc3, 0x9b, 0x50, 0x94, 0xa7, 0xc1, 0x6a, 0xf6, 0x83, 0x59, 0x79, 0x58, + 0x1c, 0xea, 0x2e, 0x1b, 0x02, 0xc8, 0x3e, 0xb5, 0x45, 0x83, 0x1d, 0xcd, 0xb3, 0xe7, 0x21, 0x03, + 0xa8, 0x9f, 0x42, 0x8c, 0x27, 0xe8, 0x12, 0xf4, 0x5e, 0x12, 0x9b, 0x39, 0x3e, 0xe1, 0xb9, 0xf7, + 0x52, 0xdf, 0xce, 0x4d, 0x62, 0x99, 0xbf, 0x61, 0xc0, 0x99, 0x4c, 0x3b, 0x80, 0x7d, 0x95, 0x1b, + 0x1c, 0x8a, 0x44, 0xa4, 0xad, 0x0d, 0x8e, 0xf1, 0x24, 0x5e, 0x4a, 0x4f, 0x30, 0xbc, 0x9f, 0x81, + 0x62, 0xbc, 0x3f, 0x23, 0xa7, 0xe5, 0xd0, 0xe1, 0x91, 0xa1, 0xdc, 0xcc, 0xfc, 0x8d, 0x01, 0x63, + 0xac, 0x0a, 0xcf, 0x6d, 0xb8, 0x49, 0xf6, 0x01, 0x32, 0x6b, 0xd2, 0x50, 0x41, 0x26, 0xbf, 0x3b, + 0x06, 0x90, 0x20, 0x93, 0x1d, 0x98, 0x5a, 0x5b, 0xae, 0x2f, 0x2e, 0xbb, 0xb4, 0x13, 0xe1, 0x45, + 0x66, 0x2a, 0x7d, 0x02, 0xdb, 0x58, 0x06, 0x1d, 0xa7, 0x25, 0x10, 0x0e, 0x92, 0xe9, 0xe9, 0x7b, + 0x6e, 0xd3, 0xf6, 0x62, 0x3a, 0xd5, 0x20, 0xd3, 0x39, 0xb2, 0x6f, 0x34, 0xaa, 0xf7, 0x57, 0x94, + 0x6f, 0xe4, 0x86, 0xfc, 0x46, 0xe8, 0xb4, 0x5b, 0x03, 0xbe, 0xa1, 0x73, 0x24, 0x7b, 0x50, 0xbe, + 0x83, 0xba, 0x5b, 0xf9, 0x4a, 0xfe, 0xf8, 0xaf, 0xbc, 0x2c, 0xbe, 0xf2, 0x02, 0x57, 0xfa, 0xd9, + 0xdf, 0xe9, 0xe3, 0x9a, 0x48, 0xee, 0xc8, 0x89, 0x92, 0xfb, 0xf7, 0x0c, 0x18, 0xe3, 0x8b, 0x43, + 0xfc, 0x02, 0x4b, 0xe6, 0xf2, 0xb3, 0xfd, 0x6c, 0x96, 0x9f, 0x72, 0x84, 0xff, 0xa9, 0x1b, 0x70, + 0x5e, 0x46, 0xea, 0xa9, 0xe7, 0x5c, 0xe4, 0x2d, 0x01, 0x1a, 0xa6, 0xbc, 0x24, 0xf1, 0xf5, 0xe2, + 0x2f, 0xb9, 0xa8, 0x5c, 0x38, 0x86, 0xfa, 0xb8, 0xe4, 0xf8, 0x53, 0x3e, 0x2e, 0xb9, 0x02, 0x45, + 0xe1, 0xbc, 0x54, 0x3b, 0x10, 0xdb, 0x4f, 0x79, 0xc0, 0x12, 0xc3, 0x95, 0xac, 0xe2, 0x1c, 0x64, + 0xef, 0x68, 0x39, 0x01, 0x63, 0x44, 0xb2, 0x06, 0xc5, 0x24, 0x56, 0xa6, 0xa8, 0x5d, 0xf5, 0xc6, + 0x70, 0xe1, 0xdd, 0xcb, 0xc3, 0x31, 0x33, 0x43, 0x63, 0x12, 0x1e, 0xe6, 0xb7, 0x0c, 0x28, 0xa7, + 0xe5, 0x85, 0xbc, 0x03, 0x13, 0x71, 0xb8, 0x52, 0xec, 0x42, 0x81, 0x67, 0xb5, 0x49, 0x7c, 0x93, + 0xe6, 0x4c, 0xa1, 0xa2, 0x93, 0x05, 0x28, 0xb0, 0x69, 0xa7, 0x24, 0x85, 0x46, 0x7d, 0xd2, 0x13, + 0x30, 0xf5, 0xea, 0x52, 0xe2, 0x29, 0xb3, 0xf6, 0xbf, 0xe4, 0x61, 0x42, 0x19, 0x2c, 0x72, 0x15, + 0x0a, 0xcb, 0xe1, 0x8a, 0xdf, 0xdc, 0xa7, 0xae, 0xb8, 0x11, 0xc1, 0xb7, 0x43, 0xbd, 0xd0, 0x6e, + 0x21, 0xd0, 0x8a, 0x8b, 0x49, 0x0d, 0x26, 0xf9, 0x7f, 0x32, 0x2c, 0x35, 0x97, 0x9c, 0xe6, 0x72, + 0x64, 0x19, 0x90, 0xaa, 0xae, 0xb0, 0x1a, 0x09, 0xf9, 0x3a, 0x00, 0x07, 0xb0, 0xf1, 0x1d, 0xc2, + 0x77, 0x59, 0x4e, 0xe0, 0x33, 0xe2, 0x03, 0x91, 0xa7, 0xb6, 0x10, 0x45, 0x41, 0x61, 0x88, 0xef, + 0x16, 0xfa, 0xcd, 0xfd, 0xe1, 0x5f, 0x2e, 0x4d, 0xde, 0x2d, 0xf4, 0x9b, 0xfb, 0x76, 0xb6, 0x23, + 0x9b, 0xca, 0x92, 0x7c, 0xdb, 0x80, 0x0b, 0x16, 0x6d, 0xfa, 0x0f, 0x69, 0x70, 0x50, 0x8d, 0x10, + 0x4b, 0xfd, 0xe2, 0xc9, 0x5e, 0x73, 0x37, 0xc5, 0x17, 0x5f, 0x0b, 0x04, 0x17, 0x8c, 0xcf, 0x69, + 0x77, 0x23, 0xfb, 0x98, 0x2a, 0x1c, 0xf3, 0x49, 0xf3, 0xcf, 0x0c, 0x65, 0x0a, 0x90, 0x55, 0x28, + 0xc6, 0xc2, 0x22, 0x0e, 0x1c, 0x63, 0xe3, 0x48, 0xc2, 0x2d, 0xfa, 0xa0, 0xf6, 0x82, 0xb8, 0xbc, + 0x98, 0x8d, 0x45, 0x4e, 0x9b, 0x11, 0x12, 0x48, 0xbe, 0x04, 0x23, 0x38, 0x54, 0x27, 0x67, 0x11, + 0x93, 0x4b, 0xcd, 0x08, 0x1b, 0x23, 0xac, 0x35, 0x52, 0x92, 0xcf, 0x09, 0x47, 0x96, 0xbc, 0x96, + 0x9f, 0x97, 0x81, 0x58, 0x3d, 0xe2, 0x35, 0x26, 0xf1, 0x9d, 0x54, 0xa4, 0xf5, 0x6f, 0x1b, 0x30, + 0xbb, 0xb9, 0x70, 0xdb, 0xa2, 0xbb, 0x1e, 0x86, 0xf5, 0x7a, 0x3e, 0xde, 0x32, 0x93, 0xf3, 0x90, + 0xb7, 0x9c, 0x47, 0x22, 0xdb, 0x27, 0x06, 0x4e, 0x04, 0xce, 0x23, 0x8b, 0xc1, 0xc8, 0xeb, 0x50, + 0xbc, 0x47, 0x0f, 0xee, 0x3a, 0x1d, 0x57, 0x3c, 0xe2, 0x51, 0xe2, 0x99, 0x61, 0xf6, 0xe9, 0x81, + 0xbd, 0x87, 0x50, 0x2b, 0x41, 0xc0, 0x2b, 0xf4, 0xde, 0xce, 0x3d, 0xca, 0x6f, 0x1a, 0x4b, 0xe2, + 0x0a, 0xbd, 0xb7, 0x83, 0xbe, 0xf9, 0xbc, 0xc4, 0xfc, 0x83, 0x3c, 0x94, 0xd3, 0xb3, 0x9f, 0xbc, + 0x0f, 0xa5, 0x75, 0x27, 0x0c, 0x1f, 0xf9, 0x81, 0x7b, 0xd7, 0x09, 0xf7, 0x44, 0x55, 0x70, 0xe3, + 0xd9, 0x15, 0x70, 0x7b, 0xcf, 0xd1, 0x12, 0xd4, 0x69, 0x04, 0xcc, 0x2a, 0xd8, 0x10, 0x71, 0x01, + 0xca, 0x2c, 0x8e, 0xfc, 0xa8, 0x9b, 0x4a, 0x3c, 0x2a, 0xd1, 0x88, 0x0b, 0xd3, 0xa9, 0xbe, 0x88, + 0x27, 0x50, 0x1c, 0xe0, 0x98, 0xee, 0x29, 0x7e, 0x92, 0xd7, 0x5b, 0x78, 0x80, 0xc9, 0xbb, 0x64, + 0x89, 0x96, 0x74, 0x41, 0x27, 0x22, 0x6f, 0x01, 0x6c, 0x2e, 0xdc, 0xc6, 0xfd, 0x27, 0x0d, 0x84, + 0x63, 0x39, 0x1e, 0xcb, 0x30, 0x26, 0x4d, 0x0e, 0x56, 0x77, 0x0d, 0x09, 0x32, 0x79, 0x13, 0xf2, + 0x3c, 0x64, 0x52, 0x4d, 0xab, 0x75, 0xff, 0x76, 0x95, 0x47, 0x99, 0x71, 0xaf, 0x02, 0xfd, 0x7a, + 0x86, 0xe1, 0x93, 0x15, 0x25, 0xe0, 0x6e, 0x4c, 0x4b, 0x2f, 0x24, 0xc1, 0x71, 0xef, 0x0f, 0x11, + 0x79, 0xf7, 0xfb, 0x79, 0x28, 0xc6, 0xdf, 0x24, 0x04, 0xd0, 0x12, 0x13, 0x2e, 0x01, 0xf8, 0x3f, + 0x39, 0x0f, 0x05, 0x69, 0x7c, 0x09, 0xb7, 0x80, 0xf1, 0x50, 0x18, 0x5e, 0x73, 0x20, 0xad, 0x2c, + 0x6e, 0x78, 0x59, 0xf2, 0x27, 0xb9, 0x01, 0xb1, 0x09, 0x35, 0xc8, 0xd6, 0x1a, 0x61, 0xa2, 0x6c, + 0xc5, 0x68, 0x64, 0x0a, 0x72, 0x1e, 0x77, 0x44, 0x2f, 0x5a, 0x39, 0xcf, 0x25, 0xef, 0x43, 0xc1, + 0x71, 0x5d, 0xea, 0xda, 0x4e, 0x34, 0xc4, 0xfb, 0xba, 0x05, 0xc6, 0x8d, 0xaf, 0x75, 0x48, 0x55, + 0x8d, 0x48, 0x15, 0x8a, 0xf8, 0xbc, 0x6a, 0x2f, 0x1c, 0xea, 0x4d, 0xd6, 0x84, 0x43, 0x81, 0x91, + 0x6d, 0x86, 0xd4, 0x25, 0xaf, 0xc1, 0x08, 0x13, 0x31, 0xb1, 0x52, 0xc6, 0x09, 0x12, 0xd7, 0x36, + 0xd6, 0x79, 0x87, 0xdd, 0x3d, 0x65, 0x21, 0x02, 0x79, 0x05, 0xf2, 0xbd, 0x85, 0x07, 0x62, 0x0d, + 0x2c, 0x27, 0x02, 0x16, 0xa3, 0xb1, 0x62, 0x72, 0x13, 0x0a, 0x8f, 0xf4, 0x58, 0xc9, 0x33, 0xa9, + 0xa1, 0x8b, 0xf1, 0x63, 0xc4, 0x5a, 0x01, 0xc6, 0x78, 0x64, 0xa2, 0xf9, 0x12, 0x40, 0xf2, 0xe9, + 0x7e, 0xef, 0x0d, 0xf3, 0xeb, 0x50, 0x8c, 0x3f, 0x49, 0x5e, 0x04, 0x65, 0x0e, 0xf3, 0xf9, 0x66, + 0x15, 0xf7, 0xe3, 0x99, 0x7c, 0x0e, 0xc6, 0xbb, 0x6c, 0x54, 0x65, 0x2e, 0x5f, 0x8b, 0x4d, 0x63, + 0x36, 0x6d, 0xe6, 0x60, 0x5c, 0x88, 0x2d, 0x0f, 0xaa, 0xb0, 0xe4, 0x4f, 0xf3, 0xb7, 0x73, 0x98, + 0x1d, 0x41, 0xa9, 0x27, 0x79, 0x19, 0x26, 0x9b, 0x01, 0xc5, 0x85, 0xda, 0x61, 0x06, 0xa3, 0xf8, + 0x4e, 0x29, 0x01, 0x2e, 0xbb, 0xe4, 0x32, 0x4c, 0x27, 0xc9, 0x85, 0xed, 0xe6, 0x8e, 0x88, 0x94, + 0x2e, 0x59, 0x93, 0x5d, 0x99, 0x5d, 0x78, 0x71, 0x07, 0x03, 0x28, 0xca, 0x6a, 0x54, 0x62, 0x24, + 0x13, 0x05, 0x17, 0xad, 0x69, 0x05, 0x8e, 0x17, 0x32, 0x67, 0x61, 0xcc, 0x71, 0x76, 0x7b, 0x1e, + 0x77, 0xe6, 0x2e, 0x59, 0xe2, 0x17, 0xf9, 0x2c, 0xcc, 0x84, 0xde, 0x6e, 0xc7, 0x89, 0x7a, 0x01, + 0x95, 0xb3, 0x0f, 0x45, 0x6a, 0xd2, 0x2a, 0xc7, 0x05, 0x72, 0xfe, 0xbd, 0x01, 0x44, 0xfd, 0x9e, + 0xbf, 0xf3, 0x11, 0x6d, 0x72, 0x51, 0x2b, 0x59, 0x33, 0x4a, 0xc9, 0x1a, 0x16, 0x90, 0xcf, 0x40, + 0x29, 0xa0, 0x21, 0x1a, 0xab, 0xd8, 0x6d, 0x98, 0x3c, 0xc8, 0x9a, 0x90, 0x30, 0xa6, 0xfa, 0x6a, + 0x30, 0xd3, 0x37, 0x07, 0xc9, 0x1b, 0x7c, 0xdf, 0x23, 0x2c, 0x97, 0x12, 0xdf, 0xe6, 0x31, 0xf5, + 0x9d, 0x7a, 0x91, 0x9b, 0x23, 0x99, 0x1d, 0x28, 0xa9, 0x2b, 0xcf, 0x09, 0x31, 0xe8, 0x67, 0xd1, + 0xab, 0x94, 0x6b, 0xc4, 0xb1, 0xa3, 0xc3, 0x4a, 0xce, 0x73, 0xd1, 0x97, 0xf4, 0x0a, 0x14, 0xa4, + 0xfd, 0xa4, 0xbe, 0x3c, 0x23, 0x4c, 0xed, 0x03, 0x2b, 0x2e, 0x35, 0x5f, 0x83, 0x71, 0xb1, 0xb8, + 0x1c, 0x7f, 0xc0, 0x65, 0x7e, 0x33, 0x07, 0xd3, 0x16, 0x65, 0x13, 0x5c, 0xbc, 0xe9, 0xf2, 0x29, + 0x4b, 0xb3, 0xac, 0xb5, 0xed, 0x98, 0x94, 0x0f, 0x7f, 0x68, 0xc0, 0x6c, 0x06, 0xee, 0xc7, 0xca, + 0xcb, 0x76, 0x0b, 0x8a, 0x75, 0xcf, 0x69, 0x55, 0x5d, 0x37, 0xf6, 0x8e, 0x45, 0x3b, 0xd9, 0x65, + 0xd3, 0xc9, 0x61, 0x50, 0xd5, 0xcc, 0x88, 0x51, 0xc9, 0x35, 0x21, 0x14, 0x49, 0xe6, 0x48, 0x99, + 0xc8, 0x19, 0x78, 0x9d, 0x92, 0x34, 0xce, 0x18, 0x83, 0xc8, 0x81, 0xc9, 0xad, 0xef, 0x73, 0x3b, + 0x74, 0xd9, 0x31, 0x88, 0xe9, 0xe6, 0x0d, 0xb5, 0x21, 0xff, 0x56, 0x0e, 0xce, 0x66, 0x13, 0x7e, + 0xdc, 0x14, 0x7b, 0x98, 0x6f, 0x43, 0xc9, 0x95, 0x8d, 0x86, 0x14, 0x4f, 0xce, 0x81, 0xf8, 0x09, + 0x02, 0x79, 0x00, 0x93, 0x2b, 0x4e, 0x18, 0xdd, 0xa5, 0x4e, 0x10, 0xed, 0x50, 0x27, 0x1a, 0xc2, + 0xb6, 0x8f, 0xdf, 0xbd, 0xc6, 0x45, 0x6d, 0x4f, 0x52, 0xa6, 0xdf, 0xbd, 0xd6, 0xd8, 0xc6, 0x82, + 0x32, 0x32, 0x84, 0xa0, 0x7c, 0x03, 0xa6, 0x1b, 0xb4, 0xed, 0x74, 0xf7, 0xfc, 0x80, 0x8a, 0x33, + 0xf9, 0xeb, 0x30, 0x19, 0x83, 0x32, 0xa5, 0x45, 0x2f, 0xd6, 0xf0, 0x95, 0x8e, 0x48, 0x54, 0x89, 0x5e, 0x6c, 0xfe, 0x66, 0x0e, 0xce, 0x55, 0x9b, 0xe2, 0x02, 0x43, 0x14, 0xc8, 0x7b, 0xd6, 0x4f, - 0xf8, 0xdb, 0x64, 0x1e, 0x8a, 0xf7, 0x9d, 0xc7, 0x2b, 0x14, 0x1f, 0x87, 0xe5, 0x39, 0x86, 0xb8, - 0xca, 0xe5, 0x3c, 0xb6, 0xe3, 0x03, 0x41, 0x2b, 0xc1, 0x51, 0xcd, 0xf0, 0x91, 0xa7, 0x34, 0xc3, - 0x4d, 0x18, 0xbb, 0xeb, 0xb7, 0x5c, 0xb1, 0x39, 0x89, 0xfb, 0x90, 0x3d, 0x84, 0x58, 0xa2, 0xc4, - 0xfc, 0xa1, 0x01, 0x53, 0x71, 0x8d, 0xb1, 0x0a, 0x9f, 0x78, 0x97, 0x5c, 0x86, 0x71, 0xfc, 0x50, - 0xfc, 0x34, 0x11, 0x6e, 0x1a, 0x2d, 0x06, 0xb2, 0x3d, 0xd7, 0x92, 0x85, 0x6a, 0x4f, 0x8c, 0x3e, - 0x5d, 0x4f, 0x98, 0xff, 0x14, 0xaf, 0x5a, 0xd4, 0x56, 0xb2, 0x9d, 0x48, 0xa9, 0x88, 0x31, 0x64, - 0x45, 0x72, 0xcf, 0x6c, 0x48, 0xf2, 0x03, 0x87, 0xe4, 0x97, 0x72, 0x30, 0x11, 0x57, 0xf6, 0x53, - 0x16, 0x3f, 0x1f, 0xb7, 0x6b, 0x28, 0xef, 0xf8, 0x86, 0x22, 0x2b, 0x84, 0x13, 0xfa, 0x97, 0x60, - 0x4c, 0x2c, 0x26, 0x23, 0x75, 0xdf, 0x98, 0x1a, 0xdd, 0xe4, 0xf9, 0x60, 0x1c, 0xd0, 0xd0, 0x12, - 0x74, 0x18, 0x7e, 0xb0, 0x4d, 0x77, 0xc4, 0xcd, 0xdb, 0x73, 0xbb, 0x47, 0x65, 0x87, 0x1f, 0x24, - 0x0d, 0x1b, 0x6a, 0x77, 0xfa, 0x47, 0x23, 0x50, 0x4e, 0x93, 0x9c, 0x9c, 0xa1, 0x60, 0xbd, 0xb7, - 0x23, 0x8c, 0x78, 0xb4, 0xf2, 0xbb, 0xbd, 0x1d, 0x8b, 0xc1, 0xc8, 0x65, 0x18, 0x59, 0x0f, 0xbc, - 0x87, 0xc2, 0x6a, 0x47, 0x0f, 0xb6, 0x6e, 0xe0, 0x3d, 0x54, 0xfd, 0x70, 0x59, 0x39, 0x5a, 0xd9, - 0x2b, 0x0d, 0xe5, 0xe5, 0x76, 0x6e, 0x65, 0xb7, 0xc2, 0x74, 0x12, 0x1b, 0x89, 0xc6, 0xb6, 0xca, - 0x1a, 0x75, 0x02, 0x11, 0x4d, 0x2f, 0xc4, 0x19, 0x6e, 0x95, 0x3b, 0x08, 0xe6, 0x99, 0x76, 0x2d, - 0x15, 0x89, 0xb4, 0x80, 0x28, 0x3f, 0xe5, 0x02, 0x3e, 0xd9, 0xc6, 0x93, 0x8f, 0x5a, 0x9d, 0x56, - 0x59, 0xdb, 0xea, 0x6a, 0xce, 0xe0, 0xfb, 0x2c, 0x4f, 0x4f, 0xd7, 0xa1, 0x88, 0x87, 0x81, 0x78, - 0xc4, 0x53, 0x38, 0x91, 0x99, 0xf4, 0x79, 0x06, 0xf4, 0x53, 0xb0, 0xe3, 0x83, 0x9e, 0x84, 0x09, - 0x79, 0x0f, 0x26, 0x54, 0x47, 0x5d, 0xee, 0x4e, 0x7a, 0x91, 0x47, 0x68, 0x0d, 0x48, 0xf6, 0xa6, - 0x12, 0x98, 0x9f, 0x53, 0x67, 0x89, 0xd8, 0xb4, 0x8f, 0x9d, 0x25, 0xe6, 0xaf, 0xa3, 0x1a, 0xdf, - 0xf6, 0x23, 0x2a, 0xb4, 0x97, 0xe7, 0x56, 0x8e, 0x25, 0x87, 0xeb, 0xa3, 0x9a, 0xaf, 0x8c, 0xd6, - 0xba, 0x27, 0x78, 0xb3, 0xfc, 0xf7, 0x0c, 0x38, 0x93, 0x49, 0x4b, 0xae, 0x03, 0x24, 0x3a, 0xa2, - 0xe8, 0x25, 0x9e, 0xc2, 0x38, 0x86, 0x5a, 0x0a, 0x06, 0xf9, 0x5a, 0x5a, 0xbb, 0x3b, 0x79, 0x73, - 0x92, 0x0f, 0x7d, 0x4c, 0xe9, 0xda, 0x5d, 0x86, 0x4e, 0x67, 0xfe, 0x61, 0x1e, 0x66, 0xfa, 0x1e, - 0x88, 0x3c, 0xc1, 0x3b, 0x61, 0x3f, 0xf5, 0xfc, 0x18, 0xbf, 0x08, 0xba, 0x36, 0xe8, 0x79, 0xca, - 0x8c, 0xc7, 0xc8, 0xf0, 0xac, 0x4e, 0x64, 0xcf, 0x3e, 0xe1, 0x4d, 0xb2, 0x30, 0xfb, 0xe1, 0xba, - 0xcf, 0x0e, 0xfc, 0xda, 0x33, 0x78, 0xc0, 0xee, 0xc7, 0xf8, 0x7d, 0xaf, 0x5f, 0xcf, 0xc1, 0x6c, - 0x5f, 0x9b, 0x9f, 0xdb, 0x55, 0xf7, 0x25, 0x6d, 0x77, 0x7b, 0x69, 0xd0, 0x98, 0x0e, 0xa5, 0x45, - 0xfc, 0x4f, 0x03, 0xce, 0x0d, 0xa0, 0x24, 0x07, 0xe9, 0x49, 0xc4, 0xb5, 0x8a, 0x1b, 0xc7, 0x7f, - 0xf0, 0x99, 0x4c, 0xa5, 0x4f, 0x6c, 0x26, 0x7c, 0x33, 0x07, 0xb0, 0x4d, 0x77, 0x9e, 0xef, 0xf4, - 0x4b, 0x5f, 0xd0, 0x26, 0x80, 0x72, 0x80, 0x39, 0x7c, 0xf6, 0xa5, 0x35, 0x3c, 0x48, 0x1c, 0x3e, - 0xf7, 0x52, 0xfc, 0x98, 0x49, 0x2e, 0xfb, 0x31, 0x13, 0x73, 0x07, 0x4e, 0xdf, 0xa1, 0x51, 0xb2, - 0x13, 0x4a, 0x1b, 0xf2, 0x78, 0xb6, 0xaf, 0x43, 0x51, 0xe0, 0xeb, 0x89, 0xe9, 0xa5, 0xab, 0x9d, - 0xe7, 0x5a, 0x09, 0x82, 0x49, 0xe1, 0x5c, 0x9d, 0xb6, 0x68, 0x44, 0x3f, 0xd9, 0xcf, 0x34, 0x80, - 0xf0, 0xa6, 0xf0, 0x37, 0x2e, 0x86, 0xfa, 0xc2, 0x89, 0xfd, 0xb3, 0x05, 0x67, 0xe2, 0xba, 0x3f, - 0x4b, 0xbe, 0xf3, 0x4c, 0x97, 0x10, 0xb1, 0x8e, 0x09, 0xc7, 0x63, 0x0e, 0x11, 0x1f, 0xc3, 0x05, - 0x49, 0xb0, 0xed, 0xc5, 0xd7, 0x43, 0x43, 0xd1, 0x92, 0x77, 0x60, 0x42, 0xa1, 0x11, 0x81, 0xd3, - 0x78, 0x0f, 0xfc, 0xc8, 0x8b, 0xf6, 0xec, 0x90, 0xc3, 0xd5, 0x7b, 0x60, 0x05, 0xdd, 0xfc, 0x2a, - 0xbc, 0x10, 0x7b, 0xf4, 0x64, 0x7c, 0x3a, 0xc5, 0xdc, 0x78, 0x32, 0xe6, 0xab, 0x49, 0xb3, 0x96, - 0x3b, 0xb1, 0x67, 0xbd, 0xe4, 0x4d, 0xd4, 0x66, 0x89, 0xc6, 0x5c, 0x54, 0xd2, 0xd2, 0x89, 0xbd, - 0x28, 0x01, 0x98, 0x6f, 0x2b, 0x95, 0xcd, 0x60, 0xa8, 0x11, 0x1b, 0x69, 0xe2, 0x6f, 0xe6, 0x60, - 0x7a, 0x6d, 0xb9, 0xbe, 0x18, 0x1f, 0x23, 0x7f, 0xca, 0x72, 0x43, 0x69, 0x6d, 0x1b, 0x2c, 0x6f, - 0xcc, 0x4d, 0x98, 0x4d, 0x75, 0x03, 0x3e, 0xe1, 0xf3, 0x1e, 0xf7, 0xbc, 0x89, 0xc1, 0x72, 0x67, - 0x39, 0x9b, 0xc5, 0x7e, 0xeb, 0xa6, 0x95, 0xc2, 0x36, 0xff, 0xd5, 0x78, 0x8a, 0xaf, 0x10, 0x61, - 0xaf, 0x43, 0x71, 0x39, 0x0c, 0x7b, 0x34, 0xd8, 0xb4, 0x56, 0x54, 0x1d, 0xd1, 0x43, 0xa0, 0xdd, - 0x0b, 0x5a, 0x56, 0x82, 0x40, 0xae, 0x42, 0x41, 0xc4, 0xd7, 0x49, 0x99, 0x80, 0x8e, 0x04, 0x71, - 0x78, 0x9e, 0x15, 0x17, 0x93, 0x37, 0xa1, 0xc4, 0xff, 0xe7, 0xb3, 0x4d, 0x74, 0x38, 0x9e, 0x55, - 0x09, 0x74, 0x3e, 0x3b, 0x2d, 0x0d, 0x8d, 0xbc, 0x06, 0x13, 0xf2, 0x8d, 0x50, 0x56, 0x23, 0x7e, - 0x02, 0x28, 0xa2, 0x3f, 0xd4, 0x12, 0x72, 0x0d, 0xf2, 0xd5, 0x45, 0x4b, 0x4d, 0x5c, 0xed, 0x34, - 0x03, 0x9e, 0xc0, 0x5e, 0x7b, 0x7b, 0xab, 0xba, 0x68, 0x91, 0x05, 0x28, 0xe0, 0xdb, 0x2a, 0x2e, - 0x0d, 0x84, 0x0b, 0x2d, 0x4e, 0x95, 0xae, 0x80, 0xa9, 0x57, 0x8c, 0x12, 0x8f, 0xcc, 0xc3, 0x78, - 0xdd, 0x0b, 0xbb, 0x2d, 0xe7, 0x40, 0x64, 0x82, 0xc1, 0x1b, 0x10, 0x97, 0x83, 0xd4, 0xc9, 0x25, - 0xb0, 0xc8, 0x55, 0x18, 0x6d, 0x34, 0xfd, 0x2e, 0x33, 0xb1, 0x62, 0x4f, 0x9f, 0x90, 0x01, 0xb4, - 0x74, 0x12, 0x0c, 0x80, 0x71, 0xde, 0x3c, 0x5c, 0xad, 0xa8, 0xc4, 0x79, 0xa7, 0xc3, 0xd4, 0x04, - 0x4e, 0xbf, 0x3b, 0x24, 0x3c, 0x4b, 0x77, 0xc8, 0x1d, 0x38, 0x77, 0x07, 0xf5, 0xfb, 0x06, 0x0d, - 0x30, 0x55, 0x27, 0x7f, 0xa7, 0x69, 0xd3, 0x5a, 0x16, 0x21, 0x7a, 0x18, 0x30, 0xc5, 0x4d, 0x00, - 0x3b, 0xe4, 0x38, 0xf2, 0x89, 0xa7, 0xd4, 0xe3, 0x14, 0x83, 0x18, 0x91, 0x2f, 0xc3, 0xe9, 0xac, - 0x22, 0x11, 0xac, 0x87, 0x4e, 0xf2, 0xd9, 0x1f, 0x50, 0xbd, 0xd4, 0xb3, 0x38, 0x90, 0x15, 0x28, - 0x73, 0x78, 0xd5, 0x6d, 0x7b, 0x9d, 0xa5, 0xb6, 0xe3, 0xb5, 0x30, 0x74, 0x4f, 0xc4, 0x5f, 0x0a, - 0xae, 0x0e, 0x2b, 0xb4, 0x29, 0x2b, 0xd5, 0x9c, 0xb5, 0x52, 0x94, 0xe4, 0x3b, 0x06, 0x94, 0x94, - 0x39, 0x16, 0x8a, 0x70, 0x86, 0x41, 0x0f, 0x7e, 0x6c, 0x3c, 0xa3, 0x07, 0x3f, 0x4a, 0xf2, 0xa5, - 0x5c, 0x5c, 0x6e, 0x5a, 0x0d, 0x50, 0x2c, 0x36, 0xaa, 0xf7, 0x57, 0x92, 0xb5, 0xfd, 0xe9, 0xba, - 0xbf, 0xd2, 0xda, 0x76, 0xcc, 0xfd, 0xd5, 0x26, 0xcc, 0xa6, 0xba, 0x41, 0x8a, 0x45, 0x0d, 0x9c, - 0x16, 0x8b, 0x29, 0x1a, 0x2b, 0x85, 0x6d, 0xfe, 0xe7, 0xb1, 0x14, 0x5f, 0x71, 0x66, 0x65, 0xc2, - 0x18, 0x97, 0x7a, 0x6a, 0xf6, 0x3a, 0x2e, 0x13, 0x2d, 0x51, 0x42, 0xce, 0x43, 0xbe, 0xd1, 0x58, - 0x53, 0x73, 0x6b, 0x86, 0xa1, 0x6f, 0x31, 0x18, 0x1b, 0x21, 0x3c, 0x8e, 0x52, 0x42, 0xe8, 0x9a, - 0x34, 0x88, 0xc4, 0x63, 0xb6, 0xaf, 0x26, 0xa2, 0x65, 0x24, 0xe9, 0x6f, 0x21, 0x5a, 0x12, 0x81, - 0xb2, 0x08, 0x73, 0xd5, 0x30, 0xa4, 0x41, 0xc4, 0x53, 0xf9, 0x87, 0xbd, 0x36, 0x0d, 0xc4, 0xf4, - 0x17, 0x62, 0x8f, 0x3f, 0x85, 0xdf, 0x0c, 0xad, 0x81, 0x88, 0xe4, 0x0a, 0x14, 0xaa, 0x3d, 0xd7, + 0xf8, 0xdb, 0x64, 0x1e, 0x8a, 0xf7, 0x9d, 0xc7, 0x2b, 0x14, 0x5f, 0xa6, 0xe5, 0x09, 0x8e, 0xb8, + 0xc9, 0xe5, 0x3c, 0xb6, 0xe3, 0x03, 0x41, 0x2b, 0xc1, 0x51, 0xb7, 0xe1, 0x23, 0x4f, 0xb9, 0x0d, + 0x37, 0x61, 0xec, 0xae, 0xdf, 0x72, 0xc5, 0xe2, 0x24, 0xee, 0x43, 0xf6, 0x10, 0x62, 0x89, 0x12, + 0xf3, 0x87, 0x06, 0x4c, 0xc5, 0x35, 0xc6, 0x2a, 0x7c, 0xe2, 0x5d, 0x72, 0x19, 0xc6, 0xf1, 0x43, + 0xf1, 0xbb, 0x48, 0xb8, 0x68, 0xb4, 0x18, 0xc8, 0xf6, 0x5c, 0x4b, 0x16, 0xaa, 0x3d, 0x31, 0xfa, + 0x74, 0x3d, 0x61, 0xfe, 0x73, 0xbc, 0x6a, 0x51, 0x5b, 0xc9, 0x56, 0x22, 0xa5, 0x22, 0xc6, 0x90, + 0x15, 0xc9, 0x3d, 0xb3, 0x21, 0xc9, 0x0f, 0x1c, 0x92, 0x5f, 0xca, 0xc1, 0x44, 0x5c, 0xd9, 0x4f, + 0x59, 0xf0, 0x7e, 0xdc, 0xae, 0xa1, 0x5c, 0xf3, 0x1b, 0x8a, 0xae, 0x10, 0x1e, 0xf0, 0x5f, 0x82, + 0x31, 0x31, 0x99, 0x8c, 0xd4, 0x7d, 0x63, 0x6a, 0x74, 0x93, 0xb7, 0x8b, 0x71, 0x40, 0x43, 0x4b, + 0xd0, 0x61, 0xec, 0xc3, 0x36, 0xdd, 0x11, 0x37, 0x6f, 0xcf, 0xed, 0x1a, 0x95, 0x1d, 0xfb, 0x90, + 0x34, 0x6c, 0xa8, 0xd5, 0xe9, 0x9f, 0x8c, 0x40, 0x39, 0x4d, 0x72, 0x72, 0x7a, 0x84, 0xf5, 0xde, + 0x8e, 0xd8, 0xc4, 0xe3, 0x2e, 0xbf, 0xdb, 0xdb, 0xb1, 0x18, 0x8c, 0x5c, 0x86, 0x91, 0xf5, 0xc0, + 0x7b, 0x28, 0x76, 0xed, 0xe8, 0x3e, 0xd7, 0x0d, 0xbc, 0x87, 0xaa, 0x13, 0x30, 0x2b, 0xc7, 0x5d, + 0xf6, 0x4a, 0x43, 0x79, 0x36, 0x9e, 0xef, 0xb2, 0x5b, 0x61, 0x3a, 0x83, 0x8e, 0x44, 0x63, 0x4b, + 0x65, 0x8d, 0x3a, 0x81, 0x08, 0xe5, 0x17, 0xea, 0x0c, 0x97, 0xca, 0x1d, 0x04, 0xf3, 0x34, 0xbf, + 0x96, 0x8a, 0x44, 0x5a, 0x40, 0x94, 0x9f, 0x72, 0x02, 0x9f, 0xbc, 0xc7, 0x93, 0x2f, 0x6a, 0x9d, + 0x56, 0x59, 0xdb, 0xea, 0x6c, 0xce, 0xe0, 0xfb, 0x2c, 0x4f, 0x4f, 0xd7, 0xa1, 0x88, 0x87, 0x81, + 0x78, 0xc4, 0x53, 0x38, 0x91, 0x99, 0x74, 0xb8, 0x06, 0xf4, 0x53, 0xb0, 0xe3, 0x83, 0x9e, 0x84, + 0x09, 0x79, 0x0f, 0x26, 0x54, 0x2f, 0x61, 0xee, 0xcb, 0x7a, 0x91, 0x87, 0x87, 0x0d, 0xc8, 0x34, + 0xa7, 0x12, 0x98, 0x9f, 0x53, 0xa5, 0x44, 0x2c, 0xda, 0xc7, 0x4a, 0x89, 0xf9, 0xeb, 0x68, 0xc6, + 0xb7, 0xfd, 0x88, 0x0a, 0xeb, 0xe5, 0xb9, 0xd5, 0x63, 0xc9, 0xe1, 0xfa, 0xa8, 0xe6, 0x2b, 0xa3, + 0xb5, 0xee, 0x09, 0x1e, 0x4c, 0xff, 0x3d, 0x03, 0xce, 0x64, 0xd2, 0x92, 0xeb, 0x00, 0x89, 0x8d, + 0x28, 0x7a, 0x89, 0xe7, 0x4f, 0x8e, 0xa1, 0x96, 0x82, 0x41, 0xbe, 0x96, 0xb6, 0xee, 0x4e, 0x5e, + 0x9c, 0xe4, 0x2b, 0x23, 0x53, 0xba, 0x75, 0x97, 0x61, 0xd3, 0x99, 0x7f, 0x98, 0x87, 0x99, 0xbe, + 0xd7, 0x29, 0x4f, 0xf0, 0x4e, 0xd8, 0x4f, 0xbd, 0x7d, 0xc6, 0x2f, 0x82, 0xae, 0x0d, 0x7a, 0x1b, + 0x33, 0xe3, 0x25, 0x34, 0x3c, 0xab, 0x13, 0xa9, 0xbb, 0x4f, 0x78, 0x10, 0x2d, 0xcc, 0x7e, 0x35, + 0xef, 0xb3, 0x03, 0xbf, 0xf6, 0x0c, 0x5e, 0xcf, 0xfb, 0x31, 0x7e, 0x5c, 0xec, 0xd7, 0x73, 0x30, + 0xdb, 0xd7, 0xe6, 0xe7, 0x76, 0xd6, 0x7d, 0x49, 0x5b, 0xdd, 0x5e, 0x1a, 0x34, 0xa6, 0x43, 0x59, + 0x11, 0xff, 0xdb, 0x80, 0x73, 0x03, 0x28, 0xc9, 0x41, 0x5a, 0x88, 0xb8, 0x55, 0x71, 0xe3, 0xf8, + 0x0f, 0x3e, 0x13, 0x51, 0xfa, 0xc4, 0x24, 0xe1, 0x9b, 0x39, 0x80, 0x6d, 0xba, 0xf3, 0x7c, 0xe7, + 0x7e, 0xfa, 0x82, 0x26, 0x00, 0xca, 0x01, 0xe6, 0xf0, 0xa9, 0x9f, 0xd6, 0xf0, 0x20, 0x71, 0xf8, + 0xc4, 0x4f, 0xf1, 0x4b, 0x2a, 0xb9, 0xec, 0x97, 0x54, 0xcc, 0x1d, 0x38, 0x7d, 0x87, 0x46, 0xc9, + 0x4a, 0x28, 0xf7, 0x90, 0xc7, 0xb3, 0x7d, 0x1d, 0x8a, 0x02, 0x5f, 0xcf, 0x8a, 0x2f, 0x5d, 0xed, + 0x3c, 0xd7, 0x4a, 0x10, 0x4c, 0x0a, 0xe7, 0xea, 0xb4, 0x45, 0x23, 0xfa, 0xc9, 0x7e, 0xa6, 0x01, + 0x84, 0x37, 0x85, 0x3f, 0xb0, 0x31, 0xd4, 0x17, 0x4e, 0xec, 0x9f, 0x2d, 0x38, 0x13, 0xd7, 0xfd, + 0x59, 0xf2, 0x9d, 0x67, 0xb6, 0x84, 0x08, 0xb4, 0x4c, 0x38, 0x1e, 0x73, 0x88, 0xf8, 0x18, 0x2e, + 0x48, 0x82, 0x6d, 0x2f, 0xbe, 0x1e, 0x1a, 0x8a, 0x96, 0xbc, 0x03, 0x13, 0x0a, 0x8d, 0x88, 0xda, + 0xc6, 0x7b, 0xe0, 0x47, 0x5e, 0xb4, 0x67, 0x87, 0x1c, 0xae, 0xde, 0x03, 0x2b, 0xe8, 0xe6, 0x57, + 0xe1, 0x85, 0xd8, 0xa3, 0x27, 0xe3, 0xd3, 0x29, 0xe6, 0xc6, 0x93, 0x31, 0x5f, 0x4d, 0x9a, 0xb5, + 0xdc, 0x89, 0x3d, 0xeb, 0x25, 0x6f, 0xa2, 0x36, 0x4b, 0x34, 0xe6, 0xa2, 0x92, 0x13, 0x4f, 0xac, + 0x45, 0x09, 0xc0, 0x7c, 0x5b, 0xa9, 0x6c, 0x06, 0x43, 0x8d, 0xd8, 0x48, 0x13, 0x7f, 0x33, 0x07, + 0xd3, 0x6b, 0xcb, 0xf5, 0xc5, 0xf8, 0x18, 0xf9, 0x53, 0x96, 0x98, 0x4a, 0x6b, 0xdb, 0x60, 0x7d, + 0x63, 0x6e, 0xc2, 0x6c, 0xaa, 0x1b, 0xf0, 0xfd, 0xa0, 0xf7, 0xb8, 0xe7, 0x4d, 0x0c, 0x96, 0x2b, + 0xcb, 0xd9, 0x2c, 0xf6, 0x5b, 0x37, 0xad, 0x14, 0xb6, 0xf9, 0x6f, 0xc6, 0x53, 0x7c, 0x85, 0x0a, + 0x7b, 0x1d, 0x8a, 0xcb, 0x61, 0xd8, 0xa3, 0xc1, 0xa6, 0xb5, 0xa2, 0xda, 0x88, 0x1e, 0x02, 0xed, + 0x5e, 0xd0, 0xb2, 0x12, 0x04, 0x72, 0x15, 0x0a, 0x22, 0xb8, 0x4f, 0xea, 0x04, 0x74, 0x24, 0x88, + 0x63, 0x03, 0xad, 0xb8, 0x98, 0xbc, 0x09, 0x25, 0xfe, 0x3f, 0x97, 0x36, 0xd1, 0xe1, 0x78, 0x56, + 0x25, 0xd0, 0xb9, 0x74, 0x5a, 0x1a, 0x1a, 0x79, 0x0d, 0x26, 0xe4, 0x03, 0xa5, 0xac, 0x46, 0xfc, + 0x04, 0x50, 0x44, 0x7f, 0xa8, 0x25, 0xe4, 0x1a, 0xe4, 0xab, 0x8b, 0x96, 0x9a, 0x35, 0xdb, 0x69, + 0x06, 0x3c, 0x7b, 0xbe, 0xf6, 0xf0, 0x57, 0x75, 0xd1, 0x22, 0x0b, 0x50, 0xc0, 0x87, 0x5d, 0x5c, + 0x1a, 0x08, 0x17, 0x5a, 0x14, 0x95, 0xae, 0x80, 0xa9, 0x57, 0x8c, 0x12, 0x8f, 0xcc, 0xc3, 0x78, + 0xdd, 0x0b, 0xbb, 0x2d, 0xe7, 0x40, 0xa4, 0xa1, 0xc1, 0x1b, 0x10, 0x97, 0x83, 0x54, 0xe1, 0x12, + 0x58, 0xe4, 0x2a, 0x8c, 0x36, 0x9a, 0x7e, 0x97, 0x6d, 0xb1, 0x62, 0x4f, 0x9f, 0x90, 0x01, 0xb4, + 0x5c, 0x16, 0x0c, 0x80, 0x41, 0xe6, 0x3c, 0x56, 0xae, 0xa8, 0x04, 0x99, 0xa7, 0x63, 0xe4, 0x04, + 0x4e, 0xbf, 0x3b, 0x24, 0x3c, 0x4b, 0x77, 0xc8, 0x1d, 0x38, 0x77, 0x07, 0xed, 0xfb, 0x06, 0x0d, + 0x30, 0x4f, 0x28, 0x7f, 0x24, 0x6a, 0xd3, 0x5a, 0x16, 0xf1, 0x81, 0x18, 0xad, 0xc5, 0xb7, 0x00, + 0x76, 0xc8, 0x71, 0xe4, 0xfb, 0x52, 0xa9, 0x97, 0x31, 0x06, 0x31, 0x22, 0x5f, 0x86, 0xd3, 0x59, + 0x45, 0x22, 0x52, 0x10, 0x9d, 0xe4, 0xb3, 0x3f, 0xa0, 0x7a, 0xa9, 0x67, 0x71, 0x20, 0x2b, 0x50, + 0xe6, 0xf0, 0xaa, 0xdb, 0xf6, 0x3a, 0x4b, 0x6d, 0xc7, 0x6b, 0x61, 0xdc, 0xa0, 0x08, 0xfe, 0x14, + 0x5c, 0x1d, 0x56, 0x68, 0x53, 0x56, 0xaa, 0x39, 0x6b, 0xa5, 0x28, 0xc9, 0x77, 0x0c, 0x28, 0x29, + 0x32, 0x16, 0x8a, 0x70, 0x86, 0x41, 0xaf, 0x8d, 0x6c, 0x3c, 0xa3, 0xd7, 0x46, 0x4a, 0xf2, 0x99, + 0x5e, 0x9c, 0x6e, 0x5a, 0x0d, 0x50, 0x2d, 0x36, 0xaa, 0xf7, 0x57, 0x92, 0xb9, 0xfd, 0xe9, 0xba, + 0xbf, 0xd2, 0xda, 0x76, 0xcc, 0xfd, 0xd5, 0x26, 0xcc, 0xa6, 0xba, 0x41, 0xaa, 0x45, 0x0d, 0x9c, + 0x56, 0x8b, 0x29, 0x1a, 0x2b, 0x85, 0x6d, 0xfe, 0xd7, 0xb1, 0x14, 0x5f, 0x71, 0x66, 0x65, 0xc2, + 0x18, 0xd7, 0x7a, 0x6a, 0xea, 0x3c, 0xae, 0x13, 0x2d, 0x51, 0x42, 0xce, 0x43, 0xbe, 0xd1, 0x58, + 0x53, 0x13, 0x7b, 0x86, 0xa1, 0x6f, 0x31, 0x18, 0x1b, 0x21, 0x3c, 0x8e, 0x52, 0x42, 0xe8, 0x9a, + 0x34, 0x88, 0xc4, 0x4b, 0xba, 0xaf, 0x26, 0xaa, 0x65, 0x24, 0xe9, 0x6f, 0xa1, 0x5a, 0x12, 0x85, + 0xb2, 0x08, 0x73, 0xd5, 0x30, 0xa4, 0x41, 0xc4, 0xdf, 0x11, 0x08, 0x7b, 0x6d, 0x1a, 0x08, 0xf1, + 0x17, 0x6a, 0x8f, 0xbf, 0xc3, 0xdf, 0x0c, 0xad, 0x81, 0x88, 0xe4, 0x0a, 0x14, 0xaa, 0x3d, 0xd7, 0xa3, 0x9d, 0xa6, 0x16, 0x3d, 0xe0, 0x08, 0x98, 0x15, 0x97, 0x92, 0x0f, 0xe1, 0x8c, 0x20, 0x92, - 0x32, 0x50, 0xf4, 0x00, 0x17, 0x7f, 0xdc, 0x92, 0x16, 0xcb, 0x53, 0x4a, 0x4e, 0x5b, 0x74, 0x49, - 0x36, 0x25, 0xa9, 0x42, 0x79, 0x09, 0xef, 0x6b, 0xe5, 0x93, 0xd6, 0x7e, 0x20, 0x52, 0x36, 0xa3, - 0x30, 0xe5, 0x77, 0xb9, 0xb6, 0x1b, 0x17, 0x5a, 0x7d, 0xe8, 0xe4, 0x1e, 0xcc, 0xa6, 0x61, 0x6c, - 0x5f, 0x28, 0x26, 0x4f, 0xce, 0xf5, 0x71, 0xc1, 0x55, 0x94, 0x45, 0x45, 0x76, 0x60, 0xa6, 0x1a, - 0x45, 0x81, 0xb7, 0xd3, 0x8b, 0x68, 0x4a, 0x9a, 0xca, 0x03, 0xcf, 0xb8, 0x5c, 0x4a, 0xd4, 0x17, - 0xc4, 0x64, 0x9c, 0x75, 0x62, 0xca, 0x58, 0xaa, 0x5a, 0xfd, 0xec, 0x88, 0x1b, 0xbf, 0x5a, 0x29, - 0x5e, 0x76, 0x14, 0x21, 0x5f, 0xf2, 0x60, 0xb9, 0x1a, 0x1e, 0xb4, 0xdb, 0x34, 0x0a, 0xd0, 0x83, - 0x00, 0x5f, 0x7e, 0x34, 0x85, 0x97, 0xd6, 0x05, 0xe5, 0xb1, 0x56, 0x7c, 0xdd, 0x53, 0x73, 0x60, - 0xd5, 0x78, 0x6a, 0x3b, 0x5a, 0x69, 0xc8, 0x1d, 0xad, 0x05, 0x33, 0x4b, 0x9d, 0x66, 0x70, 0x80, - 0xb1, 0xa7, 0xb2, 0x72, 0x93, 0x27, 0x54, 0x4e, 0x3e, 0xeb, 0x72, 0xd1, 0x91, 0x33, 0x2c, 0xab, - 0x7a, 0xfd, 0x8c, 0xcd, 0x9f, 0x83, 0x72, 0xba, 0x2f, 0x9f, 0xf2, 0xa9, 0xee, 0x27, 0x71, 0x9e, - 0x67, 0x23, 0x9d, 0x6e, 0x0b, 0x99, 0xd7, 0xde, 0x63, 0x36, 0x92, 0xec, 0x04, 0xca, 0xcb, 0xc9, - 0xda, 0x2b, 0xcc, 0x72, 0x19, 0xe7, 0xb2, 0x96, 0xb1, 0xf9, 0xcb, 0x39, 0x98, 0xe1, 0xfe, 0xbe, - 0xcf, 0xbf, 0xce, 0xfa, 0x9e, 0x26, 0x9c, 0xe5, 0x99, 0x64, 0xaa, 0x75, 0xc7, 0x68, 0xad, 0x5f, - 0x87, 0x33, 0x7d, 0x5d, 0x81, 0x02, 0xba, 0x2e, 0x3d, 0xad, 0xfb, 0x44, 0xf4, 0x5c, 0xf6, 0x47, - 0xb6, 0x6e, 0x5a, 0x7d, 0x14, 0xe6, 0x3f, 0xce, 0xf5, 0xf1, 0x17, 0xfa, 0xab, 0xaa, 0x91, 0x1a, - 0x4f, 0xa6, 0x91, 0xe6, 0x86, 0xd3, 0x48, 0x17, 0x74, 0x8d, 0x34, 0x9f, 0xdc, 0x15, 0x68, 0xdb, - 0xb6, 0xa6, 0x9c, 0x7e, 0x08, 0x93, 0x1b, 0xd4, 0x61, 0x4a, 0x96, 0x08, 0x07, 0x1c, 0xd1, 0xde, - 0x4a, 0x66, 0x65, 0x52, 0xbe, 0xc4, 0xa1, 0xc4, 0x11, 0x23, 0x60, 0xa2, 0x85, 0xc7, 0x07, 0x5a, - 0x3a, 0x07, 0x75, 0xd3, 0x18, 0x1d, 0xbc, 0x69, 0x98, 0xdf, 0xca, 0xc1, 0x84, 0xc2, 0x9e, 0x7c, - 0x1e, 0x4a, 0x6b, 0xc1, 0xae, 0xd3, 0xf1, 0x7e, 0xc6, 0x51, 0x8e, 0x81, 0xb1, 0xfa, 0xbe, 0x02, - 0xb7, 0x34, 0x2c, 0x74, 0xdf, 0xa1, 0x4e, 0x5b, 0x9d, 0xf8, 0xac, 0x7a, 0x16, 0x42, 0x95, 0x70, - 0xee, 0xfc, 0x10, 0xe1, 0xdc, 0x7a, 0x2c, 0xf4, 0xc8, 0x93, 0xc7, 0x42, 0x6b, 0xa1, 0xcb, 0xa3, - 0x4f, 0x18, 0xba, 0x6c, 0xfe, 0x6a, 0x0e, 0xca, 0xe2, 0x55, 0x61, 0x79, 0x88, 0xf9, 0xe9, 0x7a, - 0xbd, 0x43, 0x6f, 0xdc, 0x31, 0xd7, 0x74, 0x23, 0xbf, 0xf1, 0x3b, 0x15, 0x7c, 0x23, 0x36, 0xdd, - 0x1d, 0xf2, 0x8d, 0x58, 0x1d, 0x9e, 0x8e, 0xed, 0x48, 0x53, 0x59, 0x69, 0x7c, 0xf3, 0x4f, 0x73, - 0x69, 0xde, 0x42, 0x9b, 0x7a, 0x15, 0xc6, 0xf9, 0xa3, 0x70, 0xd2, 0xfd, 0x5c, 0xe4, 0xf0, 0x42, - 0x90, 0x25, 0xcb, 0x9e, 0x24, 0xca, 0xe7, 0xa4, 0x87, 0x82, 0xc9, 0x2d, 0x28, 0xa1, 0xdf, 0x4a, - 0xd5, 0x75, 0x03, 0x1a, 0x86, 0x42, 0xd1, 0xc2, 0x3b, 0xc4, 0x47, 0x74, 0xc7, 0xe6, 0xfe, 0x2d, - 0x8e, 0xeb, 0x06, 0x96, 0x86, 0x47, 0x16, 0xe1, 0xb4, 0xe6, 0x26, 0x25, 0xe9, 0x47, 0x93, 0xdd, - 0x22, 0xc2, 0x02, 0x4e, 0x9c, 0x89, 0xfc, 0xec, 0x1e, 0x49, 0x37, 0xff, 0x97, 0xc1, 0xd6, 0x5a, - 0x73, 0xff, 0x53, 0x16, 0x7f, 0xc4, 0x9a, 0x74, 0x8c, 0xb2, 0xff, 0x1f, 0x0d, 0x1e, 0x41, 0x20, - 0xa6, 0xcf, 0x5b, 0x30, 0xc6, 0x9f, 0xa0, 0x13, 0xbe, 0xee, 0x2a, 0x17, 0x5e, 0x90, 0xdc, 0x93, - 0xf1, 0x87, 0xec, 0x2c, 0x41, 0xc0, 0xac, 0x78, 0x3d, 0x90, 0x01, 0x15, 0xcf, 0xfe, 0x08, 0x06, - 0x89, 0xa5, 0xe6, 0xa7, 0x1d, 0x2e, 0xef, 0xb9, 0x71, 0x72, 0x7e, 0x5a, 0xf3, 0xff, 0xe6, 0x78, - 0x7b, 0x44, 0xa5, 0x86, 0x4d, 0xbc, 0x78, 0x19, 0x46, 0xf0, 0xb1, 0x63, 0x25, 0xbb, 0x65, 0xea, - 0xa1, 0x63, 0x2c, 0x67, 0xeb, 0x06, 0x65, 0xad, 0x1a, 0xf2, 0x86, 0xe2, 0x58, 0x5d, 0x37, 0x88, - 0x81, 0x59, 0xc5, 0x7d, 0x97, 0xaa, 0xcb, 0xa1, 0xa3, 0x27, 0x80, 0xc7, 0x72, 0x72, 0x4b, 0xf1, - 0xaf, 0x56, 0xcf, 0x58, 0xda, 0x0f, 0x1c, 0x9b, 0xfb, 0xf5, 0xaa, 0xd2, 0x36, 0x71, 0xc5, 0x5e, - 0x82, 0x29, 0x3d, 0x9a, 0x5c, 0x18, 0x1d, 0x18, 0x94, 0x9f, 0x8a, 0x44, 0x57, 0xd5, 0x5b, 0x9d, - 0x88, 0xd4, 0x60, 0x52, 0x0b, 0x19, 0x56, 0x93, 0xf1, 0xf2, 0x2c, 0x41, 0x76, 0x7f, 0xae, 0x0b, - 0x9d, 0x44, 0x39, 0xb8, 0xff, 0x1c, 0x94, 0xc5, 0xca, 0x8c, 0xa3, 0x0f, 0x51, 0xb5, 0x5b, 0xae, - 0x5b, 0xea, 0x6a, 0x6a, 0x7a, 0x6e, 0x60, 0x21, 0xd4, 0xfc, 0x9e, 0x01, 0xe7, 0xc5, 0x93, 0x74, - 0x16, 0x0d, 0x99, 0x0e, 0x89, 0x21, 0x8b, 0x38, 0x1f, 0x3f, 0x4f, 0xde, 0x91, 0x09, 0xc8, 0x74, - 0x01, 0x99, 0xfe, 0x46, 0x6d, 0x52, 0x4c, 0xca, 0x51, 0x4c, 0x41, 0x26, 0x13, 0x8f, 0xbd, 0x25, - 0x12, 0x8f, 0xe5, 0x8e, 0x27, 0x8e, 0xd7, 0x85, 0x4b, 0x3b, 0x32, 0xe1, 0xd8, 0x77, 0x73, 0x70, - 0x26, 0xa3, 0x5a, 0x5b, 0x9f, 0x7f, 0x4e, 0x85, 0x43, 0x4d, 0x13, 0x0e, 0x32, 0x33, 0xe5, 0xc0, - 0x8e, 0xcf, 0x94, 0x15, 0xbf, 0x65, 0xc0, 0x39, 0x7d, 0xf6, 0x08, 0x5b, 0x74, 0xeb, 0x26, 0x79, - 0x1b, 0xc6, 0xee, 0x52, 0xc7, 0xa5, 0x32, 0x48, 0x26, 0xce, 0xf2, 0x26, 0x4e, 0xa9, 0x79, 0x21, - 0x67, 0xfb, 0xa7, 0x7c, 0x29, 0x9f, 0xb2, 0x04, 0x09, 0xa9, 0x8b, 0xca, 0xf1, 0x6b, 0x32, 0x53, - 0xde, 0x18, 0x65, 0x7d, 0xea, 0x18, 0xc5, 0xf8, 0xf7, 0x0c, 0x78, 0xe1, 0x18, 0x1a, 0x36, 0x70, - 0x6c, 0xe8, 0xd5, 0x81, 0xc3, 0x8d, 0x05, 0xa1, 0xe4, 0x3d, 0x98, 0xde, 0x10, 0x47, 0x48, 0x72, - 0x38, 0x94, 0x2c, 0xff, 0xf2, 0x74, 0xc9, 0x96, 0xe3, 0x92, 0x46, 0x66, 0xd6, 0xff, 0x5d, 0x3f, - 0x8c, 0x3a, 0x49, 0x3a, 0x1f, 0xb4, 0xfe, 0xf7, 0x04, 0xcc, 0x8a, 0x4b, 0x99, 0x5a, 0xa0, 0x57, - 0x53, 0xb8, 0x65, 0xbc, 0x0c, 0x63, 0x0c, 0x27, 0xd6, 0xae, 0x71, 0x1e, 0xe0, 0x83, 0x69, 0x9e, - 0x6b, 0x89, 0xa2, 0xd8, 0xae, 0xcb, 0x65, 0xde, 0x9e, 0x7c, 0xcb, 0x80, 0xb2, 0xce, 0xfb, 0x69, - 0x87, 0xe6, 0x5d, 0x6d, 0x68, 0x5e, 0xc8, 0x1e, 0x9a, 0xc1, 0x63, 0xd2, 0x97, 0x59, 0x63, 0xa8, - 0xb1, 0x30, 0x61, 0xac, 0xee, 0xb7, 0x1d, 0xaf, 0xa3, 0x66, 0x83, 0x70, 0x11, 0x62, 0x89, 0x12, - 0xa5, 0xb7, 0xf2, 0x03, 0x7b, 0xcb, 0xfc, 0xce, 0x08, 0x9c, 0xe7, 0xd1, 0x38, 0x34, 0xd8, 0x0c, - 0xbd, 0xce, 0xae, 0x76, 0xb7, 0x65, 0xa6, 0x3a, 0x5c, 0x78, 0xf4, 0x31, 0x48, 0xdc, 0xdf, 0x57, - 0xa1, 0xc0, 0xa4, 0xb4, 0xd2, 0xe7, 0x68, 0xf4, 0x60, 0x4e, 0x23, 0x3e, 0xae, 0xb2, 0x98, 0x5c, - 0x13, 0x7b, 0x88, 0xe2, 0x73, 0xcd, 0xf6, 0x90, 0xd4, 0xbb, 0xf5, 0x7c, 0x1f, 0x89, 0x95, 0xaa, - 0x91, 0x01, 0x4a, 0xd5, 0x7d, 0x38, 0x5d, 0x75, 0xb9, 0x7c, 0x72, 0x5a, 0xeb, 0x81, 0xd7, 0x69, - 0x7a, 0x5d, 0xa7, 0x25, 0x95, 0x72, 0xfe, 0x8a, 0x7f, 0x5c, 0x6e, 0x77, 0x63, 0x04, 0x2b, 0x93, - 0x8c, 0x35, 0xa3, 0xbe, 0xda, 0xe0, 0x29, 0x6b, 0xc6, 0x90, 0x05, 0x36, 0xc3, 0xed, 0x84, 0x3c, - 0x67, 0x8d, 0x15, 0x17, 0xa3, 0x3a, 0x87, 0x61, 0x17, 0x1b, 0x2b, 0x8d, 0x7b, 0x22, 0x8c, 0x41, - 0xba, 0x84, 0xf1, 0x28, 0x8d, 0xa8, 0x15, 0xa2, 0xfd, 0xae, 0xe1, 0x25, 0x74, 0x8d, 0xc6, 0x5d, - 0x46, 0x57, 0xe8, 0xa3, 0x0b, 0xc3, 0x3d, 0x95, 0x8e, 0xe3, 0x91, 0x79, 0x00, 0xee, 0x54, 0x83, - 0x13, 0xa2, 0x98, 0x28, 0x7f, 0x01, 0x42, 0xb9, 0xf2, 0xa7, 0xa0, 0x90, 0x77, 0x60, 0x76, 0x69, - 0x71, 0x41, 0xc6, 0x27, 0xd4, 0xfd, 0x66, 0xaf, 0x4d, 0x3b, 0x11, 0xc6, 0xcb, 0x88, 0x80, 0x33, - 0xda, 0x5c, 0x60, 0xb3, 0x20, 0x0b, 0x4d, 0x44, 0x29, 0xf0, 0xe8, 0xbf, 0x45, 0xdf, 0xa5, 0xe1, - 0xd6, 0x8d, 0x4f, 0x59, 0x94, 0x82, 0xd2, 0x36, 0x5c, 0x6d, 0x37, 0x32, 0x57, 0xe6, 0xdf, 0xc3, - 0x28, 0x85, 0x3e, 0x5c, 0xf2, 0x13, 0x30, 0x8a, 0x3f, 0xc5, 0x8e, 0x3b, 0x9b, 0xc1, 0x36, 0xd9, - 0x6d, 0x9b, 0x3c, 0xfb, 0x08, 0x12, 0x90, 0xe5, 0xe4, 0x11, 0x83, 0x27, 0xf0, 0xb5, 0x15, 0x21, - 0xc4, 0xfa, 0xeb, 0x35, 0x2e, 0x94, 0xd4, 0x0f, 0xb2, 0x39, 0x72, 0xd7, 0x09, 0xf7, 0xa8, 0xbb, - 0x28, 0x5f, 0xab, 0x2c, 0xf1, 0x39, 0xb2, 0x87, 0x50, 0x7c, 0x59, 0xc7, 0x52, 0x50, 0x98, 0x74, - 0x58, 0x0e, 0x37, 0x43, 0x51, 0x15, 0x61, 0x05, 0x79, 0x68, 0xbd, 0xba, 0x96, 0x28, 0x42, 0x69, - 0x29, 0x73, 0x78, 0x06, 0x4e, 0x73, 0x9f, 0x06, 0x5b, 0x37, 0x3e, 0x09, 0x69, 0xa9, 0x7f, 0xe3, - 0x98, 0x31, 0xf9, 0x26, 0xc4, 0xc9, 0x73, 0x34, 0x64, 0xa6, 0x23, 0x26, 0x1e, 0x02, 0x46, 0xa2, - 0x23, 0x26, 0x1e, 0x02, 0xaa, 0x8e, 0x18, 0xa3, 0xc6, 0x19, 0x94, 0x73, 0x27, 0x64, 0x50, 0x1e, - 0x90, 0xb4, 0x5d, 0x3a, 0x97, 0x7e, 0x8a, 0xde, 0xaf, 0xf8, 0x22, 0x94, 0xaa, 0x51, 0xe4, 0x34, - 0xf7, 0xa8, 0x8b, 0x99, 0xba, 0x95, 0x3b, 0x4a, 0x47, 0xc0, 0x55, 0xb7, 0x35, 0x15, 0x57, 0x79, - 0xbf, 0x66, 0x7c, 0x88, 0xf7, 0x6b, 0xe6, 0x61, 0x7c, 0xb9, 0xf3, 0xd0, 0x63, 0x7d, 0x52, 0x48, - 0xd2, 0x77, 0x78, 0x1c, 0xa4, 0x3f, 0x7a, 0x82, 0x20, 0xf2, 0x96, 0xa2, 0x41, 0x14, 0x13, 0x55, - 0x5e, 0xbc, 0x17, 0x2e, 0x15, 0x09, 0xf5, 0xbc, 0x59, 0xa2, 0x93, 0x5b, 0x30, 0x2e, 0xad, 0x67, - 0x48, 0xd4, 0x77, 0x41, 0xe9, 0xf0, 0x12, 0x2d, 0x63, 0x88, 0xb0, 0x9e, 0xdf, 0xd1, 0xe3, 0x58, - 0x26, 0x94, 0xc8, 0x79, 0x25, 0x8e, 0x45, 0x8b, 0x9c, 0x57, 0x22, 0x5a, 0x62, 0x63, 0xa8, 0x74, - 0xa2, 0x31, 0xb4, 0x05, 0xa5, 0x75, 0x27, 0x88, 0x3c, 0xb6, 0x1d, 0x75, 0x22, 0x9e, 0xf8, 0x2c, - 0xb1, 0xd5, 0x95, 0xa2, 0xda, 0x4b, 0x32, 0x82, 0xbc, 0xab, 0xe0, 0xeb, 0x51, 0xbf, 0x09, 0x9c, - 0xac, 0x66, 0x78, 0x3a, 0x8a, 0x64, 0xa0, 0x78, 0x2b, 0xa9, 0x66, 0x9e, 0xe4, 0xa5, 0xea, 0x51, - 0x7a, 0xbf, 0x93, 0xe4, 0x4d, 0x3e, 0x06, 0x68, 0x33, 0x4e, 0x23, 0x1b, 0xcc, 0xff, 0x82, 0x7a, - 0x45, 0xca, 0x70, 0x8c, 0x11, 0xc9, 0xd7, 0xa0, 0xc4, 0xfe, 0xc7, 0x2c, 0x50, 0x1e, 0xe5, 0x89, - 0xcd, 0x12, 0xcf, 0x37, 0x7d, 0x41, 0xf3, 0x54, 0x51, 0x0d, 0x1a, 0xf1, 0x05, 0x8c, 0x8c, 0xd3, - 0x07, 0x2f, 0x1a, 0x37, 0xf2, 0x3e, 0x94, 0xd4, 0x2c, 0x72, 0x73, 0x33, 0xc9, 0x1d, 0x91, 0x2b, - 0xe0, 0xe9, 0x51, 0xd2, 0x08, 0xd8, 0xfe, 0x55, 0xed, 0x76, 0x91, 0x96, 0x28, 0xb3, 0xbd, 0xdb, - 0x4d, 0x93, 0x49, 0x34, 0xf2, 0x25, 0x28, 0x55, 0xbb, 0xdd, 0x44, 0xe2, 0xcc, 0x2a, 0x26, 0x61, - 0xb7, 0x6b, 0x67, 0x4a, 0x1d, 0x8d, 0x82, 0x4d, 0x2c, 0xa1, 0xf0, 0xe1, 0x77, 0x4f, 0x27, 0x13, - 0x4b, 0xe6, 0x46, 0x4b, 0x4f, 0x2c, 0x05, 0xdd, 0xfc, 0xa1, 0x01, 0xe7, 0x06, 0x74, 0xdb, 0xd0, - 0x8f, 0x75, 0xcd, 0x27, 0x7b, 0xb0, 0x72, 0x1c, 0x21, 0xf6, 0x60, 0xb5, 0xd1, 0x72, 0x37, 0xce, - 0xce, 0xc2, 0x96, 0xff, 0xc4, 0xb2, 0xb0, 0x99, 0x87, 0x06, 0x4c, 0x28, 0x93, 0xf9, 0x19, 0xbe, - 0xc1, 0x71, 0x59, 0xa4, 0x23, 0xcd, 0x27, 0x78, 0xa9, 0x87, 0xba, 0x79, 0xfa, 0xd1, 0xaf, 0x03, - 0xac, 0x38, 0x61, 0x54, 0x6d, 0x46, 0xde, 0x43, 0x3a, 0x84, 0xe4, 0x4e, 0xb2, 0x47, 0x38, 0x98, - 0x43, 0x98, 0x91, 0xf5, 0x65, 0x8f, 0x88, 0x19, 0x9a, 0xab, 0x30, 0xd6, 0xf0, 0x83, 0xa8, 0x76, - 0xc0, 0xb7, 0xe3, 0x3a, 0x0d, 0x9b, 0xea, 0xa1, 0xa4, 0x87, 0xc7, 0x13, 0x4d, 0x4b, 0x14, 0x31, - 0x9d, 0xf8, 0xb6, 0x47, 0x5b, 0xae, 0x7a, 0x29, 0xf5, 0x80, 0x01, 0x2c, 0x0e, 0xbf, 0xf6, 0x3e, - 0x4c, 0xcb, 0x89, 0xbd, 0xb1, 0xd2, 0xc0, 0x16, 0x4c, 0xc3, 0xc4, 0xd6, 0x92, 0xb5, 0x7c, 0xfb, - 0x2b, 0xf6, 0xed, 0xcd, 0x95, 0x95, 0xf2, 0x29, 0x32, 0x09, 0x45, 0x01, 0x58, 0xac, 0x96, 0x0d, - 0x52, 0x82, 0xc2, 0xf2, 0x6a, 0x63, 0x69, 0x71, 0xd3, 0x5a, 0x2a, 0xe7, 0xae, 0xbd, 0x0a, 0x53, - 0xc9, 0x95, 0x13, 0x86, 0xc7, 0x8e, 0x43, 0xde, 0xaa, 0x6e, 0x97, 0x4f, 0x11, 0x80, 0xb1, 0xf5, - 0x7b, 0x8b, 0x8d, 0x1b, 0x37, 0xca, 0xc6, 0xb5, 0xcf, 0x65, 0xbc, 0x8e, 0xce, 0x38, 0x35, 0x68, - 0xd7, 0x09, 0x9c, 0x88, 0xf2, 0xcf, 0xdc, 0xef, 0xb5, 0x22, 0xaf, 0xdb, 0xa2, 0x8f, 0xcb, 0xc6, - 0xb5, 0xb7, 0xfa, 0x1e, 0x39, 0x27, 0x67, 0x60, 0x66, 0x73, 0xb5, 0x7a, 0xbf, 0xb6, 0x7c, 0x67, - 0x73, 0x6d, 0xb3, 0x61, 0xdf, 0xaf, 0x6e, 0x2c, 0xde, 0x2d, 0x9f, 0x62, 0x15, 0xbe, 0xbf, 0xd6, - 0xd8, 0xb0, 0xad, 0xa5, 0xc5, 0xa5, 0xd5, 0x8d, 0xb2, 0x71, 0xed, 0x57, 0x0c, 0x98, 0xd2, 0x1f, - 0x82, 0x24, 0x97, 0xe0, 0xe2, 0x66, 0x63, 0xc9, 0xb2, 0x37, 0xd6, 0xee, 0x2d, 0xad, 0xda, 0x9b, - 0x8d, 0xea, 0x9d, 0x25, 0x7b, 0x73, 0xb5, 0xb1, 0xbe, 0xb4, 0xb8, 0x7c, 0x7b, 0x79, 0xa9, 0x5e, - 0x3e, 0x45, 0x2a, 0xf0, 0x82, 0x82, 0x61, 0x2d, 0x2d, 0xae, 0x6d, 0x2d, 0x59, 0xf6, 0x7a, 0xb5, - 0xd1, 0xd8, 0x5e, 0xb3, 0xea, 0x65, 0x83, 0x5c, 0x80, 0xb3, 0x19, 0x08, 0xf7, 0x6f, 0x57, 0xcb, - 0xb9, 0xbe, 0xb2, 0xd5, 0xa5, 0xed, 0xea, 0x8a, 0x5d, 0x5b, 0xdb, 0x28, 0xe7, 0xaf, 0xbd, 0xcf, - 0x14, 0xaf, 0xe4, 0xa5, 0x16, 0x52, 0x80, 0x91, 0xd5, 0xb5, 0xd5, 0xa5, 0xf2, 0x29, 0x32, 0x01, - 0xe3, 0xeb, 0x4b, 0xab, 0xf5, 0xe5, 0xd5, 0x3b, 0xbc, 0x5b, 0xab, 0xeb, 0xeb, 0xd6, 0xda, 0xd6, - 0x52, 0xbd, 0x9c, 0x63, 0x7d, 0x57, 0x5f, 0x5a, 0x65, 0x35, 0xcb, 0x5f, 0x33, 0x79, 0x76, 0x72, - 0x2d, 0xed, 0x2d, 0xeb, 0xad, 0xa5, 0x2f, 0x6f, 0x2c, 0xad, 0x36, 0x96, 0xd7, 0x56, 0xcb, 0xa7, - 0xae, 0x5d, 0x4c, 0xe1, 0xc8, 0x91, 0x68, 0x34, 0xee, 0x96, 0x4f, 0x5d, 0xfb, 0x1a, 0x94, 0x54, - 0xbd, 0x83, 0x9c, 0x83, 0x59, 0xf5, 0xf7, 0x3a, 0xed, 0xb8, 0x5e, 0x67, 0xb7, 0x7c, 0x2a, 0x5d, - 0x60, 0xf5, 0x3a, 0x1d, 0x56, 0x80, 0x8d, 0x57, 0x0b, 0x36, 0x68, 0xd0, 0xf6, 0x3a, 0x4c, 0xa5, - 0x28, 0xe7, 0x6a, 0xe5, 0xef, 0xff, 0xc5, 0x4b, 0xa7, 0xbe, 0xff, 0x83, 0x97, 0x8c, 0x3f, 0xfd, - 0xc1, 0x4b, 0xc6, 0x7f, 0xff, 0xc1, 0x4b, 0xc6, 0xce, 0x18, 0x4e, 0xf4, 0x9b, 0xff, 0x3f, 0x00, - 0x00, 0xff, 0xff, 0xaa, 0xc7, 0x23, 0x5f, 0x3d, 0xba, 0x00, 0x00, + 0x3a, 0x50, 0xf4, 0x00, 0x57, 0x7f, 0x7c, 0x27, 0x2d, 0xa6, 0xa7, 0xd4, 0x9c, 0xb6, 0xe8, 0x92, + 0x6c, 0x4a, 0x52, 0x85, 0xf2, 0x12, 0xde, 0xd7, 0xca, 0xf7, 0xb4, 0xfd, 0x40, 0xe4, 0x8b, 0x46, + 0x65, 0xca, 0xef, 0x72, 0x6d, 0x37, 0x2e, 0xb4, 0xfa, 0xd0, 0xc9, 0x3d, 0x98, 0x4d, 0xc3, 0xd8, + 0xba, 0x50, 0x4c, 0xde, 0xbb, 0xeb, 0xe3, 0x82, 0xb3, 0x28, 0x8b, 0x8a, 0xec, 0xc0, 0x4c, 0x35, + 0x8a, 0x02, 0x6f, 0xa7, 0x17, 0xd1, 0x94, 0x36, 0x95, 0x07, 0x9e, 0x71, 0xb9, 0xd4, 0xa8, 0x2f, + 0x08, 0x61, 0x9c, 0x75, 0x62, 0xca, 0x58, 0xab, 0x5a, 0xfd, 0xec, 0x88, 0x1b, 0x3f, 0x99, 0x29, + 0x9e, 0x95, 0x14, 0x21, 0x5f, 0xf2, 0x60, 0xb9, 0x1a, 0x1e, 0xb4, 0xdb, 0x34, 0x0a, 0xd0, 0x83, + 0x00, 0x9f, 0x9d, 0x34, 0x85, 0x97, 0xd6, 0x05, 0xe5, 0xa5, 0x58, 0x7c, 0x5a, 0x54, 0x73, 0x60, + 0xd5, 0x78, 0x6a, 0x2b, 0x5a, 0x69, 0xc8, 0x15, 0xad, 0x05, 0x33, 0x4b, 0x9d, 0x66, 0x70, 0x80, + 0xb1, 0xa7, 0xb2, 0x72, 0x93, 0x27, 0x54, 0x4e, 0xbe, 0x29, 0x73, 0xd1, 0x91, 0x12, 0x96, 0x55, + 0xbd, 0x7e, 0xc6, 0xe6, 0xcf, 0x41, 0x39, 0xdd, 0x97, 0x4f, 0xf9, 0x4e, 0xf8, 0x93, 0x38, 0xcf, + 0xb3, 0x91, 0x4e, 0xb7, 0x85, 0xcc, 0x6b, 0x8f, 0x41, 0x1b, 0x49, 0x6a, 0x04, 0xe5, 0xd9, 0x66, + 0xed, 0x09, 0x68, 0x39, 0x8d, 0x73, 0x59, 0xd3, 0xd8, 0xfc, 0xe5, 0x1c, 0xcc, 0x70, 0x7f, 0xdf, + 0xe7, 0xdf, 0x66, 0x7d, 0x4f, 0x53, 0xce, 0xf2, 0x4c, 0x32, 0xd5, 0xba, 0x63, 0xac, 0xd6, 0xaf, + 0xc3, 0x99, 0xbe, 0xae, 0x40, 0x05, 0x5d, 0x97, 0x9e, 0xd6, 0x7d, 0x2a, 0x7a, 0x2e, 0xfb, 0x23, + 0x5b, 0x37, 0xad, 0x3e, 0x0a, 0xf3, 0x9f, 0xe6, 0xfa, 0xf8, 0x0b, 0xfb, 0x55, 0xb5, 0x48, 0x8d, + 0x27, 0xb3, 0x48, 0x73, 0xc3, 0x59, 0xa4, 0x0b, 0xba, 0x45, 0x9a, 0x4f, 0xee, 0x0a, 0xb4, 0x65, + 0x5b, 0x33, 0x4e, 0x3f, 0x84, 0xc9, 0x0d, 0xea, 0x30, 0x23, 0x4b, 0x84, 0x03, 0x8e, 0x68, 0x0f, + 0x35, 0xb3, 0x32, 0xa9, 0x5f, 0xe2, 0x50, 0xe2, 0x88, 0x11, 0x30, 0xd5, 0xc2, 0xe3, 0x03, 0x2d, + 0x9d, 0x83, 0xba, 0x68, 0x8c, 0x0e, 0x5e, 0x34, 0xcc, 0x6f, 0xe5, 0x60, 0x42, 0x61, 0x4f, 0x3e, + 0x0f, 0xa5, 0xb5, 0x60, 0xd7, 0xe9, 0x78, 0x3f, 0xe3, 0x28, 0xc7, 0xc0, 0x58, 0x7d, 0x5f, 0x81, + 0x5b, 0x1a, 0x16, 0xba, 0xef, 0x50, 0xa7, 0xad, 0x0a, 0x3e, 0xab, 0x9e, 0x85, 0x50, 0x25, 0x9c, + 0x3b, 0x3f, 0x44, 0x38, 0xb7, 0x1e, 0x0b, 0x3d, 0xf2, 0xe4, 0xb1, 0xd0, 0x5a, 0xe8, 0xf2, 0xe8, + 0x13, 0x86, 0x2e, 0x9b, 0xbf, 0x9a, 0x83, 0xb2, 0x78, 0xd2, 0x58, 0x1e, 0x62, 0x7e, 0xba, 0x9e, + 0x0e, 0xd1, 0x1b, 0x77, 0xcc, 0x35, 0xdd, 0xc8, 0x6f, 0xfc, 0x4e, 0x05, 0x1f, 0xa8, 0x4d, 0x77, + 0x87, 0x7c, 0xa0, 0x56, 0x87, 0xa7, 0x63, 0x3b, 0xd2, 0x54, 0x56, 0x1a, 0xdf, 0xfc, 0xd3, 0x5c, + 0x9a, 0xb7, 0xb0, 0xa6, 0x5e, 0x85, 0x71, 0xfe, 0x22, 0x9d, 0x74, 0x3f, 0x17, 0x09, 0xc4, 0x10, + 0x64, 0xc9, 0xb2, 0x27, 0x89, 0xf2, 0x39, 0xe9, 0x95, 0x62, 0x72, 0x0b, 0x4a, 0xe8, 0xb7, 0x52, + 0x75, 0xdd, 0x80, 0x86, 0xa1, 0x30, 0xb4, 0xf0, 0x0e, 0xf1, 0x11, 0xdd, 0xb1, 0xb9, 0x7f, 0x8b, + 0xe3, 0xba, 0x81, 0xa5, 0xe1, 0x91, 0x45, 0x38, 0xad, 0xb9, 0x49, 0x49, 0xfa, 0xd1, 0x64, 0xb5, + 0x88, 0xb0, 0x80, 0x13, 0x67, 0x22, 0x3f, 0xbb, 0x17, 0xda, 0xcd, 0xbf, 0x32, 0xd8, 0x5c, 0x6b, + 0xee, 0x7f, 0xca, 0xe2, 0x8f, 0x58, 0x93, 0x8e, 0x31, 0xf6, 0xff, 0xb3, 0xc1, 0x23, 0x08, 0x84, + 0xf8, 0xbc, 0x05, 0x63, 0xfc, 0xfd, 0x3b, 0xe1, 0xeb, 0xae, 0x72, 0xe1, 0x05, 0xc9, 0x3d, 0x19, + 0x7f, 0x45, 0xcf, 0x12, 0x04, 0x6c, 0x17, 0xaf, 0x07, 0x32, 0xa0, 0xe1, 0xd9, 0x1f, 0xc1, 0x20, + 0xb1, 0xd4, 0xe4, 0xb8, 0xc3, 0x25, 0x5d, 0x37, 0x4e, 0x4e, 0x8e, 0x6b, 0xfe, 0xdf, 0x1c, 0x6f, + 0x8f, 0xa8, 0xd4, 0xb0, 0x59, 0x1f, 0x2f, 0xc3, 0x08, 0xbe, 0xb4, 0xac, 0xa4, 0xd6, 0x4c, 0xbd, + 0xb2, 0x8c, 0xe5, 0x6c, 0xde, 0xa0, 0xae, 0x55, 0x43, 0xde, 0x50, 0x1d, 0xab, 0xf3, 0x06, 0x31, + 0x30, 0xa5, 0xb9, 0xef, 0x52, 0x75, 0x3a, 0x74, 0xf4, 0xec, 0xf3, 0x58, 0x4e, 0x6e, 0x29, 0xfe, + 0xd5, 0xea, 0x19, 0x4b, 0xfb, 0x81, 0x63, 0x73, 0xbf, 0x5e, 0x55, 0xdb, 0x26, 0xae, 0xd8, 0x4b, + 0x30, 0xa5, 0x47, 0x93, 0x8b, 0x4d, 0x07, 0x06, 0xe5, 0xa7, 0x22, 0xd1, 0x55, 0xf3, 0x56, 0x27, + 0x22, 0x35, 0x98, 0xd4, 0x42, 0x86, 0xd5, 0x4c, 0xc0, 0x3c, 0x45, 0x91, 0xdd, 0x9f, 0xeb, 0x42, + 0x27, 0x51, 0x0e, 0xee, 0x3f, 0x07, 0x65, 0x31, 0x33, 0xe3, 0xe8, 0x43, 0x34, 0xed, 0x96, 0xeb, + 0x96, 0x3a, 0x9b, 0x9a, 0x9e, 0x1b, 0x58, 0x08, 0x35, 0xbf, 0x67, 0xc0, 0x79, 0xf1, 0x1e, 0x9e, + 0x45, 0x43, 0x66, 0x43, 0x62, 0xc8, 0x22, 0xca, 0xe3, 0xe7, 0xc9, 0x3b, 0x32, 0xfb, 0x99, 0xae, + 0x20, 0xd3, 0xdf, 0xa8, 0x4d, 0x0a, 0xa1, 0x1c, 0xc5, 0xfc, 0x67, 0x32, 0xeb, 0xd9, 0x5b, 0x22, + 0xeb, 0x59, 0xee, 0x78, 0xe2, 0x78, 0x5e, 0xb8, 0xb4, 0x23, 0xb3, 0x9d, 0x7d, 0x37, 0x07, 0x67, + 0x32, 0xaa, 0xb5, 0xf5, 0xf9, 0xe7, 0x54, 0x39, 0xd4, 0x34, 0xe5, 0x20, 0xd3, 0x62, 0x0e, 0xec, + 0xf8, 0x4c, 0x5d, 0xf1, 0x5b, 0x06, 0x9c, 0xd3, 0xa5, 0x47, 0xec, 0x45, 0xb7, 0x6e, 0x92, 0xb7, + 0x61, 0xec, 0x2e, 0x75, 0x5c, 0x2a, 0x83, 0x64, 0xe2, 0x14, 0x73, 0xe2, 0x94, 0x9a, 0x17, 0x72, + 0xb6, 0x7f, 0xca, 0xa7, 0xf2, 0x29, 0x4b, 0x90, 0x90, 0xba, 0xa8, 0x1c, 0xbf, 0x26, 0x33, 0xe5, + 0x8d, 0x51, 0xd6, 0xa7, 0x8e, 0x31, 0x8c, 0x7f, 0xcf, 0x80, 0x17, 0x8e, 0xa1, 0x61, 0x03, 0xc7, + 0x86, 0x5e, 0x1d, 0x38, 0x5c, 0x58, 0x10, 0x4a, 0xde, 0x83, 0xe9, 0x0d, 0x71, 0x84, 0x24, 0x87, + 0x43, 0x79, 0x62, 0x40, 0x9e, 0x2e, 0xd9, 0x72, 0x5c, 0xd2, 0xc8, 0x6c, 0xf7, 0x7f, 0xd7, 0x0f, + 0xa3, 0x4e, 0x92, 0xce, 0x07, 0x77, 0xff, 0x7b, 0x02, 0x66, 0xc5, 0xa5, 0xcc, 0x2c, 0xd0, 0xab, + 0x29, 0xdc, 0x32, 0x5e, 0x86, 0x31, 0x86, 0x13, 0x5b, 0xd7, 0x28, 0x07, 0xf8, 0x5a, 0x9b, 0xe7, + 0x5a, 0xa2, 0x28, 0xde, 0xd7, 0xe5, 0x32, 0x6f, 0x4f, 0xbe, 0x65, 0x40, 0x59, 0xe7, 0xfd, 0xb4, + 0x43, 0xf3, 0xae, 0x36, 0x34, 0x2f, 0x64, 0x0f, 0xcd, 0xe0, 0x31, 0xe9, 0xcb, 0xac, 0x31, 0xd4, + 0x58, 0x98, 0x30, 0x56, 0xf7, 0xdb, 0x8e, 0xd7, 0x51, 0xb3, 0x41, 0xb8, 0x08, 0xb1, 0x44, 0x89, + 0xd2, 0x5b, 0xf9, 0x81, 0xbd, 0x65, 0x7e, 0x67, 0x04, 0xce, 0xf3, 0x68, 0x1c, 0x1a, 0x6c, 0x86, + 0x5e, 0x67, 0x57, 0xbb, 0xdb, 0x32, 0x53, 0x1d, 0x2e, 0x3c, 0xfa, 0x18, 0x24, 0xee, 0xef, 0xab, + 0x50, 0x60, 0x5a, 0x5a, 0xe9, 0x73, 0xdc, 0xf4, 0x60, 0x4e, 0x23, 0x3e, 0xae, 0xb2, 0x98, 0x5c, + 0x13, 0x6b, 0x88, 0xe2, 0x73, 0xcd, 0xd6, 0x90, 0xd4, 0xa3, 0xf9, 0x7c, 0x1d, 0x89, 0x8d, 0xaa, + 0x91, 0x01, 0x46, 0xd5, 0x7d, 0x38, 0x5d, 0x75, 0xb9, 0x7e, 0x72, 0x5a, 0xeb, 0x81, 0xd7, 0x69, + 0x7a, 0x5d, 0xa7, 0x25, 0x8d, 0x72, 0x3c, 0x8d, 0x71, 0xe2, 0x72, 0xbb, 0x1b, 0x23, 0x58, 0x99, + 0x64, 0xac, 0x19, 0xf5, 0xd5, 0x06, 0x4f, 0x59, 0x33, 0x86, 0x2c, 0xb0, 0x19, 0x6e, 0x27, 0xe4, + 0x39, 0x6b, 0xac, 0xb8, 0x18, 0xcd, 0x39, 0x0c, 0xbb, 0xd8, 0x58, 0x69, 0xdc, 0x13, 0x61, 0x0c, + 0xd2, 0x25, 0x8c, 0x47, 0x69, 0x44, 0xad, 0x10, 0xf7, 0xef, 0x1a, 0x5e, 0x42, 0xd7, 0x68, 0xdc, + 0x65, 0x74, 0x85, 0x3e, 0xba, 0x30, 0xdc, 0x53, 0xe9, 0x38, 0x1e, 0x99, 0x07, 0xe0, 0x4e, 0x35, + 0x28, 0x10, 0xc5, 0xc4, 0xf8, 0x0b, 0x10, 0xca, 0x8d, 0x3f, 0x05, 0x85, 0xbc, 0x03, 0xb3, 0x4b, + 0x8b, 0x0b, 0x32, 0x3e, 0xa1, 0xee, 0x37, 0x7b, 0x6d, 0xda, 0x89, 0x30, 0x5e, 0x46, 0x04, 0x9c, + 0xd1, 0xe6, 0x02, 0x93, 0x82, 0x2c, 0x34, 0x11, 0xa5, 0xc0, 0xa3, 0xff, 0x16, 0x7d, 0x97, 0x86, + 0x5b, 0x37, 0x3e, 0x65, 0x51, 0x0a, 0x4a, 0xdb, 0x70, 0xb6, 0xdd, 0xc8, 0x9c, 0x99, 0xff, 0x00, + 0xa3, 0x14, 0xfa, 0x70, 0xc9, 0x4f, 0xc0, 0x28, 0xfe, 0x14, 0x2b, 0xee, 0x6c, 0x06, 0xdb, 0x64, + 0xb5, 0x6d, 0xf2, 0xec, 0x23, 0x48, 0x40, 0x96, 0x93, 0x17, 0x14, 0x9e, 0xc0, 0xd7, 0x56, 0x84, + 0x10, 0xeb, 0x4f, 0xe7, 0xb8, 0x50, 0x52, 0x3f, 0xc8, 0x64, 0xe4, 0xae, 0x13, 0xee, 0x51, 0x77, + 0x51, 0x3e, 0x95, 0x59, 0xe2, 0x32, 0xb2, 0x87, 0x50, 0x7c, 0xd6, 0xc7, 0x52, 0x50, 0x98, 0x76, + 0x58, 0x0e, 0x37, 0x43, 0x51, 0x15, 0xb1, 0x0b, 0xf2, 0x70, 0xf7, 0xea, 0x5a, 0xa2, 0x08, 0xb5, + 0xa5, 0x4c, 0x20, 0x1a, 0x38, 0xcd, 0x7d, 0x1a, 0x6c, 0xdd, 0xf8, 0x24, 0xb4, 0xa5, 0xfe, 0x8d, + 0x63, 0xc6, 0xe4, 0x9b, 0x10, 0x27, 0xcf, 0xd1, 0x90, 0x99, 0x8d, 0x98, 0x78, 0x08, 0x18, 0x89, + 0x8d, 0x98, 0x78, 0x08, 0xa8, 0x36, 0x62, 0x8c, 0x1a, 0xa7, 0x6f, 0xce, 0x9d, 0x90, 0xbe, 0x79, + 0x40, 0xc6, 0x78, 0xe9, 0x5c, 0xfa, 0x29, 0x7a, 0x3c, 0xe3, 0x8b, 0x50, 0xaa, 0x46, 0x91, 0xd3, + 0xdc, 0xa3, 0x2e, 0xa6, 0x09, 0x57, 0xee, 0x28, 0x1d, 0x01, 0x57, 0xdd, 0xd6, 0x54, 0x5c, 0xe5, + 0xf1, 0x9c, 0xf1, 0x21, 0x1e, 0xcf, 0x99, 0x87, 0xf1, 0xe5, 0xce, 0x43, 0x8f, 0xf5, 0x49, 0x21, + 0x49, 0xdf, 0xe1, 0x71, 0x90, 0xfe, 0xe2, 0x0a, 0x82, 0xc8, 0x5b, 0x8a, 0x05, 0x51, 0x4c, 0x4c, + 0x79, 0xf1, 0x58, 0xb9, 0x34, 0x24, 0xd4, 0xf3, 0x66, 0x89, 0x4e, 0x6e, 0xc1, 0xb8, 0xdc, 0x3d, + 0x43, 0x62, 0xbe, 0x0b, 0x4a, 0x87, 0x97, 0x68, 0x19, 0x43, 0xc4, 0xee, 0xf9, 0x1d, 0x3d, 0x8e, + 0x65, 0x42, 0x89, 0x9c, 0x57, 0xe2, 0x58, 0xb4, 0xc8, 0x79, 0x25, 0xa2, 0x25, 0xde, 0x0c, 0x95, + 0x4e, 0xdc, 0x0c, 0x6d, 0x41, 0x69, 0xdd, 0x09, 0x22, 0x8f, 0x2d, 0x47, 0x9d, 0x88, 0x27, 0x3e, + 0x4b, 0xf6, 0xea, 0x4a, 0x51, 0xed, 0x25, 0x19, 0x41, 0xde, 0x55, 0xf0, 0xf5, 0xa8, 0xdf, 0x04, + 0x4e, 0x56, 0x33, 0x3c, 0x1d, 0x45, 0x26, 0x52, 0xbc, 0x95, 0x54, 0xd3, 0x5e, 0xf2, 0x52, 0xf5, + 0x28, 0xbd, 0xdf, 0x49, 0xf2, 0x26, 0x1f, 0x03, 0xdc, 0x33, 0x4e, 0x23, 0x1b, 0xcc, 0xff, 0x82, + 0x76, 0x45, 0x6a, 0xe3, 0x18, 0x23, 0x92, 0xaf, 0x41, 0x89, 0xfd, 0x8f, 0x59, 0xa0, 0x3c, 0xca, + 0x13, 0x9b, 0x25, 0x9e, 0x6f, 0xfa, 0x84, 0xe6, 0xa9, 0xa2, 0x1a, 0x34, 0xe2, 0x13, 0x18, 0x19, + 0xa7, 0x0f, 0x5e, 0x34, 0x6e, 0xe4, 0x7d, 0x28, 0xa9, 0x59, 0xe4, 0xe6, 0x66, 0x92, 0x3b, 0x22, + 0x57, 0xc0, 0xd3, 0xa3, 0xa4, 0x11, 0xb0, 0xf5, 0xab, 0xda, 0xed, 0x22, 0x2d, 0x51, 0xa4, 0xbd, + 0xdb, 0x4d, 0x93, 0x49, 0x34, 0xf2, 0x25, 0x28, 0x55, 0xbb, 0xdd, 0x44, 0xe3, 0xcc, 0x2a, 0x5b, + 0xc2, 0x6e, 0xd7, 0xce, 0xd4, 0x3a, 0x1a, 0x05, 0x13, 0x2c, 0x61, 0xf0, 0xe1, 0x77, 0x4f, 0x27, + 0x82, 0x25, 0x73, 0xa3, 0xa5, 0x05, 0x4b, 0x41, 0x37, 0x7f, 0x68, 0xc0, 0xb9, 0x01, 0xdd, 0x36, + 0xf4, 0x4b, 0x61, 0xf3, 0xc9, 0x1a, 0xac, 0x1c, 0x47, 0x88, 0x35, 0x58, 0x6d, 0xb4, 0x5c, 0x8d, + 0xb3, 0xb3, 0xb0, 0xe5, 0x3f, 0xb1, 0x2c, 0x6c, 0xe6, 0xa1, 0x01, 0x13, 0x8a, 0x30, 0x3f, 0xc3, + 0x07, 0x40, 0x2e, 0x8b, 0x74, 0xa4, 0xf9, 0x04, 0x2f, 0xf5, 0x4a, 0x38, 0x4f, 0x3f, 0xfa, 0x75, + 0x80, 0x15, 0x27, 0x8c, 0xaa, 0xcd, 0xc8, 0x7b, 0x48, 0x87, 0xd0, 0xdc, 0x49, 0xf6, 0x08, 0x07, + 0x13, 0x18, 0x33, 0xb2, 0xbe, 0xec, 0x11, 0x31, 0x43, 0x73, 0x15, 0xc6, 0x1a, 0x7e, 0x10, 0xd5, + 0x0e, 0xf8, 0x72, 0x5c, 0xa7, 0x61, 0x53, 0x3d, 0x94, 0xf4, 0xf0, 0x78, 0xa2, 0x69, 0x89, 0x22, + 0x66, 0x13, 0xdf, 0xf6, 0x68, 0xcb, 0x55, 0x2f, 0xa5, 0x1e, 0x30, 0x80, 0xc5, 0xe1, 0xd7, 0xde, + 0x87, 0x69, 0x29, 0xd8, 0x1b, 0x2b, 0x0d, 0x6c, 0xc1, 0x34, 0x4c, 0x6c, 0x2d, 0x59, 0xcb, 0xb7, + 0xbf, 0x62, 0xdf, 0xde, 0x5c, 0x59, 0x29, 0x9f, 0x22, 0x93, 0x50, 0x14, 0x80, 0xc5, 0x6a, 0xd9, + 0x20, 0x25, 0x28, 0x2c, 0xaf, 0x36, 0x96, 0x16, 0x37, 0xad, 0xa5, 0x72, 0xee, 0xda, 0xab, 0x30, + 0x95, 0x5c, 0x39, 0x61, 0x78, 0xec, 0x38, 0xe4, 0xad, 0xea, 0x76, 0xf9, 0x14, 0x01, 0x18, 0x5b, + 0xbf, 0xb7, 0xd8, 0xb8, 0x71, 0xa3, 0x6c, 0x5c, 0xfb, 0x5c, 0xc6, 0xd3, 0xec, 0x8c, 0x53, 0x83, + 0x76, 0x9d, 0xc0, 0x89, 0x28, 0xff, 0xcc, 0xfd, 0x5e, 0x2b, 0xf2, 0xba, 0x2d, 0xfa, 0xb8, 0x6c, + 0x5c, 0x7b, 0xab, 0xef, 0x85, 0x75, 0x72, 0x06, 0x66, 0x36, 0x57, 0xab, 0xf7, 0x6b, 0xcb, 0x77, + 0x36, 0xd7, 0x36, 0x1b, 0xf6, 0xfd, 0xea, 0xc6, 0xe2, 0xdd, 0xf2, 0x29, 0x56, 0xe1, 0xfb, 0x6b, + 0x8d, 0x0d, 0xdb, 0x5a, 0x5a, 0x5c, 0x5a, 0xdd, 0x28, 0x1b, 0xd7, 0x7e, 0xc5, 0x80, 0x29, 0xfd, + 0x15, 0x4a, 0x72, 0x09, 0x2e, 0x6e, 0x36, 0x96, 0x2c, 0x7b, 0x63, 0xed, 0xde, 0xd2, 0xaa, 0xbd, + 0xd9, 0xa8, 0xde, 0x59, 0xb2, 0x37, 0x57, 0x1b, 0xeb, 0x4b, 0x8b, 0xcb, 0xb7, 0x97, 0x97, 0xea, + 0xe5, 0x53, 0xa4, 0x02, 0x2f, 0x28, 0x18, 0xd6, 0xd2, 0xe2, 0xda, 0xd6, 0x92, 0x65, 0xaf, 0x57, + 0x1b, 0x8d, 0xed, 0x35, 0xab, 0x5e, 0x36, 0xc8, 0x05, 0x38, 0x9b, 0x81, 0x70, 0xff, 0x76, 0xb5, + 0x9c, 0xeb, 0x2b, 0x5b, 0x5d, 0xda, 0xae, 0xae, 0xd8, 0xb5, 0xb5, 0x8d, 0x72, 0xfe, 0xda, 0xfb, + 0xcc, 0xf0, 0x4a, 0x9e, 0x89, 0x21, 0x05, 0x18, 0x59, 0x5d, 0x5b, 0x5d, 0x2a, 0x9f, 0x22, 0x13, + 0x30, 0xbe, 0xbe, 0xb4, 0x5a, 0x5f, 0x5e, 0xbd, 0xc3, 0xbb, 0xb5, 0xba, 0xbe, 0x6e, 0xad, 0x6d, + 0x2d, 0xd5, 0xcb, 0x39, 0xd6, 0x77, 0xf5, 0xa5, 0x55, 0x56, 0xb3, 0xfc, 0x35, 0x93, 0xa7, 0x46, + 0xd7, 0xd2, 0xde, 0xb2, 0xde, 0x5a, 0xfa, 0xf2, 0xc6, 0xd2, 0x6a, 0x63, 0x79, 0x6d, 0xb5, 0x7c, + 0xea, 0xda, 0xc5, 0x14, 0x8e, 0x1c, 0x89, 0x46, 0xe3, 0x6e, 0xf9, 0xd4, 0xb5, 0xaf, 0x41, 0x49, + 0xb5, 0x3b, 0xc8, 0x39, 0x98, 0x55, 0x7f, 0xaf, 0xd3, 0x8e, 0xeb, 0x75, 0x76, 0xcb, 0xa7, 0xd2, + 0x05, 0x56, 0xaf, 0xd3, 0x61, 0x05, 0xd8, 0x78, 0xb5, 0x60, 0x83, 0x06, 0x6d, 0xaf, 0xc3, 0x4c, + 0x8a, 0x72, 0xae, 0x56, 0xfe, 0xfe, 0x5f, 0xbc, 0x74, 0xea, 0xfb, 0x3f, 0x78, 0xc9, 0xf8, 0xd3, + 0x1f, 0xbc, 0x64, 0xfc, 0xcf, 0x1f, 0xbc, 0x64, 0xec, 0x8c, 0xa1, 0xa0, 0xdf, 0xfc, 0xff, 0x01, + 0x00, 0x00, 0xff, 0xff, 0x8f, 0x01, 0x48, 0x78, 0xba, 0xba, 0x00, 0x00, } func (m *KeepAlive) Marshal() (dAtA []byte, err error) { @@ -15103,6 +15109,20 @@ func (m *RoleOptions) MarshalToSizedBuffer(dAtA []byte) (int, error) { i -= len(m.XXX_unrecognized) copy(dAtA[i:], m.XXX_unrecognized) } + if m.DesktopDirectorySharing != nil { + { + size := m.DesktopDirectorySharing.Size() + i -= size + if _, err := m.DesktopDirectorySharing.MarshalTo(dAtA[i:]); err != nil { + return 0, err + } + i = encodeVarintTypes(dAtA, i, uint64(size)) + } + i-- + dAtA[i] = 0x1 + i-- + dAtA[i] = 0x9a + } if m.MaxKubernetesConnections != 0 { i = encodeVarintTypes(dAtA, i, uint64(m.MaxKubernetesConnections)) i-- @@ -16187,12 +16207,12 @@ func (m *UserSpecV2) MarshalToSizedBuffer(dAtA []byte) (int, error) { } i-- dAtA[i] = 0x42 - n119, err119 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) - if err119 != nil { - return 0, err119 + n120, err120 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) + if err120 != nil { + return 0, err120 } - i -= n119 - i = encodeVarintTypes(dAtA, i, uint64(n119)) + i -= n120 + i = encodeVarintTypes(dAtA, i, uint64(n120)) i-- dAtA[i] = 0x3a { @@ -16334,29 +16354,29 @@ func (m *LoginStatus) MarshalToSizedBuffer(dAtA []byte) (int, error) { i -= len(m.XXX_unrecognized) copy(dAtA[i:], m.XXX_unrecognized) } - n122, err122 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.RecoveryAttemptLockExpires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.RecoveryAttemptLockExpires):]) - if err122 != nil { - return 0, err122 - } - i -= n122 - i = encodeVarintTypes(dAtA, i, uint64(n122)) - i-- - dAtA[i] = 0x2a - n123, err123 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LockExpires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LockExpires):]) + n123, err123 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.RecoveryAttemptLockExpires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.RecoveryAttemptLockExpires):]) if err123 != nil { return 0, err123 } i -= n123 i = encodeVarintTypes(dAtA, i, uint64(n123)) i-- - dAtA[i] = 0x22 - n124, err124 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LockedTime, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LockedTime):]) + dAtA[i] = 0x2a + n124, err124 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LockExpires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LockExpires):]) if err124 != nil { return 0, err124 } i -= n124 i = encodeVarintTypes(dAtA, i, uint64(n124)) i-- + dAtA[i] = 0x22 + n125, err125 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LockedTime, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LockedTime):]) + if err125 != nil { + return 0, err125 + } + i -= n125 + i = encodeVarintTypes(dAtA, i, uint64(n125)) + i-- dAtA[i] = 0x1a if len(m.LockedMessage) > 0 { i -= len(m.LockedMessage) @@ -16412,12 +16432,12 @@ func (m *CreatedBy) MarshalToSizedBuffer(dAtA []byte) (int, error) { } i-- dAtA[i] = 0x1a - n126, err126 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Time, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Time):]) - if err126 != nil { - return 0, err126 + n127, err127 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Time, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Time):]) + if err127 != nil { + return 0, err127 } - i -= n126 - i = encodeVarintTypes(dAtA, i, uint64(n126)) + i -= n127 + i = encodeVarintTypes(dAtA, i, uint64(n127)) i-- dAtA[i] = 0x12 if m.Connector != nil { @@ -16600,21 +16620,21 @@ func (m *MFADevice) MarshalToSizedBuffer(dAtA []byte) (int, error) { } } } - n130, err130 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastUsed, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastUsed):]) - if err130 != nil { - return 0, err130 - } - i -= n130 - i = encodeVarintTypes(dAtA, i, uint64(n130)) - i-- - dAtA[i] = 0x3a - n131, err131 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.AddedAt, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.AddedAt):]) + n131, err131 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastUsed, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastUsed):]) if err131 != nil { return 0, err131 } i -= n131 i = encodeVarintTypes(dAtA, i, uint64(n131)) i-- + dAtA[i] = 0x3a + n132, err132 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.AddedAt, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.AddedAt):]) + if err132 != nil { + return 0, err132 + } + i -= n132 + i = encodeVarintTypes(dAtA, i, uint64(n132)) + i-- dAtA[i] = 0x32 if len(m.Id) > 0 { i -= len(m.Id) @@ -17210,12 +17230,12 @@ func (m *TunnelConnectionSpecV2) MarshalToSizedBuffer(dAtA []byte) (int, error) i-- dAtA[i] = 0x22 } - n140, err140 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastHeartbeat, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastHeartbeat):]) - if err140 != nil { - return 0, err140 + n141, err141 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastHeartbeat, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastHeartbeat):]) + if err141 != nil { + return 0, err141 } - i -= n140 - i = encodeVarintTypes(dAtA, i, uint64(n140)) + i -= n141 + i = encodeVarintTypes(dAtA, i, uint64(n141)) i-- dAtA[i] = 0x1a if len(m.ProxyName) > 0 { @@ -17307,12 +17327,12 @@ func (m *AcquireSemaphoreRequest) MarshalToSizedBuffer(dAtA []byte) (int, error) i-- dAtA[i] = 0x2a } - n141, err141 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) - if err141 != nil { - return 0, err141 + n142, err142 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) + if err142 != nil { + return 0, err142 } - i -= n141 - i = encodeVarintTypes(dAtA, i, uint64(n141)) + i -= n142 + i = encodeVarintTypes(dAtA, i, uint64(n142)) i-- dAtA[i] = 0x22 if m.MaxLeases != 0 { @@ -17361,12 +17381,12 @@ func (m *SemaphoreLease) MarshalToSizedBuffer(dAtA []byte) (int, error) { i -= len(m.XXX_unrecognized) copy(dAtA[i:], m.XXX_unrecognized) } - n142, err142 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) - if err142 != nil { - return 0, err142 + n143, err143 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) + if err143 != nil { + return 0, err143 } - i -= n142 - i = encodeVarintTypes(dAtA, i, uint64(n142)) + i -= n143 + i = encodeVarintTypes(dAtA, i, uint64(n143)) i-- dAtA[i] = 0x2a if len(m.LeaseID) > 0 { @@ -17424,12 +17444,12 @@ func (m *SemaphoreLeaseRef) MarshalToSizedBuffer(dAtA []byte) (int, error) { i-- dAtA[i] = 0x1a } - n143, err143 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) - if err143 != nil { - return 0, err143 + n144, err144 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) + if err144 != nil { + return 0, err144 } - i -= n143 - i = encodeVarintTypes(dAtA, i, uint64(n143)) + i -= n144 + i = encodeVarintTypes(dAtA, i, uint64(n144)) i-- dAtA[i] = 0x12 if len(m.LeaseID) > 0 { @@ -17648,29 +17668,29 @@ func (m *WebSessionSpecV2) MarshalToSizedBuffer(dAtA []byte) (int, error) { i-- dAtA[i] = 0x48 } - n148, err148 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LoginTime, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LoginTime):]) - if err148 != nil { - return 0, err148 - } - i -= n148 - i = encodeVarintTypes(dAtA, i, uint64(n148)) - i-- - dAtA[i] = 0x42 - n149, err149 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) + n149, err149 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LoginTime, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LoginTime):]) if err149 != nil { return 0, err149 } i -= n149 i = encodeVarintTypes(dAtA, i, uint64(n149)) i-- - dAtA[i] = 0x3a - n150, err150 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.BearerTokenExpires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.BearerTokenExpires):]) + dAtA[i] = 0x42 + n150, err150 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) if err150 != nil { return 0, err150 } i -= n150 i = encodeVarintTypes(dAtA, i, uint64(n150)) i-- + dAtA[i] = 0x3a + n151, err151 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.BearerTokenExpires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.BearerTokenExpires):]) + if err151 != nil { + return 0, err151 + } + i -= n151 + i = encodeVarintTypes(dAtA, i, uint64(n151)) + i-- dAtA[i] = 0x32 if len(m.BearerToken) > 0 { i -= len(m.BearerToken) @@ -17836,12 +17856,12 @@ func (m *RemoteClusterStatusV3) MarshalToSizedBuffer(dAtA []byte) (int, error) { i -= len(m.XXX_unrecognized) copy(dAtA[i:], m.XXX_unrecognized) } - n153, err153 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastHeartbeat, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastHeartbeat):]) - if err153 != nil { - return 0, err153 + n154, err154 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastHeartbeat, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastHeartbeat):]) + if err154 != nil { + return 0, err154 } - i -= n153 - i = encodeVarintTypes(dAtA, i, uint64(n153)) + i -= n154 + i = encodeVarintTypes(dAtA, i, uint64(n154)) i-- dAtA[i] = 0x12 if len(m.Connection) > 0 { @@ -19623,12 +19643,12 @@ func (m *LockSpecV2) MarshalToSizedBuffer(dAtA []byte) (int, error) { copy(dAtA[i:], m.XXX_unrecognized) } if m.Expires != nil { - n173, err173 := github_com_gogo_protobuf_types.StdTimeMarshalTo(*m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(*m.Expires):]) - if err173 != nil { - return 0, err173 + n174, err174 := github_com_gogo_protobuf_types.StdTimeMarshalTo(*m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(*m.Expires):]) + if err174 != nil { + return 0, err174 } - i -= n173 - i = encodeVarintTypes(dAtA, i, uint64(n173)) + i -= n174 + i = encodeVarintTypes(dAtA, i, uint64(n174)) i-- dAtA[i] = 0x1a } @@ -20309,12 +20329,12 @@ func (m *RecoveryCodesSpecV1) MarshalToSizedBuffer(dAtA []byte) (int, error) { i -= len(m.XXX_unrecognized) copy(dAtA[i:], m.XXX_unrecognized) } - n183, err183 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Created, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Created):]) - if err183 != nil { - return 0, err183 + n184, err184 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Created, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Created):]) + if err184 != nil { + return 0, err184 } - i -= n183 - i = encodeVarintTypes(dAtA, i, uint64(n183)) + i -= n184 + i = encodeVarintTypes(dAtA, i, uint64(n184)) i-- dAtA[i] = 0x12 if len(m.Codes) > 0 { @@ -20580,21 +20600,21 @@ func (m *SessionTrackerSpecV1) MarshalToSizedBuffer(dAtA []byte) (int, error) { i-- dAtA[i] = 0x32 } - n186, err186 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) - if err186 != nil { - return 0, err186 - } - i -= n186 - i = encodeVarintTypes(dAtA, i, uint64(n186)) - i-- - dAtA[i] = 0x2a - n187, err187 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Created, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Created):]) + n187, err187 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Expires, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Expires):]) if err187 != nil { return 0, err187 } i -= n187 i = encodeVarintTypes(dAtA, i, uint64(n187)) i-- + dAtA[i] = 0x2a + n188, err188 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.Created, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.Created):]) + if err188 != nil { + return 0, err188 + } + i -= n188 + i = encodeVarintTypes(dAtA, i, uint64(n188)) + i-- dAtA[i] = 0x22 if m.State != 0 { i = encodeVarintTypes(dAtA, i, uint64(m.State)) @@ -20697,12 +20717,12 @@ func (m *Participant) MarshalToSizedBuffer(dAtA []byte) (int, error) { i -= len(m.XXX_unrecognized) copy(dAtA[i:], m.XXX_unrecognized) } - n188, err188 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastActive, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastActive):]) - if err188 != nil { - return 0, err188 + n189, err189 := github_com_gogo_protobuf_types.StdTimeMarshalTo(m.LastActive, dAtA[i-github_com_gogo_protobuf_types.SizeOfStdTime(m.LastActive):]) + if err189 != nil { + return 0, err189 } - i -= n188 - i = encodeVarintTypes(dAtA, i, uint64(n188)) + i -= n189 + i = encodeVarintTypes(dAtA, i, uint64(n189)) i-- dAtA[i] = 0x22 if len(m.Mode) > 0 { @@ -23132,6 +23152,10 @@ func (m *RoleOptions) Size() (n int) { if m.MaxKubernetesConnections != 0 { n += 2 + sovTypes(uint64(m.MaxKubernetesConnections)) } + if m.DesktopDirectorySharing != nil { + l = m.DesktopDirectorySharing.Size() + n += 2 + l + sovTypes(uint64(l)) + } if m.XXX_unrecognized != nil { n += len(m.XXX_unrecognized) } @@ -42110,6 +42134,42 @@ func (m *RoleOptions) Unmarshal(dAtA []byte) error { break } } + case 19: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field DesktopDirectorySharing", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowTypes + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return ErrInvalidLengthTypes + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return ErrInvalidLengthTypes + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.DesktopDirectorySharing == nil { + m.DesktopDirectorySharing = &BoolOption{} + } + if err := m.DesktopDirectorySharing.Unmarshal(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex default: iNdEx = preIndex skippy, err := skipTypes(dAtA[iNdEx:]) diff --git a/api/types/types.proto b/api/types/types.proto index 82c0d3f7372c4..4aa5040caca26 100644 --- a/api/types/types.proto +++ b/api/types/types.proto @@ -1613,6 +1613,15 @@ message RoleOptions { // Kubernetes sessions a user may hold. int64 MaxKubernetesConnections = 18 [ (gogoproto.jsontag) = "max_kubernetes_connections,omitempty" ]; + + // DesktopDirectorySharing indicates whether directory sharing is allowed between the user's + // workstation and the remote desktop. It defaults to false unless explicitly set to + // true. + BoolValue DesktopDirectorySharing = 19 [ + (gogoproto.nullable) = true, + (gogoproto.jsontag) = "desktop_directory_sharing", + (gogoproto.customtype) = "BoolOption" + ]; } message RecordSession { diff --git a/lib/services/role.go b/lib/services/role.go index d053d467b5a2c..a47e6032d4393 100644 --- a/lib/services/role.go +++ b/lib/services/role.go @@ -686,6 +686,10 @@ type AccessChecker interface { // RecordDesktopSession returns true if a role in the role set has enabled // desktop session recoring. RecordDesktopSession() bool + // DesktopDirectorySharing returns true if the role set has directory sharing + // enabled. This setting is enabled if one or more of the roles in the set has + // enabled it. + DesktopDirectorySharing() bool // MaybeCanReviewRequests attempts to guess if this RoleSet belongs // to a user who should be submitting access reviews. Because not all rolesets @@ -2110,6 +2114,18 @@ func (set RoleSet) DesktopClipboard() bool { return true } +// DesktopDirectorySharing returns true if the role set has directory sharing +// enabled. This setting is enabled if one or more of the roles in the set has +// enabled it. +func (set RoleSet) DesktopDirectorySharing() bool { + for _, role := range set { + if !types.BoolDefaultTrue(role.GetOptions().DesktopDirectorySharing) { + return false + } + } + return true +} + // MaybeCanReviewRequests attempts to guess if this RoleSet belongs // to a user who should be submitting access reviews. Because not all rolesets // are derived from statically assigned roles, this may return false positives. diff --git a/lib/services/role_test.go b/lib/services/role_test.go index 0466d3cdc7654..2a758b9051e5f 100644 --- a/lib/services/role_test.go +++ b/lib/services/role_test.go @@ -190,12 +190,13 @@ func TestRoleParse(t *testing.T) { }, Spec: types.RoleSpecV5{ Options: types.RoleOptions{ - CertificateFormat: constants.CertificateFormatStandard, - MaxSessionTTL: types.NewDuration(apidefaults.MaxCertDuration), - PortForwarding: types.NewBoolOption(true), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, - BPF: apidefaults.EnhancedEvents(), - DesktopClipboard: types.NewBoolOption(true), + CertificateFormat: constants.CertificateFormatStandard, + MaxSessionTTL: types.NewDuration(apidefaults.MaxCertDuration), + PortForwarding: types.NewBoolOption(true), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, + BPF: apidefaults.EnhancedEvents(), + DesktopClipboard: types.NewBoolOption(true), + DesktopDirectorySharing: types.NewBoolOption(true), }, Allow: types.RoleConditions{ NodeLabels: types.Labels{}, @@ -223,12 +224,13 @@ func TestRoleParse(t *testing.T) { }, Spec: types.RoleSpecV5{ Options: types.RoleOptions{ - CertificateFormat: constants.CertificateFormatStandard, - MaxSessionTTL: types.NewDuration(apidefaults.MaxCertDuration), - PortForwarding: types.NewBoolOption(true), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, - BPF: apidefaults.EnhancedEvents(), - DesktopClipboard: types.NewBoolOption(true), + CertificateFormat: constants.CertificateFormatStandard, + MaxSessionTTL: types.NewDuration(apidefaults.MaxCertDuration), + PortForwarding: types.NewBoolOption(true), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, + BPF: apidefaults.EnhancedEvents(), + DesktopClipboard: types.NewBoolOption(true), + DesktopDirectorySharing: types.NewBoolOption(true), }, Allow: types.RoleConditions{ Namespaces: []string{apidefaults.Namespace}, @@ -254,7 +256,8 @@ func TestRoleParse(t *testing.T) { "client_idle_timeout": "17m", "disconnect_expired_cert": "yes", "enhanced_recording": ["command", "network"], - "desktop_clipboard": true + "desktop_clipboard": true, + "desktop_directory_sharing": true }, "allow": { "node_labels": {"a": "b", "c-d": "e"}, @@ -290,14 +293,15 @@ func TestRoleParse(t *testing.T) { }, Spec: types.RoleSpecV5{ Options: types.RoleOptions{ - CertificateFormat: constants.CertificateFormatStandard, - MaxSessionTTL: types.NewDuration(20 * time.Hour), - PortForwarding: types.NewBoolOption(true), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, - ClientIdleTimeout: types.NewDuration(17 * time.Minute), - DisconnectExpiredCert: types.NewBool(true), - BPF: apidefaults.EnhancedEvents(), - DesktopClipboard: types.NewBoolOption(true), + CertificateFormat: constants.CertificateFormatStandard, + MaxSessionTTL: types.NewDuration(20 * time.Hour), + PortForwarding: types.NewBoolOption(true), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, + ClientIdleTimeout: types.NewDuration(17 * time.Minute), + DisconnectExpiredCert: types.NewBool(true), + BPF: apidefaults.EnhancedEvents(), + DesktopClipboard: types.NewBoolOption(true), + DesktopDirectorySharing: types.NewBoolOption(true), }, Allow: types.RoleConditions{ NodeLabels: types.Labels{"a": []string{"b"}, "c-d": []string{"e"}}, @@ -341,7 +345,8 @@ func TestRoleParse(t *testing.T) { "client_idle_timeout": "never", "disconnect_expired_cert": "no", "enhanced_recording": ["command", "network"], - "desktop_clipboard": true + "desktop_clipboard": true, + "desktop_directory_sharing": true }, "allow": { "node_labels": {"a": "b"}, @@ -374,15 +379,16 @@ func TestRoleParse(t *testing.T) { }, Spec: types.RoleSpecV5{ Options: types.RoleOptions{ - CertificateFormat: constants.CertificateFormatStandard, - ForwardAgent: types.NewBool(true), - MaxSessionTTL: types.NewDuration(20 * time.Hour), - PortForwarding: types.NewBoolOption(true), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, - ClientIdleTimeout: types.NewDuration(0), - DisconnectExpiredCert: types.NewBool(false), - BPF: apidefaults.EnhancedEvents(), - DesktopClipboard: types.NewBoolOption(true), + CertificateFormat: constants.CertificateFormatStandard, + ForwardAgent: types.NewBool(true), + MaxSessionTTL: types.NewDuration(20 * time.Hour), + PortForwarding: types.NewBoolOption(true), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, + ClientIdleTimeout: types.NewDuration(0), + DisconnectExpiredCert: types.NewBool(false), + BPF: apidefaults.EnhancedEvents(), + DesktopClipboard: types.NewBoolOption(true), + DesktopDirectorySharing: types.NewBoolOption(true), }, Allow: types.RoleConditions{ NodeLabels: types.Labels{"a": []string{"b"}}, @@ -424,7 +430,8 @@ func TestRoleParse(t *testing.T) { "client_idle_timeout": "never", "disconnect_expired_cert": "no", "enhanced_recording": ["command", "network"], - "desktop_clipboard": true + "desktop_clipboard": true, + "desktop_directory_sharing": true }, "allow": { "node_labels": {"a": "b", "key": ["val"], "key2": ["val2", "val3"]}, @@ -446,15 +453,16 @@ func TestRoleParse(t *testing.T) { }, Spec: types.RoleSpecV5{ Options: types.RoleOptions{ - CertificateFormat: constants.CertificateFormatStandard, - ForwardAgent: types.NewBool(true), - MaxSessionTTL: types.NewDuration(20 * time.Hour), - PortForwarding: types.NewBoolOption(true), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, - ClientIdleTimeout: types.NewDuration(0), - DisconnectExpiredCert: types.NewBool(false), - BPF: apidefaults.EnhancedEvents(), - DesktopClipboard: types.NewBoolOption(true), + CertificateFormat: constants.CertificateFormatStandard, + ForwardAgent: types.NewBool(true), + MaxSessionTTL: types.NewDuration(20 * time.Hour), + PortForwarding: types.NewBoolOption(true), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, + ClientIdleTimeout: types.NewDuration(0), + DisconnectExpiredCert: types.NewBool(false), + BPF: apidefaults.EnhancedEvents(), + DesktopClipboard: types.NewBoolOption(true), + DesktopDirectorySharing: types.NewBoolOption(true), }, Allow: types.RoleConditions{ NodeLabels: types.Labels{ @@ -2380,48 +2388,55 @@ func TestExtractFrom(t *testing.T) { // port forwarding) can be disabled in a role. func TestBoolOptions(t *testing.T) { var tests = []struct { - inOptions types.RoleOptions - outCanPortForward bool - outCanForwardAgents bool - outRecordDesktopSessions bool - outDesktopClipboard bool + inOptions types.RoleOptions + outCanPortForward bool + outCanForwardAgents bool + outRecordDesktopSessions bool + outDesktopClipboard bool + outDesktopDirectorySharing bool }{ // Setting options explicitly off should remain off. { inOptions: types.RoleOptions{ - ForwardAgent: types.NewBool(false), - PortForwarding: types.NewBoolOption(false), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(false)}, - DesktopClipboard: types.NewBoolOption(false), - }, - outCanPortForward: false, - outCanForwardAgents: false, - outRecordDesktopSessions: false, - outDesktopClipboard: false, + ForwardAgent: types.NewBool(false), + PortForwarding: types.NewBoolOption(false), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(false)}, + DesktopClipboard: types.NewBoolOption(false), + DesktopDirectorySharing: types.NewBoolOption(false), + }, + outCanPortForward: false, + outCanForwardAgents: false, + outRecordDesktopSessions: false, + outDesktopClipboard: false, + outDesktopDirectorySharing: false, }, // Not setting options should set port forwarding to true (default enabled), // agent forwarding false (default disabled), // desktop session recording to true (default enabled), - // and desktop clipboard sharing to true (default enabled). + // desktop clipboard sharing to true (default enabled), + // and desktop directory sharing to true (default enabled). { - inOptions: types.RoleOptions{}, - outCanPortForward: true, - outCanForwardAgents: false, - outRecordDesktopSessions: true, - outDesktopClipboard: true, + inOptions: types.RoleOptions{}, + outCanPortForward: true, + outCanForwardAgents: false, + outRecordDesktopSessions: true, + outDesktopClipboard: true, + outDesktopDirectorySharing: true, }, // Explicitly enabling should enable them. { inOptions: types.RoleOptions{ - ForwardAgent: types.NewBool(true), - PortForwarding: types.NewBoolOption(true), - RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, - DesktopClipboard: types.NewBoolOption(true), - }, - outCanPortForward: true, - outCanForwardAgents: true, - outRecordDesktopSessions: true, - outDesktopClipboard: true, + ForwardAgent: types.NewBool(true), + PortForwarding: types.NewBoolOption(true), + RecordSession: &types.RecordSession{Desktop: types.NewBoolOption(true)}, + DesktopClipboard: types.NewBoolOption(true), + DesktopDirectorySharing: types.NewBoolOption(true), + }, + outCanPortForward: true, + outCanForwardAgents: true, + outRecordDesktopSessions: true, + outDesktopClipboard: true, + outDesktopDirectorySharing: true, }, } for _, tt := range tests { @@ -2440,6 +2455,7 @@ func TestBoolOptions(t *testing.T) { require.Equal(t, tt.outCanForwardAgents, set.CanForwardAgents()) require.Equal(t, tt.outRecordDesktopSessions, set.RecordDesktopSession()) require.Equal(t, tt.outDesktopClipboard, set.DesktopClipboard()) + require.Equal(t, tt.outDesktopDirectorySharing, set.DesktopDirectorySharing()) } } @@ -3691,8 +3707,8 @@ func TestDesktopClipboard(t *testing.T) { } { t.Run(test.desc, func(t *testing.T) { var roles []types.Role - for _, r := range test.roles { - roles = append(roles, &r) + for i := range test.roles { + roles = append(roles, &test.roles[i]) } rs := NewRoleSet(roles...) require.Equal(t, test.hasClipboard, rs.DesktopClipboard()) @@ -3700,6 +3716,56 @@ func TestDesktopClipboard(t *testing.T) { } } +func TestDesktopDirectorySharing(t *testing.T) { + for _, test := range []struct { + desc string + roles []types.RoleV5 + hasDirectorySharing bool + }{ + { + desc: "single role, unspecified, defaults true", + roles: []types.RoleV5{newRole(func(r *types.RoleV5) {})}, + hasDirectorySharing: true, + }, + { + desc: "single role, explicitly disabled", + roles: []types.RoleV5{ + newRole(func(r *types.RoleV5) { + r.SetOptions(types.RoleOptions{ + DesktopDirectorySharing: types.NewBoolOption(false), + }) + }), + }, + hasDirectorySharing: false, + }, + { + desc: "multiple conflicting roles, disable wins", + roles: []types.RoleV5{ + newRole(func(r *types.RoleV5) { + r.SetOptions(types.RoleOptions{ + DesktopDirectorySharing: types.NewBoolOption(false), + }) + }), + newRole(func(r *types.RoleV5) { + r.SetOptions(types.RoleOptions{ + DesktopDirectorySharing: types.NewBoolOption(true), + }) + }), + }, + hasDirectorySharing: false, + }, + } { + t.Run(test.desc, func(t *testing.T) { + roles := []types.Role{} + for i := range test.roles { + roles = append(roles, &test.roles[i]) + } + rs := NewRoleSet(roles...) + require.Equal(t, test.hasDirectorySharing, rs.DesktopDirectorySharing()) + }) + } +} + func TestCheckAccessToWindowsDesktop(t *testing.T) { desktopNoLabels := &types.WindowsDesktopV3{ ResourceHeader: types.ResourceHeader{ diff --git a/lib/srv/desktop/dir_sharing_disabled.go b/lib/srv/desktop/dir_sharing_disabled.go new file mode 100644 index 0000000000000..2e2253d692e38 --- /dev/null +++ b/lib/srv/desktop/dir_sharing_disabled.go @@ -0,0 +1,24 @@ +//go:build !directory_sharing +// +build !directory_sharing + +/* +Copyright 2022 Gravitational, Inc. + +Licensed 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 desktop + +func allowDirectorySharing() bool { + return false +} diff --git a/lib/srv/desktop/dir_sharing_enabled.go b/lib/srv/desktop/dir_sharing_enabled.go new file mode 100644 index 0000000000000..627d8d97eb50e --- /dev/null +++ b/lib/srv/desktop/dir_sharing_enabled.go @@ -0,0 +1,24 @@ +//go:build directory_sharing +// +build directory_sharing + +/* +Copyright 2022 Gravitational, Inc. + +Licensed 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 desktop + +func allowDirectorySharing() bool { + return true +} diff --git a/lib/srv/desktop/rdp/rdpclient/Cargo.toml b/lib/srv/desktop/rdp/rdpclient/Cargo.toml index abdfeacf731ea..494780ec028b4 100644 --- a/lib/srv/desktop/rdp/rdpclient/Cargo.toml +++ b/lib/srv/desktop/rdp/rdpclient/Cargo.toml @@ -19,6 +19,8 @@ num-derive = "0.3.3" num-traits = "0.2.14" rand = { version = "0.8.5", features = ["getrandom"] } rand_chacha = "0.3.1" -rsa = { version = "0.6.1" } +rsa = "0.6.1" rdp-rs = { git = "https://github.com/gravitational/rdp-rs", rev = "6075679e7c9bd8e3c2136a87f097d05c7db3235f" } -uuid = { version = "0.8.2", features = ["v4"] } +uuid = { version = "1.1.2", features = ["v4"] } +utf16string = "0.2.0" + diff --git a/lib/srv/desktop/rdp/rdpclient/client.go b/lib/srv/desktop/rdp/rdpclient/client.go index 831833c128a7c..d85f567ad33a9 100644 --- a/lib/srv/desktop/rdp/rdpclient/client.go +++ b/lib/srv/desktop/rdp/rdpclient/client.go @@ -240,6 +240,7 @@ func (c *Client) connect(ctx context.Context) error { C.uint16_t(c.clientWidth), C.uint16_t(c.clientHeight), C.bool(c.cfg.AllowClipboard), + C.bool(c.cfg.AllowDirectorySharing), ) if res.err != C.ErrCodeSuccess { return trace.ConnectionProblem(nil, "RDP connection failed") @@ -260,8 +261,8 @@ func (c *Client) start() { // C.read_rdp_output blocks for the duration of the RDP connection and // calls handle_bitmap repeatedly with the incoming bitmaps. - if err := C.read_rdp_output(c.rustClient); err != C.ErrCodeSuccess { - c.cfg.Log.Warningf("Failed reading RDP output frame: %v", err) + if errCode := C.read_rdp_output(c.rustClient); errCode != C.ErrCodeSuccess { + c.cfg.Log.Warningf("Failed reading RDP output frame: %v", errCode) // close the TDP connection to the browser // (without this the input streaming goroutine will hang @@ -299,7 +300,7 @@ func (c *Client) start() { switch m := msg.(type) { case tdp.MouseMove: mouseX, mouseY = m.X, m.Y - if err := C.write_rdp_pointer( + if errCode := C.write_rdp_pointer( c.rustClient, C.CGOMousePointerEvent{ x: C.uint16_t(m.X), @@ -307,7 +308,7 @@ func (c *Client) start() { button: C.PointerButtonNone, wheel: C.PointerWheelNone, }, - ); err != C.ErrCodeSuccess { + ); errCode != C.ErrCodeSuccess { return } case tdp.MouseButton: @@ -323,7 +324,7 @@ func (c *Client) start() { default: button = C.PointerButtonNone } - if err := C.write_rdp_pointer( + if errCode := C.write_rdp_pointer( c.rustClient, C.CGOMousePointerEvent{ x: C.uint16_t(mouseX), @@ -332,7 +333,7 @@ func (c *Client) start() { down: m.State == tdp.ButtonPressed, wheel: C.PointerWheelNone, }, - ); err != C.ErrCodeSuccess { + ); errCode != C.ErrCodeSuccess { return } case tdp.MouseWheel: @@ -351,7 +352,7 @@ func (c *Client) start() { default: wheel = C.PointerWheelNone } - if err := C.write_rdp_pointer( + if errCode := C.write_rdp_pointer( c.rustClient, C.CGOMousePointerEvent{ x: C.uint16_t(mouseX), @@ -360,31 +361,61 @@ func (c *Client) start() { wheel: uint32(wheel), wheel_delta: C.int16_t(m.Delta), }, - ); err != C.ErrCodeSuccess { + ); errCode != C.ErrCodeSuccess { return } case tdp.KeyboardButton: - if err := C.write_rdp_keyboard( + if errCode := C.write_rdp_keyboard( c.rustClient, C.CGOKeyboardEvent{ code: C.uint16_t(m.KeyCode), down: m.State == tdp.ButtonPressed, }, - ); err != C.ErrCodeSuccess { + ); errCode != C.ErrCodeSuccess { return } case tdp.ClipboardData: if len(m) > 0 { - if err := C.update_clipboard( + if errCode := C.update_clipboard( c.rustClient, (*C.uint8_t)(unsafe.Pointer(&m[0])), C.uint32_t(len(m)), - ); err != C.ErrCodeSuccess { + ); errCode != C.ErrCodeSuccess { return } } else { c.cfg.Log.Warning("Recieved an empty clipboard message") } + case tdp.SharedDirectoryAnnounce: + if c.cfg.AllowDirectorySharing { + driveName := C.CString(m.Name) + defer C.free(unsafe.Pointer(driveName)) + if errCode := C.handle_tdp_sd_announce(c.rustClient, C.CGOSharedDirectoryAnnounce{ + directory_id: C.uint32_t(m.DirectoryID), + name: driveName, + }); errCode != C.ErrCodeSuccess { + c.cfg.Log.Errorf("Device announce failed: %v", errCode) + return + } + } + case tdp.SharedDirectoryInfoResponse: + if c.cfg.AllowDirectorySharing { + path := C.CString(m.Fso.Path) + defer C.free(unsafe.Pointer(path)) + if errCode := C.handle_tdp_sd_info_response(c.rustClient, C.CGOSharedDirectoryInfoResponse{ + completion_id: C.uint32_t(m.CompletionID), + err_code: C.uint32_t(m.ErrCode), + fso: C.CGOFileSystemObject{ + last_modified: C.uint64_t(m.Fso.LastModified), + size: C.uint64_t(m.Fso.Size), + file_type: C.uint32_t(m.Fso.FileType), + path: path, + }, + }); errCode != C.ErrCodeSuccess { + c.cfg.Log.Errorf("SharedDirectoryInfoResponse failed: %v", errCode) + return + } + } default: c.cfg.Log.Warningf("Skipping unimplemented TDP message type %T", msg) } @@ -450,6 +481,44 @@ func (c *Client) handleRemoteCopy(data []byte) C.CGOErrCode { return C.ErrCodeSuccess } +//export tdp_sd_acknowledge +func tdp_sd_acknowledge(handle C.uintptr_t, ack *C.CGOSharedDirectoryAcknowledge) C.CGOErrCode { + return cgo.Handle(handle).Value().(*Client).sharedDirectoryAcknowledge(tdp.SharedDirectoryAcknowledge{ + ErrCode: uint32(ack.err_code), + DirectoryID: uint32(ack.directory_id), + }) +} + +// sharedDirectoryAcknowledge acknowledges that a `Shared Directory Announce` TDP message was processed. +func (c *Client) sharedDirectoryAcknowledge(ack tdp.SharedDirectoryAcknowledge) C.CGOErrCode { + if c.cfg.AllowDirectorySharing { + if err := c.cfg.Conn.OutputMessage(ack); err != nil { + c.cfg.Log.Errorf("failed to send SharedDirectoryAcknowledge: %v", err) + return C.ErrCodeFailure + } + } + return C.ErrCodeSuccess +} + +//export tdp_sd_info_request +func tdp_sd_info_request(handle C.uintptr_t, req *C.CGOSharedDirectoryInfoRequest) C.CGOErrCode { + return cgo.Handle(handle).Value().(*Client).sharedDirectoryInfoRequest(tdp.SharedDirectoryInfoRequest{ + CompletionID: uint32(req.completion_id), + DirectoryID: uint32(req.directory_id), + Path: C.GoString(req.path), + }) +} + +func (c *Client) sharedDirectoryInfoRequest(req tdp.SharedDirectoryInfoRequest) C.CGOErrCode { + if c.cfg.AllowDirectorySharing { + if err := c.cfg.Conn.OutputMessage(req); err != nil { + c.cfg.Log.Errorf("failed to send SharedDirectoryAcknowledge: %v", err) + return C.ErrCodeFailure + } + } + return C.ErrCodeSuccess +} + // close frees the memory of the cgo.Handle, // closes the RDP client connection, // and frees the Rust client. diff --git a/lib/srv/desktop/rdp/rdpclient/client_common.go b/lib/srv/desktop/rdp/rdpclient/client_common.go index 9ad356d094361..9373a699862b0 100644 --- a/lib/srv/desktop/rdp/rdpclient/client_common.go +++ b/lib/srv/desktop/rdp/rdpclient/client_common.go @@ -49,6 +49,10 @@ type Config struct { // clipboard sharing. AllowClipboard bool + // AllowDirectorySharing indicates whether the RDP connection should enable + // directory sharing. + AllowDirectorySharing bool + // Log is the logger for status messages. Log logrus.FieldLogger } diff --git a/lib/srv/desktop/rdp/rdpclient/librdprs.h b/lib/srv/desktop/rdp/rdpclient/librdprs.h index a8c6278d77a18..ab0283121c796 100644 --- a/lib/srv/desktop/rdp/rdpclient/librdprs.h +++ b/lib/srv/desktop/rdp/rdpclient/librdprs.h @@ -7,6 +7,20 @@ #define SPECIAL_NO_RESPONSE 4294967295 +#define SCARD_DEVICE_ID 1 + +#define VERSION_MAJOR 1 + +#define VERSION_MINOR 12 + +#define SMARTCARD_CAPABILITY_VERSION_01 1 + +#define DRIVE_CAPABILITY_VERSION_02 2 + +#define GENERAL_CAPABILITY_VERSION_01 1 + +#define GENERAL_CAPABILITY_VERSION_02 2 + /** * The default maximum chunk size for virtual channel data. * @@ -56,6 +70,24 @@ typedef struct ClientOrError { enum CGOErrCode err; } ClientOrError; +typedef struct CGOSharedDirectoryAnnounce { + uint32_t directory_id; + const char *name; +} CGOSharedDirectoryAnnounce; + +typedef struct CGOFileSystemObject { + uint64_t last_modified; + uint64_t size; + uint32_t file_type; + const char *path; +} CGOFileSystemObject; + +typedef struct CGOSharedDirectoryInfoResponse { + uint32_t completion_id; + uint32_t err_code; + struct CGOFileSystemObject fso; +} CGOSharedDirectoryInfoResponse; + /** * CGOMousePointerEvent is a CGO-compatible version of PointerEvent that we pass back to Go. * PointerEvent is a mouse move or click update from the user. @@ -95,6 +127,17 @@ typedef struct CGOBitmap { uintptr_t data_cap; } CGOBitmap; +typedef struct CGOSharedDirectoryAcknowledge { + uint32_t err_code; + uint32_t directory_id; +} CGOSharedDirectoryAcknowledge; + +typedef struct CGOSharedDirectoryInfoRequest { + uint32_t completion_id; + uint32_t directory_id; + const char *path; +} CGOSharedDirectoryInfoRequest; + void init(void); /** @@ -108,15 +151,16 @@ void init(void); * to their corresponding parameters. */ struct ClientOrError connect_rdp(uintptr_t go_ref, - char *go_addr, - char *go_username, + const char *go_addr, + const char *go_username, uint32_t cert_der_len, uint8_t *cert_der, uint32_t key_der_len, uint8_t *key_der, uint16_t screen_width, uint16_t screen_height, - bool allow_clipboard); + bool allow_clipboard, + bool allow_directory_sharing); /** * `update_clipboard` is called from Go, and caches data that was copied @@ -128,6 +172,28 @@ struct ClientOrError connect_rdp(uintptr_t go_ref, */ enum CGOErrCode update_clipboard(struct Client *client_ptr, uint8_t *data, uint32_t len); +/** + * handle_tdp_sd_announce announces a new drive that's ready to be + * redirected over RDP. + * + * # Safety + * + * The caller must ensure that sd_announce.name points to a valid buffer. + */ +enum CGOErrCode handle_tdp_sd_announce(struct Client *client_ptr, + struct CGOSharedDirectoryAnnounce sd_announce); + +/** + * handle_tdp_sd_info_response handles a TDP Shared Directory Info Response + * message + * + * # Safety + * + * The caller must ensure that res.fso.path points to a valid buffer. + */ +enum CGOErrCode handle_tdp_sd_info_response(struct Client *client_ptr, + struct CGOSharedDirectoryInfoResponse res); + /** * `read_rdp_output` reads incoming RDP bitmap frames from client at client_ref and forwards them to * handle_bitmap. @@ -169,13 +235,12 @@ enum CGOErrCode close_rdp(struct Client *client_ptr); */ void free_rdp(struct Client *client_ptr); -/** - * # Safety - * - * The passed pointer must point to a C-style string allocated by Rust. - */ -void free_rust_string(char *s); - extern enum CGOErrCode handle_bitmap(uintptr_t client_ref, struct CGOBitmap *b); extern enum CGOErrCode handle_remote_copy(uintptr_t client_ref, uint8_t *data, uint32_t len); + +extern enum CGOErrCode tdp_sd_acknowledge(uintptr_t client_ref, + struct CGOSharedDirectoryAcknowledge *ack); + +extern enum CGOErrCode tdp_sd_info_request(uintptr_t client_ref, + struct CGOSharedDirectoryInfoRequest *req); diff --git a/lib/srv/desktop/rdp/rdpclient/src/cliprdr.rs b/lib/srv/desktop/rdp/rdpclient/src/cliprdr.rs index 3f046b6e87e4b..5f61ed30ff44f 100644 --- a/lib/srv/desktop/rdp/rdpclient/src/cliprdr.rs +++ b/lib/srv/desktop/rdp/rdpclient/src/cliprdr.rs @@ -566,7 +566,7 @@ fn encode_clipboard(mut data: String) -> (Vec, ClipboardFormat) { (data.into_bytes(), ClipboardFormat::CF_TEXT) } else { - let encoded = util::to_unicode(&data); + let encoded = util::to_unicode(&data, true); (encoded, ClipboardFormat::CF_UNICODETEXT) } } @@ -678,7 +678,7 @@ impl FormatName for LongFormatName { // must be encoded as a single Unicode null character (two zero bytes) None => w.write_u16::(0)?, Some(name) => { - w.append(&mut util::to_unicode(name)); + w.append(&mut util::to_unicode(name, true)); } }; @@ -1039,7 +1039,7 @@ mod tests { #[test] fn responds_to_format_data_request_hasdata() { // a null-terminated utf-16 string, represented as a Vec - let test_data = util::to_unicode("test"); + let test_data = util::to_unicode("test", true); let mut c: Client = Default::default(); c.clipboard diff --git a/lib/srv/desktop/rdp/rdpclient/src/errors.rs b/lib/srv/desktop/rdp/rdpclient/src/errors.rs index 7e1ec566a19bf..4965c90ad3ac6 100644 --- a/lib/srv/desktop/rdp/rdpclient/src/errors.rs +++ b/lib/srv/desktop/rdp/rdpclient/src/errors.rs @@ -19,10 +19,18 @@ pub fn invalid_data_error(msg: &str) -> Error { Error::RdpError(RdpError::new(RdpErrorKind::InvalidData, msg)) } +pub fn not_implemented_error(msg: &str) -> Error { + Error::RdpError(RdpError::new(RdpErrorKind::NotImplemented, msg)) +} + pub fn try_error(msg: &str) -> Error { Error::TryError(msg.to_string()) } +pub fn rejected_by_server_error(msg: &str) -> Error { + Error::RdpError(RdpError::new(RdpErrorKind::RejectedByServer, msg)) +} + // NTSTATUS_OK is a Windows NTStatus value that means "success". pub const NTSTATUS_OK: u32 = 0; // SPECIAL_NO_RESPONSE is our custom (not defined by Windows) NTStatus value that means "don't send diff --git a/lib/srv/desktop/rdp/rdpclient/src/lib.rs b/lib/srv/desktop/rdp/rdpclient/src/lib.rs index 2e7ae0e089586..8d879081b1f80 100644 --- a/lib/srv/desktop/rdp/rdpclient/src/lib.rs +++ b/lib/srv/desktop/rdp/rdpclient/src/lib.rs @@ -12,13 +12,12 @@ // See the License for the specific language governing permissions and // limitations under the License. -pub mod cliprdr; -pub mod errors; -pub mod piv; -pub mod rdpdr; -pub mod scard; -pub mod util; -pub mod vchan; +mod cliprdr; +mod errors; +mod piv; +mod rdpdr; +mod util; +mod vchan; #[macro_use] extern crate log; @@ -37,6 +36,7 @@ use rdp::core::tpkt; use rdp::core::x224; use rdp::model::error::{Error as RdpError, RdpError as RdpProtocolError, RdpErrorKind, RdpResult}; use rdp::model::link::{Link, Stream}; +use rdpdr::ServerCreateDriveRequest; use std::convert::TryFrom; use std::ffi::{CStr, CString}; use std::io::Error as IoError; @@ -121,8 +121,8 @@ impl From> for ClientOrError { #[no_mangle] pub unsafe extern "C" fn connect_rdp( go_ref: usize, - go_addr: *mut c_char, - go_username: *mut c_char, + go_addr: *const c_char, + go_username: *const c_char, cert_der_len: u32, cert_der: *mut u8, key_der_len: u32, @@ -130,6 +130,7 @@ pub unsafe extern "C" fn connect_rdp( screen_width: u16, screen_height: u16, allow_clipboard: bool, + allow_directory_sharing: bool, ) -> ClientOrError { // Convert from C to Rust types. let addr = from_go_string(go_addr); @@ -147,6 +148,7 @@ pub unsafe extern "C" fn connect_rdp( screen_width, screen_height, allow_clipboard, + allow_directory_sharing, }, ) .into() @@ -181,6 +183,7 @@ struct ConnectParams { screen_width: u16, screen_height: u16, allow_clipboard: bool, + allow_directory_sharing: bool, } fn connect_rdp_inner( @@ -249,8 +252,62 @@ fn connect_rdp_inner( KeyboardLayout::US, "rdp-rs", ); - // Client for the "rdpdr" channel - smartcard emulation. - let rdpdr = rdpdr::Client::new(params.cert_der, params.key_der, pin); + + let tdp_sd_acknowledge = Box::new(move |ack: SharedDirectoryAcknowledge| -> RdpResult<()> { + debug!("sending: {:?}", ack); + unsafe { + if tdp_sd_acknowledge(go_ref, &mut CGOSharedDirectoryAcknowledge::from(ack)) + != CGOErrCode::ErrCodeSuccess + { + return Err(RdpError::TryError(String::from( + "call to tdp_sd_acknowledge failed", + ))); + } + } + Ok(()) + }); + + let tdp_sd_info_request = Box::new(move |req: SharedDirectoryInfoRequest| -> RdpResult<()> { + debug!("sending: {:?}", req); + // Create C compatible string from req.path + match CString::new(req.path.clone()) { + Ok(c_string) => { + unsafe { + let err = tdp_sd_info_request( + go_ref, + &mut CGOSharedDirectoryInfoRequest { + completion_id: req.completion_id, + directory_id: req.directory_id, + path: c_string.as_ptr(), + }, + ); + if err != CGOErrCode::ErrCodeSuccess { + return Err(RdpError::TryError(String::from( + "call to tdp_sd_info_request failed", + ))); + }; + } + Ok(()) + } + Err(_) => { + // TODO(isaiah): change TryError to TeleportError for a generic error caused by Teleport specific code. + return Err(RdpError::TryError(format!( + "path contained characters that couldn't be converted to a C string: {}", + req.path + ))); + } + } + }); + + // Client for the "rdpdr" channel - smartcard emulation and drive redirection. + let rdpdr = rdpdr::Client::new( + params.cert_der, + params.key_der, + pin, + params.allow_directory_sharing, + tdp_sd_acknowledge, + tdp_sd_info_request, + ); // Client for the "cliprdr" channel - clipboard sharing. let cliprdr = if params.allow_clipboard { @@ -329,6 +386,21 @@ impl RdpClient { } } + pub fn write_client_device_list_announce( + &mut self, + req: rdpdr::ClientDeviceListAnnounce, + ) -> RdpResult<()> { + self.rdpdr + .write_client_device_list_announce(req, &mut self.mcs) + } + + pub fn handle_tdp_sd_info_response( + &mut self, + res: SharedDirectoryInfoResponse, + ) -> RdpResult<()> { + self.rdpdr.handle_tdp_sd_info_response(res, &mut self.mcs) + } + pub fn shutdown(&mut self) -> RdpResult<()> { self.mcs.shutdown() } @@ -451,6 +523,66 @@ pub unsafe extern "C" fn update_clipboard( } } +/// handle_tdp_sd_announce announces a new drive that's ready to be +/// redirected over RDP. +/// +/// # Safety +/// +/// The caller must ensure that sd_announce.name points to a valid buffer. +#[no_mangle] +pub unsafe extern "C" fn handle_tdp_sd_announce( + client_ptr: *mut Client, + sd_announce: CGOSharedDirectoryAnnounce, +) -> CGOErrCode { + let client = match Client::from_ptr(client_ptr) { + Ok(client) => client, + Err(cgo_error) => { + return cgo_error; + } + }; + + let drive_name = from_go_string(sd_announce.name); + let new_drive = + rdpdr::ClientDeviceListAnnounce::new_drive(sd_announce.directory_id, drive_name); + + let mut rdp_client = client.rdp_client.lock().unwrap(); + match rdp_client.write_client_device_list_announce(new_drive) { + Ok(()) => CGOErrCode::ErrCodeSuccess, + Err(e) => { + error!("failed to announce new drive: {:?}", e); + CGOErrCode::ErrCodeFailure + } + } +} + +/// handle_tdp_sd_info_response handles a TDP Shared Directory Info Response +/// message +/// +/// # Safety +/// +/// The caller must ensure that res.fso.path points to a valid buffer. +#[no_mangle] +pub unsafe extern "C" fn handle_tdp_sd_info_response( + client_ptr: *mut Client, + res: CGOSharedDirectoryInfoResponse, +) -> CGOErrCode { + let client = match Client::from_ptr(client_ptr) { + Ok(client) => client, + Err(cgo_error) => { + return cgo_error; + } + }; + + let mut rdp_client = client.rdp_client.lock().unwrap(); + match rdp_client.handle_tdp_sd_info_response(SharedDirectoryInfoResponse::from(res)) { + Ok(()) => CGOErrCode::ErrCodeSuccess, + Err(e) => { + error!("failed to handle Shared Directory Info Response: {:?}", e); + CGOErrCode::ErrCodeFailure + } + } +} + /// `read_rdp_output` reads incoming RDP bitmap frames from client at client_ref and forwards them to /// handle_bitmap. /// @@ -518,7 +650,7 @@ fn read_rdp_output_inner(client: &Client) -> Option { match res { Err(RdpError::Io(io_err)) if io_err.kind() == ErrorKind::UnexpectedEof => return None, Err(e) => { - return Some(format!("failed forwarding RDP bitmap frame: {:?}", e)); + return Some(format!("RDP read failed: {:?}", e)); } _ => {} } @@ -686,18 +818,12 @@ pub unsafe extern "C" fn free_rdp(client_ptr: *mut Client) { drop(Client::from_raw(client_ptr)) } -/// # Safety -/// -/// The passed pointer must point to a C-style string allocated by Rust. -#[no_mangle] -pub unsafe extern "C" fn free_rust_string(s: *mut c_char) { - let _ = CString::from_raw(s); -} - /// # Safety /// /// s must be a C-style null terminated string. -unsafe fn from_go_string(s: *mut c_char) -> String { +/// s is cloned here, and the caller is responsible for +/// ensuring its memory is freed. +unsafe fn from_go_string(s: *const c_char) -> String { CStr::from_ptr(s).to_string_lossy().into_owned() } @@ -715,11 +841,126 @@ pub enum CGOErrCode { ErrCodeFailure = 1, } +#[repr(C)] +pub struct CGOSharedDirectoryAnnounce { + pub directory_id: u32, + pub name: *const c_char, +} + +/// SharedDirectoryAcknowledge is a CGO-compatible version of +/// the TDP Shared Directory Knowledge message that we pass back to Go. +#[derive(Debug)] +pub struct SharedDirectoryAcknowledge { + pub err_code: u32, + pub directory_id: u32, +} + +#[repr(C)] +pub struct CGOSharedDirectoryAcknowledge { + pub err_code: u32, + pub directory_id: u32, +} + +impl From for CGOSharedDirectoryAcknowledge { + fn from(ack: SharedDirectoryAcknowledge) -> CGOSharedDirectoryAcknowledge { + CGOSharedDirectoryAcknowledge { + err_code: ack.err_code, + directory_id: ack.directory_id, + } + } +} + +#[derive(Debug)] +pub struct SharedDirectoryInfoRequest { + completion_id: u32, + directory_id: u32, + path: String, +} + +#[repr(C)] +pub struct CGOSharedDirectoryInfoRequest { + pub completion_id: u32, + pub directory_id: u32, + pub path: *const c_char, +} + +impl From for SharedDirectoryInfoRequest { + fn from(req: ServerCreateDriveRequest) -> SharedDirectoryInfoRequest { + SharedDirectoryInfoRequest { + completion_id: req.device_io_request.completion_id, + directory_id: req.device_io_request.device_id, + path: req.path, + } + } +} + +#[derive(Debug)] +#[allow(dead_code)] +pub struct SharedDirectoryInfoResponse { + completion_id: u32, + err_code: u32, + fso: FileSystemObject, +} + +#[repr(C)] +pub struct CGOSharedDirectoryInfoResponse { + pub completion_id: u32, + pub err_code: u32, + pub fso: CGOFileSystemObject, +} + +impl From for SharedDirectoryInfoResponse { + fn from(cgo_res: CGOSharedDirectoryInfoResponse) -> SharedDirectoryInfoResponse { + SharedDirectoryInfoResponse { + completion_id: cgo_res.completion_id, + err_code: cgo_res.err_code, + fso: FileSystemObject::from(cgo_res.fso), + } + } +} + +#[derive(Debug)] +#[allow(dead_code)] +pub struct FileSystemObject { + last_modified: u64, + size: u64, + file_type: u32, // TODO(isaiah): make an enum + path: String, +} + +#[repr(C)] +pub struct CGOFileSystemObject { + pub last_modified: u64, + pub size: u64, + pub file_type: u32, // TODO(isaiah): make an enum + pub path: *const c_char, +} + +impl From for FileSystemObject { + fn from(cgo_fso: CGOFileSystemObject) -> FileSystemObject { + unsafe { + FileSystemObject { + last_modified: cgo_fso.last_modified, + size: cgo_fso.size, + file_type: cgo_fso.file_type, + path: from_go_string(cgo_fso.path), + } + } + } +} + // These functions are defined on the Go side. Look for functions with '//export funcname' // comments. extern "C" { fn handle_bitmap(client_ref: usize, b: *mut CGOBitmap) -> CGOErrCode; fn handle_remote_copy(client_ref: usize, data: *mut u8, len: u32) -> CGOErrCode; + + fn tdp_sd_acknowledge(client_ref: usize, ack: *mut CGOSharedDirectoryAcknowledge) + -> CGOErrCode; + fn tdp_sd_info_request( + client_ref: usize, + req: *mut CGOSharedDirectoryInfoRequest, + ) -> CGOErrCode; } /// Payload is a generic type used to represent raw incoming RDP messages for parsing. diff --git a/lib/srv/desktop/rdp/rdpclient/src/rdpdr.rs b/lib/srv/desktop/rdp/rdpclient/src/rdpdr.rs deleted file mode 100644 index 6c3e37195e8d6..0000000000000 --- a/lib/srv/desktop/rdp/rdpclient/src/rdpdr.rs +++ /dev/null @@ -1,725 +0,0 @@ -// Copyright 2021 Gravitational, Inc -// -// Licensed 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. - -use crate::errors::{invalid_data_error, NTSTATUS_OK, SPECIAL_NO_RESPONSE}; -use crate::Payload; -use crate::{scard, vchan}; -use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt}; -use num_traits::{FromPrimitive, ToPrimitive}; -use rdp::core::mcs; -use rdp::core::tpkt; -use rdp::model::data::Message; -use rdp::model::error::*; -use std::io::{Read, Write}; - -pub const CHANNEL_NAME: &str = "rdpdr"; - -/// Client implements a device redirection (RDPDR) client, as defined in -/// https://winprotocoldoc.blob.core.windows.net/productionwindowsarchives/MS-RDPEFS/%5bMS-RDPEFS%5d.pdf -/// -/// This client only supports a single smartcard device. -pub struct Client { - vchan: vchan::Client, - scard: scard::Client, -} - -impl Client { - pub fn new(cert_der: Vec, key_der: Vec, pin: String) -> Self { - Client { - vchan: vchan::Client::new(), - scard: scard::Client::new(cert_der, key_der, pin), - } - } - /// Reads raw RDP messages sent on the rdpdr virtual channel and replies as necessary. - pub fn read_and_reply( - &mut self, - payload: tpkt::Payload, - mcs: &mut mcs::Client, - ) -> RdpResult<()> { - if let Some(mut payload) = self.vchan.read(payload)? { - let header = SharedHeader::decode(&mut payload)?; - if let Component::RDPDR_CTYP_PRN = header.component { - warn!("got {:?} RDPDR header from RDP server, ignoring because we're not redirecting any printers", header); - return Ok(()); - } - let responses = match header.packet_id { - PacketId::PAKID_CORE_SERVER_ANNOUNCE => { - self.handle_server_announce(&mut payload)? - } - PacketId::PAKID_CORE_SERVER_CAPABILITY => { - self.handle_server_capability(&mut payload)? - } - PacketId::PAKID_CORE_CLIENTID_CONFIRM => { - self.handle_client_id_confirm(&mut payload)? - } - PacketId::PAKID_CORE_DEVICE_REPLY => self.handle_device_reply(&mut payload)?, - // Device IO request is where communication with the smartcard actually happens. - // Everything up to this point was negotiation and smartcard device registration. - PacketId::PAKID_CORE_DEVICE_IOREQUEST => { - self.handle_device_io_request(&mut payload)? - } - _ => { - // We don't implement the full set of messages. Only the ones necessary for initial - // negotiation and registration of a smartcard device. - error!( - "RDPDR packets {:?} are not implemented yet, ignoring", - header.packet_id - ); - vec![] - } - }; - - let chan = &CHANNEL_NAME.to_string(); - for resp in responses { - mcs.write(chan, resp)?; - } - } - Ok(()) - } - - fn handle_server_announce(&self, payload: &mut Payload) -> RdpResult>> { - let req = ServerAnnounceRequest::decode(payload)?; - debug!("got ServerAnnounceRequest {:?}", req); - - let resp = self.add_headers_and_chunkify( - PacketId::PAKID_CORE_CLIENTID_CONFIRM, - ClientAnnounceReply::new(req).encode()?, - )?; - debug!("sending client announce reply"); - Ok(resp) - } - - fn handle_server_capability(&self, payload: &mut Payload) -> RdpResult>> { - let req = ServerCoreCapabilityRequest::decode(payload)?; - debug!("got {:?}", req); - - let resp = self.add_headers_and_chunkify( - PacketId::PAKID_CORE_CLIENT_CAPABILITY, - ClientCoreCapabilityResponse::new_response().encode()?, - )?; - debug!("sending client core capability response"); - Ok(resp) - } - - fn handle_client_id_confirm(&self, payload: &mut Payload) -> RdpResult>> { - let req = ServerClientIdConfirm::decode(payload)?; - debug!("got ServerClientIdConfirm {:?}", req); - - let resp = self.add_headers_and_chunkify( - PacketId::PAKID_CORE_DEVICELIST_ANNOUNCE, - ClientDeviceListAnnounceRequest::new_smartcard().encode()?, - )?; - debug!("sending client device list announce request"); - Ok(resp) - } - - fn handle_device_reply(&self, payload: &mut Payload) -> RdpResult>> { - let req = ServerDeviceAnnounceResponse::decode(payload)?; - debug!("got {:?}", req); - - if req.device_id != SCARD_DEVICE_ID { - Err(invalid_data_error(&format!( - "got ServerDeviceAnnounceResponse for unknown device_id {}", - &req.device_id - ))) - } else if req.result_code != NTSTATUS_OK { - Err(invalid_data_error(&format!( - "got unsuccessful ServerDeviceAnnounceResponse result code NTSTATUS({})", - &req.result_code - ))) - } else { - Ok(vec![]) - } - } - - fn handle_device_io_request(&mut self, payload: &mut Payload) -> RdpResult>> { - let req = DeviceIoRequest::decode(payload)?; - debug!("got {:?}", req); - - if let MajorFunction::IRP_MJ_DEVICE_CONTROL = req.major_function { - let ioctl = DeviceControlRequest::decode(req, payload)?; - debug!("got {:?}", ioctl); - - let (code, res) = self.scard.ioctl(ioctl.io_control_code, payload)?; - if code == SPECIAL_NO_RESPONSE { - return Ok(vec![]); - } - let resp = self.add_headers_and_chunkify( - PacketId::PAKID_CORE_DEVICE_IOCOMPLETION, - DeviceControlResponse::new(&ioctl, code, res).encode()?, - )?; - debug!("sending device IO response"); - Ok(resp) - } else { - Err(invalid_data_error(&format!( - "got unsupported major_function in DeviceIoRequest: {:?}", - &req.major_function - ))) - } - } - - /// add_headers_and_chunkify takes an encoded PDU ready to be sent over a virtual channel (payload), - /// adds on the Shared Header based the passed packet_id, adds the appropriate (virtual) Channel PDU Header, - /// and splits the entire payload into chunks if the payload exceeds the maximum size. - fn add_headers_and_chunkify( - &self, - packet_id: PacketId, - payload: Vec, - ) -> RdpResult>> { - let mut inner = SharedHeader::new(Component::RDPDR_CTYP_CORE, packet_id).encode()?; - inner.extend_from_slice(&payload); - self.vchan.add_header_and_chunkify(None, inner) - } -} - -/// 2.2.1.1 Shared Header (RDPDR_HEADER) -/// This header is present at the beginning of every message in sent over the rdpdr virtual channel. -/// The purpose of this header is to describe the type of the message. -/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/29d4108f-8163-4a67-8271-e48c4b9c2a7c -#[derive(Debug)] -struct SharedHeader { - component: Component, - packet_id: PacketId, -} - -impl SharedHeader { - fn new(component: Component, packet_id: PacketId) -> Self { - Self { - component, - packet_id, - } - } - fn decode(payload: &mut Payload) -> RdpResult { - let component = payload.read_u16::()?; - let packet_id = payload.read_u16::()?; - Ok(Self { - component: Component::from_u16(component).ok_or_else(|| { - invalid_data_error(&format!("invalid component value {:#06x}", component)) - })?, - packet_id: PacketId::from_u16(packet_id).ok_or_else(|| { - invalid_data_error(&format!("invalid packet_id value {:#06x}", packet_id)) - })?, - }) - } - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u16::(self.component.to_u16().unwrap())?; - w.write_u16::(self.packet_id.to_u16().unwrap())?; - Ok(w) - } -} - -#[derive(Debug, FromPrimitive, ToPrimitive)] -#[allow(non_camel_case_types)] -enum Component { - RDPDR_CTYP_CORE = 0x4472, - RDPDR_CTYP_PRN = 0x5052, -} - -#[derive(Debug, FromPrimitive, ToPrimitive)] -#[allow(non_camel_case_types)] -enum PacketId { - PAKID_CORE_SERVER_ANNOUNCE = 0x496E, - PAKID_CORE_CLIENTID_CONFIRM = 0x4343, - PAKID_CORE_CLIENT_NAME = 0x434E, - PAKID_CORE_DEVICELIST_ANNOUNCE = 0x4441, - PAKID_CORE_DEVICE_REPLY = 0x6472, - PAKID_CORE_DEVICE_IOREQUEST = 0x4952, - PAKID_CORE_DEVICE_IOCOMPLETION = 0x4943, - PAKID_CORE_SERVER_CAPABILITY = 0x5350, - PAKID_CORE_CLIENT_CAPABILITY = 0x4350, - PAKID_CORE_DEVICELIST_REMOVE = 0x444D, - PAKID_PRN_CACHE_DATA = 0x5043, - PAKID_CORE_USER_LOGGEDON = 0x554C, - PAKID_PRN_USING_XPS = 0x5543, -} - -type ServerAnnounceRequest = ClientIdMessage; -type ClientAnnounceReply = ClientIdMessage; -type ServerClientIdConfirm = ClientIdMessage; - -const VERSION_MAJOR: u16 = 0x0001; -const VERSION_MINOR: u16 = 0x000c; - -#[derive(Debug)] -struct ClientIdMessage { - version_major: u16, - version_minor: u16, - client_id: u32, -} - -impl ClientIdMessage { - fn new(req: ServerAnnounceRequest) -> Self { - Self { - version_major: VERSION_MAJOR, - version_minor: VERSION_MINOR, - client_id: req.client_id, - } - } - - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u16::(self.version_major)?; - w.write_u16::(self.version_minor)?; - w.write_u32::(self.client_id)?; - Ok(w) - } - - fn decode(payload: &mut Payload) -> RdpResult { - Ok(Self { - version_major: payload.read_u16::()?, - version_minor: payload.read_u16::()?, - client_id: payload.read_u32::()?, - }) - } -} - -#[derive(Debug)] -struct ServerCoreCapabilityRequest { - num_capabilities: u16, - padding: u16, - capabilities: Vec, -} - -impl ServerCoreCapabilityRequest { - fn new_response() -> Self { - // Clients are always required to send the "general" capability set. - // In addition, we also send the optional smartcard capability. - Self { - num_capabilities: 2, - padding: 0, - capabilities: vec![ - CapabilitySet { - header: CapabilityHeader { - cap_type: CapabilityType::CAP_GENERAL_TYPE, - length: 8 + 36, // 8 byte header + 36 byte capability descriptor - version: GENERAL_CAPABILITY_VERSION_02, - }, - data: Capability::General(GeneralCapabilitySet { - os_type: 0, - os_version: 0, - protocol_major_version: VERSION_MAJOR, - protocol_minor_version: VERSION_MINOR, - io_code_1: 0x00007fff, // Combination of all the required bits. - io_code_2: 0, - extended_pdu: 0x00000001 | 0x00000002, // RDPDR_DEVICE_REMOVE_PDUS | RDPDR_CLIENT_DISPLAY_NAME_PDU - extra_flags_1: 0, - extra_flags_2: 0, - special_type_device_cap: 1, // Request redirection of 1 special device - smartcard. - }), - }, - CapabilitySet { - header: CapabilityHeader { - cap_type: CapabilityType::CAP_SMARTCARD_TYPE, - length: 8, // 8 byte header + empty capability descriptor - version: SMARTCARD_CAPABILITY_VERSION_01, - }, - data: Capability::Smartcard, - }, - ], - } - } - - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u16::(self.num_capabilities)?; - w.write_u16::(self.padding)?; - for cap in self.capabilities.iter() { - w.extend_from_slice(&cap.encode()?); - } - Ok(w) - } - - fn decode(payload: &mut Payload) -> RdpResult { - let num_capabilities = payload.read_u16::()?; - let padding = payload.read_u16::()?; - let mut capabilities = vec![]; - for _ in 0..num_capabilities { - capabilities.push(CapabilitySet::decode(payload)?); - } - - Ok(Self { - num_capabilities, - padding, - capabilities, - }) - } -} - -#[derive(Debug)] -struct CapabilitySet { - header: CapabilityHeader, - data: Capability, -} - -impl CapabilitySet { - fn encode(&self) -> RdpResult> { - let mut w = self.header.encode()?; - w.extend_from_slice(&self.data.encode()?); - Ok(w) - } - fn decode(payload: &mut Payload) -> RdpResult { - let header = CapabilityHeader::decode(payload)?; - let data = Capability::decode(payload, &header)?; - - Ok(Self { header, data }) - } -} - -const SMARTCARD_CAPABILITY_VERSION_01: u32 = 0x00000001; -#[allow(dead_code)] -const GENERAL_CAPABILITY_VERSION_01: u32 = 0x00000001; -const GENERAL_CAPABILITY_VERSION_02: u32 = 0x00000002; - -#[derive(Debug)] -struct CapabilityHeader { - cap_type: CapabilityType, - length: u16, - version: u32, -} - -impl CapabilityHeader { - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u16::(self.cap_type.to_u16().unwrap())?; - w.write_u16::(self.length)?; - w.write_u32::(self.version)?; - Ok(w) - } - fn decode(payload: &mut Payload) -> RdpResult { - let cap_type = payload.read_u16::()?; - Ok(Self { - cap_type: CapabilityType::from_u16(cap_type).ok_or_else(|| { - invalid_data_error(&format!("invalid capability type {:#06x}", cap_type)) - })?, - length: payload.read_u16::()?, - version: payload.read_u32::()?, - }) - } -} - -#[derive(Debug, FromPrimitive, ToPrimitive)] -#[allow(non_camel_case_types)] -enum CapabilityType { - CAP_GENERAL_TYPE = 0x0001, - CAP_PRINTER_TYPE = 0x0002, - CAP_PORT_TYPE = 0x0003, - CAP_DRIVE_TYPE = 0x0004, - CAP_SMARTCARD_TYPE = 0x0005, -} - -#[derive(Debug)] -enum Capability { - General(GeneralCapabilitySet), - Printer, - Port, - Drive, - Smartcard, -} - -impl Capability { - fn encode(&self) -> RdpResult> { - match self { - Capability::General(general) => Ok(general.encode()?), - _ => Ok(vec![]), - } - } - - fn decode(payload: &mut Payload, header: &CapabilityHeader) -> RdpResult { - match header.cap_type { - CapabilityType::CAP_GENERAL_TYPE => Ok(Capability::General( - GeneralCapabilitySet::decode(payload, header.version)?, - )), - CapabilityType::CAP_PRINTER_TYPE => Ok(Capability::Printer), - CapabilityType::CAP_PORT_TYPE => Ok(Capability::Port), - CapabilityType::CAP_DRIVE_TYPE => Ok(Capability::Drive), - CapabilityType::CAP_SMARTCARD_TYPE => Ok(Capability::Smartcard), - } - } -} - -#[derive(Debug)] -struct GeneralCapabilitySet { - os_type: u32, - os_version: u32, - protocol_major_version: u16, - protocol_minor_version: u16, - io_code_1: u32, - io_code_2: u32, - extended_pdu: u32, - extra_flags_1: u32, - extra_flags_2: u32, - special_type_device_cap: u32, -} - -impl GeneralCapabilitySet { - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u32::(self.os_type)?; - w.write_u32::(self.os_version)?; - w.write_u16::(self.protocol_major_version)?; - w.write_u16::(self.protocol_minor_version)?; - w.write_u32::(self.io_code_1)?; - w.write_u32::(self.io_code_2)?; - w.write_u32::(self.extended_pdu)?; - w.write_u32::(self.extra_flags_1)?; - w.write_u32::(self.extra_flags_2)?; - w.write_u32::(self.special_type_device_cap)?; - Ok(w) - } - - fn decode(payload: &mut Payload, version: u32) -> RdpResult { - Ok(Self { - os_type: payload.read_u32::()?, - os_version: payload.read_u32::()?, - protocol_major_version: payload.read_u16::()?, - protocol_minor_version: payload.read_u16::()?, - io_code_1: payload.read_u32::()?, - io_code_2: payload.read_u32::()?, - extended_pdu: payload.read_u32::()?, - extra_flags_1: payload.read_u32::()?, - extra_flags_2: payload.read_u32::()?, - special_type_device_cap: if version == GENERAL_CAPABILITY_VERSION_02 { - payload.read_u32::()? - } else { - 0 - }, - }) - } -} - -type ClientCoreCapabilityResponse = ServerCoreCapabilityRequest; - -// If there were multiple redirected devices, they would need unique IDs. In our case there is only -// one permanent smartcard device, so we hardcode an ID 1. -const SCARD_DEVICE_ID: u32 = 1; - -#[derive(Debug)] -struct ClientDeviceListAnnounceRequest { - count: u32, - devices: Vec, -} - -impl ClientDeviceListAnnounceRequest { - fn new_smartcard() -> Self { - Self { - count: 1, - devices: vec![DeviceAnnounceHeader { - device_type: DeviceType::RDPDR_DTYP_SMARTCARD, - device_id: SCARD_DEVICE_ID, - // This name is a constant defined by the spec. - preferred_dos_name: "SCARD".to_string(), - device_data_length: 0, - device_data: vec![], - }], - } - } - - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u32::(self.count)?; - for dev in self.devices.iter() { - w.extend_from_slice(&dev.encode()?); - } - Ok(w) - } -} - -#[derive(Debug)] -struct DeviceAnnounceHeader { - device_type: DeviceType, - device_id: u32, - preferred_dos_name: String, - device_data_length: u32, - device_data: Vec, -} - -impl DeviceAnnounceHeader { - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u32::(self.device_type.to_u32().unwrap())?; - w.write_u32::(self.device_id)?; - let mut name: &str = &self.preferred_dos_name; - if name.len() > 8 { - name = &name[..8]; - } - w.extend_from_slice(&format!("{:\x00<8}", name).into_bytes()); - w.write_u32::(self.device_data_length)?; - w.extend_from_slice(&self.device_data); - Ok(w) - } -} - -#[derive(Debug, FromPrimitive, ToPrimitive)] -#[allow(non_camel_case_types)] -enum DeviceType { - RDPDR_DTYP_SERIAL = 0x00000001, - RDPDR_DTYP_PARALLEL = 0x00000002, - RDPDR_DTYP_PRINT = 0x00000004, - RDPDR_DTYP_FILESYSTEM = 0x00000008, - RDPDR_DTYP_SMARTCARD = 0x00000020, -} - -#[derive(Debug)] -struct ServerDeviceAnnounceResponse { - device_id: u32, - result_code: u32, -} - -impl ServerDeviceAnnounceResponse { - fn decode(payload: &mut Payload) -> RdpResult { - Ok(Self { - device_id: payload.read_u32::()?, - result_code: payload.read_u32::()?, - }) - } -} - -#[derive(Debug)] -#[allow(dead_code)] -struct DeviceIoRequest { - device_id: u32, - file_id: u32, - completion_id: u32, - major_function: MajorFunction, - minor_function: MinorFunction, -} - -impl DeviceIoRequest { - fn decode(payload: &mut Payload) -> RdpResult { - let device_id = payload.read_u32::()?; - let file_id = payload.read_u32::()?; - let completion_id = payload.read_u32::()?; - let major_function = payload.read_u32::()?; - let minor_function = payload.read_u32::()?; - Ok(Self { - device_id, - file_id, - completion_id, - major_function: MajorFunction::from_u32(major_function).ok_or_else(|| { - invalid_data_error(&format!( - "invalid major function value {:#010x}", - major_function - )) - })?, - minor_function: MinorFunction::from_u32(minor_function).ok_or_else(|| { - invalid_data_error(&format!( - "invalid minor function value {:#010x}", - minor_function - )) - })?, - }) - } -} - -#[derive(Debug, FromPrimitive, ToPrimitive)] -#[allow(non_camel_case_types)] -enum MajorFunction { - IRP_MJ_CREATE = 0x00000000, - IRP_MJ_CLOSE = 0x00000002, - IRP_MJ_READ = 0x00000003, - IRP_MJ_WRITE = 0x00000004, - IRP_MJ_DEVICE_CONTROL = 0x0000000E, - IRP_MJ_QUERY_VOLUME_INFORMATION = 0x0000000A, - IRP_MJ_SET_VOLUME_INFORMATION = 0x0000000B, - IRP_MJ_QUERY_INFORMATION = 0x00000005, - IRP_MJ_SET_INFORMATION = 0x00000006, - IRP_MJ_DIRECTORY_CONTROL = 0x0000000C, - IRP_MJ_LOCK_CONTROL = 0x00000011, -} - -#[derive(Debug, FromPrimitive, ToPrimitive)] -#[allow(non_camel_case_types)] -enum MinorFunction { - IRP_MN_NONE = 0x00000000, - IRP_MN_QUERY_DIRECTORY = 0x00000001, - IRP_MN_NOTIFY_CHANGE_DIRECTORY = 0x00000002, -} - -#[derive(Debug)] -#[allow(dead_code)] -struct DeviceControlRequest { - header: DeviceIoRequest, - output_buffer_length: u32, - input_buffer_length: u32, - io_control_code: u32, - padding: [u8; 20], -} - -impl DeviceControlRequest { - fn decode(header: DeviceIoRequest, payload: &mut Payload) -> RdpResult { - let output_buffer_length = payload.read_u32::()?; - let input_buffer_length = payload.read_u32::()?; - let io_control_code = payload.read_u32::()?; - let mut padding: [u8; 20] = [0; 20]; - payload.read_exact(&mut padding)?; - Ok(Self { - header, - output_buffer_length, - input_buffer_length, - io_control_code, - padding, - }) - } -} - -#[derive(Debug)] -struct DeviceIoResponse { - device_id: u32, - completion_id: u32, - io_status: u32, -} - -impl DeviceIoResponse { - fn new(req: &DeviceIoRequest, io_status: u32) -> Self { - Self { - device_id: req.device_id, - completion_id: req.completion_id, - io_status, - } - } - - fn encode(&self) -> RdpResult> { - let mut w = vec![]; - w.write_u32::(self.device_id)?; - w.write_u32::(self.completion_id)?; - w.write_u32::(self.io_status)?; - Ok(w) - } -} - -#[derive(Debug)] -struct DeviceControlResponse { - header: DeviceIoResponse, - output_buffer_length: u32, - output_buffer: Vec, -} - -impl DeviceControlResponse { - fn new(req: &DeviceControlRequest, io_status: u32, output: Vec) -> Self { - Self { - header: DeviceIoResponse::new(&req.header, io_status), - output_buffer_length: output.length() as u32, - output_buffer: output, - } - } - - fn encode(&mut self) -> RdpResult> { - let mut w = vec![]; - w.extend_from_slice(&self.header.encode()?); - w.write_u32::(self.output_buffer_length)?; - w.extend_from_slice(&self.output_buffer); - Ok(w) - } -} diff --git a/lib/srv/desktop/rdp/rdpclient/src/rdpdr/consts.rs b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/consts.rs new file mode 100644 index 0000000000000..b8d572febd3cd --- /dev/null +++ b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/consts.rs @@ -0,0 +1,167 @@ +// Copyright 2022 Gravitational, Inc +// +// Licensed 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. + +pub const CHANNEL_NAME: &str = "rdpdr"; + +// Each redirected device requires a unique ID. We only share +// one permanent smartcard device, so we can give it hardcoded ID 1. +pub const SCARD_DEVICE_ID: u32 = 1; + +pub const VERSION_MAJOR: u16 = 0x0001; +pub const VERSION_MINOR: u16 = 0x000c; + +pub const SMARTCARD_CAPABILITY_VERSION_01: u32 = 0x00000001; +pub const DRIVE_CAPABILITY_VERSION_02: u32 = 0x00000002; +#[allow(dead_code)] +pub const GENERAL_CAPABILITY_VERSION_01: u32 = 0x00000001; +pub const GENERAL_CAPABILITY_VERSION_02: u32 = 0x00000002; + +#[derive(Debug, FromPrimitive, ToPrimitive)] +#[allow(non_camel_case_types)] +pub enum Component { + RDPDR_CTYP_CORE = 0x4472, + RDPDR_CTYP_PRN = 0x5052, +} + +#[derive(Debug, FromPrimitive, ToPrimitive)] +#[allow(non_camel_case_types)] +pub enum PacketId { + PAKID_CORE_SERVER_ANNOUNCE = 0x496E, + PAKID_CORE_CLIENTID_CONFIRM = 0x4343, + PAKID_CORE_CLIENT_NAME = 0x434E, + PAKID_CORE_DEVICELIST_ANNOUNCE = 0x4441, + PAKID_CORE_DEVICE_REPLY = 0x6472, + PAKID_CORE_DEVICE_IOREQUEST = 0x4952, + PAKID_CORE_DEVICE_IOCOMPLETION = 0x4943, + PAKID_CORE_SERVER_CAPABILITY = 0x5350, + PAKID_CORE_CLIENT_CAPABILITY = 0x4350, + PAKID_CORE_DEVICELIST_REMOVE = 0x444D, + PAKID_PRN_CACHE_DATA = 0x5043, + PAKID_CORE_USER_LOGGEDON = 0x554C, + PAKID_PRN_USING_XPS = 0x5543, +} + +#[derive(Debug, FromPrimitive, ToPrimitive)] +#[allow(non_camel_case_types)] +pub enum CapabilityType { + CAP_GENERAL_TYPE = 0x0001, + CAP_PRINTER_TYPE = 0x0002, + CAP_PORT_TYPE = 0x0003, + CAP_DRIVE_TYPE = 0x0004, + CAP_SMARTCARD_TYPE = 0x0005, +} + +#[derive(Debug, FromPrimitive, ToPrimitive)] +#[allow(non_camel_case_types)] +pub enum DeviceType { + RDPDR_DTYP_SERIAL = 0x00000001, + RDPDR_DTYP_PARALLEL = 0x00000002, + RDPDR_DTYP_PRINT = 0x00000004, + RDPDR_DTYP_FILESYSTEM = 0x00000008, + RDPDR_DTYP_SMARTCARD = 0x00000020, +} + +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/a087ffa8-d0d5-4874-ac7b-0494f63e2d5d +#[derive(Debug, FromPrimitive, ToPrimitive, PartialEq, Clone)] +#[allow(non_camel_case_types)] +pub enum MajorFunction { + IRP_MJ_CREATE = 0x00000000, + IRP_MJ_CLOSE = 0x00000002, + IRP_MJ_READ = 0x00000003, + IRP_MJ_WRITE = 0x00000004, + IRP_MJ_DEVICE_CONTROL = 0x0000000E, + IRP_MJ_QUERY_VOLUME_INFORMATION = 0x0000000A, + IRP_MJ_SET_VOLUME_INFORMATION = 0x0000000B, + IRP_MJ_QUERY_INFORMATION = 0x00000005, + IRP_MJ_SET_INFORMATION = 0x00000006, + IRP_MJ_DIRECTORY_CONTROL = 0x0000000C, + IRP_MJ_LOCK_CONTROL = 0x00000011, +} + +#[derive(Debug, FromPrimitive, ToPrimitive, Clone)] +#[allow(non_camel_case_types)] +pub enum MinorFunction { + IRP_MN_NONE = 0x00000000, + IRP_MN_QUERY_DIRECTORY = 0x00000001, + IRP_MN_NOTIFY_CHANGE_DIRECTORY = 0x00000002, +} + +/// Windows defines an absolutely massive list of potential NTSTATUS values. +/// This enum includes the basic ones we support for communicating with the windows machine. +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-erref/596a1078-e883-4972-9bbc-49e60bebca55 +#[derive(ToPrimitive, Debug)] +#[repr(u32)] +#[allow(non_camel_case_types)] +#[allow(dead_code)] +pub enum NTSTATUS { + STATUS_SUCCESS = 0x00000000, + STATUS_UNSUCCESSFUL = 0xC0000001, + STATUS_NOT_IMPLEMENTED = 0xC0000002, + STATUS_NO_MORE_FILES = 0x80000006, +} + +/// 2.4 File Information Classes [MS-FSCC] +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/4718fc40-e539-4014-8e33-b675af74e3e1 +#[derive(FromPrimitive, Debug, PartialEq)] +#[repr(u32)] +#[allow(clippy::enum_variant_names)] +pub enum FsInformationClassLevel { + FileAccessInformation = 8, + FileAlignmentInformation = 17, + FileAllInformation = 18, + FileAllocationInformation = 19, + FileAlternateNameInformation = 21, + FileAttributeTagInformation = 35, + FileBasicInformation = 4, + FileBothDirectoryInformation = 3, + FileCompressionInformation = 28, + FileDirectoryInformation = 1, + FileDispositionInformation = 13, + FileEaInformation = 7, + FileEndOfFileInformation = 20, + FileFullDirectoryInformation = 2, + FileFullEaInformation = 15, + FileHardLinkInformation = 46, + FileIdBothDirectoryInformation = 37, + FileIdExtdDirectoryInformation = 60, + FileIdFullDirectoryInformation = 38, + FileIdGlobalTxDirectoryInformation = 50, + FileIdInformation = 59, + FileInternalInformation = 6, + FileLinkInformation = 11, + FileMailslo = 26, + FileMailslotSetInformation = 27, + FileModeInformation = 16, + FileMoveClusterInformation = 31, + FileNameInformation = 9, + FileNamesInformation = 12, + FileNetworkOpenInformation = 34, + FileNormalizedNameInformation = 48, + FileObjectIdInformation = 29, + FilePipeInformation = 23, + FilePipInformation = 24, + FilePipeRemoteInformation = 25, + FilePositionInformation = 14, + FileQuotaInformation = 32, + FileRenameInformation = 10, + FileReparsePointInformation = 33, + FileSfioReserveInformation = 44, + FileSfioVolumeInformation = 45, + FileShortNameInformation = 40, + FileStandardInformation = 5, + FileStandardLinkInformation = 54, + FileStreamInformation = 22, + FileTrackingInformation = 36, + FileValidDataLengthInformation = 39, +} diff --git a/lib/srv/desktop/rdp/rdpclient/src/rdpdr/flags.rs b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/flags.rs new file mode 100644 index 0000000000000..6bff374d8da2e --- /dev/null +++ b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/flags.rs @@ -0,0 +1,200 @@ +// Copyright 2022 Gravitational, Inc +// +// Licensed 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. + +use bitflags::bitflags; + +bitflags! { + /// DesiredAccess can be interpreted as either + /// 2.2.13.1.1 File_Pipe_Printer_Access_Mask [MS-SMB2] (https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-smb2/77b36d0f-6016-458a-a7a0-0f4a72ae1534) + /// or + /// 2.2.13.1.2 Directory_Access_Mask [MS-SMB2] (https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-smb2/0a5934b1-80f1-4da0-b1bf-5e021c309b71) + /// + /// This implements the combination of the two. For flags where the names and/or functions are distinct between the two, + /// the names are appended with an "_OR_", and the File_Pipe_Printer_Access_Mask functionality is described on the top line comment, + /// and the Directory_Access_Mask functionality is described on the bottom (2nd) line comment. + pub struct DesiredAccess: u32 { + /// This value indicates the right to read data from the file or named pipe. + /// This value indicates the right to enumerate the contents of the directory. + const FILE_READ_DATA_OR_FILE_LIST_DIRECTORY = 0x00000001; + /// This value indicates the right to write data into the file or named pipe beyond the end of the file. + /// This value indicates the right to create a file under the directory. + const FILE_WRITE_DATA_OR_FILE_ADD_FILE = 0x00000002; + /// This value indicates the right to append data into the file or named pipe. + /// This value indicates the right to add a sub-directory under the directory. + const FILE_APPEND_DATA_OR_FILE_ADD_SUBDIRECTORY = 0x00000004; + /// This value indicates the right to read the extended attributes of the file or named pipe. + const FILE_READ_EA = 0x00000008; + /// This value indicates the right to write or change the extended attributes to the file or named pipe. + const FILE_WRITE_EA = 0x00000010; + /// This value indicates the right to traverse this directory if the server enforces traversal checking. + const FILE_TRAVERSE = 0x00000020; + /// This value indicates the right to delete entries within a directory. + const FILE_DELETE_CHILD = 0x00000040; + /// This value indicates the right to execute the file/directory. + const FILE_EXECUTE = 0x00000020; + /// This value indicates the right to read the attributes of the file/directory. + const FILE_READ_ATTRIBUTES = 0x00000080; + /// This value indicates the right to change the attributes of the file/directory. + const FILE_WRITE_ATTRIBUTES = 0x00000100; + /// This value indicates the right to delete the file/directory. + const DELETE = 0x00010000; + /// This value indicates the right to read the security descriptor for the file/directory or named pipe. + const READ_CONTROL = 0x00020000; + /// This value indicates the right to change the discretionary access control list (DACL) in the security descriptor for the file/directory or named pipe. For the DACL data pub structure, see ACL in [MS-DTYP]. + const WRITE_DAC = 0x00040000; + /// This value indicates the right to change the owner in the security descriptor for the file/directory or named pipe. + const WRITE_OWNER = 0x00080000; + /// SMB2 clients set this flag to any value. SMB2 servers SHOULD ignore this flag. + const SYNCHRONIZE = 0x00100000; + /// This value indicates the right to read or change the system access control list (SACL) in the security descriptor for the file/directory or named pipe. For the SACL data pub structure, see ACL in [MS-DTYP]. + const ACCESS_SYSTEM_SECURITY = 0x01000000; + /// This value indicates that the client is requesting an open to the file with the highest level of access the client has on this file. If no access is granted for the client on this file, the server MUST fail the open with STATUS_ACCESS_DENIED. + const MAXIMUM_ALLOWED = 0x02000000; + /// This value indicates a request for all the access flags that are previously listed except MAXIMUM_ALLOWED and ACCESS_SYSTEM_SECURITY. + const GENERIC_ALL = 0x10000000; + /// This value indicates a request for the following combination of access flags listed above: FILE_READ_ATTRIBUTES| FILE_EXECUTE| SYNCHRONIZE| READ_CONTROL. + const GENERIC_EXECUTE = 0x20000000; + /// This value indicates a request for the following combination of access flags listed above: FILE_WRITE_DATA| FILE_APPEND_DATA| FILE_WRITE_ATTRIBUTES| FILE_WRITE_EA| SYNCHRONIZE| READ_CONTROL. + const GENERIC_WRITE = 0x40000000; + /// This value indicates a request for the following combination of access flags listed above: FILE_READ_DATA| FILE_READ_ATTRIBUTES| FILE_READ_EA| SYNCHRONIZE| READ_CONTROL. + const GENERIC_READ = 0x80000000; + } +} + +bitflags! { + /// 2.6 File Attributes [MS-FSCC] + /// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/ca28ec38-f155-4768-81d6-4bfeb8586fc9 + pub struct FileAttributes: u32 { + const FILE_ATTRIBUTE_READONLY = 0x00000001; + const FILE_ATTRIBUTE_HIDDEN = 0x00000002; + const FILE_ATTRIBUTE_SYSTEM = 0x00000004; + const FILE_ATTRIBUTE_DIRECTORY = 0x00000010; + const FILE_ATTRIBUTE_ARCHIVE = 0x00000020; + const FILE_ATTRIBUTE_NORMAL = 0x00000080; + const FILE_ATTRIBUTE_TEMPORARY = 0x00000100; + const FILE_ATTRIBUTE_SPARSE_FILE = 0x00000200; + const FILE_ATTRIBUTE_REPARSE_POINT = 0x00000400; + const FILE_ATTRIBUTE_COMPRESSED = 0x00000800; + const FILE_ATTRIBUTE_OFFLINE = 0x00001000; + const FILE_ATTRIBUTE_NOT_CONTENT_INDEXED = 0x00002000; + const FILE_ATTRIBUTE_ENCRYPTED = 0x00004000; + const FILE_ATTRIBUTE_INTEGRITY_STREAM = 0x00008000; + const FILE_ATTRIBUTE_NO_SCRUB_DATA = 0x00020000; + const FILE_ATTRIBUTE_RECALL_ON_OPEN = 0x00040000; + const FILE_ATTRIBUTE_PINNED = 0x00080000; + const FILE_ATTRIBUTE_UNPINNED = 0x00100000; + const FILE_ATTRIBUTE_RECALL_ON_DATA_ACCESS = 0x00400000; + } +} + +bitflags! { + /// Specifies the sharing mode for the open. If ShareAccess values of FILE_SHARE_READ, FILE_SHARE_WRITE and FILE_SHARE_DELETE are set for a printer file or a named pipe, the server SHOULD<35> ignore these values. The field MUST be pub constructed using a combination of zero or more of the following bit values. + /// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-smb2/e8fb45c1-a03d-44ca-b7ae-47385cfd7997 + pub struct SharedAccess: u32 { + const FILE_SHARE_READ = 0x00000001; + const FILE_SHARE_WRITE = 0x00000002; + const FILE_SHARE_DELETE = 0x00000004; + } +} + +bitflags! { + /// Defines the action the server MUST take if the file that is specified in the name field already exists. For opening named pipes, this field can be set to any value by the client and MUST be ignored by the server. For other files, this field MUST contain one of the following values. + /// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-smb2/e8fb45c1-a03d-44ca-b7ae-47385cfd7997 + /// See https://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_file.c#L207 + /// for information about how these should be interpreted. + pub struct CreateDisposition: u32 { + const FILE_SUPERSEDE = 0x00000000; + const FILE_OPEN = 0x00000001; + const FILE_CREATE = 0x00000002; + const FILE_OPEN_IF = 0x00000003; + const FILE_OVERWRITE = 0x00000004; + const FILE_OVERWRITE_IF = 0x00000005; + } +} + +bitflags! { + /// Specifies the options to be applied when creating or opening the file. Combinations of the bit positions listed below are valid, unless otherwise noted. This field MUST be pub constructed using the following values. + /// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-smb2/e8fb45c1-a03d-44ca-b7ae-47385cfd7997 + pub struct CreateOptions: u32 { + const FILE_DIRECTORY_FILE = 0x00000001; + const FILE_WRITE_THROUGH = 0x00000002; + const FILE_SEQUENTIAL_ONLY = 0x00000004; + const FILE_NO_INTERMEDIATE_BUFFERING = 0x00000008; + const FILE_SYNCHRONOUS_IO_ALERT = 0x00000010; + const FILE_SYNCHRONOUS_IO_NONALERT = 0x00000020; + const FILE_NON_DIRECTORY_FILE = 0x00000040; + const FILE_COMPLETE_IF_OPLOCKED = 0x00000100; + const FILE_NO_EA_KNOWLEDGE = 0x00000200; + const FILE_RANDOM_ACCESS = 0x00000800; + const FILE_DELETE_ON_CLOSE = 0x00001000; + const FILE_OPEN_BY_FILE_ID = 0x00002000; + const FILE_OPEN_FOR_BACKUP_INTENT = 0x00004000; + const FILE_NO_COMPRESSION = 0x00008000; + const FILE_OPEN_REMOTE_INSTANCE = 0x00000400; + const FILE_OPEN_REQUIRING_OPLOCK = 0x00010000; + const FILE_DISALLOW_EXCLUSIVE = 0x00020000; + const FILE_RESERVE_OPFILTER = 0x00100000; + const FILE_OPEN_REPARSE_POINT = 0x00200000; + const FILE_OPEN_NO_RECALL = 0x00400000; + const FILE_OPEN_FOR_FREE_SPACE_QUERY = 0x00800000; + } +} + +bitflags! { + /// An unsigned 8-bit integer. This field indicates the success of the Device Create Request (section 2.2.1.4.1). + /// The value of the Information field depends on the value of CreateDisposition field in the Device Create Request + /// (section 2.2.1.4.1). If the IoStatus field is set to 0x00000000, this field MAY be skipped, in which case the + /// server MUST assume that the Information field is set to 0x00. + /// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/99e5fca5-b37a-41e4-bc69-8d7da7860f76 + pub struct Information: u8 { + /// A new file was created. + const FILE_SUPERSEDED = 0x00000000; + /// An existing file was opened. + const FILE_OPENED = 0x00000001; + /// An existing file was overwritten. + const FILE_OVERWRITTEN = 0x00000003; + } +} + +bitflags! { + /// Specifies the types of changes to monitor. It is valid to choose multiple trigger conditions. + /// In this case, if any condition is met, the client is notified of the change and the CHANGE_NOTIFY operation is completed. + /// See CompletionFilter at: https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-smb2/598f395a-e7a2-4cc8-afb3-ccb30dd2df7c + pub struct CompletionFilter: u32 { + /// The client is notified if a file-name changes. + const FILE_NOTIFY_CHANGE_FILE_NAME = 0x00000001; + /// The client is notified if a directory name changes. + const FILE_NOTIFY_CHANGE_DIR_NAME = 0x00000002; + /// The client is notified if a file's attributes change. Possible file attribute values are specified in [MS-FSCC] section 2.6. + const FILE_NOTIFY_CHANGE_ATTRIBUTES = 0x00000004; + /// The client is notified if a file's size changes. + const FILE_NOTIFY_CHANGE_SIZE = 0x00000008; + /// The client is notified if the last write time of a file changes. + const FILE_NOTIFY_CHANGE_LAST_WRITE = 0x00000010; + /// The client is notified if the last access time of a file changes. + const FILE_NOTIFY_CHANGE_LAST_ACCESS = 0x00000020; + /// The client is notified if the creation time of a file changes. + const FILE_NOTIFY_CHANGE_CREATION = 0x00000040; + /// The client is notified if a file's extended attributes (EAs) change. + const FILE_NOTIFY_CHANGE_EA = 0x00000080; + /// The client is notified of a file's access control list (ACL) settings change. + const FILE_NOTIFY_CHANGE_SECURITY = 0x00000100; + /// The client is notified if a named stream is added to a file. + const FILE_NOTIFY_CHANGE_STREAM_NAME = 0x00000200; + /// The client is notified if the size of a named stream is changed. + const FILE_NOTIFY_CHANGE_STREAM_SIZE = 0x00000400; + /// The client is notified if a named stream is modified. + const FILE_NOTIFY_CHANGE_STREAM_WRITE = 0x00000800; + } +} diff --git a/lib/srv/desktop/rdp/rdpclient/src/rdpdr/mod.rs b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/mod.rs new file mode 100644 index 0000000000000..3c40557ca2fb8 --- /dev/null +++ b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/mod.rs @@ -0,0 +1,1655 @@ +// Copyright 2021 Gravitational, Inc +// +// Licensed 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. + +mod consts; +mod flags; +mod scard; + +use crate::errors::{ + invalid_data_error, not_implemented_error, rejected_by_server_error, try_error, NTSTATUS_OK, + SPECIAL_NO_RESPONSE, +}; +use crate::util; +use crate::vchan; +use crate::{ + Payload, SharedDirectoryAcknowledge, SharedDirectoryInfoRequest, SharedDirectoryInfoResponse, +}; + +use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt}; +use consts::{ + CapabilityType, Component, DeviceType, FsInformationClassLevel, MajorFunction, MinorFunction, + PacketId, DRIVE_CAPABILITY_VERSION_02, GENERAL_CAPABILITY_VERSION_02, NTSTATUS, + SCARD_DEVICE_ID, SMARTCARD_CAPABILITY_VERSION_01, VERSION_MAJOR, VERSION_MINOR, +}; +use num_traits::{FromPrimitive, ToPrimitive}; +use rdp::core::mcs; +use rdp::core::tpkt; +use rdp::model::data::Message; +use rdp::model::error::Error as RdpError; +use rdp::model::error::*; +use std::collections::HashMap; +use std::convert::{TryFrom, TryInto}; +use std::io::{Read, Write}; + +pub use consts::CHANNEL_NAME; + +/// Client implements a device redirection (RDPDR) client, as defined in +/// https://winprotocoldoc.blob.core.windows.net/productionwindowsarchives/MS-RDPEFS/%5bMS-RDPEFS%5d.pdf +/// +/// This client only supports a single smartcard device. +pub struct Client { + vchan: vchan::Client, + scard: scard::Client, + + allow_directory_sharing: bool, + active_device_ids: Vec, + + // Functions for sending tdp messages to the browser client. + tdp_sd_acknowledge: Box RdpResult<()>>, + tdp_sd_info_request: Box RdpResult<()>>, + + // Completion-id-indexed maps of handlers for tdp messages coming from the browser client. + pending_sd_info_resp_handlers: HashMap, +} + +impl Client { + pub fn new( + cert_der: Vec, + key_der: Vec, + pin: String, + allow_directory_sharing: bool, + + tdp_sd_acknowledge: Box RdpResult<()>>, + tdp_sd_info_request: Box RdpResult<()>>, + ) -> Self { + if allow_directory_sharing { + debug!("creating rdpdr client with directory sharing enabled") + } else { + debug!("creating rdpdr client with directory sharing disabled") + } + Client { + vchan: vchan::Client::new(), + scard: scard::Client::new(cert_der, key_der, pin), + active_device_ids: vec![], + allow_directory_sharing, + + tdp_sd_acknowledge, + tdp_sd_info_request, + + pending_sd_info_resp_handlers: HashMap::new(), + } + } + /// Reads raw RDP messages sent on the rdpdr virtual channel and replies as necessary. + pub fn read_and_reply( + &mut self, + payload: tpkt::Payload, + mcs: &mut mcs::Client, + ) -> RdpResult<()> { + if let Some(mut payload) = self.vchan.read(payload)? { + let header = SharedHeader::decode(&mut payload)?; + if let Component::RDPDR_CTYP_PRN = header.component { + warn!("got {:?} RDPDR header from RDP server, ignoring because we're not redirecting any printers", header); + return Ok(()); + } + let responses = match header.packet_id { + PacketId::PAKID_CORE_SERVER_ANNOUNCE => { + self.handle_server_announce(&mut payload)? + } + PacketId::PAKID_CORE_SERVER_CAPABILITY => { + self.handle_server_capability(&mut payload)? + } + PacketId::PAKID_CORE_CLIENTID_CONFIRM => { + self.handle_client_id_confirm(&mut payload)? + } + PacketId::PAKID_CORE_DEVICE_REPLY => self.handle_device_reply(&mut payload)?, + // Device IO request is where communication with the smartcard and shared drive actually happens. + // Everything up to this point was negotiation (and smartcard device registration). + PacketId::PAKID_CORE_DEVICE_IOREQUEST => { + self.handle_device_io_request(&mut payload)? + } + _ => { + // We don't implement the full set of messages. + error!( + "RDPDR packets {:?} are not implemented yet, ignoring", + header.packet_id + ); + vec![] + } + }; + + let chan = &CHANNEL_NAME.to_string(); + for resp in responses { + mcs.write(chan, resp)?; + } + } + Ok(()) + } + + fn handle_server_announce(&self, payload: &mut Payload) -> RdpResult>> { + let req = ServerAnnounceRequest::decode(payload)?; + debug!("got ServerAnnounceRequest {:?}", req); + + let resp = self.add_headers_and_chunkify( + PacketId::PAKID_CORE_CLIENTID_CONFIRM, + ClientAnnounceReply::new(req).encode()?, + )?; + debug!("sending client announce reply"); + Ok(resp) + } + + fn handle_server_capability(&self, payload: &mut Payload) -> RdpResult>> { + let req = ServerCoreCapabilityRequest::decode(payload)?; + debug!("got {:?}", req); + + let resp = self.add_headers_and_chunkify( + PacketId::PAKID_CORE_CLIENT_CAPABILITY, + ClientCoreCapabilityResponse::new_response(self.allow_directory_sharing).encode()?, + )?; + debug!("sending client core capability response"); + Ok(resp) + } + + fn handle_client_id_confirm(&mut self, payload: &mut Payload) -> RdpResult>> { + let req = ServerClientIdConfirm::decode(payload)?; + debug!("got ServerClientIdConfirm {:?}", req); + + // The smartcard initialization sequence that contains this message happens once at session startup, + // and once when login succeeds. We only need to announce the smartcard once. + let resp = if !self.active_device_ids.contains(&SCARD_DEVICE_ID) { + self.push_active_device_id(SCARD_DEVICE_ID)?; + self.add_headers_and_chunkify( + PacketId::PAKID_CORE_DEVICELIST_ANNOUNCE, + ClientDeviceListAnnounceRequest::new_smartcard(SCARD_DEVICE_ID).encode()?, + )? + } else { + self.add_headers_and_chunkify( + PacketId::PAKID_CORE_DEVICELIST_ANNOUNCE, + ClientDeviceListAnnounceRequest::new_empty().encode()?, + )? + }; + debug!("replying with: {:?}", resp); + Ok(resp) + } + + fn handle_device_reply(&self, payload: &mut Payload) -> RdpResult>> { + let req = ServerDeviceAnnounceResponse::decode(payload)?; + debug!("got ServerDeviceAnnounceResponse: {:?}", req); + + if self.active_device_ids.contains(&req.device_id) { + if req.device_id != self.get_scard_device_id()? { + // This was for a directory we're sharing over TDP + let mut err_code: u32 = 0; + if req.result_code != NTSTATUS_OK { + err_code = 1; + debug!("ServerDeviceAnnounceResponse for smartcard redirection failed with result code NTSTATUS({})", &req.result_code); + } else { + debug!("ServerDeviceAnnounceResponse for shared directory succeeded") + } + + (self.tdp_sd_acknowledge)(SharedDirectoryAcknowledge { + err_code, + directory_id: req.device_id, + })?; + } else { + // This was for the smart card + if req.result_code != NTSTATUS_OK { + // End the session, we cannot continue without + // the smart card being redirected. + return Err(rejected_by_server_error(&format!( + "ServerDeviceAnnounceResponse for smartcard redirection failed with result code NTSTATUS({})", + &req.result_code + ))); + } + debug!("ServerDeviceAnnounceResponse for smartcard redirection succeeded"); + } + } else { + return Err(invalid_data_error(&format!( + "got ServerDeviceAnnounceResponse for unknown device_id {}", + &req.device_id + ))); + } + Ok(vec![]) + } + + fn handle_device_io_request(&mut self, payload: &mut Payload) -> RdpResult>> { + let device_io_request = DeviceIoRequest::decode(payload)?; + let major_function = device_io_request.major_function.clone(); + + // Smartcard control only uses IRP_MJ_DEVICE_CONTROL; directory control uses IRP_MJ_DEVICE_CONTROL along with + // all the other MajorFunctions supported by this Client. Therefore if we receive any major function when drive + // redirection is not allowed, something has gone wrong. In such a case, we return an error as a security measure + // to ensure directories are never shared when RBAC doesn't permit it. + if major_function != MajorFunction::IRP_MJ_DEVICE_CONTROL && !self.allow_directory_sharing { + return Err(Error::TryError( + "received a drive redirection major function when drive redirection was not allowed" + .to_string(), + )); + } + + match major_function { + MajorFunction::IRP_MJ_DEVICE_CONTROL => { + let ioctl = DeviceControlRequest::decode(device_io_request, payload)?; + let is_smart_card_op = ioctl.header.device_id == self.get_scard_device_id()?; + debug!("got: {:?}", ioctl); + + // IRP_MJ_DEVICE_CONTROL is the one major function used by both the smartcard controller (always enabled) + // and shared directory controller (potentially disabled by RBAC). Here we check that directory sharing + // is enabled here before proceeding with any shared directory controls as an additional security measure. + if !is_smart_card_op && !self.allow_directory_sharing { + return Err(Error::TryError("received a drive redirection major function when drive redirection was not allowed".to_string())); + } + let resp = if is_smart_card_op { + // Smart card control + let (code, res) = self.scard.ioctl(ioctl.io_control_code, payload)?; + if code == SPECIAL_NO_RESPONSE { + return Ok(vec![]); + } + DeviceControlResponse::new(&ioctl, code, res) + } else { + // Drive redirection, mimic FreeRDP's "no-op" + // https://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_main.c#L677-L684 + DeviceControlResponse::new( + &ioctl, + NTSTATUS::STATUS_SUCCESS.to_u32().unwrap(), + vec![], + ) + }; + debug!("replying with: {:?}", resp); + let resp = self.add_headers_and_chunkify( + PacketId::PAKID_CORE_DEVICE_IOCOMPLETION, + resp.encode()?, + )?; + debug!("sending device IO response"); + Ok(resp) + } + MajorFunction::IRP_MJ_CREATE => { + let rdp_req = ServerCreateDriveRequest::decode(device_io_request, payload)?; + debug!("got: {:?}", rdp_req); + + // Send a TDP Shared Directory Info Request + (self.tdp_sd_info_request)(SharedDirectoryInfoRequest::from(rdp_req.clone()))?; + + // Add a TDP Shared Directory Info Response handler to the handler cache. + // When we receive a TDP Shared Directory Info Response with this completion_id, + // this handler will be called. + self.pending_sd_info_resp_handlers.insert( + rdp_req.device_io_request.completion_id, + Box::new( + |_cli: &mut Self, + res: SharedDirectoryInfoResponse| + -> RdpResult>> { + let _rdp_req = rdp_req; + debug!("got {:?}", res); + // TODO(isaiah): see https://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_file.c#L207 + + Ok(vec![]) + }, + ), + ); + Ok(vec![]) + } + _ => Err(invalid_data_error(&format!( + // TODO(isaiah): send back a not implemented response(?) + "got unsupported major_function in DeviceIoRequest: {:?}", + &major_function + ))), + } + } + + /// This is called from Go (in effect) to announce a new directory + /// for sharing. + pub fn write_client_device_list_announce( + &mut self, + req: ClientDeviceListAnnounce, + mcs: &mut mcs::Client, + ) -> RdpResult<()> { + self.push_active_device_id(req.device_list[0].device_id)?; + debug!("sending new drive for redirection: {:?}", req); + + let responses = + self.add_headers_and_chunkify(PacketId::PAKID_CORE_DEVICELIST_ANNOUNCE, req.encode()?)?; + let chan = &CHANNEL_NAME.to_string(); + for resp in responses { + mcs.write(chan, resp)?; + } + + Ok(()) + } + + pub fn handle_tdp_sd_info_response( + &mut self, + res: SharedDirectoryInfoResponse, + mcs: &mut mcs::Client, + ) -> RdpResult<()> { + if let Some(tdp_resp_handler) = self + .pending_sd_info_resp_handlers + .remove(&res.completion_id) + { + let rdp_responses = tdp_resp_handler(self, res)?; + let chan = &CHANNEL_NAME.to_string(); + for resp in rdp_responses { + mcs.write(chan, resp)?; + } + Ok(()) + } else { + return Err(try_error(&format!( + "received invalid completion id: {}", + res.completion_id + ))); + } + } + + /// add_headers_and_chunkify takes an encoded PDU ready to be sent over a virtual channel (payload), + /// adds on the Shared Header based the passed packet_id, adds the appropriate (virtual) Channel PDU Header, + /// and splits the entire payload into chunks if the payload exceeds the maximum size. + fn add_headers_and_chunkify( + &self, + packet_id: PacketId, + payload: Vec, + ) -> RdpResult>> { + let mut inner = SharedHeader::new(Component::RDPDR_CTYP_CORE, packet_id).encode()?; + inner.extend_from_slice(&payload); + self.vchan.add_header_and_chunkify(None, inner) + } + + fn push_active_device_id(&mut self, device_id: u32) -> RdpResult<()> { + if self.active_device_ids.contains(&device_id) { + return Err(RdpError::TryError(format!( + "attempted to add a duplicate device_id {} to active_device_ids {:?}", + device_id, self.active_device_ids + ))); + } + self.active_device_ids.push(device_id); + Ok(()) + } + + fn get_scard_device_id(&self) -> RdpResult { + // We always push it into the list first + if !self.active_device_ids.is_empty() { + return Ok(self.active_device_ids[0]); + } + Err(RdpError::TryError("no active device ids".to_string())) + } +} + +/// 2.2.1.1 Shared Header (RDPDR_HEADER) +/// This header is present at the beginning of every message in sent over the rdpdr virtual channel. +/// The purpose of this header is to describe the type of the message. +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/29d4108f-8163-4a67-8271-e48c4b9c2a7c +#[derive(Debug)] +struct SharedHeader { + component: Component, + packet_id: PacketId, +} + +impl SharedHeader { + fn new(component: Component, packet_id: PacketId) -> Self { + Self { + component, + packet_id, + } + } + fn decode(payload: &mut Payload) -> RdpResult { + let component = payload.read_u16::()?; + let packet_id = payload.read_u16::()?; + Ok(Self { + component: Component::from_u16(component).ok_or_else(|| { + invalid_data_error(&format!("invalid component value {:#06x}", component)) + })?, + packet_id: PacketId::from_u16(packet_id).ok_or_else(|| { + invalid_data_error(&format!("invalid packet_id value {:#06x}", packet_id)) + })?, + }) + } + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u16::(self.component.to_u16().unwrap())?; + w.write_u16::(self.packet_id.to_u16().unwrap())?; + Ok(w) + } +} + +type ServerAnnounceRequest = ClientIdMessage; +type ClientAnnounceReply = ClientIdMessage; +type ServerClientIdConfirm = ClientIdMessage; + +#[derive(Debug)] +struct ClientIdMessage { + version_major: u16, + version_minor: u16, + client_id: u32, +} + +impl ClientIdMessage { + fn new(req: ServerAnnounceRequest) -> Self { + Self { + version_major: VERSION_MAJOR, + version_minor: VERSION_MINOR, + client_id: req.client_id, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u16::(self.version_major)?; + w.write_u16::(self.version_minor)?; + w.write_u32::(self.client_id)?; + Ok(w) + } + + fn decode(payload: &mut Payload) -> RdpResult { + Ok(Self { + version_major: payload.read_u16::()?, + version_minor: payload.read_u16::()?, + client_id: payload.read_u32::()?, + }) + } +} + +#[derive(Debug)] +struct ServerCoreCapabilityRequest { + num_capabilities: u16, + padding: u16, + capabilities: Vec, +} + +impl ServerCoreCapabilityRequest { + fn new_response(allow_directory_sharing: bool) -> Self { + // Clients are always required to send the "general" capability set. + // In addition, we also send the optional smartcard capability (CAP_SMARTCARD_TYPE) + // and drive capability (CAP_DRIVE_TYPE). + let mut capabilities = vec![ + CapabilitySet { + header: CapabilityHeader { + cap_type: CapabilityType::CAP_GENERAL_TYPE, + length: 8 + 36, // 8 byte header + 36 byte capability descriptor + version: GENERAL_CAPABILITY_VERSION_02, + }, + data: Capability::General(GeneralCapabilitySet { + os_type: 0, + os_version: 0, + protocol_major_version: VERSION_MAJOR, + protocol_minor_version: VERSION_MINOR, + io_code_1: 0x00007fff, // Combination of all the required bits. + io_code_2: 0, + extended_pdu: 0x00000001 | 0x00000002, // RDPDR_DEVICE_REMOVE_PDUS | RDPDR_CLIENT_DISPLAY_NAME_PDU + extra_flags_1: 0, + extra_flags_2: 0, + special_type_device_cap: 1, // Request redirection of 1 special device - smartcard. + }), + }, + CapabilitySet { + header: CapabilityHeader { + cap_type: CapabilityType::CAP_SMARTCARD_TYPE, + length: 8, // 8 byte header + empty capability descriptor + version: SMARTCARD_CAPABILITY_VERSION_01, + }, + data: Capability::Smartcard, + }, + ]; + + if allow_directory_sharing { + capabilities.push(CapabilitySet { + header: CapabilityHeader { + cap_type: CapabilityType::CAP_DRIVE_TYPE, + length: 8, // 8 byte header + empty capability descriptor + version: DRIVE_CAPABILITY_VERSION_02, + }, + data: Capability::Drive, + }); + } + + Self { + padding: 0, + num_capabilities: capabilities.len() as u16, + capabilities, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u16::(self.num_capabilities)?; + w.write_u16::(self.padding)?; + for cap in self.capabilities.iter() { + w.extend_from_slice(&cap.encode()?); + } + Ok(w) + } + + fn decode(payload: &mut Payload) -> RdpResult { + let num_capabilities = payload.read_u16::()?; + let padding = payload.read_u16::()?; + let mut capabilities = vec![]; + for _ in 0..num_capabilities { + capabilities.push(CapabilitySet::decode(payload)?); + } + + Ok(Self { + num_capabilities, + padding, + capabilities, + }) + } +} + +#[derive(Debug)] +struct CapabilitySet { + header: CapabilityHeader, + data: Capability, +} + +impl CapabilitySet { + fn encode(&self) -> RdpResult> { + let mut w = self.header.encode()?; + w.extend_from_slice(&self.data.encode()?); + Ok(w) + } + fn decode(payload: &mut Payload) -> RdpResult { + let header = CapabilityHeader::decode(payload)?; + let data = Capability::decode(payload, &header)?; + + Ok(Self { header, data }) + } +} + +#[derive(Debug)] +struct CapabilityHeader { + cap_type: CapabilityType, + length: u16, + version: u32, +} + +impl CapabilityHeader { + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u16::(self.cap_type.to_u16().unwrap())?; + w.write_u16::(self.length)?; + w.write_u32::(self.version)?; + Ok(w) + } + fn decode(payload: &mut Payload) -> RdpResult { + let cap_type = payload.read_u16::()?; + Ok(Self { + cap_type: CapabilityType::from_u16(cap_type).ok_or_else(|| { + invalid_data_error(&format!("invalid capability type {:#06x}", cap_type)) + })?, + length: payload.read_u16::()?, + version: payload.read_u32::()?, + }) + } +} + +#[derive(Debug)] +enum Capability { + General(GeneralCapabilitySet), + Printer, + Port, + Drive, + Smartcard, +} + +impl Capability { + fn encode(&self) -> RdpResult> { + match self { + Capability::General(general) => Ok(general.encode()?), + _ => Ok(vec![]), + } + } + + fn decode(payload: &mut Payload, header: &CapabilityHeader) -> RdpResult { + match header.cap_type { + CapabilityType::CAP_GENERAL_TYPE => Ok(Capability::General( + GeneralCapabilitySet::decode(payload, header.version)?, + )), + CapabilityType::CAP_PRINTER_TYPE => Ok(Capability::Printer), + CapabilityType::CAP_PORT_TYPE => Ok(Capability::Port), + CapabilityType::CAP_DRIVE_TYPE => Ok(Capability::Drive), + CapabilityType::CAP_SMARTCARD_TYPE => Ok(Capability::Smartcard), + } + } +} + +#[derive(Debug)] +struct GeneralCapabilitySet { + os_type: u32, + os_version: u32, + protocol_major_version: u16, + protocol_minor_version: u16, + io_code_1: u32, + io_code_2: u32, + extended_pdu: u32, + extra_flags_1: u32, + extra_flags_2: u32, + special_type_device_cap: u32, +} + +impl GeneralCapabilitySet { + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u32::(self.os_type)?; + w.write_u32::(self.os_version)?; + w.write_u16::(self.protocol_major_version)?; + w.write_u16::(self.protocol_minor_version)?; + w.write_u32::(self.io_code_1)?; + w.write_u32::(self.io_code_2)?; + w.write_u32::(self.extended_pdu)?; + w.write_u32::(self.extra_flags_1)?; + w.write_u32::(self.extra_flags_2)?; + w.write_u32::(self.special_type_device_cap)?; + Ok(w) + } + + fn decode(payload: &mut Payload, version: u32) -> RdpResult { + Ok(Self { + os_type: payload.read_u32::()?, + os_version: payload.read_u32::()?, + protocol_major_version: payload.read_u16::()?, + protocol_minor_version: payload.read_u16::()?, + io_code_1: payload.read_u32::()?, + io_code_2: payload.read_u32::()?, + extended_pdu: payload.read_u32::()?, + extra_flags_1: payload.read_u32::()?, + extra_flags_2: payload.read_u32::()?, + special_type_device_cap: if version == GENERAL_CAPABILITY_VERSION_02 { + payload.read_u32::()? + } else { + 0 + }, + }) + } +} + +type ClientCoreCapabilityResponse = ServerCoreCapabilityRequest; + +#[derive(Debug)] +pub struct ClientDeviceListAnnounceRequest { + device_count: u32, + device_list: Vec, +} + +pub type ClientDeviceListAnnounce = ClientDeviceListAnnounceRequest; + +impl ClientDeviceListAnnounceRequest { + // We only need to announce the smartcard in this Client Device List Announce Request. + // Drives (directories) can be announced at any time with a Client Drive Device List Announce. + fn new_smartcard(device_id: u32) -> Self { + Self { + device_count: 1, + device_list: vec![DeviceAnnounceHeader { + device_type: DeviceType::RDPDR_DTYP_SMARTCARD, + device_id, + // This name is a constant defined by the spec. + preferred_dos_name: "SCARD".to_string(), + device_data_length: 0, + device_data: vec![], + }], + } + } + + /// Creates a ClientDeviceListAnnounceRequest for announcing a new shared drive (directory). + /// A new drive can be announced at any time during RDP's operation. It is up to the caller + /// to ensure that the passed device_id is unique from that of any previously shared devices. + pub fn new_drive(device_id: u32, drive_name: String) -> Self { + // According to the spec: + // + // If the client supports DRIVE_CAPABILITY_VERSION_02 in the Drive Capability Set, + // then the full name MUST also be specified in the DeviceData field, as a null-terminated + // Unicode string. If the DeviceDataLength field is nonzero, the content of the + // PreferredDosName field is ignored. + // + // In the RDP spec, Unicode typically means null-terminated UTF-16LE, however empirically it + // appears that this field expects null-terminated UTF-8. + let device_data = util::to_utf8(&drive_name); + + Self { + device_count: 1, + device_list: vec![DeviceAnnounceHeader { + device_type: DeviceType::RDPDR_DTYP_FILESYSTEM, + device_id, + preferred_dos_name: drive_name, + device_data_length: device_data.len() as u32, + device_data, + }], + } + } + + fn new_empty() -> Self { + Self { + device_count: 0, + device_list: vec![], + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u32::(self.device_count)?; + for dev in self.device_list.iter() { + w.extend_from_slice(&dev.encode()?); + } + Ok(w) + } +} + +/// 2.2.1.3 Device Announce Header (DEVICE_ANNOUNCE) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/32e34332-774b-4ead-8c9d-5d64720d6bf9 +#[derive(Debug)] +struct DeviceAnnounceHeader { + device_type: DeviceType, + device_id: u32, + preferred_dos_name: String, + device_data_length: u32, + device_data: Vec, +} + +impl DeviceAnnounceHeader { + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u32::(self.device_type.to_u32().unwrap())?; + w.write_u32::(self.device_id)?; + let mut name: &str = &self.preferred_dos_name; + // See "PreferredDosName" at + // https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/32e34332-774b-4ead-8c9d-5d64720d6bf9 + if name.len() > 7 { + name = &name[..7]; + } + w.extend_from_slice(&format!("{:\x00<8}", name).into_bytes()); + w.write_u32::(self.device_data_length)?; + w.extend_from_slice(&self.device_data); + Ok(w) + } +} + +#[derive(Debug)] +struct ServerDeviceAnnounceResponse { + device_id: u32, + result_code: u32, +} + +impl ServerDeviceAnnounceResponse { + fn decode(payload: &mut Payload) -> RdpResult { + Ok(Self { + device_id: payload.read_u32::()?, + result_code: payload.read_u32::()?, + }) + } +} + +/// 2.2.1.4 Device I/O Request (DR_DEVICE_IOREQUEST) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/a087ffa8-d0d5-4874-ac7b-0494f63e2d5d +#[derive(Debug, Clone)] +#[allow(dead_code)] +pub struct DeviceIoRequest { + pub device_id: u32, + file_id: u32, + pub completion_id: u32, + major_function: MajorFunction, + minor_function: MinorFunction, +} + +impl DeviceIoRequest { + fn decode(payload: &mut Payload) -> RdpResult { + let device_id = payload.read_u32::()?; + let file_id = payload.read_u32::()?; + let completion_id = payload.read_u32::()?; + let major_function = payload.read_u32::()?; + let major_function = MajorFunction::from_u32(major_function).ok_or_else(|| { + invalid_data_error(&format!( + "invalid major function value {:#010x}", + major_function + )) + })?; + let minor_function = payload.read_u32::()?; + // From the spec (2.2.1.4 Device I/O Request (DR_DEVICE_IOREQUEST)): + // "This field [MinorFunction] is valid only when the MajorFunction field + // is set to IRP_MJ_DIRECTORY_CONTROL. If the MajorFunction field is set + // to another value, the MinorFunction field value SHOULD be 0x00000000."" + // + // SHOULD means implementations are not guaranteed to give us 0x00000000, + // so handle that possibility here. + let minor_function = if major_function == MajorFunction::IRP_MJ_DIRECTORY_CONTROL { + minor_function + } else { + 0x00000000 + }; + let minor_function = MinorFunction::from_u32(minor_function).ok_or_else(|| { + invalid_data_error(&format!( + "invalid minor function value {:#010x}", + minor_function + )) + })?; + + Ok(Self { + device_id, + file_id, + completion_id, + major_function, + minor_function, + }) + } +} + +/// 2.2.1.4.5 Device Control Request (DR_CONTROL_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/30662c80-ec6e-4ed1-9004-2e6e367bb59f +#[derive(Debug)] +#[allow(dead_code)] +struct DeviceControlRequest { + header: DeviceIoRequest, + output_buffer_length: u32, + input_buffer_length: u32, + io_control_code: u32, + padding: [u8; 20], +} + +impl DeviceControlRequest { + fn decode(header: DeviceIoRequest, payload: &mut Payload) -> RdpResult { + let output_buffer_length = payload.read_u32::()?; + let input_buffer_length = payload.read_u32::()?; + let io_control_code = payload.read_u32::()?; + let mut padding: [u8; 20] = [0; 20]; + payload.read_exact(&mut padding)?; + Ok(Self { + header, + output_buffer_length, + input_buffer_length, + io_control_code, + padding, + }) + } +} + +/// 2.2.1.5 Device I/O Response (DR_DEVICE_IOCOMPLETION) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/1c412a84-0776-4984-b35c-3f0445fcae65 +#[derive(Debug)] +struct DeviceIoResponse { + device_id: u32, + completion_id: u32, + io_status: u32, +} + +impl DeviceIoResponse { + fn new(req: &DeviceIoRequest, io_status: u32) -> Self { + Self { + device_id: req.device_id, + completion_id: req.completion_id, + io_status, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_u32::(self.device_id)?; + w.write_u32::(self.completion_id)?; + w.write_u32::(self.io_status)?; + Ok(w) + } +} + +#[derive(Debug)] +struct DeviceControlResponse { + header: DeviceIoResponse, + output_buffer_length: u32, + output_buffer: Vec, +} + +impl DeviceControlResponse { + fn new(req: &DeviceControlRequest, io_status: u32, output: Vec) -> Self { + Self { + header: DeviceIoResponse::new(&req.header, io_status), + output_buffer_length: output.length() as u32, + output_buffer: output, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.extend_from_slice(&self.header.encode()?); + w.write_u32::(self.output_buffer_length)?; + w.extend_from_slice(&self.output_buffer); + Ok(w) + } +} + +/// 2.2.3.3.1 Server Create Drive Request (DR_DRIVE_CREATE_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/95b16fd0-d530-407c-a310-adedc85e9897 +pub type ServerCreateDriveRequest = DeviceCreateRequest; + +/// 2.2.1.4.1 Device Create Request (DR_CREATE_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/5f71f6d2-d9ff-40c2-bdb5-a739447d3c3e +#[derive(Debug, Clone)] +#[allow(dead_code)] +pub struct DeviceCreateRequest { + /// The MajorFunction field in this header MUST be set to IRP_MJ_CREATE. + pub device_io_request: DeviceIoRequest, + desired_access: flags::DesiredAccess, + allocation_size: u64, + file_attributes: flags::FileAttributes, + shared_access: flags::SharedAccess, + create_disposition: flags::CreateDisposition, + create_options: flags::CreateOptions, + path_length: u32, + pub path: String, +} + +#[allow(dead_code)] +impl DeviceCreateRequest { + fn decode(device_io_request: DeviceIoRequest, payload: &mut Payload) -> RdpResult { + let invalid_flags = || invalid_data_error("invalid flags in Device Create Request"); + + let desired_access = flags::DesiredAccess::from_bits(payload.read_u32::()?) + .ok_or_else(invalid_flags)?; + let allocation_size = payload.read_u64::()?; + let file_attributes = flags::FileAttributes::from_bits(payload.read_u32::()?) + .ok_or_else(invalid_flags)?; + let shared_access = flags::SharedAccess::from_bits(payload.read_u32::()?) + .ok_or_else(invalid_flags)?; + let create_disposition = + flags::CreateDisposition::from_bits(payload.read_u32::()?) + .ok_or_else(invalid_flags)?; + let create_options = flags::CreateOptions::from_bits(payload.read_u32::()?) + .ok_or_else(invalid_flags)?; + let path_length = payload.read_u32::()?; + + // usize is 32 bits on a 32 bit target and 64 on a 64, so we can safely say try_into().unwrap() + // for a u32 will never panic on the machines that run teleport. + let mut path = vec![0u8; path_length.try_into().unwrap()]; + payload.read_exact(&mut path)?; + let path = util::from_unicode(path)?; + + Ok(Self { + device_io_request, + desired_access, + allocation_size, + file_attributes, + shared_access, + create_disposition, + create_options, + path_length, + path, + }) + } +} + +/// 2.2.1.5.1 Device Create Response (DR_CREATE_RSP) +/// A message with this header describes a response to a Device Create Request (section 2.2.1.4.1). +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/99e5fca5-b37a-41e4-bc69-8d7da7860f76 +#[derive(Debug)] +#[allow(dead_code)] +struct DeviceCreateResponse { + device_io_reply: DeviceIoResponse, + file_id: u32, + /// The values of the CreateDisposition field in the Device Create Request (section 2.2.1.4.1) that determine the value + /// of the Information field are associated as follows: + /// +---------------------+--------------------+ + /// | CreateDisposition | Information | + /// +---------------------+--------------------+ + /// | FILE_SUPERSEDE | FILE_SUPERSEDED | + /// | FILE_OPEN | | + /// | FILE_CREATE | | + /// | FILE_OVERWRITE | | + /// +---------------------+--------------------+ + /// | FILE_OPEN_IF | FILE_OPENED | + /// +---------------------+--------------------+ + /// | FILE_OVERWRITE_IF | FILE_OVERWRITTEN | + /// +---------------------+--------------------+ + information: flags::Information, +} + +#[allow(dead_code)] +impl DeviceCreateResponse { + fn new(device_create_request: &DeviceCreateRequest, io_status: NTSTATUS) -> Self { + let device_io_request = &device_create_request.device_io_request; + + let information: flags::Information; + if device_create_request.create_disposition.intersects( + flags::CreateDisposition::FILE_SUPERSEDE + | flags::CreateDisposition::FILE_OPEN + | flags::CreateDisposition::FILE_CREATE + | flags::CreateDisposition::FILE_OVERWRITE, + ) { + information = flags::Information::FILE_SUPERSEDED; + } else if device_create_request.create_disposition == flags::CreateDisposition::FILE_OPEN_IF + { + information = flags::Information::FILE_OPENED; + } else if device_create_request.create_disposition + == flags::CreateDisposition::FILE_OVERWRITE_IF + { + information = flags::Information::FILE_OVERWRITTEN; + } else { + panic!("program error, CreateDispositionFlags check should be exhaustive"); + } + + Self { + device_io_reply: DeviceIoResponse::new( + device_io_request, + NTSTATUS::to_u32(&io_status).unwrap(), + ), + file_id: device_io_request.file_id, // TODO(isaiah): this is false, the client should be generating the file_id here + information, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.extend_from_slice(&self.device_io_reply.encode()?); + w.write_u32::(self.file_id)?; + w.write_u8(self.information.bits())?; + Ok(w) + } +} + +/// 2.2.3.3.8 Server Drive Query Information Request (DR_DRIVE_QUERY_INFORMATION_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/e43dcd68-2980-40a9-9238-344b6cf94946 +#[derive(Debug)] +#[allow(dead_code)] +struct ServerDriveQueryInformationRequest { + /// A DR_DEVICE_IOREQUEST (section 2.2.1.4) header. The MajorFunction field in the DR_DEVICE_IOREQUEST header MUST be set to IRP_MJ_QUERY_INFORMATION. + device_io_request: DeviceIoRequest, + /// A 32-bit unsigned integer. + /// This field MUST contain one of the following values: + /// FileBasicInformation + /// This information class is used to query a file for the times of creation, last access, last write, and change, in addition to file attribute information. The Reserved field of the FileBasicInformation structure ([MS-FSCC] section 2.4.7) MUST NOT be present. + /// + /// FileStandardInformation + /// This information class is used to query for file information such as allocation size, end-of-file position, and number of links. The Reserved field of the FileStandardInformation structure ([MS-FSCC] section 2.4.41) MUST NOT be present. + /// + /// FileAttributeTagInformation + /// This information class is used to query for file attribute and reparse tag information. + fs_information_class_lvl: FsInformationClassLevel, + // Length, Padding, and QueryBuffer appear to be vestigial fields and can safely be ignored. Their description + // is provided below for documentation purposes. + // + // Length (4 bytes): A 32-bit unsigned integer that specifies the number of bytes in the QueryBuffer field. + // + // Padding (24 bytes): An array of 24 bytes. This field is unused and MUST be ignored. + // + // QueryBuffer (variable): A variable-length array of bytes. The size of the array is specified by the Length field. + // The content of this field is based on the value of the FsInformationClass field, which determines the different + // structures that MUST be contained in the QueryBuffer field. For a complete list of these structures, see [MS-FSCC] + // section 2.4. The "File information class" table defines all the possible values for the FsInformationClass field. +} + +#[allow(dead_code)] +impl ServerDriveQueryInformationRequest { + fn decode(device_io_request: DeviceIoRequest, payload: &mut Payload) -> RdpResult { + if let Some(fs_information_class_lvl) = + FsInformationClassLevel::from_u32(payload.read_u32::()?) + { + Ok(Self { + device_io_request, + fs_information_class_lvl, + }) + } else { + Err(invalid_data_error( + "received invalid FsInformationClass in ServerDriveQueryInformationRequest", + )) + } + } +} + +/// 2.4 File Information Classes [MS-FSCC] +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/4718fc40-e539-4014-8e33-b675af74e3e1 +#[derive(Debug)] +#[allow(dead_code, clippy::enum_variant_names)] +enum FsInformationClass { + FileBasicInformation(FileBasicInformation), + FileStandardInformation(FileStandardInformation), + FileBothDirectoryInformation(FileBothDirectoryInformation), +} + +#[allow(dead_code)] +impl FsInformationClass { + fn encode(&self) -> RdpResult> { + match self { + Self::FileBasicInformation(file_basic_info) => file_basic_info.encode(), + Self::FileStandardInformation(file_standard_info) => file_standard_info.encode(), + Self::FileBothDirectoryInformation(fil_both_dir_info) => fil_both_dir_info.encode(), // TODO(isaiah) + } + } +} + +/// 2.4.7 FileBasicInformation [MS-FSCC] +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/16023025-8a78-492f-8b96-c873b042ac50 +#[derive(Debug)] +struct FileBasicInformation { + creation_time: i64, + last_access_time: i64, + last_write_time: i64, + change_time: i64, + file_attributes: flags::FileAttributes, + // NOTE: The `reserved` field in the spec MUST not be serialized and sent over RDP, or it will break the server implementation. + // FreeRDP does the same: https://github.com/FreeRDP/FreeRDP/blob/1adb263813ca2e76a893ef729a04db8f94b5d757/channels/drive/client/drive_file.c#L508 + //reserved: u32, +} + +#[allow(dead_code)] +/// 4 i64's and 1 u32's = (4 * 8) + 4 +const FILE_BASIC_INFORMATION_SIZE: u32 = (4 * 8) + 4; + +impl FileBasicInformation { + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_i64::(self.creation_time)?; + w.write_i64::(self.last_access_time)?; + w.write_i64::(self.last_write_time)?; + w.write_i64::(self.change_time)?; + w.write_u32::(self.file_attributes.bits())?; + Ok(w) + } +} + +/// 2.4.41 FileStandardInformation [MS-FSCC] +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/5afa7f66-619c-48f3-955f-68c4ece704ae +#[derive(Debug)] +struct FileStandardInformation { + /// A 64-bit signed integer that contains the file allocation size, in bytes. The value of this field MUST be an + /// integer multiple of the cluster size. + /// Cluster size is the size of the logical minimal unit of disk space used by the operating system. FreeRDP + /// doesn't give the actual size here, but rather just gives the file size itself, which we will mimic. + /// (ttps://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_file.c#L518-L519). + /// + /// When FileStandardInformation is requested for a directory, its not entirely clear what "file size" means. + /// FreeRDP derives this value from the st_size field of a stat struct (https://linux.die.net/man/2/lstat), which says + /// "The st_size field gives the size of the file (if it is a regular file or a symbolic link) in bytes. The size of + /// a symbolic link is the length of the pathname it contains, without a terminating null byte." Since it's not + /// entirely clear what is offered here in the case of a directory, we will just use 0. + allocation_size: i64, + /// A 64-bit signed integer that contains the absolute end-of-file position as a byte offset from the start of the + /// file. EndOfFile specifies the offset to the byte immediately following the last valid byte in the file. Because + /// this value is zero-based, it actually refers to the first free byte in the file. That is, it is the offset from + /// the beginning of the file at which new bytes appended to the file will be written. The value of this field MUST + /// be greater than or equal to 0. + end_of_file: i64, + /// A 32-bit unsigned integer that contains the number of non-deleted [hard] links to this file. + /// NOTE: this information is not available to us in the browser, so we will simply set this field to 0. + number_of_links: u32, + /// Set to TRUE to indicate that a file deletion has been requested; set to FALSE + /// otherwise. + delete_pending: Boolean, + /// Set to TRUE to indicate that the file is a directory; set to FALSE otherwise. + directory: Boolean, + // NOTE: `reserved` field omitted, see NOTE in FileBasicInformation struct. + // reserved: u16, +} + +impl FileStandardInformation { + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.write_i64::(self.allocation_size)?; + w.write_i64::(self.end_of_file)?; + w.write_u32::(self.number_of_links)?; + w.write_u8(Boolean::to_u8(&self.delete_pending).unwrap())?; + w.write_u8(Boolean::to_u8(&self.directory).unwrap())?; + Ok(w) + } +} + +#[allow(dead_code)] +// 2 i64's + 1 u32 + 2 Boolean (u8) = (2 * 8) + 4 + 2 +const FILE_STANDARD_INFORMATION_SIZE: u32 = (2 * 8) + 4 + 2; + +/// 2.1.8 Boolean +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/8ce7b38c-d3cc-415d-ab39-944000ea77ff +#[derive(Debug, ToPrimitive)] +#[repr(u8)] +#[allow(dead_code)] +enum Boolean { + True = 1, + False = 0, +} + +/// 2.4.8 FileBothDirectoryInformation +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-fscc/270df317-9ba5-4ccb-ba00-8d22be139bc5 +/// Fields are omitted based on those omitted by FreeRDP: https://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_file.c#L871 +#[derive(Debug)] +struct FileBothDirectoryInformation { + // next_entry_offset: u32, + // file_index: u32, + creation_time: i64, + last_access_time: i64, + last_write_time: i64, + change_time: i64, + end_of_file: i64, + allocation_size: i64, + file_attributes: flags::FileAttributes, + file_name_length: u32, + // ea_size: u32, + // short_name_length: i8, + // reserved: u8: MUST NOT be added, + // see https://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_file.c#L907 + // short_name: String, // 24 bytes + file_name: String, +} + +#[allow(dead_code)] +/// Base size of the FileBothDirectoryInformation, not accounting for variably sized file_name. +/// Note that file_name's size should be calculated as if it were a Unicode string. +/// 5 u32's (including FileAttributesFlags) + 6 i64's + 1 i8 + 24 bytes +const FILE_BOTH_DIRECTORY_INFORMATION_BASE_SIZE: u32 = (5 * 4) + (6 * 8) + 1 + 24; // 93 + +#[allow(dead_code)] +impl FileBothDirectoryInformation { + fn new( + creation_time: i64, + last_access_time: i64, + last_write_time: i64, + change_time: i64, + file_size: i64, + file_attributes: flags::FileAttributes, + file_name: String, + ) -> Self { + Self { + creation_time, + last_access_time, + last_write_time, + change_time, + end_of_file: file_size, + allocation_size: file_size, + file_attributes, + file_name_length: u32::try_from(util::to_unicode(&file_name, false).len()).unwrap(), + file_name, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + // next_entry_offset + w.write_u32::(0)?; + // file_index + w.write_u32::(0)?; + w.write_i64::(self.creation_time)?; + w.write_i64::(self.last_access_time)?; + w.write_i64::(self.last_write_time)?; + w.write_i64::(self.change_time)?; + w.write_i64::(self.end_of_file)?; + w.write_i64::(self.allocation_size)?; + w.write_u32::(self.file_attributes.bits())?; + w.write_u32::(self.file_name_length)?; + // ea_size + w.write_u32::(0)?; + // short_name_length + w.write_i8(0)?; + // reserved u8, MUST NOT be added! + // short_name + w.extend_from_slice(&[0; 24]); + // When working with this field, use file_name_length to determine the length of the file name rather + // than assuming the presence of a trailing null delimiter. Dot directory names are valid for this field. + w.extend_from_slice(&util::to_unicode(&self.file_name, false)); + Ok(w) + } +} + +/// 2.2.3.4.8 Client Drive Query Information Response (DR_DRIVE_QUERY_INFORMATION_RSP) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/37ef4fb1-6a95-4200-9fbf-515464f034a4 +#[derive(Debug)] +#[allow(dead_code)] + +struct ClientDriveQueryInformationResponse { + device_io_response: DeviceIoResponse, + length: u32, + buffer: FsInformationClass, +} + +#[allow(dead_code)] +impl ClientDriveQueryInformationResponse { + /// Constructs a ClientDriveQueryInformationResponse from a ServerDriveQueryInformationRequest and an NTSTATUS. + /// If the ServerDriveQueryInformationRequest.fs_information_class_lvl is currently unsupported, the program will panic. + /// TODO(isaiah): We will pass some sort of file structure into here. + fn new(req: &ServerDriveQueryInformationRequest, io_status: NTSTATUS) -> RdpResult { + let (length, buffer) = match req.fs_information_class_lvl { + FsInformationClassLevel::FileBasicInformation => ( + FILE_BASIC_INFORMATION_SIZE, + FsInformationClass::FileBasicInformation(FileBasicInformation { + creation_time: 1, + last_access_time: 2, + last_write_time: 3, + change_time: 4, + file_attributes: flags::FileAttributes::FILE_ATTRIBUTE_DIRECTORY, + }), + ), + FsInformationClassLevel::FileStandardInformation => ( + FILE_STANDARD_INFORMATION_SIZE, + FsInformationClass::FileStandardInformation(FileStandardInformation { + allocation_size: 0, + end_of_file: 0, + number_of_links: 0, + delete_pending: Boolean::False, + directory: Boolean::True, + }), + ), + _ => { + return Err(not_implemented_error(&format!( + "received unsupported NTSTATUS: {:?}", + io_status + ))) + } + }; + + Ok(Self { + device_io_response: DeviceIoResponse::new( + &req.device_io_request, + NTSTATUS::to_u32(&io_status).unwrap(), + ), + length, + buffer, + }) + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.extend_from_slice(&self.device_io_response.encode()?); + w.write_u32::(self.length)?; + w.extend_from_slice(&self.buffer.encode()?); + Ok(w) + } +} + +/// 2.2.1.4.2 Device Close Request (DR_CLOSE_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/3ec6627f-9e0f-4941-a828-3fc6ed63d9e7 +#[derive(Debug)] +#[allow(dead_code)] +struct DeviceCloseRequest { + device_io_request: DeviceIoRequest, + // Padding (32 bytes): An array of 32 bytes. Reserved. This field can be set to any value, and MUST be ignored. +} + +#[allow(dead_code)] +impl DeviceCloseRequest { + fn decode(device_io_request: DeviceIoRequest) -> Self { + Self { device_io_request } + } +} + +/// 2.2.1.5.2 Device Close Response (DR_CLOSE_RSP) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/0dae7031-cfd8-4f14-908c-ec06e14997b5 +#[derive(Debug)] +#[allow(dead_code)] +struct DeviceCloseResponse { + /// The CompletionId field of this header MUST match a Device I/O Request (section 2.2.1.4) message that had the MajorFunction field set to IRP_MJ_CLOSE. + device_io_response: DeviceIoResponse, + /// This field can be set to any value and MUST be ignored. + padding: u32, +} +#[allow(dead_code)] +impl DeviceCloseResponse { + fn new(device_close_request: DeviceCloseRequest, io_status: NTSTATUS) -> Self { + Self { + device_io_response: DeviceIoResponse::new( + &device_close_request.device_io_request, + NTSTATUS::to_u32(&io_status).unwrap(), + ), + padding: 0, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.extend_from_slice(&self.device_io_response.encode()?); + w.write_u32::(self.padding)?; + Ok(w) + } +} + +/// 2.2.3.3.11 Server Drive NotifyChange Directory Request (DR_DRIVE_NOTIFY_CHANGE_DIRECTORY_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/ed05e73d-e53e-4261-a1e1-365a70ba6512 +#[derive(Debug)] +#[allow(dead_code)] +struct ServerDriveNotifyChangeDirectoryRequest { + /// The MajorFunction field in the DR_DEVICE_IOREQUEST header MUST be set to IRP_MJ_DIRECTORY_CONTROL, + /// and the MinorFunction field MUST be set to IRP_MN_NOTIFY_CHANGE_DIRECTORY. + device_io_request: DeviceIoRequest, + /// If nonzero, a change anywhere within the tree MUST trigger the notification response; otherwise, only a change in the root directory will do so. + watch_tree: u8, + completion_filter: flags::CompletionFilter, + // Padding (27 bytes): An array of 27 bytes. This field is unused and MUST be ignored. +} + +#[allow(dead_code)] +impl ServerDriveNotifyChangeDirectoryRequest { + fn decode(device_io_request: DeviceIoRequest, payload: &mut Payload) -> RdpResult { + let invalid_flags = + || invalid_data_error("invalid flags in Server Drive NotifyChange Directory Request"); + + let watch_tree = payload.read_u8()?; + let completion_filter = + flags::CompletionFilter::from_bits(payload.read_u32::()?) + .ok_or_else(invalid_flags)?; + + Ok(Self { + device_io_request, + watch_tree, + completion_filter, + }) + } +} + +/// 2.2.1.4.3 Device Read Request (DR_READ_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/3192516d-36a6-47c5-987a-55c214aa0441 +#[derive(Debug)] +#[allow(dead_code)] +struct DeviceReadRequest { + /// The MajorFunction field in this header MUST be set to IRP_MJ_READ. + device_io_request: DeviceIoRequest, + /// This field specifies the maximum number of bytes to be read from the device. + length: u32, + /// This field specifies the file offset where the read operation is performed. + offset: u64, + // Padding (20 bytes): An array of 20 bytes. Reserved. This field can be set to any value and MUST be ignored. +} + +#[allow(dead_code)] +impl DeviceReadRequest { + fn decode(device_io_request: DeviceIoRequest, payload: &mut Payload) -> RdpResult { + Ok(Self { + device_io_request, + length: payload.read_u32::()?, + offset: payload.read_u64::()?, + }) + } +} + +/// 2.2.1.5.3 Device Read Response (DR_READ_RSP) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/d35d3f91-fc5b-492b-80be-47f483ad1dc9 +#[derive(Debug)] +#[allow(dead_code)] +struct DeviceReadResponse { + /// The CompletionId field of this header MUST match a Device I/O Request (section 2.2.1.4) message that had the MajorFunction field set to IRP_MJ_READ. + device_io_reply: DeviceIoResponse, + /// Specifies the number of bytes in the ReadData field. + length: u32, + /// A variable-length array of bytes that specifies the output data from the read request. + read_data: Vec, +} + +#[allow(dead_code)] +impl DeviceReadResponse { + fn new( + device_read_request: &DeviceReadRequest, + io_status: NTSTATUS, + read_data: Vec, + ) -> Self { + let device_io_request = &device_read_request.device_io_request; + + Self { + device_io_reply: DeviceIoResponse::new( + device_io_request, + NTSTATUS::to_u32(&io_status).unwrap(), + ), + length: u32::try_from(read_data.len()).unwrap(), + read_data, + } + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.extend_from_slice(&self.device_io_reply.encode()?); + w.write_u32::(self.length)?; + w.extend_from_slice(&self.read_data); + Ok(w) + } +} + +/// 2.2.3.3.10 Server Drive Query Directory Request (DR_DRIVE_QUERY_DIRECTORY_REQ) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/458019d2-5d5a-4fd4-92ef-8c05f8d7acb1 +#[derive(Debug)] +#[allow(dead_code)] +struct ServerDriveQueryDirectoryRequest { + /// The MajorFunction field in the DR_DEVICE_IOREQUEST header MUST be set to IRP_MJ_DIRECTORY_CONTROL, + /// and the MinorFunction field MUST be set to IRP_MN_QUERY_DIRECTORY. + device_io_request: DeviceIoRequest, + /// Must contain one of FileDirectoryInformation, FileFullDirectoryInformation, FileBothDirectoryInformation, FileNamesInformation + fs_information_class_lvl: FsInformationClassLevel, + /// If the value of this field is zero, the request is for the next file in the directory that was specified in a previous + /// Server Drive Query Directory Request. If such a file does not exist, the client MUST complete this request with STATUS_NO_MORE_FILES + /// in the IoStatus field of the Client Drive I/O Response packet (section 2.2.3.4). If the value of this field is non-zero and such a + /// file does not exist, the client MUST complete this request with STATUS_NO_SUCH_FILE in the IoStatus field of the Client Drive I/O Response. + initial_query: u8, + /// Specifies the number of bytes in the Path field, including the null-terminator. + path_length: u32, + // Padding (23 bytes): An array of 23 bytes. This field is unused and MUST be ignored. + /// A variable-length array of Unicode characters (we will store this as a regular rust String) that specifies the directory + /// on which this operation will be performed. The Path field MUST be null-terminated. If the value of the InitialQuery field + /// is zero, then the contents of the Path field MUST be ignored, irrespective of the value specified in the PathLength field. + path: String, +} + +#[allow(dead_code)] +impl ServerDriveQueryDirectoryRequest { + fn decode(device_io_request: DeviceIoRequest, payload: &mut Payload) -> RdpResult { + let fs_information_class_lvl = + FsInformationClassLevel::from_u32(payload.read_u32::()?) + .ok_or_else(|| invalid_data_error("failed to read FsInformationClassLevel"))?; + if fs_information_class_lvl != FsInformationClassLevel::FileDirectoryInformation + && fs_information_class_lvl != FsInformationClassLevel::FileFullDirectoryInformation + && fs_information_class_lvl != FsInformationClassLevel::FileBothDirectoryInformation + && fs_information_class_lvl != FsInformationClassLevel::FileNamesInformation + { + return Err(invalid_data_error(&format!( + "read invalid FsInformationClassLevel: {:?}, expected one of {:?}", + fs_information_class_lvl, + vec![ + FsInformationClassLevel::FileDirectoryInformation, + FsInformationClassLevel::FileFullDirectoryInformation, + FsInformationClassLevel::FileBothDirectoryInformation, + FsInformationClassLevel::FileNamesInformation + ] + ))); + } + let initial_query = payload.read_u8()?; + let mut path_length: u32 = 0; + let mut path = String::from(""); + if initial_query != 0 { + path_length = payload.read_u32::()?; + + // TODO(isaiah): make a payload.skip(n) + let mut padding: [u8; 23] = [0; 23]; + payload.read_exact(&mut padding)?; + + // TODO(isaiah): make a from_unicode_exact + let mut path_as_vec = vec![0u8; path_length.try_into().unwrap()]; + payload.read_exact(&mut path_as_vec)?; + path = util::from_unicode(path_as_vec)?; + } + + Ok(Self { + device_io_request, + fs_information_class_lvl, + initial_query, + path_length, + path, + }) + } +} + +/// 2.2.3.4.10 Client Drive Query Directory Response (DR_DRIVE_QUERY_DIRECTORY_RSP) +/// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-rdpefs/9c929407-a833-4893-8f20-90c984756140 +#[derive(Debug)] +#[allow(dead_code)] +struct ClientDriveQueryDirectoryResponse { + /// The CompletionId field of the DR_DEVICE_IOCOMPLETION header MUST match a Device I/O Request (section 2.2.1.4) that + /// has the MajorFunction field set to IRP_MJ_DIRECTORY_CONTROL and the MinorFunction field set to IRP_MN_QUERY_DIRECTORY. + device_io_reply: DeviceIoResponse, + /// Specifies the number of bytes in the Buffer field. + length: u32, + /// The content of this field is based on the value of the FsInformationClass field in the Server Drive Query Directory Request + /// message, which determines the different structures that MUST be contained in the Buffer field. + buffer: Option, + // Padding (1 byte): This field is unused and MUST be ignored. +} + +#[allow(dead_code)] +impl ClientDriveQueryDirectoryResponse { + fn new( + req: &ServerDriveQueryDirectoryRequest, + io_status: NTSTATUS, + buffer: Option, + ) -> RdpResult { + let device_io_request = &req.device_io_request; + let length = match buffer { + Some(ref fs_information_class) => match fs_information_class { + FsInformationClass::FileBothDirectoryInformation( + file_both_directory_information, + ) => { + FILE_BOTH_DIRECTORY_INFORMATION_BASE_SIZE + + file_both_directory_information.file_name_length + } + _ => { + return Err(not_implemented_error(&format!("ClientDriveQueryDirectoryResponse not implemented for fs_information_class {:?}", fs_information_class))); + } + }, + None => 0, + }; + + Ok(Self { + device_io_reply: DeviceIoResponse::new( + device_io_request, + NTSTATUS::to_u32(&io_status).unwrap(), + ), + length, + buffer, + }) + } + + fn encode(&self) -> RdpResult> { + let mut w = vec![]; + w.extend_from_slice(&self.device_io_reply.encode()?); + + if self.device_io_reply.io_status == NTSTATUS::to_u32(&NTSTATUS::STATUS_SUCCESS).unwrap() { + w.write_u32::(self.length)?; + w.extend_from_slice( + &self + .buffer.as_ref() + .ok_or_else(|| invalid_data_error( + "ClientDriveQueryDirectoryResponse with NTSTATUS::STATUS_SUCCESS expects a FsInformationClass" + ))? + .encode()?, + ); + } else if self.device_io_reply.io_status + == NTSTATUS::to_u32(&NTSTATUS::STATUS_NO_MORE_FILES).unwrap() + { + // https://github.com/FreeRDP/FreeRDP/blob/511444a65e7aa2f537c5e531fa68157a50c1bd4d/channels/drive/client/drive_file.c#L935-L937 + w.write_u32::(0)?; + w.write_u8(0)?; + } else { + return Err(invalid_data_error(&format!( + "Found ClientDriveQueryDirectoryResponse with invalid or unhandled NTSTATUS: {:?}", + self.device_io_reply.io_status + ))); + } + + Ok(w) + } +} + +type SharedDirectoryInfoResponseHandler = + Box RdpResult>>>; diff --git a/lib/srv/desktop/rdp/rdpclient/src/scard.rs b/lib/srv/desktop/rdp/rdpclient/src/rdpdr/scard.rs similarity index 100% rename from lib/srv/desktop/rdp/rdpclient/src/scard.rs rename to lib/srv/desktop/rdp/rdpclient/src/rdpdr/scard.rs diff --git a/lib/srv/desktop/rdp/rdpclient/src/util.rs b/lib/srv/desktop/rdp/rdpclient/src/util.rs index 1f412b8721988..6bb326d15daf3 100644 --- a/lib/srv/desktop/rdp/rdpclient/src/util.rs +++ b/lib/srv/desktop/rdp/rdpclient/src/util.rs @@ -12,6 +12,10 @@ // See the License for the specific language governing permissions and // limitations under the License. +use crate::errors::invalid_data_error; +use rdp::model::error::RdpResult; +use utf16string::{WString, LE}; + /// According to [MS-RDPEFS] 1.1 Glossary: /// Unless otherwise specified, all Unicode strings follow the UTF-16LE /// encoding scheme with no Byte Order Mark (BOM). @@ -20,9 +24,48 @@ /// UTF-16LE encoded Vec, which is useful in cases where we want /// to handle some data in the code as a &str (or String), and later /// convert it to RDP's preferred format and send it over the wire. -pub fn to_unicode(s: &str) -> Vec { - let mut buf: Vec = s.encode_utf16().flat_map(|v| v.to_le_bytes()).collect(); - let mut null_terminator: Vec = vec![0, 0]; - buf.append(&mut null_terminator); +pub fn to_unicode(s: &str, with_null_term: bool) -> Vec { + let mut buf = WString::::from(s).as_bytes().to_vec(); + if with_null_term { + let mut null_terminator: Vec = vec![0, 0]; + buf.append(&mut null_terminator); + } buf } + +#[allow(clippy::bind_instead_of_map)] +pub fn from_unicode(s: Vec) -> RdpResult { + let mut with_null_terminator = WString::from_utf16le(s) + .or_else(|_| Err(invalid_data_error("invalid Unicode")))? + .to_utf8(); + with_null_terminator.pop(); + let without_null_terminator = with_null_terminator; + Ok(without_null_terminator) +} + +/// Converts a &str into a null-terminated UTF-8 encoded Vec +pub fn to_utf8(s: &str) -> Vec { + format!("{}\x00", s).into_bytes() +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn to_and_from() { + let hello_vec = to_unicode("hello", true); + assert_eq!( + hello_vec, + vec![104, 0, 101, 0, 108, 0, 108, 0, 111, 0, 0, 0] + ); + + let hello_string = from_unicode(hello_vec).unwrap(); + assert_eq!(hello_string, "hello"); + } + + #[test] + fn from_unicode_empty_vector() { + assert_eq!(from_unicode(vec![]).unwrap(), ""); + } +} diff --git a/lib/srv/desktop/tdp/proto.go b/lib/srv/desktop/tdp/proto.go index 06bea594c9730..aa751104416ea 100644 --- a/lib/srv/desktop/tdp/proto.go +++ b/lib/srv/desktop/tdp/proto.go @@ -46,16 +46,20 @@ type MessageType byte // For descriptions of each message type see: // https://github.com/gravitational/teleport/blob/master/rfd/0037-desktop-access-protocol.md#message-types const ( - TypeClientScreenSpec = MessageType(1) - TypePNGFrame = MessageType(2) - TypeMouseMove = MessageType(3) - TypeMouseButton = MessageType(4) - TypeKeyboardButton = MessageType(5) - TypeClipboardData = MessageType(6) - TypeClientUsername = MessageType(7) - TypeMouseWheel = MessageType(8) - TypeError = MessageType(9) - TypeMFA = MessageType(10) + TypeClientScreenSpec = MessageType(1) + TypePNGFrame = MessageType(2) + TypeMouseMove = MessageType(3) + TypeMouseButton = MessageType(4) + TypeKeyboardButton = MessageType(5) + TypeClipboardData = MessageType(6) + TypeClientUsername = MessageType(7) + TypeMouseWheel = MessageType(8) + TypeError = MessageType(9) + TypeMFA = MessageType(10) + TypeSharedDirectoryAnnounce = MessageType(11) + TypeSharedDirectoryAcknowledge = MessageType(12) + TypeSharedDirectoryInfoRequest = MessageType(13) + TypeSharedDirectoryInfoResponse = MessageType(14) ) // Message is a Go representation of a desktop protocol message. @@ -108,6 +112,14 @@ func decode(in peekReader) (Message, error) { return decodeError(in) case TypeMFA: return DecodeMFA(in) + case TypeSharedDirectoryAnnounce: + return decodeSharedDirectoryAnnounce(in) + case TypeSharedDirectoryAcknowledge: + return decodeSharedDirectoryAcknowledge(in) + case TypeSharedDirectoryInfoRequest: + return decodeSharedDirectoryInfoRequest(in) + case TypeSharedDirectoryInfoResponse: + return decodeSharedDirectoryInfoResponse(in) default: return nil, trace.BadParameter("unsupported desktop protocol message type %d", t) } @@ -609,6 +621,218 @@ func DecodeMFAChallenge(in peekReader) (*MFA, error) { }, nil } +type SharedDirectoryAnnounce struct { + DirectoryID uint32 + Name string +} + +func (s SharedDirectoryAnnounce) Encode() ([]byte, error) { + buf := new(bytes.Buffer) + buf.WriteByte(byte(TypeSharedDirectoryAnnounce)) + binary.Write(buf, binary.BigEndian, s.DirectoryID) + if err := encodeString(buf, s.Name); err != nil { + return nil, trace.Wrap(err) + } + return buf.Bytes(), nil +} + +func decodeSharedDirectoryAnnounce(in peekReader) (SharedDirectoryAnnounce, error) { + t, err := in.ReadByte() + if err != nil { + return SharedDirectoryAnnounce{}, trace.Wrap(err) + } + if t != byte(TypeSharedDirectoryAnnounce) { + return SharedDirectoryAnnounce{}, trace.BadParameter("got message type %v, expected SharedDirectoryAnnounce(%v)", t, TypeSharedDirectoryAnnounce) + } + var completionID, directoryID uint32 + err = binary.Read(in, binary.BigEndian, &completionID) + if err != nil { + return SharedDirectoryAnnounce{}, trace.Wrap(err) + } + err = binary.Read(in, binary.BigEndian, &directoryID) + if err != nil { + return SharedDirectoryAnnounce{}, trace.Wrap(err) + } + name, err := decodeString(in, windowsMaxUsernameLength) + if err != nil { + return SharedDirectoryAnnounce{}, trace.Wrap(err) + } + + return SharedDirectoryAnnounce{ + DirectoryID: directoryID, + Name: name, + }, nil +} + +type SharedDirectoryAcknowledge struct { + ErrCode uint32 + DirectoryID uint32 +} + +func decodeSharedDirectoryAcknowledge(in peekReader) (SharedDirectoryAcknowledge, error) { + t, err := in.ReadByte() + if err != nil { + return SharedDirectoryAcknowledge{}, trace.Wrap(err) + } + if t != byte(TypeSharedDirectoryAcknowledge) { + return SharedDirectoryAcknowledge{}, trace.BadParameter("got message type %v, expected SharedDirectoryAcknowledge(%v)", t, TypeSharedDirectoryAcknowledge) + } + + var s SharedDirectoryAcknowledge + err = binary.Read(in, binary.BigEndian, &s) + return s, trace.Wrap(err) +} + +func (s SharedDirectoryAcknowledge) Encode() ([]byte, error) { + buf := new(bytes.Buffer) + buf.WriteByte(byte(TypeSharedDirectoryAcknowledge)) + binary.Write(buf, binary.BigEndian, s.ErrCode) + binary.Write(buf, binary.BigEndian, s.DirectoryID) + return buf.Bytes(), nil +} + +type SharedDirectoryInfoRequest struct { + CompletionID uint32 + DirectoryID uint32 + Path string +} + +func (s SharedDirectoryInfoRequest) Encode() ([]byte, error) { + buf := new(bytes.Buffer) + buf.WriteByte(byte(TypeSharedDirectoryInfoRequest)) + binary.Write(buf, binary.BigEndian, s.CompletionID) + binary.Write(buf, binary.BigEndian, s.DirectoryID) + if err := encodeString(buf, s.Path); err != nil { + return nil, trace.Wrap(err) + } + return buf.Bytes(), nil +} + +func decodeSharedDirectoryInfoRequest(in peekReader) (SharedDirectoryInfoRequest, error) { + t, err := in.ReadByte() + if err != nil { + return SharedDirectoryInfoRequest{}, trace.Wrap(err) + } + if t != byte(TypeSharedDirectoryInfoRequest) { + return SharedDirectoryInfoRequest{}, trace.BadParameter("got message type %v, expected SharedDirectoryInfoRequest(%v)", t, TypeSharedDirectoryInfoRequest) + } + var completionID, directoryID uint32 + err = binary.Read(in, binary.BigEndian, &completionID) + if err != nil { + return SharedDirectoryInfoRequest{}, trace.Wrap(err) + } + err = binary.Read(in, binary.BigEndian, &directoryID) + if err != nil { + return SharedDirectoryInfoRequest{}, trace.Wrap(err) + } + path, err := decodeString(in, tdpMaxPathLength) + if err != nil { + return SharedDirectoryInfoRequest{}, trace.Wrap(err) + } + + return SharedDirectoryInfoRequest{ + CompletionID: completionID, + DirectoryID: directoryID, + Path: path, + }, nil +} + +type SharedDirectoryInfoResponse struct { + CompletionID uint32 + ErrCode uint32 + Fso FileSystemObject +} + +func (s SharedDirectoryInfoResponse) Encode() ([]byte, error) { + buf := new(bytes.Buffer) + buf.WriteByte(byte(TypeSharedDirectoryInfoResponse)) + binary.Write(buf, binary.BigEndian, s.CompletionID) + binary.Write(buf, binary.BigEndian, s.ErrCode) + fso, err := s.Fso.Encode() + if err != nil { + return nil, trace.Wrap(err) + } + binary.Write(buf, binary.BigEndian, fso) + return buf.Bytes(), nil +} + +func decodeSharedDirectoryInfoResponse(in peekReader) (SharedDirectoryInfoResponse, error) { + t, err := in.ReadByte() + if err != nil { + return SharedDirectoryInfoResponse{}, trace.Wrap(err) + } + if t != byte(TypeSharedDirectoryInfoResponse) { + return SharedDirectoryInfoResponse{}, trace.BadParameter("got message type %v, expected SharedDirectoryInfoResponse(%v)", t, TypeSharedDirectoryInfoResponse) + } + var completionID, errCode uint32 + err = binary.Read(in, binary.BigEndian, &completionID) + if err != nil { + return SharedDirectoryInfoResponse{}, trace.Wrap(err) + } + err = binary.Read(in, binary.BigEndian, &errCode) + if err != nil { + return SharedDirectoryInfoResponse{}, trace.Wrap(err) + } + fso, err := decodeFileSystemObject(in) + if err != nil { + return SharedDirectoryInfoResponse{}, trace.Wrap(err) + } + + return SharedDirectoryInfoResponse{ + CompletionID: completionID, + ErrCode: errCode, + Fso: fso, + }, nil +} + +const tdpMaxPathLength = tdpMaxErrorMessageLength + +type FileSystemObject struct { + LastModified uint64 + Size uint64 + FileType uint32 + Path string +} + +func (s FileSystemObject) Encode() ([]byte, error) { + buf := new(bytes.Buffer) + binary.Write(buf, binary.BigEndian, s.LastModified) + binary.Write(buf, binary.BigEndian, s.Size) + binary.Write(buf, binary.BigEndian, s.FileType) + if err := encodeString(buf, s.Path); err != nil { + return nil, trace.Wrap(err) + } + return buf.Bytes(), nil +} + +func decodeFileSystemObject(in peekReader) (FileSystemObject, error) { + var lastModified, size uint64 + var fileType uint32 + err := binary.Read(in, binary.BigEndian, &lastModified) + if err != nil { + return FileSystemObject{}, trace.Wrap(err) + } + err = binary.Read(in, binary.BigEndian, &size) + if err != nil { + return FileSystemObject{}, trace.Wrap(err) + } + err = binary.Read(in, binary.BigEndian, &fileType) + if err != nil { + return FileSystemObject{}, trace.Wrap(err) + } + path, err := decodeString(in, tdpMaxPathLength) + if err != nil { + return FileSystemObject{}, trace.Wrap(err) + } + + return FileSystemObject{ + LastModified: lastModified, + Size: size, + FileType: fileType, + Path: path, + }, nil +} + // encodeString encodes strings for TDP. Strings are encoded as UTF-8 with // a 32-bit length prefix (in bytes): // https://github.com/gravitational/teleport/blob/master/rfd/0037-desktop-access-protocol.md#field-types diff --git a/lib/srv/desktop/windows_server.go b/lib/srv/desktop/windows_server.go index 295d8eb0b81be..de4d253d5dcf1 100644 --- a/lib/srv/desktop/windows_server.go +++ b/lib/srv/desktop/windows_server.go @@ -838,6 +838,8 @@ func (s *WindowsService) connectRDP(ctx context.Context, log logrus.FieldLogger, Conn: tdpConn, AuthorizeFn: authorize, AllowClipboard: authCtx.Checker.DesktopClipboard(), + // allowDirectorySharing() ensures this setting is modulated by build flag while in development + AllowDirectorySharing: authCtx.Checker.DesktopDirectorySharing() && allowDirectorySharing(), }) if err != nil { s.onSessionStart(ctx, sw, &identity, sessionStartTime, windowsUser, string(sessionID), desktop, err) diff --git a/lib/web/desktop/playback.go b/lib/web/desktop/playback.go index 79f0a7c86d1d7..1f22df9d3ed2b 100644 --- a/lib/web/desktop/playback.go +++ b/lib/web/desktop/playback.go @@ -107,8 +107,6 @@ const ( // actionPlayPause toggles the playback state // between playing and paused actionPlayPause = playbackAction("play/pause") - - // TODO(isaiah): support playbackAction("seek") ) // actionMessage is a message passed from the playback client diff --git a/lib/web/resources_test.go b/lib/web/resources_test.go index 63fb8d706dd6d..161a5a4efc81f 100644 --- a/lib/web/resources_test.go +++ b/lib/web/resources_test.go @@ -115,6 +115,7 @@ spec: options: cert_format: standard desktop_clipboard: true + desktop_directory_sharing: true enhanced_recording: - command - network