/// <summary> /// Sorts packet's bytes into easily accessible structures. Manipulates endianess. /// </summary> /// <param name="data">Expects bytes, composed of fully decoded bits (Decode, then SeparatePackets, then DestuffZeros, then PackIntoBytes).</param> /// <param name="time">x10nS</param> /// <param name="littleEndian">See Communications.Parse</param> /// <returns></returns> public static EncoderPacket Parse(byte[] data, int time, bool littleEndian = false) { //bool containsSub = data.Length > OrdinaryPacketLength; byte[][] result = new byte[FieldLength.Count][]; int dataLen = data.Length - FieldLength[Fields.FCS]; List <byte> fcs = new List <byte>(dataLen > -1 ? dataLen : 0); int current = 0; try { for (int i = 0; i < FieldLength.Count; i++) { if (OrdinaryPacketFieldsToExclude.Any(x => x == (Fields)i)) { continue; } int l = FieldLength[(Fields)i]; if (l == -1) { l = data.Length - FieldLength.Sum(x => x.Value) - 1; //-1 stands for "variable length" } result[i] = new byte[l]; //Multibyte fields may be little-endian at physical layer (in fact they should be, but it turns out they're not...) //All in all, we'd better implement a switch for (int j = 0; j < l; j++) { result[i][j] = data[current + (littleEndian ? (l - j - 1) : (j))]; if ((Fields)i != Fields.FCS) { fcs.Add(result[i][j]); } } current += l; } } catch (OverflowException) { ErrorListener.Add(new Exception(string.Format("Packet at {0} is incomplete!", time))); } var toParse = result[(int)Fields.Unknown]; //if (containsSub) toParse = toParse.Concat(result[(int)Fields.SubcommandData]).ToArray(); var packet = new EncoderPacket(result, EncoderCommand.Parse(toParse), time); packet.ComputedFCS = HDLCManchesterDecoder.ComputeFCS(fcs.ToArray()); try { packet.FCSError = !packet.ComputedFCS.SequenceEqual(packet.ParsedData[(int)Fields.FCS]); } catch (ArgumentNullException) { packet.FCSError = true; } packet.DatabaseReport = EncoderCommandDatabase.GetReport(packet); return(packet); }
protected EncoderPacket(byte[][] d, EncoderCommand cmd, int time) : this(d, cmd) { Timestamp = time; }
protected EncoderPacket(byte[][] d, EncoderCommand cmd) { ParsedData = d.Select(x => (x != null) ? x.ToArray() : null).ToArray(); Command = cmd; }