mirror of
https://github.com/yv1ing/ShotRDP.git
synced 2025-09-16 15:10:57 +08:00
460 lines
15 KiB
Go
460 lines
15 KiB
Go
package pdu
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/hex"
|
|
|
|
"ShotRDP/grdp/core"
|
|
"ShotRDP/grdp/emission"
|
|
"ShotRDP/grdp/glog"
|
|
"ShotRDP/grdp/protocol/t125/gcc"
|
|
)
|
|
|
|
type PDULayer struct {
|
|
emission.Emitter
|
|
transport core.Transport
|
|
sharedId uint32
|
|
userId uint16
|
|
channelId uint16
|
|
serverCapabilities map[CapsType]Capability
|
|
clientCapabilities map[CapsType]Capability
|
|
fastPathSender core.FastPathSender
|
|
demandActivePDU *DemandActivePDU
|
|
}
|
|
|
|
func NewPDULayer(t core.Transport) *PDULayer {
|
|
p := &PDULayer{
|
|
Emitter: *emission.NewEmitter(),
|
|
transport: t,
|
|
sharedId: 0x103EA,
|
|
serverCapabilities: map[CapsType]Capability{
|
|
CAPSTYPE_GENERAL: &GeneralCapability{
|
|
ProtocolVersion: 0x0200,
|
|
},
|
|
CAPSTYPE_BITMAP: &BitmapCapability{
|
|
Receive1BitPerPixel: 0x0001,
|
|
Receive4BitsPerPixel: 0x0001,
|
|
Receive8BitsPerPixel: 0x0001,
|
|
BitmapCompressionFlag: 0x0001,
|
|
MultipleRectangleSupport: 0x0001,
|
|
},
|
|
CAPSTYPE_ORDER: &OrderCapability{
|
|
DesktopSaveXGranularity: 1,
|
|
DesktopSaveYGranularity: 20,
|
|
MaximumOrderLevel: 1,
|
|
OrderFlags: NEGOTIATEORDERSUPPORT,
|
|
DesktopSaveSize: 480 * 480,
|
|
},
|
|
CAPSTYPE_POINTER: &PointerCapability{ColorPointerCacheSize: 20},
|
|
CAPSTYPE_INPUT: &InputCapability{},
|
|
CAPSTYPE_VIRTUALCHANNEL: &VirtualChannelCapability{},
|
|
CAPSTYPE_FONT: &FontCapability{SupportFlags: 0x0001},
|
|
CAPSTYPE_COLORCACHE: &ColorCacheCapability{CacheSize: 0x0006},
|
|
CAPSTYPE_SHARE: &ShareCapability{},
|
|
},
|
|
clientCapabilities: map[CapsType]Capability{
|
|
CAPSTYPE_GENERAL: &GeneralCapability{
|
|
ProtocolVersion: 0x0200,
|
|
},
|
|
CAPSTYPE_BITMAP: &BitmapCapability{
|
|
Receive1BitPerPixel: 0x0001,
|
|
Receive4BitsPerPixel: 0x0001,
|
|
Receive8BitsPerPixel: 0x0001,
|
|
BitmapCompressionFlag: 0x0001,
|
|
MultipleRectangleSupport: 0x0001,
|
|
},
|
|
CAPSTYPE_ORDER: &OrderCapability{
|
|
DesktopSaveXGranularity: 1,
|
|
DesktopSaveYGranularity: 20,
|
|
MaximumOrderLevel: 1,
|
|
OrderFlags: NEGOTIATEORDERSUPPORT,
|
|
DesktopSaveSize: 480 * 480,
|
|
TextANSICodePage: 0x4e4,
|
|
},
|
|
CAPSTYPE_CONTROL: &ControlCapability{0, 0, 2, 2},
|
|
CAPSTYPE_ACTIVATION: &WindowActivationCapability{},
|
|
CAPSTYPE_POINTER: &PointerCapability{1, 20, 20},
|
|
CAPSTYPE_SHARE: &ShareCapability{},
|
|
CAPSTYPE_COLORCACHE: &ColorCacheCapability{6, 0},
|
|
CAPSTYPE_SOUND: &SoundCapability{0x0001, 0},
|
|
CAPSTYPE_INPUT: &InputCapability{},
|
|
CAPSTYPE_FONT: &FontCapability{0x0001, 0},
|
|
CAPSTYPE_BRUSH: &BrushCapability{BRUSH_COLOR_8x8},
|
|
CAPSTYPE_GLYPHCACHE: &GlyphCapability{},
|
|
CAPSETTYPE_BITMAP_CODECS: &BitmapCodecsCapability{},
|
|
CAPSTYPE_BITMAPCACHE_REV2: &BitmapCache2Capability{
|
|
BitmapCachePersist: 2,
|
|
CachesNum: 5,
|
|
BmpC0Cells: 0x258,
|
|
BmpC1Cells: 0x258,
|
|
BmpC2Cells: 0x800,
|
|
BmpC3Cells: 0x1000,
|
|
BmpC4Cells: 0x800,
|
|
},
|
|
CAPSTYPE_VIRTUALCHANNEL: &VirtualChannelCapability{0, 1600},
|
|
CAPSETTYPE_MULTIFRAGMENTUPDATE: &MultiFragmentUpdate{65535},
|
|
CAPSTYPE_RAIL: &RemoteProgramsCapability{
|
|
RailSupportLevel: RAIL_LEVEL_SUPPORTED |
|
|
RAIL_LEVEL_SHELL_INTEGRATION_SUPPORTED |
|
|
RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED |
|
|
RAIL_LEVEL_SERVER_TO_CLIENT_IME_SYNC_SUPPORTED |
|
|
RAIL_LEVEL_HIDE_MINIMIZED_APPS_SUPPORTED |
|
|
RAIL_LEVEL_WINDOW_CLOAKING_SUPPORTED |
|
|
RAIL_LEVEL_HANDSHAKE_EX_SUPPORTED |
|
|
RAIL_LEVEL_DOCKED_LANGBAR_SUPPORTED,
|
|
},
|
|
CAPSETTYPE_LARGE_POINTER: &LargePointerCapability{1},
|
|
CAPSETTYPE_SURFACE_COMMANDS: &SurfaceCommandsCapability{
|
|
CmdFlags: SURFCMDS_SET_SURFACE_BITS | SURFCMDS_STREAM_SURFACE_BITS | SURFCMDS_FRAME_MARKER,
|
|
},
|
|
CAPSSETTYPE_FRAME_ACKNOWLEDGE: &FrameAcknowledgeCapability{2},
|
|
},
|
|
}
|
|
|
|
t.On("close", func() {
|
|
p.Emit("close")
|
|
}).On("error", func(err error) {
|
|
p.Emit("error", err)
|
|
})
|
|
return p
|
|
}
|
|
|
|
func (p *PDULayer) sendPDU(message PDUMessage) {
|
|
pdu := NewPDU(p.userId, message)
|
|
p.transport.Write(pdu.serialize())
|
|
}
|
|
|
|
func (p *PDULayer) sendDataPDU(message DataPDUData) {
|
|
dataPdu := NewDataPDU(message, p.sharedId)
|
|
p.sendPDU(dataPdu)
|
|
}
|
|
|
|
func (p *PDULayer) SetFastPathSender(f core.FastPathSender) {
|
|
p.fastPathSender = f
|
|
}
|
|
|
|
type Client struct {
|
|
*PDULayer
|
|
clientCoreData *gcc.ClientCoreData
|
|
buff *bytes.Buffer
|
|
}
|
|
|
|
func NewClient(t core.Transport) *Client {
|
|
c := &Client{
|
|
PDULayer: NewPDULayer(t),
|
|
buff: &bytes.Buffer{},
|
|
}
|
|
c.transport.Once("connect", c.connect)
|
|
return c
|
|
}
|
|
|
|
func (c *Client) connect(data *gcc.ClientCoreData, userId uint16, channelId uint16) {
|
|
glog.Debug("pdu connect:", userId, ",", channelId)
|
|
c.clientCoreData = data
|
|
c.userId = userId
|
|
c.channelId = channelId
|
|
c.transport.Once("data", c.recvDemandActivePDU)
|
|
}
|
|
|
|
func (c *Client) recvDemandActivePDU(s []byte) {
|
|
glog.Trace("PDU recvDemandActivePDU", hex.EncodeToString(s))
|
|
r := bytes.NewReader(s)
|
|
pdu, err := readPDU(r)
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return
|
|
}
|
|
if pdu.ShareCtrlHeader.PDUType != PDUTYPE_DEMANDACTIVEPDU {
|
|
glog.Info("PDU ignore message during connection sequence, type is", pdu.ShareCtrlHeader.PDUType)
|
|
c.transport.Once("data", c.recvDemandActivePDU)
|
|
return
|
|
}
|
|
c.sharedId = pdu.Message.(*DemandActivePDU).SharedId
|
|
c.demandActivePDU = pdu.Message.(*DemandActivePDU)
|
|
for _, caps := range c.demandActivePDU.CapabilitySets {
|
|
glog.Debugf("serverCapabilities<%s>: %+v", caps.Type(), caps)
|
|
c.serverCapabilities[caps.Type()] = caps
|
|
}
|
|
|
|
c.sendConfirmActivePDU()
|
|
c.sendClientFinalizeSynchronizePDU()
|
|
c.transport.Once("data", c.recvServerSynchronizePDU)
|
|
}
|
|
|
|
func (c *Client) sendConfirmActivePDU() {
|
|
glog.Debug("PDU start sendConfirmActivePDU")
|
|
|
|
pdu := NewConfirmActivePDU()
|
|
generalCapa := c.clientCapabilities[CAPSTYPE_GENERAL].(*GeneralCapability)
|
|
generalCapa.OSMajorType = OSMAJORTYPE_WINDOWS
|
|
generalCapa.OSMinorType = OSMINORTYPE_WINDOWS_NT
|
|
generalCapa.ExtraFlags = LONG_CREDENTIALS_SUPPORTED | NO_BITMAP_COMPRESSION_HDR |
|
|
FASTPATH_OUTPUT_SUPPORTED | AUTORECONNECT_SUPPORTED
|
|
generalCapa.RefreshRectSupport = 0
|
|
generalCapa.SuppressOutputSupport = 0
|
|
|
|
bitmapCapa := c.clientCapabilities[CAPSTYPE_BITMAP].(*BitmapCapability)
|
|
bitmapCapa.PreferredBitsPerPixel = c.clientCoreData.HighColorDepth
|
|
bitmapCapa.DesktopWidth = c.clientCoreData.DesktopWidth
|
|
bitmapCapa.DesktopHeight = c.clientCoreData.DesktopHeight
|
|
bitmapCapa.DesktopResizeFlag = 0x0001
|
|
|
|
orderCapa := c.clientCapabilities[CAPSTYPE_ORDER].(*OrderCapability)
|
|
orderCapa.OrderFlags = NEGOTIATEORDERSUPPORT | ZEROBOUNDSDELTASSUPPORT | COLORINDEXSUPPORT | ORDERFLAGS_EXTRA_FLAGS
|
|
orderCapa.OrderSupportExFlags |= ORDERFLAGS_EX_ALTSEC_FRAME_MARKER_SUPPORT
|
|
orderCapa.OrderSupport[TS_NEG_DSTBLT_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_PATBLT_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_SCRBLT_INDEX] = 1
|
|
//orderCapa.OrderSupport[TS_NEG_LINETO_INDEX] = 1
|
|
//orderCapa.OrderSupport[TS_NEG_MEMBLT_INDEX] = 1
|
|
//orderCapa.OrderSupport[TS_NEG_MEM3BLT_INDEX] = 1
|
|
//orderCapa.OrderSupport[TS_NEG_POLYLINE_INDEX] = 1
|
|
/*orderCapa.OrderSupport[TS_NEG_MULTIOPAQUERECT_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_GLYPH_INDEX_INDEX] = 1
|
|
//orderCapa.OrderSupport[TS_NEG_DRAWNINEGRID_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_SAVEBITMAP_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_POLYGON_SC_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_POLYGON_CB_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_ELLIPSE_SC_INDEX] = 1
|
|
orderCapa.OrderSupport[TS_NEG_ELLIPSE_CB_INDEX] = 1*/
|
|
orderCapa.OrderSupport[TS_NEG_FAST_GLYPH_INDEX] = 1
|
|
|
|
inputCapa := c.clientCapabilities[CAPSTYPE_INPUT].(*InputCapability)
|
|
inputCapa.Flags = INPUT_FLAG_SCANCODES | INPUT_FLAG_MOUSEX | INPUT_FLAG_UNICODE
|
|
inputCapa.KeyboardLayout = c.clientCoreData.KbdLayout
|
|
inputCapa.KeyboardType = c.clientCoreData.KeyboardType
|
|
inputCapa.KeyboardSubType = c.clientCoreData.KeyboardSubType
|
|
inputCapa.KeyboardFunctionKey = c.clientCoreData.KeyboardFnKeys
|
|
inputCapa.ImeFileName = c.clientCoreData.ImeFileName
|
|
|
|
glyphCapa := c.clientCapabilities[CAPSTYPE_GLYPHCACHE].(*GlyphCapability)
|
|
/*glyphCapa.GlyphCache[0] = cacheEntry{254, 4}
|
|
glyphCapa.GlyphCache[1] = cacheEntry{254, 4}
|
|
glyphCapa.GlyphCache[2] = cacheEntry{254, 8}
|
|
glyphCapa.GlyphCache[3] = cacheEntry{254, 8}
|
|
glyphCapa.GlyphCache[4] = cacheEntry{254, 16}
|
|
glyphCapa.GlyphCache[5] = cacheEntry{254, 32}
|
|
glyphCapa.GlyphCache[6] = cacheEntry{254, 64}
|
|
glyphCapa.GlyphCache[7] = cacheEntry{254, 128}
|
|
glyphCapa.GlyphCache[8] = cacheEntry{254, 256}
|
|
glyphCapa.GlyphCache[9] = cacheEntry{64, 2048}
|
|
glyphCapa.FragCache = 0x01000100*/
|
|
glyphCapa.SupportLevel = GLYPH_SUPPORT_NONE
|
|
|
|
pdu.SharedId = c.sharedId
|
|
for _, v := range c.clientCapabilities {
|
|
glog.Debugf("clientCapabilities<%s>: %+v", v.Type(), v)
|
|
pdu.CapabilitySets = append(pdu.CapabilitySets, v)
|
|
}
|
|
pdu.NumberCapabilities = uint16(len(pdu.CapabilitySets))
|
|
pdu.LengthSourceDescriptor = c.demandActivePDU.LengthSourceDescriptor
|
|
pdu.SourceDescriptor = c.demandActivePDU.SourceDescriptor
|
|
pdu.LengthCombinedCapabilities = c.demandActivePDU.LengthCombinedCapabilities
|
|
|
|
c.sendPDU(pdu)
|
|
}
|
|
|
|
func (c *Client) sendClientFinalizeSynchronizePDU() {
|
|
glog.Debug("PDU start sendClientFinalizeSynchronizePDU")
|
|
c.sendDataPDU(NewSynchronizeDataPDU(c.channelId))
|
|
c.sendDataPDU(&ControlDataPDU{Action: CTRLACTION_COOPERATE})
|
|
c.sendDataPDU(&ControlDataPDU{Action: CTRLACTION_REQUEST_CONTROL})
|
|
//c.sendDataPDU(&PersistKeyPDU{BBitMask: 0x03})
|
|
c.sendDataPDU(&FontListDataPDU{ListFlags: 0x0003, EntrySize: 0x0032})
|
|
}
|
|
|
|
func (c *Client) recvServerSynchronizePDU(s []byte) {
|
|
glog.Debug("PDU recvServerSynchronizePDU")
|
|
r := bytes.NewReader(s)
|
|
pdu, err := readPDU(r)
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return
|
|
}
|
|
dataPdu, ok := pdu.Message.(*DataPDU)
|
|
if !ok || dataPdu.Header.PDUType2 != PDUTYPE2_SYNCHRONIZE {
|
|
if ok {
|
|
glog.Error("recvServerSynchronizePDU ignore datapdu type2", dataPdu.Header.PDUType2)
|
|
} else {
|
|
glog.Error("recvServerSynchronizePDU ignore message type", pdu.ShareCtrlHeader.PDUType)
|
|
}
|
|
glog.Infof("%+v", dataPdu)
|
|
c.transport.Once("data", c.recvServerSynchronizePDU)
|
|
return
|
|
}
|
|
c.transport.Once("data", c.recvServerControlCooperatePDU)
|
|
}
|
|
|
|
func (c *Client) recvServerControlCooperatePDU(s []byte) {
|
|
glog.Debug("PDU recvServerControlCooperatePDU")
|
|
r := bytes.NewReader(s)
|
|
pdu, err := readPDU(r)
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return
|
|
}
|
|
dataPdu, ok := pdu.Message.(*DataPDU)
|
|
if !ok || dataPdu.Header.PDUType2 != PDUTYPE2_CONTROL {
|
|
if ok {
|
|
glog.Error("recvServerControlCooperatePDU ignore datapdu type2", dataPdu.Header.PDUType2)
|
|
} else {
|
|
glog.Error("recvServerControlCooperatePDU ignore message type", pdu.ShareCtrlHeader.PDUType)
|
|
}
|
|
c.transport.Once("data", c.recvServerControlCooperatePDU)
|
|
return
|
|
}
|
|
if dataPdu.Data.(*ControlDataPDU).Action != CTRLACTION_COOPERATE {
|
|
glog.Error("recvServerControlCooperatePDU ignore action", dataPdu.Data.(*ControlDataPDU).Action)
|
|
c.transport.Once("data", c.recvServerControlCooperatePDU)
|
|
return
|
|
}
|
|
c.transport.Once("data", c.recvServerControlGrantedPDU)
|
|
}
|
|
|
|
func (c *Client) recvServerControlGrantedPDU(s []byte) {
|
|
glog.Debug("PDU recvServerControlGrantedPDU")
|
|
r := bytes.NewReader(s)
|
|
pdu, err := readPDU(r)
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return
|
|
}
|
|
dataPdu, ok := pdu.Message.(*DataPDU)
|
|
if !ok || dataPdu.Header.PDUType2 != PDUTYPE2_CONTROL {
|
|
if ok {
|
|
glog.Error("recvServerControlGrantedPDU ignore datapdu type2", dataPdu.Header.PDUType2)
|
|
} else {
|
|
glog.Error("recvServerControlGrantedPDU ignore message type", pdu.ShareCtrlHeader.PDUType)
|
|
}
|
|
c.transport.Once("data", c.recvServerControlGrantedPDU)
|
|
return
|
|
}
|
|
if dataPdu.Data.(*ControlDataPDU).Action != CTRLACTION_GRANTED_CONTROL {
|
|
glog.Error("recvServerControlGrantedPDU ignore action", dataPdu.Data.(*ControlDataPDU).Action)
|
|
c.transport.Once("data", c.recvServerControlGrantedPDU)
|
|
return
|
|
}
|
|
c.transport.Once("data", c.recvServerFontMapPDU)
|
|
}
|
|
|
|
func (c *Client) recvServerFontMapPDU(s []byte) {
|
|
glog.Debug("PDU recvServerFontMapPDU")
|
|
r := bytes.NewReader(s)
|
|
pdu, err := readPDU(r)
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return
|
|
}
|
|
dataPdu, ok := pdu.Message.(*DataPDU)
|
|
if !ok || dataPdu.Header.PDUType2 != PDUTYPE2_FONTMAP {
|
|
if ok {
|
|
glog.Error("recvServerFontMapPDU ignore datapdu type2", dataPdu.Header.PDUType2)
|
|
} else {
|
|
glog.Error("recvServerFontMapPDU ignore message type", pdu.ShareCtrlHeader.PDUType)
|
|
}
|
|
return
|
|
}
|
|
c.transport.On("data", c.recvPDU)
|
|
c.Emit("ready")
|
|
}
|
|
|
|
func (c *Client) recvPDU(s []byte) {
|
|
glog.Trace("PDU recvPDU", hex.EncodeToString(s))
|
|
r := bytes.NewReader(s)
|
|
if r.Len() > 0 {
|
|
p, err := readPDU(r)
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return
|
|
}
|
|
if p.ShareCtrlHeader.PDUType == PDUTYPE_DEACTIVATEALLPDU {
|
|
c.transport.Once("data", c.recvDemandActivePDU)
|
|
} else if p.ShareCtrlHeader.PDUType == PDUTYPE_DATAPDU {
|
|
d := p.Message.(*DataPDU)
|
|
if d.Header.PDUType2 == PDUTYPE2_UPDATE {
|
|
up := d.Data.(*UpdateDataPDU)
|
|
p := up.Udata
|
|
if up.UpdateType == FASTPATH_UPDATETYPE_BITMAP {
|
|
c.Emit("bitmap", p.(*BitmapUpdateDataPDU).Rectangles)
|
|
} else if up.UpdateType == FASTPATH_UPDATETYPE_ORDERS {
|
|
c.Emit("orders", p.(*FastPathOrdersPDU).OrderPdus)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *Client) RecvFastPath(secFlag byte, s []byte) {
|
|
glog.Trace("PDU RecvFastPath", hex.EncodeToString(s))
|
|
r := bytes.NewReader(s)
|
|
for r.Len() > 0 {
|
|
updateHeader, err := core.ReadUInt8(r)
|
|
if err != nil {
|
|
return
|
|
}
|
|
updateCode := updateHeader & 0x0f
|
|
fragmentation := updateHeader & 0x30
|
|
compression := updateHeader & 0xC0
|
|
|
|
var compressionFlags uint8 = 0
|
|
if compression == FASTPATH_OUTPUT_COMPRESSION_USED {
|
|
compressionFlags, err = core.ReadUInt8(r)
|
|
}
|
|
|
|
size, err := core.ReadUint16LE(r)
|
|
|
|
if err != nil {
|
|
return
|
|
}
|
|
glog.Trace("Code:", FastPathUpdateType(updateCode),
|
|
"compressionFlags:", compressionFlags,
|
|
"fragmentation:", fragmentation,
|
|
"size:", size, "len:", r.Len())
|
|
if compressionFlags&RDP_MPPC_COMPRESSED != 0 {
|
|
glog.Info("RDP_MPPC_COMPRESSED")
|
|
}
|
|
if fragmentation != FASTPATH_FRAGMENT_SINGLE {
|
|
if fragmentation == FASTPATH_FRAGMENT_FIRST {
|
|
c.buff.Reset()
|
|
}
|
|
b, _ := core.ReadBytes(r.Len(), r)
|
|
c.buff.Write(b)
|
|
if fragmentation != FASTPATH_FRAGMENT_LAST {
|
|
return
|
|
}
|
|
r = bytes.NewReader(c.buff.Bytes())
|
|
}
|
|
|
|
p, err := readFastPathUpdatePDU(r, updateCode)
|
|
if err != nil || p == nil || p.Data == nil {
|
|
glog.Debug("readFastPathUpdatePDU:", err)
|
|
return
|
|
}
|
|
|
|
if updateCode == FASTPATH_UPDATETYPE_BITMAP {
|
|
c.Emit("bitmap", p.Data.(*FastPathBitmapUpdateDataPDU).Rectangles)
|
|
} else if updateCode == FASTPATH_UPDATETYPE_COLOR {
|
|
c.Emit("color", p.Data.(*FastPathColorPdu))
|
|
} else if updateCode == FASTPATH_UPDATETYPE_ORDERS {
|
|
c.Emit("orders", p.Data.(*FastPathOrdersPDU).OrderPdus)
|
|
}
|
|
}
|
|
}
|
|
|
|
type InputEventsInterface interface {
|
|
Serialize() []byte
|
|
}
|
|
|
|
func (c *Client) SendInputEvents(msgType uint16, events []InputEventsInterface) {
|
|
p := &ClientInputEventPDU{}
|
|
p.NumEvents = uint16(len(events))
|
|
p.SlowPathInputEvents = make([]SlowPathInputEvent, 0, p.NumEvents)
|
|
for _, in := range events {
|
|
seria := in.Serialize()
|
|
s := SlowPathInputEvent{0, msgType, len(seria), seria}
|
|
p.SlowPathInputEvents = append(p.SlowPathInputEvents, s)
|
|
}
|
|
|
|
c.sendDataPDU(p)
|
|
}
|