Here is a very simple example of how we can use websockets in a client browser to interact with the JNIOR. This example will demonstrate how to register a few event callbacks, connect, and display the messages that are received from the JNIOR. To force a monitor message to get sent we added an inversion checkbox for Digital Input 1. When the Inversion state changes the input acts as though an electrical signal has forced the state change.
The WebSocket connection now behaves like any other bi-directional TCP Socket. All of the functionality of the JNIOR protocol has been implemented in the WebSocket protocol. The binary implementation has been replaced by JSON. This not only makes it easier to code and debug but it is more extensible.
Below we have a zip file containing a demo page. The page shows the incoming JSON messages in the lower part of the page. The upper part of the page contains a button that will invert input 1 in software thus simulating the input receiving an electrical state change.
The javascript for our demo page is listed below
// or comm object. The communication will be established to the device that serves this page. _comm = new Comm(); // called when the connection is successfully opened _comm.onopen = function () { logToBody('comm opened'); }; // called when the connection closed _comm.onclose = function () { _loggedIn = false; logToBody('comm closed'); }; _comm.onerror = function (error) { logToBody('comm error: ' + error); }; // called when there is an error _comm.onmessage = function (evt) { var messageJson = JSON.parse(evt.data); logToBody('comm message: ' + JSON.stringify(messageJson)); }; // called when the page has been loaded window.onload = function () { logToBody('init comm'); _comm.connect(); } // a function to create a div with a timestamped message to the messages section of the page. // The div will be prepended so that is is at the top of the section. var logToBody = function (message) { var el = document.createElement('div'); el.innerHTML = new Date().toLocaleString() + ': ' + message; el.style = 'white-space: nowrap'; var messagesElement = document.getElementById('messages'); messages.insertBefore(el, messages.firstChild); }; // called when the user clicks the inversion checkbox function invert(checkbox, channelNumber) { var registryWriteJson = { "Message": "Registry Write", "Keys": {} }; registryWriteJson.Keys["IO/Inputs/din" + channelNumber + "/Inversion"] = checkbox.checked.toString(); _comm.sendJson(registryWriteJson); }
wsdemo.zip [ Sep 19 2018, 5.80 KB, MD5: b3efe72c03f4b642c269311057613977 ]
To run the demo simply place the downloaded .zip
file in the flash/www
folder on your JNIOR. Then open your browser and navigate to http://[IP_ADDRESS]/wsdemo
.
After the connection is open, any change to I/O will send a new Monitor Packet. Clicking the “Invert Input 1” checkbox will cause a change to the I/O and send the Monitor Packet.
Using the zip file is taking advantage of another great feature of the JNIORs web server. A whole web application can be served out of a single zip file. This makes distribution and file management a lot easier.
Depending on the application, something going wrong and stopping it from running could lead to big complications. Thankfully, a watchdog can monitor your application and react how you want it to using the Watchdog class. The watchdog can stop, restart, and report on applications not responding to it and more. Watchdogs have configurable timeframes, and when they aren’t updated before the time frame expires they trigger.
The watchdog is a very powerful concept. Wikipedia says:
A watchdog timer (sometimes called a computer operating properly or COP timer, or simply a watchdog) is an electronic timer that is used to detect and recover from computer malfunctions. During normal operation, the computer regularly resets the watchdog timer to prevent it from elapsing, or “timing out”. If, due to a hardware fault or program error, the computer fails to reset the watchdog, the timer will elapse and generate a timeout signal. The timeout signal is used to initiate corrective action or actions. The corrective actions typically include placing the computer system in a safe state and restoring normal system operation.
Usage
The example below is very basic. You will need to adjust the duration and location of the watchdog for your application.
View on GitHub
This is what we see in the log after the unit has come back up.
01/15/19 08:37:36.361, FTP/10.0.0.27:62888 uploaded /flash/WatchdogExample.jar [115.3 kbps] 01/15/19 08:37:44.081, WatchdogExample started 01/15/19 08:37:44.203, WatchdogExample activated 01/15/19 08:38:14.307, WatchdogExample loop finished 01/15/19 08:38:18.451, ** Assertion: WatchdogExample watchdog triggered reboot (Line 1278) 01/15/19 08:38:18.483, ** Terminating: System 01/15/19 08:38:20.971, ** Reboot on assertion: WatchdogExample watchdog triggered reboot (Line 1278) 01/15/19 08:38:20.997, -- JANOS 410 v1.7.1 initialized (POR: 1546)
Here are the options that are available to choose from when the watchdog expires. You see that we assign the action in the setAction() method above. watchdog.setAction(Watchdog.WDT_REBOOT);
WDT_REBOOT
This is the default action. Expiration of the watchdog timer causes the unit to reboot.
WDT_APPLICATION
This action indicates that a watchdog timer expiration is to be handled by the application itself. The system takes no action.
WDT_TERMINATE
When the watchdog timer expires the system will terminate the application.
WDT_BREAK
This interrupts the current application (similar to Ctrl-C interruption) when the watchdog timer expires.
WDT_RESTART
When the watchdog timer expires this will interrupt the current application and restart it.
WDT_MESSAGE
When the watchdog timer expires a WM_WATCHDOG (0x11)
message will be sent through the system message pump. The content contains the text associated with the watchdog.
WDT_EVENT
Watchdog timer expiration will notify the first thread in the application that is waiting for notification using waitOnWatchdogNotify()
.
- Added ability to load single CA Certificate to be supplied on TLS connection
- Support TLS Client Certificate Verification on outgoing connections
- Added legacy PKCS1 support for externally generated certificates
- Corrected array issue with processing of deferred email transmissions
- Corrected memory issue with TAB use on the Command Line
This example will monitor the IO Log utilizing the IoLog
class. There are many ways to watch IO for changes. You could simply poll the IO and see if it changes. You are likely to set up a polling routing on some interval. If that interval is not fast enough then you have the potential of missing an change. Using the IO Log ensures that you see that a transition occurred. You have the added benefit of getting the exact time of that transition to the millisecond. This helps if you are measuring the time between transitions.
package com.integpg;
import com.integpg.system.IoEvent;
import com.integpg.system.Iolog;
import java.util.Date;
public class IOLogSampleMain implements Runnable {
private final Iolog _iolog = new Iolog();
private Thread _thd;
public static void main(String[] args) throws InterruptedException {
new IOLogSampleMain().start();
Thread.sleep(Integer.MAX_VALUE);
}
public void start() {
if (null == _thd) {
_thd = new Thread((Runnable) this);
_thd.setName("IoLogMonitor");
_thd.start();
}
}
@Override
public void run() {
long lastConsumedTimestamp = System.currentTimeMillis();
while (true) {
try {
// refresh the iolog object so that we only get new events since the last time we queried it
_iolog.refresh(lastConsumedTimestamp);
// get the events for the inputs only
IoEvent[] ioEvents = _iolog.getInputEvents();
// make sure there were returned events
if (ioEvents.length > 0) {
System.out.println("IoLogMonitor returned " + ioEvents.length + " events");
// go through each event and print some meaningful information about it
for (int index = ioEvents.length - 1; index >= 0; index--) {
IoEvent ioEvent = ioEvents[index];
System.out.println("ioEvent[" + index + "]: time: " + ioEvent.timestamp + " changed input mask: " + Long.toHexString(ioEvent.mask) + " input states mask: " + Long.toHexString(ioEvent.states));
// update our lastConsumedTimestamp with the date of the most recent event
if (ioEvent.timestamp > lastConsumedTimestamp) {
lastConsumedTimestamp = ioEvent.timestamp;
System.out.println("lastConsumedTimestamp: " + new Date(lastConsumedTimestamp));
}
}
}
Thread.sleep(100);
} catch (Exception ex) {
ex.printStackTrace();
}
}
}
}
This sample shows you how to read a barcode scanner using the AUX port. You must have the correct settings to communicate successfully with your device. The settings for the port may be set using the setSerialPortParams()
method. This method can only be called after the port has been successfully opened.
package barcodescanner; import com.integpg.comm.AUXSerialPort; import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; public class BarcodeScanner { private static int BAUD = 19200; public static void main(String[] args) { AUXSerialPort auxPort = null; try { // init the aux port auxPort = new AUXSerialPort(); // open the port System.out.println("Open the AUX Serial Port"); auxPort.open(); // set the parameters System.out.println("Set port parameters to " + BAUD + ", 8, none, 1"); auxPort.setSerialPortParams(BAUD, 8, 1, 0); } catch (Exception ex) { ex.printStackTrace(); // cant open the serial port so exit System.exit(-1); } try { // create our reader InputStream is = auxPort.getInputStream(); InputStreamReader isr = new InputStreamReader(is); BufferedReader br = new BufferedReader(isr); // read the barcodes System.out.println("Ready to Scan Barcodes..."); String line = null; // read the serial port in a loop to get all of the barcodes. each barcode is // terminated by a linefeed. That means we can use readLine() while ((line = br.readLine()) != null) { System.out.println("Barcode: " + line); } } catch (Exception ex) { ex.printStackTrace(); } } }
This sample requires that the MODBUS Server is running.
This sample is meant to show you how to gain access to the MODBUS scratch memory space. You can see in the MODBUS manual that the MODBUS addressing assigns a “scratch” address space starting at WORD 256.
Since WORDs are 2 bytes we know that the starting address is going to be 512. We can gain access to the “scratch” area by opening the Modbus00
immutable block. The Modbus00
immutable block does not exist until the MODBUS server has been started on the unit at least once in its lifetime. You can use the nv
command to discover whether the Modbus00
block exists or not.
If the Modbus00
block does not exist then execute the MODBUS Server. Once the MODBUS Server has been executed the Modbus00
will exist.
The immutable example shown in the Immutable Class example uses an array of long values. We will use a byte array in this example.
package modbussample;
import com.integpg.system.ArrayUtils;
import com.integpg.system.Immutable;
public class ModbusSample {
public static void main(String[] args) throws InterruptedException {
// open the Modbus00 immutable block
System.out.println("Open the MODBUS Store");
byte[] modbusStore = Immutable.getByteArray("Modbus00");
// if the block does not exist then alert the user and exit. it is not our responsibility to create it
if (modbusStore == null) {
System.out.println("MODBUS store has not been created. Please run the MODBUS Server application.");
System.exit(0);
}
// the modbus store must exist. loop so that external changes by a client are seen by our application
while (true) {
// get and print the first 4 WORDs in our scratch area
short[] value = new short[]{
ArrayUtils.getShort(modbusStore, 512),
ArrayUtils.getShort(modbusStore, 514),
ArrayUtils.getShort(modbusStore, 516),
ArrayUtils.getShort(modbusStore, 518)
};
System.out.println("WORDs @0256: " + value[0] + ", " + value[1] + ", " + value[2] + ", " + value[3]);
// get and print the first 8 bytes that represent the first 4 WORDs in our scratch area
byte[] b = new byte[8];
ArrayUtils.arraycopy(modbusStore, 512, b, 0, b.length);
System.out.println("bytes @0512: " + hexDump(b, 0, b.length));
// sleep a little before checking again
Thread.sleep(1000);
}
}
public static String hexDump(byte[] bytes, int offset, int length) {
StringBuffer sb = new StringBuffer();
StringBuffer chars = new StringBuffer();
int i;
for (i = offset; i
< length; i++) { if (i % 16 == 0) { if (chars.length() > 0) {
sb.append(" ");
sb.append(chars.toString());
chars
= new StringBuffer();
sb.append("\r\n");
}
} else if (i % 16 == 8) {
chars.append(" ");
sb.append(" ");
}
if (bytes[i] >= 32) {
chars.append((char) bytes[i]);
} else {
chars.append('.');
}
int p = sb.length();
sb.append(Integer.toHexString((bytes[i] & 0xff) / 16));
sb.append(Integer.toHexString((bytes[i] & 0xff) % 16));
sb.append(" ");
}
int mod = i % 16;
if (mod != 0) {
if (mod <= 8) { sb.append(" "); } for (i = 16 - (mod); i > 0; i--) {
sb.append(" ");
}
sb.append(" ");
sb.append(chars.toString());
}
return sb.toString();
}
}
Here is some sample output from our application. We used a PC MODBUS client to adjust the values of the registers in the MODBUS scratch address space. You can see the values changing during the execution of our application.
This sample shows you how to use the Immutable class. The immutable class represents persistent storage that can be accessed at the low level as an array of primitive types. This information will withstand reboots and application terminations. You can view Immutable data by typing the ‘nv’ command in the command line of a JNIOR. If using only one data type, you can express to create an immutable array of that data type, but if you want your array to have mixed data types inside it, you have to use a byte array.
Using Immutable Blocks
This example shows the use of an array of longs. The long values in this application are date values that represent when the application was started. The long array is part of an Immutable block, therefore the application start times don’t go away until overwritten or manual removed. This application holds up to the last five application start times.
View on GitHub
I put the built jar file of this example application into the JNIOR’s flash folder and ran it from the Web UI’s console tab. After it has successfully run, I run the application multiple times, and it shows each time it runs a new application start time added to its immutable array.
Sometimes it is vital to ensure that only one copy of an application is running. To do that JANOS has provided us the ability to register a process with the operating system. Using the registerProcess(uid) call we can get the number of processes running with the given unique identifier. This can become an issue if you have multiple run keys set to start the same application.
Here is the demo application. The one instance application is started with the & parameter. This tells the console session to run the application as a separate process. The First time the application run it returns a process count of 1. The process is allowed to proceed.
Since the application is running as a separate process from the console session we can start another instance now. The returned count is now 2. The code checks for a count greater than 1 and decides to exit.
Using this feature will help you be sure that there is only ever one instance of your application running at a time. Here is the source for the sample application.
package oneinstance; import com.integpg.system.JANOS; public class Oneinstance { public static void main(String[] args) { // we assign a unique identifier to this application. This can be ANY string. String uid = "abcdef"; // we register the process with JANOS. this call will return the number of processes // running with this uid. The result should be 1 indicating that our process is now running. int processCount = JANOS.registerProcess(uid); System.out.println("# of processes running using this UID: " + processCount); if (processCount > 1) { System.out.println("There is another copy of this application already running. exit now."); return; } // now sleep for a while to give us time to start other instances of this application as a test try { System.out.println("We are allowed to continue as the only instance of this applicaiton"); Thread.sleep(60000); } catch (InterruptedException ex) { ex.printStackTrace(); } } }
The JNIOR is a very flexible and powerful controller. Utilize our bundled or add-on software applications. If those don’t meet your needs, let INTEG quickly develop an application for you.
The JNIOR offers superb functionality with its included and available software. However, if you require a custom application to run on your JNIOR, INTEG can develop it for you, or you can develop it yourself using the JNIOR Software Development Kit (SDK).INTEG has already developed a number of custom applications for a variety of customers. Some of these applications have become our ‘add-on’ applications because they have met the needs of a large group of customers.
Other times the applications have been focused and developed to meet the needs of a specific customer. After the user requirements are gathered it doesn’t take long for INTEG to deliver something for the customer to test. Often times we dont get devices sent to the office that the customer wants to interface with. This makes it tough for INTEG to complete full testing in the office. Sometimes we write test applications to mimic the communications between the JNIOR and the end device.
If you have an application that you have in mind and want to talk to INTEG about the JNIOR please call the office, 724-933-9350, or email support@integpg.com. You can also fill out the contact form.
Thank you for your interest in the JNIOR from INTEG.
The JNIOR Model 410 looks very much like your familiar model 310 but just how similar is it? This insiders’ guide takes you on a tour of the JANOS Operating System highlighting what’s new and what’s different. It’s a technical peek into the next generation of JNIOR.
This Guide is for those who are very familiar with the JNIOR Model 310, 312 or 314. While the new series of JNIOR provides all of the capabilities that you have come to enjoy there are many (sometimes subtle) improvements. It is here that we will outline those for you. Our goal is to remove any mystery, to make you comfortable with the new generation, and to help you efficiently assimilate these devices into your installations.
1 Appearance
First of all the Model 410 looks different but not all that different from the Model 310. It is important that you be able to continue to use the JNIOR 410 in every application where previously you would the 310. For this reason the product housing has not changed; There has been no change to the connections; And, the power supply requirements remain the same.
At the same time it is important that you be able to quickly identify a 410 limiting any possible confusion. To this aim the color scheme used in the labeling has changed. The 410 provides a sleek dark look with its predominantly black front label. In addition the GREEN power LED is no longer green. The model 410 uses a BLUE LED to indicate power which can be easily identified from a distance. Other than this color difference the LEDs serve an identical purpose.
2 Connections
All of the connections to the Model 410 are 100% compatible with the Model 310. This is so you are guaranteed to be able to remove a 310 from its application and directly replace it with the new 410. There is one very subtle change which you would likely be hard-pressed to identify unless we point it out. The tab on the Sensor Port jack is now located upwards. We flipped this connection over because with the JNIOR mounted to the wall and depending on the hardware used to mount it getting your finger under the RJ-l2 plug to press the tab was sometimes difficult. This is certainly one of the least significant differences in the product if not THE least significant but we did promise to outline ALL of the differences.
3 Under the Hood
With the Model 310 there was no real need to remove the cover to access the circuit board inside. For the most part this is true for the 410 as well. Here there are two differences under the hood that should be mentioned.
3.1 Battery
All JNIORs use a 3V non-rechargeable lithium battery cell. This battery is used to maintain the content of the non-flash memory holding that portion of the file system outside of the /flash
folder. The battery has a life expectancy of several years. This is especially true when the JNIOR remains in use and powered during most of that period. With the Model 310 this battery is NOT user replaceable.
The Model 410 uses a standard 3V CR2032 coin cell which is readily available wherever batteries are sold. While we have not experienced any significant issue with the life of the batteries used in the 310 they are quite expensive and do not provide the advantages of a removable cell. If your battery has failed you would notice a lack of any entries in the jniorsys.log
file prior to the removal of power. The loss of file data may or may not impact your application. With the 410 at least you can easily replace the battery by simply sliding the old one out of the holder and slipping a new one in. The CR2032 coin cell actually has more capacity than the permanent batteries that we have used in the 31X models.
3.2 Configurable Relays
As you know our relay outputs are normally open (N.O.) dry contact. Those familiar with the Model 312, which has 12 relay outputs and 4 isolated digital inputs, may know that there are two relays which can be configured by internal jumper to alternatively function as normally closed (N.C.). We have carried this feature into the Model 410 where Relay Output 1 and Relay Output 2 can be configured to be either N.O. or N.C. simply by moving the associated jumper. All units come factory configured for N.O. operation.
4 Performance
By far the majority of differences in the new Model 410, those beyond what have been mentioned already, become apparent when power is applied. The most significant of which is product performance. The Model 410 is considerably faster and much more responsive than its predecessor.
This key difference becomes immediately obvious when power is applied. You are likely aware that the YELLOW/ORANGE Status LED to the right of the Power LED on JNIORs remains on a during product boot. This is true also for the Model 410 however try not to be distracted by the BLUE Power LED as you may miss the boot indication entirely. The Model 410 operating system (OS) literally boots in a second. The status LED correspondingly merely flashes.
After a second any configured and enabled applications have been lunched and have begun to initialize. The network connection becomes active after about 5 seconds as the hardware completes the normal auto-negotiation with the interconnected hub. This means that after only 5 seconds the Model 410 JNIOR is ready to accept browser connections, process FTP transfers or otherwise handle all network traffic. Even the most complicated applications are generally up and running in no more than 15 seconds.
4.1 Processing Core
The Model 410’s leap in performance comes from a significant upgrade in processor. The internal system clock, which governs the pace of machine instructions, is 3 times faster than the Model 310. Furthermore the new processor uses leading-edge technology to combine a variable length instruction set with an advanced instruction pipeline to achieve a one instruction per clock cycle execution rate. On average the Model 310 requires about 3 clock cycles per instruction. On this basis alone we would expect a 10X speed improvement with the Model 410.
The new processor is 32-bit where the Model 310’s processor was only 8-bit. Numbers are generally represented by programming languages as 4-byte (32-bit) integers. The simple act of adding two integers in an 8-bit system requires dozens of machine instructions whereas with a 32-bit processor it takes just one. In our case this difference is perhaps on average a factor of 20. This would imply that the Model 410 might be as much as 200 times faster than the 310.
In addition, on the Model 410, the operating system code is run out of flash memory internal to the processor which has a 10 nano-second access time. This is extremely fast for flash especially when compared to the external flash used by the Model 310 where access requires 70 nanoseconds. In order for the prior 310 to avoid having to wait for memory, programs are copied to RAM and executed there. The impact on performance in having to copy code and the increased load on memory management cannot be directly quantified.
As a further advantage the new processor has built-in functionality to perform multiplications, divisions and floating point calculations. The benefit of all of this is also difficult to quantify. The bottom line is that on the basis of hardware alone the Model 410 should be some 250 times faster than the 310 if not much more.
4.2 JANOS Operating System
As with any product the operating system brings life to the hardware. This is complex programming that is tightly coupled with the hardware design. In order for INTEG to move the JNIOR line to a new processor platform a new operating system had to be developed. For this new operating system to retain functionality closely matching (and indeed indistinguishable from) the existing 31X line we had to create it completely in-house. Therefore with the new Model 410 we introduce the JNIOR Automation Network Operating System of “JANOS” for short.
In ancient Roman religion and myth, Janus is the god of beginnings and transitions, and also of gates, doors, passages, endings and time. With this in mind the acronym JANOS is appropriate for an I/O system controlling inputs and outputs.
Unlike its predecessor the JniorOS (built upon the Dallas/Maxim TINI OS) which was written in Java, JANOS is written in C language directly generating efficient machine instructions which are optimized for the new processor core. Where previously the Java code interpreted sequences of bytecodes pacing OS performance, the new operating system performs dramatically faster and significantly more efficiently executing directly in machine language.
As a result the new Model 410 out-performs the Model 310 by orders of magnitude opening up possibilities for new and exciting applications.
5 File Storage
Beyond the improvement in processing speed the Model 410 also has the capability to store more file data. By default the area preserved for the /flash
folder is 32MB which is over 40 times that provided by the Model 310. This providing ample space to store a fully featured website inclusive of graphics or for the storage of data covering long periods of time in data logging applications. The internal flash memory may even be expanded by special order to as much as 128MB and perhaps beyond.
6 JANOS Command Line
As with its predecessor the Model 410’s command line is accessed through a serial connection to the RS-232 port or via the network through a Telnet connection or application support such a connection. The prompt should be very familiar and all of the commands that you have used to manage your JNIORs are there. There are some differences which will be high-lighted in the next section.
6.1 Editing
The entry of commands has been improved. In addition to the UP/DOWN ARROW command history that JNIOR provides you may now use a number of other editing keys. The RIGHT/LEFT ARROW keys together with the HOME, END, BACKSPACE, DEL and INS keys can be used to flexibly edit and enter commands. This is welcome relief and a definite improvement over the previously limited editing available in the Model 31X series.
6.2 Auto-fill (TAB)
The TAB key is used to auto-fill file names. While entering any command you may type one or more characters representing the beginning of a file or folder name and then use the TAB keystroke to to toggle through potential files and folders matching that criteria. This is very useful to avoid having to enter lengthy file names in their entirety. The TAB key can even be used to construct paths to files located deep in the file system. For instance the sequencecat f[TAB]/j[TAB][RETURN]
will likely execute the following as if entered directly.cat flash/jnior.ini[RETURN]
The TAB key also enhances the registry
command discussed in the next section.
6.3 Prompt Abbreviation (DEL)
The prompt itself displays the current Hostname together with the current working folder (directory). If you have used the cd
command to move about folders in the file system and depending on the length of the Hostname defined, your prompts can be quite lengthy. You might be starting to enter commands 1/3rd of the way across a line and depending on the application used for access may have to contend with command line wrap.
If this becomes an issue, you may now use the DEL key immediately at a new prompt to squelch the display of the Hostname. If you do so, all subsequent command line prompts will omit the Hostname. Once another character is type after the prompt the DEL key will perform its editing function as expected.
6.4 Command History
As with its predecessor the Model 410 provides a command history wherein the UP/DOWN ARROW keys may be used to access and repeat a previously entered command. This is very useful.
The 410 however remembers up to 8 unique command entries sorted so that the UP ARROW supplies the most recently entered commands first. By doubling the number of remembered lines, eliminating duplicates, and sorting by age the command history becomes a more effective time saver. Add this to other editing and auto-fill functionality and you find that the command line is much easier to use.
6.5 Custom Command Creation
The java
command is used to execute application programs. As will be covered later these are stored in .JAR files. The Model 410 allows you to execute a program using simply its name. So these two command lines equivalently execute the target application. Note also that file names are case-independent.java MyApp.jar -debug
myapp -debug
This in effect allows you to create a custom command.
7 Commands
All of the commands available with the Model 310 are also provided in the new Model 410. In many cases the output may be formatted slightly differently, some additional information might be provided, or there may be additional functionality. We will review each command here focusing on significant differences.
7.1 The arp
Command (New!)
7.2 The bye
Command (New!)
The bye
command terminates the current Command session. It is equivalent to the exit
command.
7.3 The cat
Command
Ctrl-C
can be used to interrupt the listing of a lengthy file. If you cat
a lengthy file and decide that you do not need to see the whole thing, hit the Control-C key combination to stop the listing.
The cat
command can be used with the -h
option to dump the content of a file in hexadecimal. This allows you to view the binary content of a file. For example:
JANOS_Rev04 /> cat -h jniorboot.log 00000000 30 38 2f 31 32 2f 31 33 20 31 39 3a 33 36 3a 31 08/12/13 .19:36:1 00000010 31 2e 35 39 38 2c 20 4d 6f 64 65 6c 20 34 31 30 1.598,.M odel.410 00000020 20 2d 20 4a 41 4e 4f 53 20 76 30 2e 38 2e 35 2d .-.JANOS .v0.8.5- 00000030 72 63 34 2e 31 0d 0a 30 38 2f 31 32 2f 31 33 20 rc4.1..0 8/12/13. 00000040 31 39 3a 33 36 3a 31 31 2e 35 39 39 2c 20 43 6f 19:36:11 .599,.Co 00000050 70 79 72 69 67 68 74 20 28 63 29 20 32 30 31 32 pyright. (c).2012 00000060 2d 32 30 31 33 20 49 4e 54 45 47 20 50 72 6f 63 -2013.IN TEG.Proc . . .
This is useful in debugging applications that may store information in a binary form.
7.4 The date
Command
The date
command has 3 new options. The -s
option disables the use of Daylight Savings Time for the current Timezone; Correspondingly the -d
option enables Daylight Savings Time; And, the -v
(verbose) option provides additional detail regarding the current time. For example.
JANOS_Rev04 /> date -v utc: 1376337540 Mon Aug 12 15:59:00 EDT 2013 Current Timezone is EST for the America/New_York area. Abbrieviated EDT when Daylist Savings is in effect. Daylight Savings Time begins at 02:00 on the Sun on or after Mar 8th. Daylight Savings Time ends at 02:00 on the Sun on or after Nov 1st. When in effect Daylight Savings Time sets clocks ahead by 1 hour. Daylight Savings Time is currently in effect. JANOS_Rev04 />
7.5 The extern
Command
The extern
command manages external devices. In JANOS it remembers and displays the addressing and type for each module used with the unit.
JANOS_Rev04 /> extern TypeFB_1 = CD111090708109FB present TypeFB_2 = BE111120220410FB not present TypeFB_3 = 79111130517082FB not present TypeFA_1 = C7100511100083FA not present TypeFE_1 = 23111130619007FE not present TypeFD_1 = 4B111110510241FD not present JANOS_Rev04 />
Note that the first two TypeFB (4ROUT) devices (TypeFB_1 and TypeFB_2) extend the Relay Output functionality of the unit. On the 410 the first represents Relay Outputs 9 through 12 and the second 13 through 16. JANOS like its predecessor works to maintain the proper association between the Relay Output and the physical module. This is done through this addressing.
Unlike its predecessor, the Model 410 does not automatically forget modules should they be removed and the unit rebooted. This greatly reduces the risk that the order of 4ROUTs and their associated Relay Outputs be confused and improperly assigned. If you do need to reset this addressing, use the -r
option as follows:extern -r
This will ‘remove’ any modules that are no longer present. The 4ROUTs will be reassigned in the order that they are detected.
The Model 410 also scans for new modules and verifies existing modules every 5 seconds. A reboot is no longer required (or use of the extern
command) to detect newly connected devices.
Procedure for Assigning 4ROUT Modules
1) Remove all modules.
2) Issue the extern -r
command.
3) Connect the 4ROUT module to be associated with Relay Output channels 6 through 12.
4) Wait 5 – 10 seconds (or use extern
command until you see that the module is assigned).
5) Connect the 4ROUT for channels 13 through 16.
The 4ROUT and Power 4ROUT modules are TypeFB and are interchangeable as far as channel assignments are concerned. All other modules are addressed directly by their address in all protocols.
7.6 The iolog
Command (New!)
7.7 The jar
Command (New!)
7.8 The jrupdate
Command (New!)
7.9 The manifest
Command (New!)
7.10 The mode
Command (New!)
7.11 The nv
Command (New!)
7.12 The reg
Command (New!)
The reg
command is an alias (abbreviation) for the registry
command.
7.13 The registry
Command
Listing Registry Content
The registry
command can now be used with wildcards to list matching Registry entries. Wildcards adhere to the DOS standards using ‘?’ and ‘*’ wilds. For example:
JANOS_Rev04 /> reg Ip* IpConfig/Hostname = JANOS_Rev04 IpConfig/DHCP = disabled IpConfig/IPAddress = 10.0.0.71 IpConfig/SubnetMask = 255.255.255.0 IpConfig/GatewayIP = 10.0.0.1 IpConfig/PrimaryDNS = 10.0.0.4 JANOS_Rev04 />
Auto-fill Registry Keys
The TAB key can be used to auto-fill Registry Key names. For example:reg Ip[TAB]/[TAB][TAB][TAB] =
This results in the following command line wherein you may complete it by adding the IP address.reg IpConfig/IPAddress =
Note the use of repeated TAB keys to toggle through the various subkeys to end up with the one you want. The keys appear in alphabetical order and if you continue through all available the original command line re-displays. You may then proceed through the list again. This only works for existing defined keys. To define a new key you will need to type it out.
Recall Current Value
You may use the TAB key immediately following the ‘=’ to recall the current value of an existing Registry key. For example:reg IpConfig/IpAddress =[TAB]
This results in the following line which may then be edited if desired.reg IpConfig/IPAddress = 10.0.0.71
Specifying Files
The TAB key functions as in any other command line when used following the the ‘=’ sign but not immediately after. For example:reg Run/TaskManager = f[TAB]/T[TAB][RETURN]
This results in the following entry and can be used to easily define the key to start TaskManager.reg Run/TaskManager = flash/TaskManager.jar
Combined with the newly available LEFT/RIGHT ARROW, BACKSPACE, DEL and INS editing this can greatly improve the command line experience.
7.14 The touch
Command (New!)
7.15 The usermod
Command (New!)
7.16 The users
Command (New!)
8 User Management
9 Registry
10 Web Server
The JANOS Web Server introduces 2 new enhancements in addition to improvements in performance. The JANOS Web Server supports built-in websocket functionality as well as a form of server-side scripting consistent with the Hypertext PreProcessor (PHP).
10.1 Websockets
JANOS allows a web connection to be promoted to the websocket protocol through the HTML port (default 80). In this case JSON formatted messages can be exchanged over a single persistent connection with the client browser providing AJAX type services in support of dynamic web pages. This offers an alternative to the older Java Applets and implements the approach which has become the norm for many websites.
10.2 Server-side Scripting
The JANOS Web Server implements a small subset of the well-known scripting language known as PHP. Documented separately this server-side scripting function complements websocket and dynamic HTML by providing the ability to generate context specific web content on demand.
11 JAVA Applications Programming
12 I/O Logging
13 External Modules
14 Firmware Updates
A key advantage with the new Model 410 is the ability to update 100% of the operating firmware. With its predecessor, the Model 31X series, a percentage of the operating firmware is supplied by a third party in binary form. It was not possible to field update that portion of the JNIOR Model 310 code. We in fact had not changed that portion of the operating system through the entire life of the Model 31X product. This has forced us to work-around some (permanent) deficiencies that had been discovered over the years.
JANOS on the other hand was completely developed by INTEG and this includes every single byte of operating code. Correspondingly we are able to update the system in its entirety. We are able to service an issue that may arise and better yet, we are able to extend the function of the operating system in any way conceivable.
Each Model 410 may be updated manually or through programs like the JNIOR Support Tool. The firmware update is supplied in a .UPD file which must first be copied into the JNIOR file system using FTP. This file is typically between 600KB and 700KB and so it is recommended that it be transferred to the /flash
folder where the 410 provides ample space. Unlike its predecessor the Model 410 does not automatically detect the .UPD file on boot nor does it automatically remove the file after updating. The jrupdate
command is used to install the update. For example:jrupdate -u flash/filename.upd
This initiates the firmware update. Note that a reboot
will be required to complete the OS replacement.
14.1 Java System Library
The Java library is stored in the /etc/JanosClasses.jar
file. This is the system built-in library residing in the Read-Only /etc
folder. This contains all of the base classes required to build Java applications to run on the JNIOR Model 410. A .UPD file my optionally carry new content for the /etc
folder. Note that the folder is replaced immediately upon execution of the jrupdate
command in the form shown above.
Since Java applications cache referenced classes, the library .JAR can be swapped while Java applications are running. An application may throw an unexpected exception if it should attempt to load a new class during the update. If this is a concern you might want to stop any running application before performing the update.
14.2 OS Update and Rollback
The Model 31X series maintained two JniorOS images, the field update and the original factory installed OS. These, of course, are images of that part of the OS that can be field updated. It is possible with the 310 to rollback to the factory installed OS. We have found that doing so is rarely desirable given that the factory installation can quickly become outdated. It is likely that JNIOR applications would fail to run under the original OS. The rollback in this case has not been recommended and practically never used.
The Model 410 on the other hand also supports two JANOS images one of which is the currently executing operating system. The other image is a copy of the previously installed version. The rollback then becomes seriously useful in that it will restore any previous version of OS which can be assumed to have been recently operational unlike a potentially aged original factory version. The jrupdate -r
command performs the rollback.
In actuality the jrupdate -r
command schedules a swap of the OS images on reboot. In this case on reboot the “Saved OS” becomes the executing OS and the previously running OS is “saved”. One can use the jrupdate -r
command to toggle between an updated version of JANOS and a prior version. The stats
command displays the current and saved versions of the operating system.
The jrupdate -u
command as described above merely overwrites the “Saved OS” image with the supplied update and schedules the OS swap on the next reboot. The jrupdate -c
command can be used to cancel any scheduled swap.
The Model 410 then performs any scheduled OS swap on boot bringing up the desired copy of JANOS in its entirety. While we recommend that the reboot be caused using the reboot
command it is not strictly required with the new JNIOR. The Model 410 has been designed to perform properly in the face of the practice of pulling power to force a reboot. This is not the case with the 31X series where issues (namely the loss of configuration changes) may result if the reboot
command is not used. We continue to recommend that the reboot
command be used with with all Model 310, 312 and 314 JNIORs.
14.3 Removing Power During Firmware Update
You might be familiar with the warning: “Updating firmware DO NOT remove power.” if you, like everyone else, use new products that support network firmware updates. The assumption is that if you remove power and their update process has not completed you will be left with a product with half an operating system which is potentially no longer operational. This is not a concern with the Model 410 JNIOR.
On reboot the Model 410 handles a scheduled OS swap. The JANOS images are indeed physically swapped in program flash memory. If a traditional memory copy operation were employed then we would indeed need to warn you. But the Model 410 uses an algorithm for the swap that insures success even if you flip power off and on as fast as you can possibly do so during the procedure. Try it!
The YELLOW/ORANGE status LED flashes slowly during the swap which typically takes only a few seconds. The updated JANOS will subsequently boot in just another second no matter the stability of power during the swap. This was a critical concern because otherwise if power is pulled to effect the reboot and not restored decisively a lazy update process would risk failure. This risk is unacceptable and unlike other products we address the issue head-on with a fault-tolerant update procedure so you can update with confidence.
15 Safe Mode
The Model 410 may be started in SAFE MODE. In this mode applications that are programmed to start automatically through Registry “Run” keys are not started.
In order to access SAFE MODE a jumper must be inserted onto the pins accessible through the small opening between the Ethernet connector and the RS-232 Command Port. The unit is then rebooted or powered up. When the command line mode is subsequently accessed either through the serial connection or via the network, “SAFE MODE” will be indicated below the welcome banner. This is the only indication that the mode has been enabled. The jumper must be removed and the unit must be rebooted in order to exit SAFE MODE.
Note that you may ‘borrow’ a jumper from the N.O./N.C. relay jumpers if you remove the unit’s cover. Do so only if disconnecting that relay will not adversely affect any system connected to it. Use a unused channel if available. Once you are done with SAFE MODE be sure to return the jumper to the original position. Jumpers placed close to the relay output connector are set for Normally Open (N.O.) operation (default).
There are two situations in which SAFE MODE is useful.
15.1 Application Generated Boot Loops
If an application that is programmed to automatically start upon boot misbehaves and immediately causes a reboot, a boot loop will result. In this case the JNIOR will rapidly reboot and it will be impossible to regain control of the unit through normal means. The solution is to insert the SAFE MODE jumper. On the next reboot the application will not be restarted and you will be able to log into command line mode. Once at the command line you can proceed with debugging. The SAFE MODE jumper should be removed and you might want to also remove the application’s “Run” key until you are certain the issue is resolved.
15.2 Forgotten Administrator Username or Password
If you have lost and forgotten the administrator’s password (‘jnior’ user for instance), you will need to contact INTEG to obtain the “backdoor” password for your unit. This password will allow you to log into all accounts (even disabled accounts) but only in SAFE MODE. Once you have logged into the administrator’s account you should use the passwd
command to change the administrator’s password.
In case you have removed the default administrator’s accounts (‘jnior’ and ‘admin’) and replaced them with your own account name which now you may have forgotten, SAFE MODE will restore a DISABLED account for ‘jnior’ with the standard default password. You may log into disabled accounts in SAFE MODE using the backdoor password. Use the ‘jnior’ account to manage your user accounts.
A unit should not be left in SAFE MODE as this enables the backdoor password wherever a password is requested. That means that it would be valid for web page login, FTP, etc. as well. This would represent a serious security concern. Remember to remove the SAFE MODE jumper once you are done and reboot!
This sample shows you how to pulse multiple outputs and a single output. The method must take two binary masks. One describing the desired states during the pulse and the other describing which channels will be pulsed.
package com.integpg;
import com.integpg.system.JANOS;
public class PulseOutputs {
public static void main(String[] args) {
// to get the states of the outputs use the JANOS class and the getOutputStates method
int outputStates = JANOS.getOutputStates();
//print the Output States through telnet (console) connection.
System.out.println("Output States are: " + outputStates);
//Pulse 8 Relay Outputs On for 5 seconds (5000 milliseconds) after which outputs will return to previous state.
//All channels (1111 1111b)
JANOS.setOutputPulsed(255, 255, 5000);
//Sleep 10 seconds to so that there is a noticable difference between on and off states.
try {
Thread.sleep(10000);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
int counter = 0;
while(counter<5){
//Pulse Channel 5 Relay Output On for 5 seconds (5000 milliseconds) after which output will return to previous state.
//Channels 8765 4321
//Channel 5 (0001 0000b)
JANOS.setOutputPulsed(16, 16, 5000);
try {
Thread.sleep(10000);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
counter++;
System.out.println("Counter: " + counter);
}
}
}
The classic Hello World application that runs on the JNIOR!
package helloworld;
public class HelloWorld {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
System.out.println("hello world");
}
}
Write Outputs
Sometimes you want to control the outputs. The outputs may be wired to things like lights, sirens, valves and maybe fans. (Note: You may need our Power 4 Relay Output Module for high loads). This example will show you how to set the output states programmatically.
package com.integpg;
import com.integpg.system.JANOS;
public class WriteOutputs {
public static void main(String[] args) {
// to get the states of the outputs use the JANOS class and the getOutputStates method
int outputStates = JANOS.getOutputStates();
//print the Output States through telnet (console) connection.
System.out.println("Output States are: " + outputStates);
//set output relay for channel 5 (channel n-1) and true top turn the relay on, false to turn relay off.
JANOS.setOutputRelay(4 , true);
//Relay Output 5 should now be on.
//print the Output States through telnet (console) connection.
System.out.println("Output States are: " + outputStates);
}
}
The JNIOR is a very flexible and powerful controller. Utilize our bundled or add-on software applications. If those don’t meet your needs, let INTEG quickly develop an application for you.
The JNIOR offers superb functionality with its included and available software. However, if you require a custom application to run on your JNIOR, INTEG can develop it for you, or you can develop it yourself using the JNIOR Software Development Kit (SDK).INTEG has already developed a number of custom applications for a variety of customers. Some of these applications have become our ‘add-on’ applications because they have met the needs of a large group of customers.
Other times the applications have been focused and developed to meet the needs of a specific customer. After the user requirements are gathered it doesn’t take long for INTEG to deliver something for the customer to test. Often times we dont get devices sent to the office that the customer wants to interface with. This makes it tough for INTEG to complete full testing in the office. Sometimes we write test applications to mimic the communications between the JNIOR and the end device.
If you have an application that you have in mind and want to talk to INTEG about the JNIOR please call the office, 724-933-9350, or email support@integpg.com. You can also fill out the contact form.
Thank you for your interest in the JNIOR from INTEG.
NOTE: Before creating the setup described in this post on your JNIOR 410, please download the update project below and install it on your 410 using the JNIOR Support Tool. This is required to get the JNIOR to create a DMX connection.
Name | Version | Release Date | Size | MD5 |
---|---|---|---|---|
DmxPort for enabling DMX on 410 | Mar 17 2021 | 3.5 KB | 299c66717c03a9c9b702716d9d56d095 |
This article is the first of two addressing the issues encountered in, and a means to simplify, the processing of the DMX Universe data stream using a standard UART serial receiver. The follow-up article is entitled JNIOR as a DMX Fixture Revisited.
The Model 412DMX generates a DMX512 Universe and allows the JNIOR to control DMX fixtures like those used in stage lighting. What if you needed a fixture with relays that can be controlled by DMX? Perhaps you need to output channels over a 4-20ma loops. Maybe you need a 10 VDC output signal to control LED house lighting. Can the JNIOR receive DMX? Can the JNIOR be a DMX Fixture?
We showed you how you could control DMX fixtures with a standard Model 410 in a White Paper available here:
AN01 DMX512 Implementation [ Jul 20 2017, 101.27 KB, MD5: e1b0203f177d1866e56cfbfdd0e221d4 ]
Now we have the 412DMX JNIOR designed for that purpose. Can the Model 410 also serve as a DMX fixture? Yes, it can. I’ll show you how here and we’ll see how we manage to accommodate some of the unique aspects of the DMX512 format with the JNIOR.
Cabling
We can use the JNIOR Model 410 because the AUX port is compatible with RS-485. In the white paper explaining how the 410 can be used to control DMX fixtures we described an adapter cable taking the DB9 output from the JNIOR and presenting the proper female XLR connector for DMX. Now since a DMX fixture always has both a male and female 5-pin XLR connectors, our cabling has to be slightly different. Note that you can do this with the 3-pin XLR (as I have) if that is appropriate for your situation.
Here is an example of one that we put together.
This can be constructed by splicing into a standard DMX extension cable. A number of DB9 adapters with screw terminals like the one pictured can be found on Amazon. Note that you will want one with large screws compatible with larger wire sizes. DMX wiring is typically of a larger diameter and you will need to successfully clamp two wires in each of three positions on the adapter.
Here is the pin numbering. Note that wire colors vary.
Signal XLR DB-9 Male -------------------- --------- ----------- Signal Ground (GND) 1 5 Data (D-) 2 2 Data (D+) 3 8 Not Used (NC) 4,5 1,3,4,6,7,9
This cable allows the JNIOR to be a DMX FIXTURE.
THE RESULTING DMX CONNECTION IS NOT ISOLATED. We recommend using an isolated power supply for the JNIOR and not sharing that voltage with other circuits. Take great care in making ground connections. Note that the JNIOR relay outputs are naturally isolated.
Serial Connection
Connect the adapter to the Model 410 AUX serial port as I have in this photo and connect this to the DMX network. Note that the 412 and 414 are not RS-485 compatible and cannot be used for this purpose.
The serial port parameters should be set as follows. This is done through the Dynamic Configuration Pages (DCP) that should come up when accessing the JNIOR using your browser. You enable the RS-485 mode here so the AUX port output doesn’t disrupt the DMX communications before you have a chance to run the DMXFIXTURE application that I will describe. That application will also configure the AUX port just to make sure that all is well.
If you encounter “Applets” instead of the DCP then your Series 4 needs to be updated or you have a Series 3. The latter also cannot be used for this application. You will need to update your JNIOR to JANOS v1.6.6 or later for the functionality to be described here.
Data
With the JNIOR Model 410 wired to the DMX network and the AUX serial port properly configured the unit should be receiving data. There is a simple way to check that. You can see data without any application running just by using the IOLOG command. Here we enter the Console (or Command Line Interface) and use this command.
InfoComm_LED /> help iolog
IOLOG
Options:
-T Indicate transitions
-R Reset logs
-A AUX Serial log
-S Sensor Port log
-O Output to stdout
Generates jniorio.log file from available logs.
InfoComm_LED /> iolog -ao
-- 07/02/18 15:42:46.098
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--80--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--FF--00--FF--80--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--83--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--80--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--FF--00--FF--80--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--83--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--80- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--FF--00--FF--80- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--83--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-80--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--FF--00--FF- ................
-80--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--83--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ................
-00--00--00--00--00--00--00--00--00--00--00--00--00--00--00- ...............
InfoComm_LED />
If you scan down in the above output and look through the data you will see that there are a couple of channels at 100% (0xFF) and a couple near half (0x80’ish). There are two pretty major issues in trying to read these bytes with standard library read()
functions.
- How do we know how to find Channel 1? There are many more than 512 bytes shown here. If you read 512 bytes what you get could start anywhere.
- The data rate at 250 Kbaud supplies 44 complete channel sets per second! That absolutely will overrun the buffer before you can process any of it. The overrun would likely further obfuscate the data.
The fact is that the standard serial communications routines that you may be used to are just not usable here. JANOS will come to the rescue. But first let’s take a look at the data stream so we understand how this is to be resolved.
DMX Format
The DMX data on the RS-485 lines conforms to standard asynchronous serial data with 8 data bits, 2 stop bits and no parity. The bits are marched out from least significant (LSB) D0 to the most significant (MSB) D7. Each byte is called a “Slot”. The standard implementation transfers a START CODE and 512 channels in a total of 513 slots. The START CODE for normal DMX data is 0x00.
The beginning of the sequence is signaled with a Break Condition. This BREAK can be detected by the fixtures which allows them to synchronize with the stream. After the BREAK comes the START CODE (0x00 – NULL START) followed by the value for Channel 1 on through to Channel 512. Not all 512 channels need to be part of the transmission. The number of channels may vary by DMX controller. The complete implementation provides all 512.
On the oscilloscope the BREAK looks like this. Here all of the channels are 0X00 and so you see only the STOP BITS. That long low pulse is the BREAK.
The issue is that the BREAK is difficult to handle with the standard serial port. It results in a FRAMING ERROR. During the break the signal is held at a low level. When the receiving serial UART expects the STOP BITS and they aren’t there it throws a FRAMING ERROR. While that can be detected and your application can be notified there still is no way to insure that the next bytes read from the port are those that follow the break. They may have been buffered some time before. They may be overrun by oncoming data.
In order to handle this and properly capture a reliable channel set, there must be a special function for that purpose in the AUX port class (AUXSerialPort). Of course, being the author of JANOS, I have implemented exactly what we need. And, those details are next…
Packet Capture
To read the DMX Packet (START CODE plus up to 512 Slots/Channels) we need to detect the Break Condition and then reliably collect as many as 513 serial bytes that immediately follow. Under many other RTOS implementations we would need to write an interrupt driven routine both to detect the Break condition and then also to collect the data. The JNIOR executes application programs written in a managed language (Java) and one does not have low level access to write things like serial interrupt routines. That is actually a good thing as the user generally does not have the programming experience. Such low level user programming often leads to unstable/unpredictable operation.
Here we rely on JANOS to maintain reliable operation. Low level interrupt routines have already been implemented to buffer incoming serial data and otherwise issue a notification of errors. Recall that the Break Condition manifests itself and one or more FRAMING ERRORS. But we have already established that reading buffered serial data and receiving asynchronous notifications is not going to be sufficient for capturing a DMX packet. This is where we benefit from having developed JANOS in-house and having authored 100% of it. Here we identify a need and are able to promptly and correctly implement a solution.
AUXSerialPort.readAfterBreak(byte[] buffer)
I have added the readAfterBreak()
method to the AUXSerialPort
class in the JANOSClasses.jar
library. From the naming its use is self-explanatory. Here you create a buffer as a byte array and pass it to JANOS. The operating system enables the capture and then blocks the thread until the data collection completes. At the low-level JANOS sets up the buffer with a pointer and goes into a kind of ‘armed’ state. The interrupt routine that detects FRAMING errors has a tiny bit of code that checks for an armed capture and ‘triggers’ the collection of data. The interrupt routine that collects and buffers serial bytes from the port has code to set each byte aside into the buffer that you have provided. Once triggered the capture passes into the a ‘collection’ mode. When the buffer is full (or when another Break Condition is detected) the capture is ‘complete’ and the application program can proceed now with a byte array containing the DMX data.
Now to benefit from this new feature, you will need to update your Series 4 to run JANOS v1.6.6 or later. At the moment this is Beta code. We would make it available if you were to want to try this before its release. All you need do is ask.
Next we need to try it out…
DMX Capture Test
Here we create a project in Netbeans (making the few settings needed to target it for JANOS) and create the following test program. This merely takes control of, and fully configures, the AUX port in case it has not been configured through the DCP. Lines 23 and 24 test our new method. The rest merely dumps the byte array content for review.
package dmxfixture; import com.integpg.comm.AUXSerialPort; import com.integpg.comm.SerialPort; public class Dmxfixture { public static void main(String[] args) throws Throwable { // AUX port access and configuration. We need to open the port to gain exclusive access and // set the proper baud rate and format. We enable RS-485 mode and make sure that the receivers // are enabled. With normal RS-485 you would disable the transmit drivers. Our adapter doesn't // bridge the transmit and receive lines anyway and the DCP configuration automatically disables // the drivers. It is here for clarity. AUXSerialPort aux = new AUXSerialPort(); aux.open(); aux.setSerialPortParams(250000, 8, 1, SerialPort.PARITY_NONE); aux.setRS485(true); aux.enableReceivers(true); aux.enableDrivers(false); // capture a complete frame using our new method byte[] data = new byte[513]; aux.readAfterBreak(data); // The remainder here is a fancy dump (skipping the START CODE). Note how JANOS implements the // printf formatting for us. for (int i = 1; i < data.length; i++) { if (i % 10 == 1) System.out.printf("%04d ", i); System.out.printf("%4d ", data[ i ] & 0xff); if (i % 10 == 0) System.out.println(""); } System.out.println(""); } }
To run this we first build it in Netbeans. Then using the DCP we open the Folders tab and select the /flash
folder. We then drag the dmxfixture.jar
file from the project to the /flash
folder (it can be executed from the root too). Then under the Console tab we log in and execute the application. The following is the result.
InfoComm_LED /> dmxfixture
0001 0 0 0 0 0 0 0 0 0 0
0011 0 0 0 0 0 0 0 0 0 0
0021 0 0 0 0 0 0 0 0 0 0
0031 0 0 0 0 0 0 0 0 0 0
0041 0 0 0 0 0 0 0 0 0 0
0051 0 0 0 0 0 0 0 0 0 0
0061 0 0 0 0 0 0 0 0 0 0
0071 0 0 0 0 0 0 0 0 0 0
0081 0 0 0 0 0 0 0 0 0 0
0091 0 0 255 0 255 128 0 0 0 0
0101 0 0 0 0 0 0 0 0 0 0
0111 0 0 0 0 0 0 0 0 0 0
0121 131 0 0 0 0 0 0 0 0 0
0131 0 0 0 0 0 0 0 0 0 0
0141 0 0 0 0 0 0 0 0 0 0
0151 0 0 0 0 0 0 0 0 0 0
0161 0 0 0 0 0 0 0 0 0 0
0171 0 0 0 0 0 0 0 0 0 0
0181 0 0 0 0 0 0 0 0 0 0
0191 0 0 0 0 0 0 0 0 0 0
0201 0 0 0 0 0 0 0 0 0 0
0211 0 0 0 0 0 0 0 0 0 0
0221 0 0 0 0 0 0 0 0 0 0
0231 0 0 0 0 0 0 0 0 0 0
0241 0 0 0 0 0 0 0 0 0 0
0251 0 0 0 0 0 0 0 0 0 0
0261 0 0 0 0 0 0 0 0 0 0
0271 0 0 0 0 0 0 0 0 0 0
0281 0 0 0 0 0 0 0 0 0 0
0291 0 0 0 0 0 0 0 0 0 0
0301 0 0 0 0 0 0 0 0 0 0
0311 0 0 0 0 0 0 0 0 0 0
0321 0 0 0 0 0 0 0 0 0 0
0331 0 0 0 0 0 0 0 0 0 0
0341 0 0 0 0 0 0 0 0 0 0
0351 0 0 0 0 0 0 0 0 0 0
0361 0 0 0 0 0 0 0 0 0 0
0371 0 0 0 0 0 0 0 0 0 0
0381 0 0 0 0 0 0 0 0 0 0
0391 0 0 0 0 0 0 0 0 0 0
0401 0 0 0 0 0 0 0 0 0 0
0411 0 0 0 0 0 0 0 0 0 0
0421 0 0 0 0 0 0 0 0 0 0
0431 0 0 0 0 0 0 0 0 0 0
0441 0 0 0 0 0 0 0 0 0 0
0451 0 0 0 0 0 0 0 0 0 0
0461 0 0 0 0 0 0 0 0 0 0
0471 0 0 0 0 0 0 0 0 0 0
0481 0 0 0 0 0 0 0 0 0 0
0491 0 0 0 0 0 0 0 0 0 0
0501 0 0 0 0 0 0 0 0 0 0
0511 0 0
InfoComm_LED />
We note that channels are correct. Here we go over to the 412DMX controlling this DMX network and check Kevin’s DMX panel page for comparison.
Putting it to Work
Now we can receive a DMX frame and read the individual channels what can we do with it? I mean other than dump it?
Well Kevin has defined an eight channel fixture starting at DMX channel 121. The idea being that each channel would correspond to a JNIOR Relay Output. Channel settings from 0-127 would result in an open/off relay and values in the range 128-255 would close the relay. You can imagine any use that you would want given the flexibility that you now have in JNIOR programming. Let’s implement this particular fixture.
The approach will be to sample a DMX packet periodically and set the relays appropriately. There is no need to catch every DMX packet and in fact we are not likely going to be able to do that. We are also going to be considerate of the JNIOR CPU and anything else that the unit might want to be doing. We will sample say every 1/4 second and sleep in between.
Here is the program. This uses an infinite loop to sample the DMX stream about 4 times a second. The starting address must be defined in the Registry. This could be cached. With this implementation you can change the starting address without rebooting or restarting the DMXFIXTURE program. It is presume that you would start the DMXFIXTURE program automatically at boot with a Registry Run key.
package dmxfixture; import com.integpg.comm.AUXSerialPort; import com.integpg.comm.SerialPort; import com.integpg.system.JANOS; public class Dmxfixture { public static void main(String[] args) throws Throwable { // AUX port access and configuration. We need to open the port to gain exclusive access and // set the proper baud rate and format. We enable RS-485 mode and make sure that the receivers // are enabled. With normal RS-485 you would disable the transmit drivers. Our adapter doesn't // bridge the transmit and receive lines anyway and the DCP configuration automatically disables // the drivers. It is here for clarity. AUXSerialPort aux = new AUXSerialPort(); aux.open(); aux.setSerialPortParams(250000, 8, 1, SerialPort.PARITY_NONE); aux.setRS485(true); aux.enableReceivers(true); aux.enableDrivers(false); // here we create an infinite loop to continuously process the DMX data byte[] data = new byte[513]; for (;;) { // capture a complete frame aux.readAfterBreak(data); // Obtain the starting address. If it is invalid or not defined no action is taken. int addr = JANOS.getRegistryInt("DMX/Address", 0); if (addr > 0 && addr < 505) { // Although we don't have to we are going to collect all of the relay states // and set them simultaneously. This will also take advantage of signed values // in Java. Values in the range 128-255 will appear to be negative if we don't // mask them with 0xff. int bits = 0; for (int i = 0; i < 8; i++) { if (data[addr++] < 0) bits += (1 << i); } JANOS.setOutputStates(bits, 0xff); } // sleep for a quarter second System.sleep(250); } } }
This program should be pretty easy to follow. Let’s test it.
Demonstration
A video can best demonstrate the operation of this program. Here we have a DMX application running on a 412DMX (10.0.0.242) allowing us to vary the channels that we associate with our 410 fixture. A separate Model 410 running our DMXFIXTURE (10.0.0.250) program can be monitored remotely through its DCP page. Here we overlap the two browser entities and we can see how modifying the channel fader results in the relay status change out across the DMX network.
Reliability
Let’s look into potential error conditions and the reliability of this approach. The DMX format typically supplies nearly 44 frames per second. If there is a communications error, due to electrical noise for instance, one and possibly up to a few frames might be in error. For a light fixture this might cause a minute flicker or some small flinch in pointing. But, given the frame rate it is quickly corrected and might not be even noticeable. If we are interpreting a frame with our program we need to be extra careful not to trigger a chain of events based upon an error packet.
Typically in data protocols we would have some form of checksum or CRC which we can use to identify an erroneous transmission so it can be ignored. There is no such thing in the DMX512 protocol. So what steps can we take?
Well to start we should verify that the START CODE is the expected NULL START 0x00 and ignore any frame with a different code. The controller might actually be inserting those and we must ignore them. I will adjust the program to check this.
Well… The START CODE is returning 128 (0x80) and the channels appear to be properly registered (e.g. in the right place). Now to look into this.
Synchronization After Break
The DMX512 specification defines the width of the Break Condition as something greater than 92 microseconds. It is important to note that it is something greater than twice that of a single slot time (the time to receive a single byte) of 44 microseconds (11 bit times – start bit, 8 data bits and 2 stop bits). It is not a precise multiple of slot times or even bit times. This forces the receiver to synchronize with each and every packet.
Given this I could make the argument that the Mark After Break should be at least one slot time of 44 microseconds in order to insure that the leading start bit of the first slot is successfully interpreted. The DMX512 specification however specifies the minimum Mark after Break of 12 microseconds. This puts us at the mercy of the UART design and its ability to synchronize following a Break Condition of arbitrary length. There are a number of possible outcomes that depend on what the UART decides is the first STOP BIT once the Break Condition passes.
- For example, if the beginning of the Mark After Break is seen as a valid STOP BIT then a 0x00 byte is received AHEAD OF the normal NULL START code 0x00. This extra 0x00 can be interpreted as a valid START CODE but all of the channel slots are off by 1. Channel 2 would have the value for Channel 1. This is an ERROR!
- If the Mark condition just slightly into to Mark After Break is interpreted as a valid start bit then an extra 0x80 is received AHEAD of the START CODE. This might be seen as a bad packet if the START CODE is verified. Channels are also shifted if values are used. This is a ERROR!
- The above continues with each bit time advance into the Mark After Break generating an initial extra byte of 0xC0, 0xE0, 0xF0, 0xF8, 0xFC, 0xFE and 0xFF depending on the length of the Mark After Break. In each case the START CODE would then be considered the Channel 1 value. ERRORs result!
- With a short Mark After Break the UART might look at a low bit value in the START CODE as a missing STOP BIT and generate yet another FRAMING ERROR. Again depending on the timing the START CODE might be returned as 0x80 with the first STOP BIT actually being interpreted as the MSB. In this case the Channel data is properly positioned. This is actually the most common mode I am seeing in the current set up. It is timing sensitive. This is also an ERROR!
If you follow this logic you might see that it is possible that it may take a couple of regular slot times before the UART grabs something it is happy about. It is all about the synchronization aspect of the hardware design.
The question is how to know when you are receiving valid data and properly aligned slots? Is there a solution to this?
A UART that requires a Marking Condition before attempting to detect a START BIT (falling edge) would function properly. Apparently they don’t work this way. At least not all of them.
UART Issue
The problem that we run into is an ancient design flaw in serial ports.
A Framing Error results when the UART (RX SCI) expects a Stop Bit and none is detected. A Stop Bit is a high (1 Marking) and during a Break Condition the signal is held low (0 Space) so a Framing Error is quickly encountered. Now most descriptions of UART logic suggest that after a Break the UART locates the next Start Bit (0 Marking) and that this is detected by a high to low transition of signal (1 -> 0). Logically it is done that way for asynchronous reception as the UART clock needs to synchronize and then sample the middle of each bit period.
In reality after a Framing Error the UART seems to see the next low (0 Space) as a Start Bit and continues to read bit data. As a result Framing Errors are repeated throughout the break period. A bogus byte value might appear to be properly read if the tail end of the Break Condition aligns with the UART in a way to make the high (1 Marking) after the Break look like a Stop Bit.
The likelihood of this bogus data byte and its content can vary depending on the length of the Break and the length of the Marking after the Break and before actual data is present. Since bytes are serialized LSB first these extra bytes look like one of 0xFF, 0xFE, 0xFC, 0xF8, 0xF0, 0xE0, 0xC0, 0x80 and even 0x00.
If the Marking after Break is brief (only a few bit times) and the alignment falls such that the UART looks at bits in the first byte of data for that magic Stop Bit, you will receive an incorrect value for the fist byte. It is conceivable that the UART might take several bytes before synchronizing and providing real data.
If the UART simply fell into a mode whereby it actually did search for the next Start Bit by looking for a valid high to low transition (1 -> 0), you would get a single Framing Error followed by the proper collection of data. But no… after 50+ years we have not addressed this issue. I half recall struggling with this exact thing maybe 30 or so years back now. The fact that it is still an problem is not impressive.
I guess I shouldn’t be surprised in that these hi-tech MCU processors all still include the Real Time Clock (RTC) circuit first designed for the very first digital watches in the late 1960’s. This forces us to parse time into Day, Month , Year, Hour, Minute and Seconds as if setting a watch on your wrist. In fact Seconds can only be reset to 00 and not directly set. On boot we have to read the time and reassemble it into Linux or Internet time as a tally of milliseconds since some epoch. Lots of work that causes loss of precision. And the ideal would be a non-volatile battery-backed 64-bit millisecond counter. Sometimes silicon space is limited and this counter would save lots of that. But no… these integrated circuit companies aren’t as swift as we would like to think.
Since DMX512 signals can have different lengths of Break and Marking after Break and these can vary depending on source, and since the protocol has no leading header that can be used in identifying valid frames, we are NOT ABLE to reliably receive data. Note that if the DMX512 Standard had forced the Mark After Break to be at least one data Slot long (> 44 microseconds) then UARTs would likely properly synchronize and reliably present the first byte of data. But the spec does not and the problem is that changing the standard now does not correct all of the DMX controllers already in use all over the world. So it is what it is.
So for us to insure that we read a valid frame, we need to resort to some trickery, filtering and indeed AI. While that can be fun, it’s unfortunate.
- Corrected FTP listing issue created by the v1.6.4 release
- Corrected
getRegistryList
method memory leak - Corrected 412DMX light Flickering
- Corrected 412DMX NAND Flash processing issue
- Corrected FTP transfer restart issue