2023-04-12 15:40:00 +00:00
|
|
|
|
using System.Text;
|
2022-08-06 10:21:00 +00:00
|
|
|
|
using Modbus.Device;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
using Modbus.Serial;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
using PluginInterface;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
using System.IO.Ports;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
using System.Net.Sockets;
|
2023-04-12 15:40:00 +00:00
|
|
|
|
using Microsoft.Extensions.Logging;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
namespace PLC.ModBusMaster
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
[DriverSupported("TCP")]
|
|
|
|
|
[DriverSupported("UDP")]
|
|
|
|
|
[DriverSupported("Rtu")]
|
|
|
|
|
[DriverSupported("Rtu Over TCP")]
|
|
|
|
|
[DriverSupported("Rtu Over UDP")]
|
|
|
|
|
[DriverSupported("Ascii")]
|
|
|
|
|
[DriverSupported("Ascii Over TCP")]
|
|
|
|
|
[DriverSupported("Ascii Over UDP")]
|
|
|
|
|
[DriverInfo("ModBusMaster", "V1.0.0", "Copyright IoTGateway.net 20230220")]
|
|
|
|
|
public class DeviceModBusMaster : IDriver
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-08-06 10:21:00 +00:00
|
|
|
|
private TcpClient? _tcpClient;
|
|
|
|
|
private UdpClient? _udpClient;
|
|
|
|
|
private SerialPort? _serialPort;
|
2023-04-12 15:40:00 +00:00
|
|
|
|
private ModbusMaster? _master;
|
2022-08-06 10:21:00 +00:00
|
|
|
|
private SerialPortAdapter? _adapter;
|
|
|
|
|
|
|
|
|
|
public ILogger _logger { get; set; }
|
2022-08-08 07:15:09 +00:00
|
|
|
|
private readonly string _device;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2021-12-12 06:55:48 +00:00
|
|
|
|
#region 配置参数
|
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("设备Id")] public string DeviceId { get; set; }
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("主站类型")] public MasterType MasterType { get; set; } = MasterType.Tcp;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("IP地址")] public string IpAddress { get; set; } = "127.0.0.1";
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("端口号")] public int Port { get; set; } = 502;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("串口名")] public string PortName { get; set; } = "COM1";
|
2021-12-15 16:16:22 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("波特率")] public int BaudRate { get; set; } = 9600;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("数据位")] public int DataBits { get; set; } = 8;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("校验位")] public Parity Parity { get; set; } = Parity.None;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("停止位")] public StopBits StopBits { get; set; } = StopBits.One;
|
2021-12-15 16:16:22 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("从站号")] public byte SlaveAddress { get; set; } = 1;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("超时时间ms")] public int Timeout { get; set; } = 3000;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
[ConfigParameter("最小通讯周期ms")] public uint MinPeriod { get; set; } = 3000;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
#region 生命周期
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 反射构造函数
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="device"></param>
|
|
|
|
|
/// <param name="logger"></param>
|
|
|
|
|
public DeviceModBusMaster(string device, ILogger logger)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_device = device;
|
|
|
|
|
_logger = logger;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2022-08-08 07:15:09 +00:00
|
|
|
|
_logger.LogInformation($"Device:[{_device}],Create()");
|
2022-08-06 10:21:00 +00:00
|
|
|
|
}
|
2021-12-12 06:55:48 +00:00
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 连接状态
|
|
|
|
|
/// </summary>
|
2021-12-12 06:55:48 +00:00
|
|
|
|
public bool IsConnected
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2022-08-10 08:52:41 +00:00
|
|
|
|
switch (MasterType)
|
2021-12-18 11:13:41 +00:00
|
|
|
|
{
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Tcp:
|
|
|
|
|
case MasterType.RtuOnTcp:
|
|
|
|
|
case MasterType.AsciiOnTcp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
return _tcpClient != null && _master != null && _tcpClient.Connected;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Udp:
|
|
|
|
|
case MasterType.RtuOnUdp:
|
|
|
|
|
case MasterType.AsciiOnUdp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
return _udpClient != null && _master != null && _udpClient.Client.Connected;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Rtu:
|
|
|
|
|
case MasterType.Ascii:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
return _serialPort != null && _master != null && _serialPort.IsOpen;
|
2021-12-18 11:13:41 +00:00
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 连接
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns></returns>
|
2021-12-12 06:55:48 +00:00
|
|
|
|
public bool Connect()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2022-08-08 07:15:09 +00:00
|
|
|
|
_logger.LogInformation($"Device:[{_device}],Connect()");
|
2022-08-10 08:52:41 +00:00
|
|
|
|
switch (MasterType)
|
2021-12-15 16:16:22 +00:00
|
|
|
|
{
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Tcp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_tcpClient = new TcpClient(IpAddress, Port);
|
|
|
|
|
_tcpClient.ReceiveTimeout = Timeout;
|
|
|
|
|
_tcpClient.SendTimeout = Timeout;
|
|
|
|
|
_master = ModbusIpMaster.CreateIp(_tcpClient);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Udp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_udpClient = new UdpClient(IpAddress, Port);
|
|
|
|
|
_udpClient.Client.ReceiveTimeout = Timeout;
|
|
|
|
|
_udpClient.Client.SendTimeout = Timeout;
|
|
|
|
|
_master = ModbusIpMaster.CreateIp(_udpClient);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Rtu:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_serialPort = new SerialPort(PortName, BaudRate, Parity, DataBits, StopBits);
|
|
|
|
|
_serialPort.ReadTimeout = Timeout;
|
|
|
|
|
_serialPort.WriteTimeout = Timeout;
|
|
|
|
|
_serialPort.Open();
|
|
|
|
|
_adapter = new SerialPortAdapter(_serialPort);
|
|
|
|
|
_master = ModbusSerialMaster.CreateRtu(_adapter);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.RtuOnTcp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_tcpClient = new TcpClient(IpAddress, Port);
|
|
|
|
|
_tcpClient.ReceiveTimeout = Timeout;
|
|
|
|
|
_tcpClient.SendTimeout = Timeout;
|
|
|
|
|
_master = ModbusSerialMaster.CreateRtu(_tcpClient);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.RtuOnUdp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_udpClient = new UdpClient(IpAddress, Port);
|
|
|
|
|
_udpClient.Client.ReceiveTimeout = Timeout;
|
|
|
|
|
_udpClient.Client.SendTimeout = Timeout;
|
|
|
|
|
_master = ModbusSerialMaster.CreateRtu(_udpClient);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.Ascii:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_serialPort = new SerialPort(PortName, BaudRate, Parity, DataBits, StopBits);
|
|
|
|
|
_serialPort.ReadTimeout = Timeout;
|
|
|
|
|
_serialPort.WriteTimeout = Timeout;
|
|
|
|
|
_serialPort.Open();
|
|
|
|
|
_adapter = new SerialPortAdapter(_serialPort);
|
|
|
|
|
_master = ModbusSerialMaster.CreateAscii(_adapter);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.AsciiOnTcp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_tcpClient = new TcpClient(IpAddress, Port);
|
|
|
|
|
_tcpClient.ReceiveTimeout = Timeout;
|
|
|
|
|
_tcpClient.SendTimeout = Timeout;
|
|
|
|
|
_master = ModbusSerialMaster.CreateAscii(_tcpClient);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
case MasterType.AsciiOnUdp:
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_udpClient = new UdpClient(IpAddress, Port);
|
|
|
|
|
_udpClient.Client.ReceiveTimeout = Timeout;
|
|
|
|
|
_udpClient.Client.SendTimeout = Timeout;
|
|
|
|
|
_master = ModbusSerialMaster.CreateAscii(_udpClient);
|
2021-12-15 16:16:22 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
_master!.Transport.ReadTimeout = Timeout;
|
|
|
|
|
_master!.Transport.WriteTimeout = Timeout;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2021-12-18 11:13:41 +00:00
|
|
|
|
catch (Exception ex)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],Connect(),Error");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return IsConnected;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 断开
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns></returns>
|
2021-12-12 06:55:48 +00:00
|
|
|
|
public bool Close()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_tcpClient?.Close();
|
|
|
|
|
_udpClient?.Close();
|
|
|
|
|
_serialPort?.Close();
|
2023-04-12 15:40:00 +00:00
|
|
|
|
_logger.LogInformation($"Device:[{_device}],Close()");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return !IsConnected;
|
|
|
|
|
}
|
2022-08-06 10:21:00 +00:00
|
|
|
|
catch (Exception ex)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],Close(),Error");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 释放
|
|
|
|
|
/// </summary>
|
2021-12-12 06:55:48 +00:00
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2022-08-06 10:21:00 +00:00
|
|
|
|
_tcpClient?.Dispose();
|
|
|
|
|
_udpClient?.Dispose();
|
|
|
|
|
_serialPort?.Dispose();
|
|
|
|
|
_master?.Dispose();
|
2023-04-12 15:40:00 +00:00
|
|
|
|
|
|
|
|
|
// Suppress finalization.
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
|
|
2022-08-08 07:15:09 +00:00
|
|
|
|
_logger.LogInformation($"Device:[{_device}],Dispose()");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-08-06 10:21:00 +00:00
|
|
|
|
catch (Exception ex)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],Dispose(),Error");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region 读写方法
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 批量读后缓存
|
|
|
|
|
/// </summary>
|
|
|
|
|
private Dictionary<string, object> _cache = new();
|
|
|
|
|
[Method("多地址读取", description: "多地址读取缓存")]
|
|
|
|
|
public DriverReturnValueModel ReadMultiple(DriverAddressIoArgModel ioArg)
|
|
|
|
|
{
|
|
|
|
|
byte slaveId = SlaveAddress;
|
|
|
|
|
if (ioArg.Address.Contains('|'))
|
|
|
|
|
{
|
|
|
|
|
slaveId = byte.Parse(ioArg.Address.Split('|')[0]);
|
|
|
|
|
ioArg.Address = ioArg.Address.Split('|')[1];
|
|
|
|
|
}
|
|
|
|
|
var args = ioArg.Address.Split(',');
|
|
|
|
|
var func = args[0];//功能码
|
|
|
|
|
var startAddress = ushort.Parse(args[1]);//开始地址
|
|
|
|
|
var length = ushort.Parse(args[2]);//读取字数
|
|
|
|
|
var cacheKey = args[3];//缓存字典名
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
switch (func)
|
|
|
|
|
{
|
|
|
|
|
case "1":
|
|
|
|
|
var coils = _master.ReadCoils(slaveId, startAddress, length);
|
|
|
|
|
_cache[cacheKey] = coils;
|
|
|
|
|
break;
|
|
|
|
|
case "2":
|
|
|
|
|
var inputs = _master.ReadInputs(slaveId, startAddress, length);
|
|
|
|
|
_cache[cacheKey] = inputs;
|
|
|
|
|
break;
|
|
|
|
|
case "3":
|
|
|
|
|
var holdingRs = _master.ReadHoldingRegisters(slaveId, startAddress, length);
|
|
|
|
|
_cache[cacheKey] = holdingRs;
|
|
|
|
|
break;
|
|
|
|
|
case "4":
|
|
|
|
|
var inputRs = _master.ReadInputRegisters(slaveId, startAddress, length);
|
|
|
|
|
_cache[cacheKey] = inputRs;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return new DriverReturnValueModel()
|
|
|
|
|
{
|
|
|
|
|
StatusType = VaribaleStatusTypeEnum.Good,
|
|
|
|
|
Value = Newtonsoft.Json.JsonConvert.SerializeObject(_cache[cacheKey])
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadMultiple(),Error");
|
|
|
|
|
_cache[cacheKey] = null;
|
|
|
|
|
return new DriverReturnValueModel()
|
|
|
|
|
{
|
|
|
|
|
StatusType = VaribaleStatusTypeEnum.Bad
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 从缓存内读
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
[Method("从缓存读取", description: "从缓存读取")]
|
|
|
|
|
public DriverReturnValueModel ReadFromCache(DriverAddressIoArgModel ioArg)
|
|
|
|
|
{
|
|
|
|
|
DriverReturnValueModel ret = new();
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
string cacheName = _cache.Any() ? _cache.FirstOrDefault().Key : "0";
|
|
|
|
|
int startIndex = 0;
|
|
|
|
|
if (ioArg.Address.Contains(","))
|
|
|
|
|
{
|
|
|
|
|
var args = ioArg.Address.Split(',');
|
|
|
|
|
cacheName = args[0];
|
|
|
|
|
startIndex = int.Parse(args[1]);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
startIndex = int.Parse(ioArg.Address);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (_cache.ContainsKey(cacheName) && _cache[cacheName] != null)
|
|
|
|
|
{
|
|
|
|
|
var cacheBuffers = (ushort[])_cache[cacheName];
|
|
|
|
|
var rawBuffers = cacheBuffers.Skip(startIndex).Take(cacheBuffers.Length - startIndex).ToArray();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var retBuffers = ChangeBuffersOrder(rawBuffers, ioArg.EndianType);
|
|
|
|
|
if (ioArg.ValueType == DataTypeEnum.AsciiString)
|
|
|
|
|
retBuffers = rawBuffers;
|
|
|
|
|
|
|
|
|
|
if (ioArg.ValueType.ToString().Contains("Uint16"))
|
|
|
|
|
ret.Value = retBuffers[0];
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Int16"))
|
|
|
|
|
ret.Value = (short)retBuffers[0];
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Bcd16"))
|
|
|
|
|
ret.Value = ModBusDataConvert.GetBCD(GetBytes(retBuffers));
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Uint32"))
|
|
|
|
|
ret.Value = (uint)(retBuffers[0] << 16) + retBuffers[1];
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Int32"))
|
|
|
|
|
ret.Value = (retBuffers[0] << 16) + retBuffers[1];
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Bcd32"))
|
|
|
|
|
{
|
|
|
|
|
var newBuffers = new ushort[2] { retBuffers[1], retBuffers[0] };
|
|
|
|
|
ret.Value = ModBusDataConvert.GetBCD(GetBytes(newBuffers));
|
|
|
|
|
}
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Float"))
|
|
|
|
|
{
|
|
|
|
|
var bytes = new[]
|
|
|
|
|
{
|
|
|
|
|
(byte)(retBuffers[1] & 0xff), (byte)((retBuffers[1] >> 8) & 0xff),
|
|
|
|
|
(byte)(retBuffers[0] & 0xff), (byte)((retBuffers[0] >> 8) & 0xff)
|
|
|
|
|
};
|
|
|
|
|
ret.Value = BitConverter.ToSingle(bytes, 0);
|
|
|
|
|
}
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("AsciiString"))
|
|
|
|
|
{
|
|
|
|
|
var str = Encoding.ASCII.GetString(GetBytes(retBuffers).ToArray());
|
|
|
|
|
if (str.Contains('\0'))
|
|
|
|
|
str = str.Split('\0')[0];
|
|
|
|
|
ret.Value = str;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Good;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadFromCache(),Error");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 功能码03
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
[Method("功能码:03", description: "HoldingRegisters读保持寄存器")]
|
|
|
|
|
public DriverReturnValueModel HoldingRegisters(DriverAddressIoArgModel ioArg)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
DriverReturnValueModel ret = new();
|
|
|
|
|
try
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
if (IsConnected)
|
2023-04-12 15:40:00 +00:00
|
|
|
|
ret = ReadRegistersBuffers(3, ioArg);
|
2022-04-15 08:49:58 +00:00
|
|
|
|
else
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
ret.Message = "TCP连接异常";
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-04-15 08:49:58 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.UnKnow;
|
|
|
|
|
ret.Message = ex.Message;
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadHoldingRegisters(),Error");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2022-04-15 08:49:58 +00:00
|
|
|
|
return ret;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 功能码04
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
[Method("功能码:04", description: "HoldingRegisters读输入寄存器")]
|
|
|
|
|
public DriverReturnValueModel InputRegisters(DriverAddressIoArgModel ioArg)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
DriverReturnValueModel ret = new();
|
|
|
|
|
try
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
if (IsConnected)
|
2023-04-12 15:40:00 +00:00
|
|
|
|
ret = ReadRegistersBuffers(4, ioArg);
|
2022-04-15 08:49:58 +00:00
|
|
|
|
else
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
ret.Message = "TCP连接异常";
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-04-15 08:49:58 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.UnKnow;
|
|
|
|
|
ret.Message = ex.Message;
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadInputRegisters(),Error");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2022-04-15 08:49:58 +00:00
|
|
|
|
return ret;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 功能码01
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
[Method("功能码:01", description: "Coil读线圈")]
|
|
|
|
|
public DriverReturnValueModel Coil(DriverAddressIoArgModel ioArg)
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
DriverReturnValueModel ret = new();
|
|
|
|
|
try
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
if (IsConnected)
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
var (slaveAddress, ioAddress) = GetSlaveAddress(ioArg);
|
|
|
|
|
var retBool = _master.ReadCoils(slaveAddress, ushort.Parse(ioAddress), 1)[0];
|
|
|
|
|
if (ioArg.ValueType == DataTypeEnum.Bit)
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
if (retBool)
|
|
|
|
|
ret.Value = 1;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
else
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.Value = 0;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.Value = retBool;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Good;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
2022-04-15 08:49:58 +00:00
|
|
|
|
else
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
ret.Message = "TCP连接异常";
|
2022-04-13 09:01:24 +00:00
|
|
|
|
}
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
2022-04-15 08:49:58 +00:00
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.UnKnow;
|
|
|
|
|
ret.Message = ex.Message;
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadCoil(),Error");
|
2022-04-15 08:49:58 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2022-04-15 08:49:58 +00:00
|
|
|
|
return ret;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 功能码02
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
[Method("功能码:02", description: "Input读输入")]
|
|
|
|
|
public DriverReturnValueModel Input(DriverAddressIoArgModel ioArg)
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
DriverReturnValueModel ret = new();
|
|
|
|
|
try
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
if (IsConnected)
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
var (slaveAddress, ioAddress) = GetSlaveAddress(ioArg);
|
|
|
|
|
var retBool = _master.ReadInputs(slaveAddress, ushort.Parse(ioAddress), 1)[0];
|
|
|
|
|
if (ioArg.ValueType == DataTypeEnum.Bit)
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
if (retBool)
|
|
|
|
|
ret.Value = 1;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
else
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.Value = 0;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.Value = retBool;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Good;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
2022-04-15 08:49:58 +00:00
|
|
|
|
else
|
2022-01-15 16:16:53 +00:00
|
|
|
|
{
|
2022-04-15 08:49:58 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
ret.Message = "TCP连接异常";
|
2022-04-13 09:01:24 +00:00
|
|
|
|
}
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
2022-04-15 08:49:58 +00:00
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.UnKnow;
|
|
|
|
|
ret.Message = ex.Message;
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadInput(),Error");
|
2022-04-15 08:49:58 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2022-04-15 08:49:58 +00:00
|
|
|
|
return ret;
|
2022-01-15 16:16:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-12-12 06:55:48 +00:00
|
|
|
|
[Method("Read方法样例", description: "Read方法样例描述")]
|
2023-04-12 15:40:00 +00:00
|
|
|
|
public DriverReturnValueModel Read(DriverAddressIoArgModel ioArg)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
|
|
|
|
DriverReturnValueModel ret = new DriverReturnValueModel
|
|
|
|
|
{
|
|
|
|
|
Message = "",
|
|
|
|
|
StatusType = VaribaleStatusTypeEnum.Good,
|
2023-04-12 15:40:00 +00:00
|
|
|
|
Value = $"{DeviceId} {DateTime.Now.ToString("O")} Read {ioArg.Address}"
|
2021-12-12 06:55:48 +00:00
|
|
|
|
};
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 写入
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="requestId"></param>
|
|
|
|
|
/// <param name="method"></param>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
public async Task<RpcResponse> WriteAsync(string requestId, string method, DriverAddressIoArgModel ioArg)
|
|
|
|
|
{
|
|
|
|
|
RpcResponse rpcResponse = new() { IsSuccess = false };
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
if (!IsConnected)
|
|
|
|
|
rpcResponse.Description = "设备连接已断开";
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
ushort address, count;
|
|
|
|
|
AnalyzeAddress(ioArg, out address, out count);
|
|
|
|
|
var (slaveAddress, ioAddress) = GetSlaveAddress(ioArg);
|
|
|
|
|
|
|
|
|
|
//功能码01
|
|
|
|
|
if (method == nameof(Coil))
|
|
|
|
|
{
|
|
|
|
|
var value = ioArg.Value.ToString() == "1" || ioArg.Value.ToString().ToLower() == "true";
|
|
|
|
|
await _master.WriteSingleCoilAsync(slaveAddress, address, value);
|
|
|
|
|
rpcResponse.IsSuccess = true;
|
|
|
|
|
return rpcResponse;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//功能码03
|
|
|
|
|
if (method == nameof(HoldingRegisters))
|
|
|
|
|
{
|
|
|
|
|
ushort[] shortArray = new ushort[count];
|
|
|
|
|
|
|
|
|
|
switch (ioArg.ValueType)
|
|
|
|
|
{
|
|
|
|
|
case DataTypeEnum.AsciiString:
|
|
|
|
|
ModBusDataConvert.SetString(shortArray, 0, ioArg.Value.ToString());
|
|
|
|
|
await _master.WriteMultipleRegistersAsync(slaveAddress, address, shortArray);
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case DataTypeEnum.Float:
|
|
|
|
|
float f = 0;
|
|
|
|
|
float.TryParse(ioArg.Value.ToString(), out f);
|
|
|
|
|
ModBusDataConvert.SetReal(shortArray, 0, f);
|
|
|
|
|
await _master.WriteMultipleRegistersAsync(slaveAddress, address, shortArray);
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
await _master.WriteSingleRegisterAsync(slaveAddress, address,
|
|
|
|
|
ushort.Parse(ioArg.Value.ToString()));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rpcResponse.IsSuccess = true;
|
|
|
|
|
return rpcResponse;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rpcResponse.Description = $"不支持写入:{method}";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
rpcResponse.Description = $"写入失败,[method]:{method},[ioArg]:{ioArg},[ex]:{ex}";
|
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],WriteAsync(),Error");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return rpcResponse;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region 私有方法
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 读功能码03、或04
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="funCode"></param>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
private DriverReturnValueModel ReadRegistersBuffers(byte funCode, DriverAddressIoArgModel ioArg)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
|
|
|
|
DriverReturnValueModel ret = new() { StatusType = VaribaleStatusTypeEnum.Good };
|
2021-12-18 11:13:41 +00:00
|
|
|
|
if (!IsConnected)
|
2021-12-12 06:55:48 +00:00
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
else
|
|
|
|
|
{
|
2022-07-17 10:03:15 +00:00
|
|
|
|
ushort startAddress, count;
|
2023-04-12 15:40:00 +00:00
|
|
|
|
ret = AnalyzeAddress(ioArg, out startAddress, out count);
|
|
|
|
|
var (slaveAddress, ioAddress) = GetSlaveAddress(ioArg);
|
|
|
|
|
|
2022-08-06 10:21:00 +00:00
|
|
|
|
if (ret.StatusType != VaribaleStatusTypeEnum.Good)
|
2022-07-17 10:03:15 +00:00
|
|
|
|
return ret;
|
|
|
|
|
try
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-07-17 10:03:15 +00:00
|
|
|
|
var rawBuffers = new ushort[] { };
|
2022-08-10 08:52:41 +00:00
|
|
|
|
if (funCode == 3)
|
2023-04-12 15:40:00 +00:00
|
|
|
|
rawBuffers = _master.ReadHoldingRegisters(slaveAddress, startAddress, count);
|
2022-08-10 08:52:41 +00:00
|
|
|
|
else if (funCode == 4)
|
2023-04-12 15:40:00 +00:00
|
|
|
|
rawBuffers = _master.ReadInputRegisters(slaveAddress, startAddress, count);
|
2022-07-17 10:03:15 +00:00
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
var retBuffers = ChangeBuffersOrder(rawBuffers, ioArg.EndianType);
|
|
|
|
|
if (ioArg.ValueType == DataTypeEnum.AsciiString)
|
2022-07-17 10:03:15 +00:00
|
|
|
|
retBuffers = rawBuffers;
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
if (ioArg.ValueType.ToString().Contains("Uint16"))
|
2022-07-17 10:03:15 +00:00
|
|
|
|
ret.Value = retBuffers[0];
|
2023-04-12 15:40:00 +00:00
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Int16"))
|
2022-07-17 10:03:15 +00:00
|
|
|
|
ret.Value = (short)retBuffers[0];
|
2023-04-12 15:40:00 +00:00
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Bcd16"))
|
|
|
|
|
ret.Value = ModBusDataConvert.GetBCD(GetBytes(retBuffers));
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Uint32"))
|
2022-08-06 10:21:00 +00:00
|
|
|
|
ret.Value = (uint)(retBuffers[0] << 16) + retBuffers[1];
|
2023-04-12 15:40:00 +00:00
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Int32"))
|
2022-08-06 10:21:00 +00:00
|
|
|
|
ret.Value = (retBuffers[0] << 16) + retBuffers[1];
|
2023-04-12 15:40:00 +00:00
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Bcd32"))
|
|
|
|
|
{
|
|
|
|
|
var newBuffers = new ushort[2] { retBuffers[1], retBuffers[0] };
|
|
|
|
|
ret.Value = ModBusDataConvert.GetBCD(GetBytes(newBuffers));
|
|
|
|
|
}
|
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("Float"))
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-08-10 08:52:41 +00:00
|
|
|
|
var bytes = new[]
|
|
|
|
|
{
|
|
|
|
|
(byte)(retBuffers[1] & 0xff), (byte)((retBuffers[1] >> 8) & 0xff),
|
|
|
|
|
(byte)(retBuffers[0] & 0xff), (byte)((retBuffers[0] >> 8) & 0xff)
|
|
|
|
|
};
|
2022-07-17 10:03:15 +00:00
|
|
|
|
ret.Value = BitConverter.ToSingle(bytes, 0);
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2023-04-12 15:40:00 +00:00
|
|
|
|
else if (ioArg.ValueType.ToString().Contains("AsciiString"))
|
2021-12-12 06:55:48 +00:00
|
|
|
|
{
|
2022-08-06 10:21:00 +00:00
|
|
|
|
var str = Encoding.ASCII.GetString(GetBytes(retBuffers).ToArray());
|
2022-07-17 10:03:15 +00:00
|
|
|
|
if (str.Contains('\0'))
|
|
|
|
|
str = str.Split('\0')[0];
|
|
|
|
|
ret.Value = str;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-07-17 10:03:15 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.Bad;
|
|
|
|
|
ret.Message = ex.Message;
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],ReadRegistersBuffers(),Error");
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-07-17 10:03:15 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
///
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="functionCode"></param>
|
|
|
|
|
/// <param name="dataType"></param>
|
|
|
|
|
/// <returns></returns>
|
2021-12-12 06:55:48 +00:00
|
|
|
|
private ushort GetModbusReadCount(uint functionCode, DataTypeEnum dataType)
|
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
if (dataType.ToString().Contains("32") || dataType.ToString().Contains("Float"))
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return 2;
|
2023-04-12 15:40:00 +00:00
|
|
|
|
if (dataType.ToString().Contains("64") || dataType.ToString().Contains("Double"))
|
2021-12-12 06:55:48 +00:00
|
|
|
|
return 4;
|
2022-08-06 10:21:00 +00:00
|
|
|
|
return 1;
|
2023-04-12 15:40:00 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// 大小端转换
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="buffers"></param>
|
|
|
|
|
/// <param name="dataType"></param>
|
|
|
|
|
/// <returns></returns>
|
2022-12-06 08:01:32 +00:00
|
|
|
|
private ushort[] ChangeBuffersOrder(ushort[] buffers, EndianEnum dataType)
|
2023-04-12 15:40:00 +00:00
|
|
|
|
{
|
|
|
|
|
int datalen = buffers.Length;
|
|
|
|
|
ushort[] newBuffers = new ushort[datalen];
|
|
|
|
|
if (dataType == EndianEnum.None)
|
|
|
|
|
{
|
|
|
|
|
newBuffers = buffers;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
if (datalen == 1)//16位
|
|
|
|
|
{
|
|
|
|
|
switch (dataType)
|
|
|
|
|
{
|
|
|
|
|
case EndianEnum.LittleEndian://BA
|
|
|
|
|
var ab = BitConverter.GetBytes(buffers[0]);
|
|
|
|
|
newBuffers[0] = BitConverter.ToUInt16(new byte[] { ab[1], ab[0] });
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.BigEndian://AB
|
|
|
|
|
default:
|
|
|
|
|
newBuffers[0] = buffers[0];
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (datalen == 2)//32位
|
|
|
|
|
{
|
|
|
|
|
newBuffers = new ushort[2];
|
|
|
|
|
var ab = BitConverter.GetBytes(buffers[0]);
|
|
|
|
|
var cd = BitConverter.GetBytes(buffers[1]);
|
|
|
|
|
var _ab = new byte[2];
|
|
|
|
|
var _cd = new byte[2];
|
|
|
|
|
switch (dataType)
|
|
|
|
|
{
|
|
|
|
|
case EndianEnum.BigEndian://ABCD
|
|
|
|
|
_ab = ab;
|
|
|
|
|
_cd = cd;
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.LittleEndian://DCBA
|
|
|
|
|
_ab[0] = cd[1];
|
|
|
|
|
_ab[1] = cd[0];
|
|
|
|
|
_cd[0] = ab[1];
|
|
|
|
|
_cd[1] = ab[0];
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.BigEndianSwap://BADC
|
|
|
|
|
_ab[0] = ab[1];
|
|
|
|
|
_ab[1] = ab[0];
|
|
|
|
|
_cd[0] = cd[1];
|
|
|
|
|
_cd[1] = cd[0];
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.LittleEndianSwap://CDAB
|
|
|
|
|
_ab[0] = cd[0];
|
|
|
|
|
_ab[1] = cd[1];
|
|
|
|
|
_cd[0] = ab[0];
|
|
|
|
|
_cd[1] = ab[1];
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
newBuffers[0] = BitConverter.ToUInt16(_ab, 0);
|
|
|
|
|
newBuffers[1] = BitConverter.ToUInt16(_cd, 0);
|
|
|
|
|
}
|
|
|
|
|
else if (datalen == 4)//64位
|
|
|
|
|
{
|
|
|
|
|
newBuffers = new ushort[2];
|
|
|
|
|
var ab = BitConverter.GetBytes(buffers[0]);
|
|
|
|
|
var cd = BitConverter.GetBytes(buffers[1]);
|
|
|
|
|
var ef = BitConverter.GetBytes(buffers[2]);
|
|
|
|
|
var gh = BitConverter.GetBytes(buffers[3]);
|
|
|
|
|
var _ab = new byte[2];
|
|
|
|
|
var _cd = new byte[2];
|
|
|
|
|
var _ef = new byte[2];
|
|
|
|
|
var _gh = new byte[2];
|
|
|
|
|
switch (dataType)
|
|
|
|
|
{
|
|
|
|
|
case EndianEnum.BigEndian://AB CD EF GH
|
|
|
|
|
_ab = ab;
|
|
|
|
|
_cd = cd;
|
|
|
|
|
_ef = ef;
|
|
|
|
|
_gh = gh;
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.LittleEndian://HG FE DC BA
|
|
|
|
|
_ab[0] = gh[1];
|
|
|
|
|
_ab[1] = gh[0];
|
|
|
|
|
_cd[0] = ef[1];
|
|
|
|
|
_cd[1] = ef[0];
|
|
|
|
|
|
|
|
|
|
_ef[0] = cd[1];
|
|
|
|
|
_ef[1] = cd[0];
|
|
|
|
|
_gh[0] = ab[1];
|
|
|
|
|
_gh[1] = ab[0];
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.BigEndianSwap://BA DC FE HG
|
|
|
|
|
_ab[0] = ab[1];
|
|
|
|
|
_ab[1] = ab[0];
|
|
|
|
|
_cd[0] = cd[1];
|
|
|
|
|
_cd[1] = cd[0];
|
|
|
|
|
|
|
|
|
|
_ef[0] = ef[1];
|
|
|
|
|
_ef[1] = ef[0];
|
|
|
|
|
_gh[0] = gh[1];
|
|
|
|
|
_gh[1] = gh[0];
|
|
|
|
|
break;
|
|
|
|
|
case EndianEnum.LittleEndianSwap://GH EF CD AB
|
|
|
|
|
_ab[0] = gh[0];
|
|
|
|
|
_ab[1] = gh[1];
|
|
|
|
|
_cd[0] = ef[0];
|
|
|
|
|
_cd[1] = ef[1];
|
|
|
|
|
|
|
|
|
|
_ef[0] = cd[0];
|
|
|
|
|
_ef[1] = cd[1];
|
|
|
|
|
_gh[0] = ab[0];
|
|
|
|
|
_gh[1] = ab[1];
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
newBuffers[0] = BitConverter.ToUInt16(_ab, 0);
|
|
|
|
|
newBuffers[1] = BitConverter.ToUInt16(_cd, 0);
|
|
|
|
|
newBuffers[2] = BitConverter.ToUInt16(_ef, 0);
|
|
|
|
|
newBuffers[3] = BitConverter.ToUInt16(_gh, 0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
newBuffers = buffers;
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return newBuffers;
|
|
|
|
|
}
|
2022-04-13 09:01:24 +00:00
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// ushort数组转byte数组
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="retBuffers"></param>
|
|
|
|
|
/// <returns></returns>
|
2022-07-17 10:03:15 +00:00
|
|
|
|
private List<byte> GetBytes(ushort[] retBuffers)
|
|
|
|
|
{
|
|
|
|
|
List<byte> vs = new();
|
2022-08-06 10:21:00 +00:00
|
|
|
|
foreach (var retBuffer in retBuffers)
|
2022-07-17 10:03:15 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
vs.Add((byte)(retBuffer & 0xFF));
|
|
|
|
|
vs.Add((byte)((retBuffer & 0xFF00) >> 8));
|
2022-07-17 10:03:15 +00:00
|
|
|
|
}
|
2023-04-12 15:40:00 +00:00
|
|
|
|
|
2022-07-17 10:03:15 +00:00
|
|
|
|
return vs;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 解析出特定站号
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <param name="startAddress"></param>
|
|
|
|
|
/// <param name="readCount"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
private DriverReturnValueModel AnalyzeAddress(DriverAddressIoArgModel ioArg, out ushort startAddress,
|
2022-08-10 08:52:41 +00:00
|
|
|
|
out ushort readCount)
|
2022-07-17 10:03:15 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
var (slaveAddress, ioAddress) = GetSlaveAddress(ioArg);
|
2022-07-17 10:03:15 +00:00
|
|
|
|
DriverReturnValueModel ret = new() { StatusType = VaribaleStatusTypeEnum.Good };
|
|
|
|
|
try
|
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
if (ioArg.ValueType == DataTypeEnum.AsciiString)
|
2022-07-17 10:03:15 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
startAddress = ushort.Parse(ioAddress.Split(',')[0]);
|
|
|
|
|
readCount = ushort.Parse(ioAddress.Split(',')[1]);
|
2022-07-17 10:03:15 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
startAddress = ushort.Parse(ioAddress);
|
|
|
|
|
readCount = GetModbusReadCount(3, ioArg.ValueType);
|
2022-07-17 10:03:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-10 08:52:41 +00:00
|
|
|
|
return ret;
|
2022-07-17 10:03:15 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
ret.StatusType = VaribaleStatusTypeEnum.AddressError;
|
|
|
|
|
ret.Message = ex.Message;
|
2022-08-10 08:52:41 +00:00
|
|
|
|
startAddress = 0;
|
|
|
|
|
readCount = 0;
|
2022-10-14 05:29:54 +00:00
|
|
|
|
_logger.LogError(ex, $"Device:[{_device}],AnalyzeAddress(),Error");
|
2022-07-17 10:03:15 +00:00
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 支持灵活读取不同SlaveId的数据
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ioArg"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
private (byte slaveAddress, string ioAddress) GetSlaveAddress(DriverAddressIoArgModel ioArg)
|
2022-04-13 09:01:24 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
byte slaveAddress = SlaveAddress;
|
|
|
|
|
string ioAddress = ioArg.Address;
|
2022-04-15 08:49:58 +00:00
|
|
|
|
try
|
2022-04-13 09:01:24 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
if (ioArg.Address.Contains('|'))
|
2022-04-13 09:01:24 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
slaveAddress = byte.Parse(ioArg.Address.Split('|')[0]);
|
|
|
|
|
ioAddress = ioArg.Address.Split('|')[1];
|
2022-04-13 09:01:24 +00:00
|
|
|
|
}
|
2023-04-12 15:40:00 +00:00
|
|
|
|
|
2022-04-13 09:01:24 +00:00
|
|
|
|
}
|
2023-04-12 15:40:00 +00:00
|
|
|
|
catch (Exception e)
|
2022-04-15 08:49:58 +00:00
|
|
|
|
{
|
2023-04-12 15:40:00 +00:00
|
|
|
|
_logger.LogError(e, $"Device:[{_device}],AnalyseAddress(),Error");
|
2022-04-15 08:49:58 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
|
2023-04-12 15:40:00 +00:00
|
|
|
|
return (slaveAddress, ioAddress);
|
2022-04-13 09:01:24 +00:00
|
|
|
|
}
|
2023-04-12 15:40:00 +00:00
|
|
|
|
#endregion
|
|
|
|
|
|
2021-12-12 06:55:48 +00:00
|
|
|
|
}
|
2022-08-10 08:52:41 +00:00
|
|
|
|
}
|