StateMachineMessageCutter.cs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. using Edge.Core.Processor.Communicator;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. namespace HengshanPaymentTerminal
  8. {
  9. public class StateMachineMessageCutter : IMessageCutter<byte[]>
  10. {
  11. #region Internal State enum
  12. private enum State
  13. {
  14. Uninitialized,
  15. LengthReady,
  16. BodyReady,
  17. EtxReady,
  18. CrcReady
  19. }
  20. #endregion
  21. #region Fields
  22. public byte[] Message { get; private set; }
  23. public event EventHandler OnMessageCut;
  24. public event EventHandler<MessageCutterInvalidMessageReadEventArg> OnInvalidMessageRead;
  25. private string loggerAppendix = "HengshanPay Terminal";
  26. private readonly SizableWindow<byte> window;
  27. //private State nextState = State.Uninitialized;
  28. private const int STX = 0xFA;
  29. #endregion
  30. #region Logger
  31. static NLog.Logger innerLogger = NLog.LogManager.LoadConfiguration("nlog.config").GetLogger("Communicator");
  32. #endregion
  33. public StateMachineMessageCutter()
  34. {
  35. window = new SizableWindow<byte>();
  36. window.OnWindowFull += (data) =>
  37. {
  38. byte[] tempBytes = window.ToArray();
  39. string receiveStr = BitConverter.ToString(tempBytes).Replace("-", " ");
  40. //innerLogger.Info($"receive data : ${receiveStr}");
  41. //查找包头位置
  42. int findIndex = -1;
  43. for (int index = 0;index < tempBytes.Length - 1;index++)
  44. {
  45. if (tempBytes[index] == STX && tempBytes[index + 1] != STX)
  46. {
  47. findIndex = index;
  48. break;
  49. }
  50. }
  51. //innerLogger.Info($"find head index = ${findIndex}");
  52. //若没有找到包头,证明目前缓存中都是垃圾数据,清掉
  53. if (findIndex == -1)
  54. {
  55. window.RemoveRange(0, tempBytes.Length);
  56. return;
  57. }
  58. //长度不够表示有粘包,部分数据还未接收到,等待下一轮接收数据再处理
  59. //1byte FA + 1 byte 目标地址 + 1 byte 源地址 + 1 byte 帧号 + 2 byte 有效数据长度 + 2 byte 数据校验 = 8byte
  60. if(findIndex + 8 > tempBytes.Length)
  61. {
  62. innerLogger.Info($"Insufficient data length,at least ,need {findIndex + 8},but length is {tempBytes.Length}");
  63. return;
  64. }
  65. //获取有效数据长度
  66. int dataLen = Bcd2Int(tempBytes[4], tempBytes[5]);
  67. byte[] reduceFAData = Reduce0xFAPair(tempBytes);
  68. string reduceFADataStr = BitConverter.ToString(reduceFAData).Replace("-", " ");
  69. //innerLogger.Info($"data length is ${dataLen},reduceFAData len is ${reduceFAData.Length},value is ${reduceFADataStr}");
  70. //去除多余FA后,实际数据不足有效数据长度,表示有粘包,仍非完整数据,等待下一轮接收数据再处理
  71. if(reduceFAData.Length < dataLen + 2)
  72. {
  73. innerLogger.Info("Insufficient data length");
  74. return;
  75. }
  76. //计算crc,若值对应不上,表示数据错误
  77. ushort calculatorValue = HengshanCRC16.ComputeChecksum(reduceFAData.AsSpan(1,reduceFAData.Length - 3).ToArray());
  78. byte highByte = reduceFAData[reduceFAData.Length - 2];
  79. byte lowByte = reduceFAData[reduceFAData.Length - 1];
  80. int crcValue = (highByte << 8) | lowByte;
  81. if (calculatorValue != crcValue)
  82. {
  83. innerLogger.Info($"crc value error,get value is {crcValue},calculator value is {calculatorValue}");
  84. window.RemoveRange(0, tempBytes.Length);
  85. return;
  86. }
  87. //获取到实际数据包,并将缓存数据删除
  88. Message = reduceFAData;
  89. if (reduceFAData[6] != 0x10) innerLogger.Info($"receive data : ${reduceFAData}");
  90. var safe = OnMessageCut;
  91. safe?.Invoke(this, null);
  92. window.RemoveRange(0, tempBytes.Length);
  93. };
  94. }
  95. public void Feed(byte[] data)
  96. {
  97. List<byte> list = new(data);
  98. window.Add(list);
  99. //for (int i = 0; i < data.Length; i++)
  100. //{
  101. // window.Add(data[i]);
  102. //}
  103. }
  104. private int Get0xFAPairCountInWindow(IEnumerable<byte> data)
  105. {
  106. return (int)Math.Round(((double)(data.Count(w => w == 0xFA)) / 2), MidpointRounding.AwayFromZero);
  107. }
  108. /// <summary>
  109. /// 删除多余的FA,在接收到的一个完整数据包中,数据包头为FA,其余数据中每一个FA会替换为两个FA,这里要删除多余的fa来获取实际数据包
  110. /// </summary>
  111. /// <param name="data">原数据包</param>
  112. /// <returns></returns>
  113. public byte[] Reduce0xFAPair(byte[] data)
  114. {
  115. List<byte> list = new();
  116. for (int index = 0; index < data.Length; index++)
  117. {
  118. if (index < data.Length -1 && data[index] == STX && data[index + 1] == STX)
  119. {
  120. list.Add(STX);
  121. index++;
  122. }
  123. else
  124. {
  125. list.Add(data[index]);
  126. }
  127. }
  128. return list.ToArray();
  129. }
  130. public int Bcd2Int(byte byte1,byte byte2)
  131. {
  132. // 提取第一个字节的高四位和低四位
  133. int digit1 = (byte1 >> 4) & 0x0F; // 高四位
  134. int digit2 = byte1 & 0x0F; // 低四位
  135. // 提取第二个字节的高四位和低四位
  136. int digit3 = (byte2 >> 4) & 0x0F; // 高四位
  137. int digit4 = byte2 & 0x0F; // 低四位
  138. // 组合成一个整数
  139. int result = digit1 * 1000 + digit2 * 100 + digit3 * 10 + digit4;
  140. return result;
  141. }
  142. }
  143. }