IsoSocket.cs 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. using System;
  2. using System.Net;
  3. using System.Net.Sockets;
  4. using System.Threading;
  5. using Houdar.PLC.Driver.Simenss.Protocol.Iso;
  6. using ByteBuffer = Houdar.Core.Communication.Transport.ByteBuffer;
  7. namespace Houdar.PLC.Driver.Simenss.Communication
  8. {
  9. /// <summary>
  10. /// 采用同步通信机制
  11. /// </summary>
  12. public class IsoSocket : IIsoSender
  13. {
  14. private Socket _socket;
  15. public bool Connected
  16. {
  17. get
  18. {
  19. try
  20. {
  21. if (_socket == null)
  22. return false;
  23. return !((_socket.Poll(1000, SelectMode.SelectRead) && (_socket.Available == 0)) || !_socket.Connected);
  24. }
  25. catch { return false; }
  26. }
  27. }
  28. private Action<MessageEvent> _messageAction;
  29. public void RegisterMessage(Action<MessageEvent> messageAction)
  30. {
  31. if (messageAction == null) throw new ArgumentNullException("messageAction");
  32. _messageAction = messageAction;
  33. }
  34. private void OnMessage(string methode, string message)
  35. {
  36. if (_messageAction != null)
  37. _messageAction.Invoke(new MessageEvent(methode, message));
  38. }
  39. private static bool _isConnectionSuccessful = false;
  40. private int _connecting;//连接中
  41. public bool Connect(string ip, int port)
  42. {
  43. try
  44. {
  45. if (Interlocked.CompareExchange(ref _connecting, 1, 0) != 0)
  46. {
  47. OnMessage("Connect", "连接正在处理中");
  48. return false;
  49. }
  50. _socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
  51. _socket.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReceiveTimeout, 1000);
  52. _socket.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.SendTimeout, 1000);
  53. IPEndPoint server = new IPEndPoint(IPAddress.Parse(ip), 102);
  54. #region 添加的代码 20170801
  55. timeoutObject = new ManualResetEvent(false);
  56. _socket.BeginConnect(server, new AsyncCallback(CallBackMethod), _socket);
  57. if (timeoutObject.WaitOne(1000, false))
  58. {
  59. if (_isConnectionSuccessful)
  60. {
  61. }
  62. else
  63. {
  64. _socket.Close();
  65. throw new Exception("连接超时");
  66. }
  67. }
  68. #endregion;
  69. #region 被注释的代码
  70. //_socket.Connect(server);
  71. #endregion;
  72. return true;
  73. }
  74. catch (Exception ex)
  75. {
  76. OnMessage("Connect", "连接异常:{ex.Message}");
  77. return false;
  78. }
  79. finally
  80. {
  81. Interlocked.CompareExchange(ref _connecting, 0, 1);
  82. }
  83. }
  84. #region 添加的代码 20170801
  85. private ManualResetEvent timeoutObject;//通知一个或多个正在等待的线程已发生事件
  86. /// <summary>
  87. /// 连接过程中的异步回调
  88. /// </summary>
  89. /// <param name="asyncresult"></param>
  90. private void CallBackMethod(IAsyncResult asyncresult)
  91. {
  92. try
  93. {
  94. _isConnectionSuccessful = false;
  95. _socket = asyncresult.AsyncState as Socket;
  96. if (_socket != null)
  97. {
  98. _socket.EndConnect(asyncresult);
  99. _isConnectionSuccessful = true;
  100. }
  101. }
  102. catch (Exception ex)
  103. {
  104. _isConnectionSuccessful = false;
  105. }
  106. finally
  107. {
  108. timeoutObject.Set();//解除被阻塞的连接线程
  109. }
  110. }
  111. #endregion;
  112. private bool IsActive()
  113. {
  114. if (!Connected)
  115. {
  116. OnMessage("SendRecive", "未连接到PLC设备");
  117. return false;
  118. }
  119. if (Interlocked.CompareExchange(ref _sending, 1, 0) != 0)
  120. {
  121. OnMessage("SendRecive", "正在处理中");
  122. return false;
  123. }
  124. return true;
  125. }
  126. private int _sending;//发送中
  127. public byte[] SendRecive(byte[] sendBytes, int length)
  128. {
  129. try
  130. {
  131. if (!IsActive()) return null;
  132. var sendStr = BitConverter.ToString(sendBytes, 0, length);
  133. //Console.WriteLine($"{DateTime.Now.ToString("HH:mm:ss ffff")}[Send]({length}):{sendStr}");
  134. _socket.Send(sendBytes, length, SocketFlags.None);
  135. byte[] bReceive = new byte[1024 * 4 * 8];
  136. var resLength = _socket.Receive(bReceive, SocketFlags.None);
  137. var recvStr = BitConverter.ToString(bReceive, 0, resLength);
  138. //Console.WriteLine($"{DateTime.Now.ToString("HH:mm:ss ffff")}[Recv]({resLength}):{recvStr}");
  139. var receiveBytes = new byte[resLength];
  140. Array.Copy(bReceive, 0, receiveBytes, 0, resLength);
  141. return receiveBytes;
  142. }
  143. catch (Exception ex)
  144. {
  145. OnMessage("Send", "发送异常:{ex.Message}");
  146. return null;
  147. }
  148. finally
  149. {
  150. Interlocked.CompareExchange(ref _sending, 0, 1);
  151. }
  152. }
  153. private TResponse GetResponse<TResponse>(byte[] resBytes, bool isContainHeader = true)
  154. where TResponse : IBuildResponse, new()
  155. {
  156. ByteBuffer buffer= ByteBuffer.Allocate();
  157. buffer.Push(resBytes);
  158. if(isContainHeader)
  159. {
  160. IsoDataPdu pdu = new IsoDataPdu();
  161. pdu.Build(buffer);
  162. if (resBytes.Length != pdu.Length)
  163. return default(TResponse);
  164. }
  165. TResponse response = new TResponse();
  166. response.Build(buffer);
  167. return response;
  168. }
  169. private ByteBuffer GetByteBuffer<TRequest>(TRequest request, bool isContainHeader=true)
  170. where TRequest : IBuildRequest, new()
  171. {
  172. if(request==null) throw new ArgumentNullException("request");
  173. request.Build();
  174. var bufferData = request.GetBuffer();
  175. ByteBuffer buffer = ByteBuffer.Allocate();
  176. if(isContainHeader)
  177. {
  178. //构建ISO头部
  179. IsoDataPdu pdu = new IsoDataPdu { Length = (ushort)bufferData.WriteIndex };
  180. pdu.Build();
  181. pdu.GetBuffer(buffer);
  182. }
  183. buffer.Push(bufferData.Buffer, bufferData.WriteIndex);
  184. return buffer;
  185. }
  186. /// <summary>
  187. /// 不含ISO头的请求
  188. /// </summary>
  189. /// <typeparam name="TRequest"></typeparam>
  190. /// <typeparam name="TResponse"></typeparam>
  191. /// <param name="request"></param>
  192. /// <returns></returns>
  193. public TResponse Send<TRequest, TResponse>(TRequest request) where TRequest : IBuildRequest, new() where TResponse : IBuildResponse, new()
  194. {
  195. var buffer = GetByteBuffer(request, false);
  196. var resByte = SendRecive(buffer.Buffer, buffer.WriteIndex);
  197. if (resByte == null || resByte.Length == 0) return default(TResponse);
  198. return GetResponse<TResponse>(resByte, false);
  199. }
  200. /// <summary>
  201. /// 包含ISO头的请求
  202. /// </summary>
  203. /// <typeparam name="TRequest"></typeparam>
  204. /// <typeparam name="TResponse"></typeparam>
  205. /// <param name="request"></param>
  206. /// <returns></returns>
  207. public TResponse IsoSend<TRequest, TResponse>(TRequest request) where TRequest : IBuildRequest, new() where TResponse : IBuildResponse, new()
  208. {
  209. try
  210. {
  211. var buffer = GetByteBuffer(request);
  212. var resByte = SendRecive(buffer.Buffer, buffer.WriteIndex);
  213. if (resByte == null) return default(TResponse);
  214. return GetResponse<TResponse>(resByte);
  215. }
  216. catch (Exception ex)
  217. {
  218. return default(TResponse);
  219. }
  220. }
  221. }
  222. }