02-03-2014 10:31 AM
No. That is a limit set by the definition of the Modbus protocol. If you need more registers send more than one request.
Regards, Jens
02-03-2014 10:52 AM
Hey Varun,
jg69 is correct, that is part of the protocol and is a limitation of the slave in this limitation. That is, the master may freely request any number of registers and it is the decision of the slave whether or not to allow that request. The slave implementation provided follows the standard. You can override this behavior by implementing a different data model. I believe I put an example somewhere earlier in this thread.
Thanks,
Daniel
02-03-2014 10:57 AM
Jozo wrote:
Hey smithd
im using Modbus Master Example.vi for communication with Modbus slave simulator between 2 serial ports interconnected through 2 USB2RS485 converters on my PC (because i dont have PLC yet).
Communication is working with following problem:
I cannot get rid of this error (Error 56 occurred at RTU Data Unit.lvclass:Read ADU Packet.vi:5030001)
Do you have any suggestions?
Thanks,
Joz
Hey Joz, what version of the code are you using?
02-03-2014 02:31 PM
I've attached a new build of the code. It should resolve issues with serial RTU related to corrupted packets resulting in error 56. It also includes a more flexible timing value for RTU.
The serial code now works in such a way that even if it guesses wrong about packet size (based on RTU 3.5 char silence times), it will re-use the old data. That is, lets say the driver tells us we've had 64 bytes on the bus for more than 3.5 char times--this means we should attempt to process the packet. If CRC fails, we put that data in a buffer and listen again. This time we have 34 bytes on the bus. Once we actually perform the read, we'll have two arrays--data1[64+34] and data2[34]. If neither data1 nor data2 is valid and timeout has not elapsed, we listen some more...lets say we get data3[21], and this is the actual end of the packet. The processor looks at data1[64+34+21], sees that this is actually a complete packet (CRC=0) and passes this to the rest of the system. In this way we can still ignore noise or invalid packets and we have a bit more leeway on how well the serial driver handles the background transfer.
xtabi2
While fixing the above issues I think we should be in a state where you don't need the ability to change how long the code waits, but I added it in anyway. New behavior is to take the normal 3.5 char times, multiply by a factor (defaults to 1) and use that. While buffer size=0, it will poll at a rate of 6 or more ms, depending on baud rate. (For example 3.5 char times at 9600 baud is 4 ms, so the code polls at 6 ms until it sees any data on the line, and then will wait at least an additional 4 ms for silence on the line). You can set this factor. However, I compromised and did not expose this on the main API constructor functions. While I was writing the code I realized this was RTU-only, while the serial master/slave functions can work with both ASCII and RTU. Rather than including something that most users won't need to touch, I decided to simply put it inside of the constructor. Then, for people like yourself who need this feature, you can make a copy of the constructor for your own use and expose the properties you need. I've included an example for the master side, which defaults to dividing the time in 2.
joz
Please try this version of the code. Let me know if it fixes your issues.
02-04-2014 02:37 AM
Hey Smithd
Thank you very much... great job!
Im using new version 1.1.3.32 and everything is working perfect for me now (no Error 56).
I hope that it will also work with real PLC.
Thanks Joz
02-04-2014 11:15 AM
I'm glad to hear that worked. I thought of a way to optimize the code slightly with no ill effects, so I'm going to post that shortly. It should be 1.1.4. However if you don't have any performance issues (ie you are not using an older target) you don't need to worry about it. Good luck with your PLC
02-04-2014 11:35 AM
Attached is a minor update which should behave very slightly more efficiently.
02-05-2014 11:22 AM
Hey All,
I've attached a special debug build of the code which might come in handy. It uses a protocol called syslog (VIPM will install it for you) to send debug messages across the network in UDP packets. I have it configured to send a significant amount of data, but if you are getting failures it may help. To enable, you need to do two things. First, set the conditional disable symbol "syslog" to "TRUE". Then you'll need to add an init and shutdown to your code. You may wish to add the same conditional disable here. When it says "RT" code below, I really mean whatever you're instrumenting.
Then, in order to recieve messages you'll need a collector. You can either use third-party collectors, or make your own. As a starting point, I attached a very basic one.
Thanks,
Daniel
02-06-2014 02:07 PM
Hi Smithd,
I have an USB 4-port RS485 converter and I am programming some prototype LabVIEW Modbus code. What I'd like to do is to use one of the RS485 ports as Modbus master and the other three as the slaves. I am trying to run master and the three slaves in the same PC. I wonder how to set up Modbus master, especially the slave RS485 address, with the LabVIEW Modbus API.
Thanks
Jason
02-07-2014 10:45 AM
Hey Jason,
So, each serial port should show up as a separate visa resource. As such it should be easy to set up a modbus serial slave (just change the Create Modbus Instance function from TCP to serial) associated with each visa address. Each slave should have a different unit ID, starting at 1 (0:=broadcast). The same thing for the single master. In the case of the master, you'll basically do the following:
1) Set unit ID to the ID of slave 1
2) Send needed requests
3) Set unit ID to the ID of slave 2
4) Send needed requests
etc...
The way the code is designed, every request is synchronous and you are expected to never assign more than one modbus instance to a given network interface, whether than is a TCP port or serial port. As a result, you use unit IDs to differentiate between different devices on a network.
Thanks,
Daniel