Make SerialPortMessageListener able to parse multi-byte binary delimiters

This commit is contained in:
hedgecrw85 2019-03-19 12:38:36 -05:00
parent bebef1f45c
commit a6a377a1bc
3 changed files with 57 additions and 36 deletions

View File

@ -2,7 +2,7 @@
* SerialPort.java * SerialPort.java
* *
* Created on: Feb 25, 2012 * Created on: Feb 25, 2012
* Last Updated on: Mar 15, 2019 * Last Updated on: Mar 19, 2019
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2019 Fazecast, Inc. * Copyright (C) 2012-2019 Fazecast, Inc.
@ -26,8 +26,8 @@
package com.fazecast.jSerialComm; package com.fazecast.jSerialComm;
import java.lang.ProcessBuilder; import java.lang.ProcessBuilder;
import java.nio.charset.Charset;
import java.io.BufferedReader; import java.io.BufferedReader;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream; import java.io.DataOutputStream;
import java.io.File; import java.io.File;
import java.io.FileNotFoundException; import java.io.FileNotFoundException;
@ -37,6 +37,7 @@ import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.io.InputStreamReader; import java.io.InputStreamReader;
import java.io.OutputStream; import java.io.OutputStream;
import java.util.Arrays;
import java.util.Date; import java.util.Date;
import java.util.regex.Pattern; import java.util.regex.Pattern;
@ -702,7 +703,7 @@ public final class SerialPort
userDataListener = listener; userDataListener = listener;
serialEventListener = ((userDataListener instanceof SerialPortPacketListener) ? new SerialPortEventListener(((SerialPortPacketListener)userDataListener).getPacketSize()) : serialEventListener = ((userDataListener instanceof SerialPortPacketListener) ? new SerialPortEventListener(((SerialPortPacketListener)userDataListener).getPacketSize()) :
((userDataListener instanceof SerialPortMessageListener) ? ((userDataListener instanceof SerialPortMessageListener) ?
new SerialPortEventListener(((SerialPortMessageListener)userDataListener).getMessageDelimiter(), ((SerialPortMessageListener)userDataListener).getCharacterEncoding()) : new SerialPortEventListener(((SerialPortMessageListener)userDataListener).getMessageDelimiter(), ((SerialPortMessageListener)userDataListener).delimiterIndicatesEndOfMessage()) :
new SerialPortEventListener())); new SerialPortEventListener()));
eventFlags = 0; eventFlags = 0;
@ -1144,16 +1145,15 @@ public final class SerialPort
private final class SerialPortEventListener private final class SerialPortEventListener
{ {
private volatile boolean isListening = false; private volatile boolean isListening = false;
private final byte[] dataPacket; private final boolean messageEndIsDelimited;
private final String delimiter, delimiterRegex; private final byte[] dataPacket, delimiters;
private final Charset messageCharset; private volatile ByteArrayOutputStream messageBytes = new ByteArrayOutputStream();
private volatile int dataPacketIndex = 0; private volatile int dataPacketIndex = 0, delimiterIndex = 0;
private String messages = "";
private Thread serialEventThread = null; private Thread serialEventThread = null;
public SerialPortEventListener() { dataPacket = new byte[0]; delimiter = delimiterRegex = ""; messageCharset = Charset.forName("UTF-8"); } public SerialPortEventListener() { dataPacket = new byte[0]; delimiters = new byte[0]; messageEndIsDelimited = true; }
public SerialPortEventListener(int packetSizeToReceive) { dataPacket = new byte[packetSizeToReceive]; delimiter = delimiterRegex = ""; messageCharset = Charset.forName("UTF-8"); } public SerialPortEventListener(int packetSizeToReceive) { dataPacket = new byte[packetSizeToReceive]; delimiters = new byte[0]; messageEndIsDelimited = true; }
public SerialPortEventListener(String messageDelimiter, Charset charset) { dataPacket = new byte[0]; delimiter = messageDelimiter; delimiterRegex = Pattern.quote(messageDelimiter); messageCharset = charset; } public SerialPortEventListener(byte[] messageDelimiters, boolean delimiterForMessageEnd) { dataPacket = new byte[0]; delimiters = messageDelimiters; messageEndIsDelimited = delimiterForMessageEnd; }
public final void startListening() public final void startListening()
{ {
@ -1203,15 +1203,28 @@ public final class SerialPort
byte[] newBytes = new byte[numBytesAvailable]; byte[] newBytes = new byte[numBytesAvailable];
newBytesIndex = 0; newBytesIndex = 0;
bytesRemaining = readBytes(portHandle, newBytes, newBytes.length, 0); bytesRemaining = readBytes(portHandle, newBytes, newBytes.length, 0);
if (!delimiter.isEmpty()) if (delimiters.length > 0)
{ {
messages += new String(newBytes, messageCharset); int startIndex = 0;
while (messages.contains(delimiter)) for (int offset = 0; offset < bytesRemaining; ++offset)
{ if (newBytes[offset] == delimiters[delimiterIndex])
String[] message = messages.split(delimiterRegex, 2); {
messages = (message.length > 1) ? message[1] : ""; if ((++delimiterIndex) == delimiters.length)
userDataListener.serialEvent(new SerialPortEvent(SerialPort.this, LISTENING_EVENT_DATA_RECEIVED, message[0].getBytes(messageCharset))); {
} messageBytes.write(newBytes, startIndex, 1 + offset - startIndex);
byte[] byteArray = (messageEndIsDelimited ? messageBytes.toByteArray() : Arrays.copyOf(messageBytes.toByteArray(), messageBytes.size() - delimiters.length));
if ((byteArray.length > 0) && (messageEndIsDelimited || (delimiters[0] == byteArray[0])))
userDataListener.serialEvent(new SerialPortEvent(SerialPort.this, LISTENING_EVENT_DATA_RECEIVED, byteArray));
startIndex = offset + 1;
messageBytes.reset();
delimiterIndex = 0;
if (!messageEndIsDelimited)
messageBytes.write(delimiters, 0, delimiters.length);
}
}
else if (delimiterIndex != 0)
delimiterIndex = (newBytes[offset] == delimiters[0]) ? 1 : 0;
messageBytes.write(newBytes, startIndex, bytesRemaining - startIndex);
} }
else if (dataPacket.length == 0) else if (dataPacket.length == 0)
userDataListener.serialEvent(new SerialPortEvent(SerialPort.this, LISTENING_EVENT_DATA_RECEIVED, newBytes.clone())); userDataListener.serialEvent(new SerialPortEvent(SerialPort.this, LISTENING_EVENT_DATA_RECEIVED, newBytes.clone()));

View File

@ -2,7 +2,7 @@
* SerialPortMessageListener.java * SerialPortMessageListener.java
* *
* Created on: Mar 14, 2019 * Created on: Mar 14, 2019
* Last Updated on: Mar 15, 2019 * Last Updated on: Mar 19, 2019
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2019 Fazecast, Inc. * Copyright (C) 2012-2019 Fazecast, Inc.
@ -25,10 +25,8 @@
package com.fazecast.jSerialComm; package com.fazecast.jSerialComm;
import java.nio.charset.Charset;
/** /**
* This interface must be implemented to enable delimited string-based message reads using event-based serial port I/O. * This interface must be implemented to enable delimited message reads using event-based serial port I/O.
* <p> * <p>
* <i>Note</i>: Using this interface will negate any serial port read timeout settings since they make no sense in an asynchronous context. * <i>Note</i>: Using this interface will negate any serial port read timeout settings since they make no sense in an asynchronous context.
* *
@ -40,16 +38,16 @@ import java.nio.charset.Charset;
public interface SerialPortMessageListener extends SerialPortDataListener public interface SerialPortMessageListener extends SerialPortDataListener
{ {
/** /**
* Must be overridden to return the expected message delimiter that <b>must</b> be encountered before the {@link #serialEvent(SerialPortEvent)} callback is triggered. * Must be overridden to return the expected message delimiter bytes that <b>must</b> be encountered before the {@link #serialEvent(SerialPortEvent)} callback is triggered.
* *
* @return A string indicating the expected message delimiter that must be encountered before the {@link #serialEvent(SerialPortEvent)} callback is triggered. * @return A byte array containing the expected message delimiters that must be encountered before the {@link #serialEvent(SerialPortEvent)} callback is triggered.
*/ */
public abstract String getMessageDelimiter(); public abstract byte[] getMessageDelimiter();
/** /**
* Must be overridden to return the expected character encoding used by your message transfer protocol. * Must be overridden to return whether the message delimiter indicates the end or the beginning of a message.
* *
* @return A {@link java.nio.charset.Charset} indicating the expected character encoding used by your serial messages. * @return A boolean indicating whether the message delimiter indicates the end or the beginning of a message.
*/ */
public abstract Charset getCharacterEncoding(); public abstract boolean delimiterIndicatesEndOfMessage();
} }

View File

@ -2,7 +2,7 @@
* SerialPortTest.java * SerialPortTest.java
* *
* Created on: Feb 27, 2015 * Created on: Feb 27, 2015
* Last Updated on: Mar 14, 2019 * Last Updated on: Mar 19, 2019
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2019 Fazecast, Inc. * Copyright (C) 2012-2019 Fazecast, Inc.
@ -58,18 +58,28 @@ public class SerialPortTest
private static final class MessageListener implements SerialPortMessageListener private static final class MessageListener implements SerialPortMessageListener
{ {
public String byteToHex(byte num)
{
char[] hexDigits = new char[2];
hexDigits[0] = Character.forDigit((num >> 4) & 0xF, 16);
hexDigits[1] = Character.forDigit((num & 0xF), 16);
return new String(hexDigits);
}
@Override @Override
public int getListeningEvents() { return SerialPort.LISTENING_EVENT_DATA_RECEIVED; } public int getListeningEvents() { return SerialPort.LISTENING_EVENT_DATA_RECEIVED; }
@Override @Override
public void serialEvent(SerialPortEvent event) public void serialEvent(SerialPortEvent event)
{ {
String newMessage = new String(event.getReceivedData()); byte[] byteArray = event.getReceivedData();
System.out.println("Received the following message: " + newMessage); StringBuffer hexStringBuffer = new StringBuffer();
for (int i = 0; i < byteArray.length; i++)
hexStringBuffer.append(byteToHex(byteArray[i]));
System.out.println("Received the following message: " + hexStringBuffer.toString());
} }
@Override @Override
public String getMessageDelimiter() { return "\n"; } public byte[] getMessageDelimiter() { return new byte[]{ (byte)0xB5, (byte)0x62 }; }
@Override @Override
public Charset getCharacterEncoding() { return Charset.forName("UTF-8"); } public boolean delimiterIndicatesEndOfMessage() { return false; }
} }
static public void main(String[] args) static public void main(String[] args)
@ -187,7 +197,7 @@ public class SerialPortTest
ubxPort.addDataListener(listener); ubxPort.addDataListener(listener);
try { Thread.sleep(5000); } catch (Exception e) {} try { Thread.sleep(5000); } catch (Exception e) {}
ubxPort.removeDataListener(); ubxPort.removeDataListener();
System.out.println("\nNow listening for newline-delimited string messages\n"); System.out.println("\nNow listening for byte-delimited binary messages\n");
MessageListener messageListener = new MessageListener(); MessageListener messageListener = new MessageListener();
ubxPort.addDataListener(messageListener); ubxPort.addDataListener(messageListener);
try { Thread.sleep(5000); } catch (Exception e) {} try { Thread.sleep(5000); } catch (Exception e) {}