AdvancedHMI Software

General Category => Feature Request => Topic started by: Godra on August 09, 2015, 06:57:12 PM

Title: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 09, 2015, 06:57:12 PM
Related to this topic: http://advancedhmi.com/forum/index.php?topic=899.0

Would it be possible to convert either of these 2 drivers to be able to access COM based PLCs over ethernet?

The SerialToIP server provides transparent access to a PLC connected to COM port so it would have to be the driver to properly access the PLC and issue correct commands.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 09, 2015, 07:09:36 PM
You would essentially have to get the protocol specifications for the Ethernet to Serial converter and write another driver layer.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 09, 2015, 07:41:34 PM
Archie,

If I try to do this on my own that could take quite some time.

Since you know your drivers "by heart" could you let me know which part of EthernetIPforPLCSLCMicroCom driver is actually establishing connection before sending a packet?
Is it possible to have that part included in DF1Com driver so it could route its packets to ethernet port instead of COM port?

Kind of trying to combine and make 1 driver out of 2 existing.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 09, 2015, 08:31:55 PM
At the lowest level in the DLL file is a EIPTransport class that ultimately handles the transfer of the packets to the Ethernet port.

What would probably need to be done is a new DF1DataLinkLayer class be made that would accept the packets, wrap them in the protocol for the particular converter, then send the via Ethernet.

The Allen Bradley 1761-NET-ENI module will do all of this. It will let you use the Ethernet drivers to talk to a serial based PLC.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: abslcplc on August 09, 2015, 09:09:21 PM
I have a Moxa N-Port Ethernet serial port on my network, and on 2 desktops & a laptop it is configured as a COM port  & I can use the regular DF1 driver to access it from either of the desktops without any modification of the driver.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 09, 2015, 09:28:08 PM
The change to the driver is not required at all.

As shown in this topic: https://www.advancedhmi.com/forum/index.php?topic=899.0 the driver works fine together with com0com and SerialToIP programs to provide access to a serial device over ethernet just by using 2 computers (most users might have an old computer sitting around that they could use for this instead of purchasing physical serial to ethernet device).

The change to the driver would only help those who would be looking to avoid hassle of installing/configuring com0com and running SerialToIP software on client computers. They would still need to run SerialToIP server program on the server computer.

Additional feature can only provide additional options to users.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 09, 2015, 09:48:45 PM
I may not be completely understanding everything here. I read the other post in more detail and this is what I am interpreting. You have 2 PCs. One is connected to the PLC via serial port (or with a USB-to-RS232 adapter). The other PC is used to communicate to the PLC via Ethernet.

The 2nd PC sends data to a virtual COM port that is then transmitted to the 1st PC over Ethernet. The 1st PC receives the packet and relays it through the COM port to the PLC.

What you want to do is to bypass the software on the 2nd PC and send the packets directly to the server on the 1st PC that is connected to the PLC?
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 09, 2015, 10:00:25 PM
That would be correct.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 09, 2015, 10:11:48 PM
The PC connected to PLC actually behaves as Allen Bradley 1761-NET-ENI module, which connects the serial device to ethernet network.

If DF1Com driver could send its packets directly then there would be no need for either com0com or SerialToIP software on client computer.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 09, 2015, 10:25:02 PM
This is what I think would have to be done. In DF1Com.vb, this defines where the packets go:

    Private Shared DLL As System.Collections.Concurrent.ConcurrentDictionary(Of Integer, MfgControl.AdvancedHMI.Drivers.DF1DataLinkLayerR1)

There is a SendData method as part of the DF1DataLinkLayerR1 class. Essentially a new class would have to be written that implements the SendData to transmit the data to the UDP port that is listening on the server side.

Part of the DF1DataLinkLayerR1 class is an event of Df1DataLinkLayerDataReceived.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Vitor on August 10, 2015, 03:40:58 AM
Godra
Ethernet/IP and DF1 are diferent protocols.
The sugested 1761-NET-ENI is a piece of skilled hardware+software that makes the translation between the two, like the 1747-UIC makes the same between DF1 and DH485.
I think its possible to do what you want at software level, but also i think in the mighty effort it will be...
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 10, 2015, 11:35:44 AM
Vitor, you might be right.

Currently, SerialToIP software is handling the transport of DF1 packets from COM port to ethernet and it works just fine so I will stick with it.

Arrchie does seem to understand the idea I suggested but for me to realize it goes currently beyond my knowledge.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 10, 2015, 08:11:42 PM
As I thought more about this, I wonder if the solution is to make a server component that can receive PCCC Ethernet/IP messages and relay them to a serial port. Essentially this would make a PC act like an ENI module.

The PCCC packet is essentially the same whether wrapped in an Ethernet/IP packet or sent over serial. The only difference is that it has 2 additional bytes for the source node and target nodes (always 0 for point to point), the checksum, and STX/ETX bytes.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 11, 2015, 01:33:23 AM
I don't completely understand your point. Would it be a server component within AHMI?

The SerialToIP software can act as a server and is already programmed and free to download (along with the source code).

This server seems to accept any packets that come to its port and just sends them to PLC. If packets were in correct format then PLC will respond. You can see the attached picture of 2 drivers that I tried and their routes as well as response of the server to both of these. The Route 1 is working fine but the Route 2 has 28 Bytes reach the server but get no response from PLC.

If DF1 driver could wrap its packets and send them directly over ethernet that would probably be the best solution. I did try just opening a server socket and/or using TCPClient from within the DF1Com driver and sending its packets <-- the packets did reach the server but I din't know how to go about getting a response (and still not sure if packets were in correct format, all I know is that 7 bytes were received by the server).
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 13, 2015, 12:46:25 PM
Quote
The PCCC packet is essentially the same whether wrapped in an Ethernet/IP packet or sent over serial. The only difference is that it has 2 additional bytes for the source node and target nodes (always 0 for point to point), the checksum, and STX/ETX bytes.

Archie,

Could you provide some more information about these additional bytes?

From your driver code, I can see that you are adding source node and target node values to "datawithnodes" but what variables are holding the checksum, STX/ETX bytes. Would TNS and MyObjectID need to be included in the packet as well?

Code: [Select]
    Friend Overrides Sub SendPacket(packet As MfgControl.AdvancedHMI.Drivers.PCCCCommandPacket, ByVal TNS As Integer)
        Dim TNSByte As Byte = Convert.ToByte(TNS And 255)
        RawRequests(TNSByte) = packet
        RawRequests(TNSByte).Responded = False

        Dim data() As Byte = packet.GetBytes
        Dim dataWithNodes(data.Length + 1) As Byte
        data.CopyTo(dataWithNodes, 2)
        dataWithNodes(0) = m_TargetNode
        dataWithNodes(1) = m_MyNode

        DLL(MyDLLInstance).SendData(dataWithNodes, TNS, MyObjectID)
    End Sub

If the following packet is captured from serial port:

HEX 10 02 01 00 06 00 AC 24 03 10 03 E0 00

Would a packet sent directly from driver to server socket have to be the same format?
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 13, 2015, 12:56:20 PM
The STX, ETX, and Checksum are all part the Data Link Layer (technically part of the transport layer).

There is a shared function that calculates the checksum:

MfgControl.AdvancedHMI.Drivers.Checksum.CalculateCRC16ForDF1

Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 13, 2015, 01:02:43 PM
I did find checksum in the following property:

Code: [Select]
    Private m_CheckSumType As MfgControl.AdvancedHMI.Drivers.DF1Transport.ChecksumOptions
    Public Property CheckSumType() As MfgControl.AdvancedHMI.Drivers.DF1Transport.ChecksumOptions
        Get
            Return m_CheckSumType
        End Get
        Set(ByVal value As MfgControl.AdvancedHMI.Drivers.DF1Transport.ChecksumOptions)
            m_CheckSumType = value
            If MyDLLInstance > 0 AndAlso DLL.Count > 0 AndAlso DLL(MyDLLInstance) IsNot Nothing Then   'AndAlso Not DLL(MyDLLInstance).IsPortOpen Then
                DLL(MyDLLInstance).ChecksumType = m_CheckSumType
            End If
        End Set
    End Property

From the packet in previous post, would the last 00 be extra byte?
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Archie on August 13, 2015, 01:25:00 PM
10 02      STX
01           Destination Node
00           Source Node
06           Command
00           STS  (error code)
AC 24      TNS (Transaction Number)
03            Function
10 03       ETX
E0 00       CRC16 checksum (2 bytes). Should only be a single byte is BCC is used
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 14, 2015, 06:13:54 PM
Archie,

With info from this last reply of yours, I did manage to modify the packet, shoot it out the socket and actually get the PLC to respond to it with several different packets (check the attached picture). I did get those packets into the receive buffer but was not sure of how to send them to AB classes for processing and returning the information to AHMI application.

This would indicate to me that this way of communication is definitly possible for DF1 driver.

All my testing was nothing but using a "brute force" instead of creating a proper code that would route packets to a destination other than COM port. I will leave that part to you, if you ever decide to include it in the driver.
Title: Re: Modify DF1Com or EthernetIPforPLCSLCMicroCom drivers to access COM over ethernet
Post by: Godra on August 21, 2015, 12:32:45 AM
For those who might be interested, here is what I did:

1) Add Import statements at the top of the file:

Code: [Select]
    Imports System.Net
    Imports System.Net.Sockets
    Imports System.Windows.Forms

2) Create 2 new properties to hold the IP address and the port of the remote computer

Code: [Select]
    Private m_IPAddress As String = "192.168.0.197"
    <System.ComponentModel.Category("Communication Settings")> _
    Public Property RemoteIPAddress() As String
        Get
            Return m_IPAddress
        End Get
        Set(ByVal value As String)
            If m_IPAddress <> value Then
                '* Stop the subscriptions and allow the last request to complete
                Dim SubscriptionStatus As Boolean = Me.DisableSubscriptions
                Me.DisableSubscriptions = True
                System.Threading.Thread.Sleep(500)

                m_IPAddress = value

                '* Restart the subscriptions
                Me.DisableSubscriptions = SubscriptionStatus

            End If
        End Set
    End Property

    Private m_remotePort As UShort = &H205A 'Port 8282
    Public Property RemotePort As Integer
        Get
            Return m_remotePort
        End Get
        Set(ByVal value As Integer)
            If m_remotePort <> value Then
                '* Limit the value to 0-65535
                m_remotePort = CUShort(Math.Max(0, Math.Min(value, 65535)))
            End If
        End Set
    End Property

3) Add some variables before and some new code inside the SendPacket sub of the DF1 driver, just after the packet was sent to serial port:

Code: [Select]
    Private socket As New Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp)
    Private remoteEP As New IPEndPoint(IPAddress.Parse(m_IPAddress), m_remotePort)
    Private buff As Byte() = New Byte(8191) {}
    Friend Overrides Sub SendPacket(ByVal packet As MfgControl.AdvancedHMI.Drivers.PCCCCommandPacket, ByVal TNS As Integer)
        Dim TNSByte As Byte = Convert.ToByte(TNS And 255)
        RawRequests(TNSByte) = packet
        RawRequests(TNSByte).Responded = False

        Dim data() As Byte = packet.GetBytes
        Dim dataWithNodes(data.Length + 1) As Byte
        data.CopyTo(dataWithNodes, 2)
        dataWithNodes(0) = m_TargetNode
        dataWithNodes(1) = m_MyNode

        DLL(MyDLLInstance).SendData(dataWithNodes, TNS, MyObjectID)

        If m_IPAddress <> Nothing AndAlso m_remotePort <> Nothing Then
            Dim CRC As UShort = MfgControl.AdvancedHMI.Drivers.Checksum.CalculateCRC16ForDF1(dataWithNodes, dataWithNodes.Length)
            Dim CRCbytes As Byte() = BitConverter.GetBytes(CRC)
            Dim dataWithNodesPacket(dataWithNodes.Length + 5) As Byte
            dataWithNodes.CopyTo(dataWithNodesPacket, 2)
            dataWithNodesPacket(0) = &H10 '
            dataWithNodesPacket(1) = &H2  ' Not sure if these 4 entries were specific to my setup
            dataWithNodesPacket(9) = &H10 '
            dataWithNodesPacket(10) = &H3 '
            dataWithNodesPacket(11) = CRCbytes(0)
            dataWithNodesPacket(12) = CRCbytes(1)

            ' *** Using socket to send/receive packets ***
            Try
                If Not socket.Connected Then socket.Connect(remoteEP)
                socket.Send(dataWithNodesPacket)
                If socket.Available > 0 Then
                    Dim bytesAvailable As Integer = Math.Min(socket.Available, 8192)
                    Dim rawData() As Byte = New Byte(bytesAvailable - 1) {}
                    socket.Receive(buff, bytesAvailable, SocketFlags.None)
                    For i As Integer = 0 To bytesAvailable - 1
                        rawData(i) = buff(i)
                    Next
                    Dim rawDataReduced(rawData.Length - 7) As Byte
                    Array.ConstrainedCopy(rawData, 6, rawDataReduced, 0, rawDataReduced.Length)
                    Dim e1 As New MfgControl.AdvancedHMI.Drivers.Common.PlcComEventArgs(rawDataReduced, "B3:0/0", TNS, MyObjectID)
                    DataLinkLayer_DataReceived(Me, e1)
                End If
            Catch ex As Exception
                MessageBox.Show(ex.ToString)
            End Try
        End If
    End Sub

The part of sending the received data to DF1DataLinkLayerDataReceived didn't work properly. I did try sending the data as it was and then stripped of 2 bytes and then stripped of 4 bytes <-- none of these would work.