ModbusSerialTransport.cs 2.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. using System.Diagnostics;
  2. using System.IO;
  3. using NModbus.Logging;
  4. namespace NModbus.IO
  5. {
  6. /// <summary>
  7. /// Transport for Serial protocols.
  8. /// Refined Abstraction - http://en.wikipedia.org/wiki/Bridge_Pattern
  9. /// </summary>
  10. public abstract class ModbusSerialTransport : ModbusTransport, IModbusSerialTransport
  11. {
  12. private bool _checkFrame = true;
  13. internal ModbusSerialTransport(IStreamResource streamResource, IModbusFactory modbusFactory, IModbusLogger logger)
  14. : base(streamResource, modbusFactory, logger)
  15. {
  16. Debug.Assert(streamResource != null, "Argument streamResource cannot be null.");
  17. }
  18. /// <summary>
  19. /// Gets or sets a value indicating whether LRC/CRC frame checking is performed on messages.
  20. /// </summary>
  21. public bool CheckFrame
  22. {
  23. get => _checkFrame;
  24. set => _checkFrame = value;
  25. }
  26. public void DiscardInBuffer()
  27. {
  28. StreamResource.DiscardInBuffer();
  29. }
  30. public override void Write(IModbusMessage message)
  31. {
  32. DiscardInBuffer();
  33. byte[] frame = BuildMessageFrame(message);
  34. Logger.LogFrameTx(frame);
  35. StreamResource.Write(frame, 0, frame.Length);
  36. }
  37. public override IModbusMessage CreateResponse<T>(byte[] frame)
  38. {
  39. IModbusMessage response = base.CreateResponse<T>(frame);
  40. // compare checksum
  41. if (CheckFrame && !ChecksumsMatch(response, frame))
  42. {
  43. string msg = $"Checksums failed to match {string.Join(", ", response.MessageFrame)} != {string.Join(", ", frame)}";
  44. Logger.Warning(msg);
  45. throw new IOException(msg);
  46. }
  47. return response;
  48. }
  49. public abstract void IgnoreResponse();
  50. public abstract bool ChecksumsMatch(IModbusMessage message, byte[] messageFrame);
  51. internal override void OnValidateResponse(IModbusMessage request, IModbusMessage response)
  52. {
  53. // no-op
  54. }
  55. }
  56. }