/**
* Copyright (c) 2010-2016 by the respective copyright holders.
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*/
package org.openhab.binding.ebus.internal.utils;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
import javax.xml.bind.DatatypeConverter;
import org.openhab.binding.ebus.internal.EBusTelegram;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* A helper class to decode all eBus data types and telegrams.
*
* @author Christian Sowada
* @since 1.7.0
*/
public class EBusUtils {
private static final Logger logger = LoggerFactory.getLogger(EBusUtils.class);
/** calculated crc values */
final static public byte CRC_TAB_8_VALUE[] = { (byte) 0x00, (byte) 0x9B, (byte) 0xAD, (byte) 0x36, (byte) 0xC1,
(byte) 0x5A, (byte) 0x6C, (byte) 0xF7, (byte) 0x19, (byte) 0x82, (byte) 0xB4, (byte) 0x2F, (byte) 0xD8,
(byte) 0x43, (byte) 0x75, (byte) 0xEE, (byte) 0x32, (byte) 0xA9, (byte) 0x9F, (byte) 0x04, (byte) 0xF3,
(byte) 0x68, (byte) 0x5E, (byte) 0xC5, (byte) 0x2B, (byte) 0xB0, (byte) 0x86, (byte) 0x1D, (byte) 0xEA,
(byte) 0x71, (byte) 0x47, (byte) 0xDC, (byte) 0x64, (byte) 0xFF, (byte) 0xC9, (byte) 0x52, (byte) 0xA5,
(byte) 0x3E, (byte) 0x08, (byte) 0x93, (byte) 0x7D, (byte) 0xE6, (byte) 0xD0, (byte) 0x4B, (byte) 0xBC,
(byte) 0x27, (byte) 0x11, (byte) 0x8A, (byte) 0x56, (byte) 0xCD, (byte) 0xFB, (byte) 0x60, (byte) 0x97,
(byte) 0x0C, (byte) 0x3A, (byte) 0xA1, (byte) 0x4F, (byte) 0xD4, (byte) 0xE2, (byte) 0x79, (byte) 0x8E,
(byte) 0x15, (byte) 0x23, (byte) 0xB8, (byte) 0xC8, (byte) 0x53, (byte) 0x65, (byte) 0xFE, (byte) 0x09,
(byte) 0x92, (byte) 0xA4, (byte) 0x3F, (byte) 0xD1, (byte) 0x4A, (byte) 0x7C, (byte) 0xE7, (byte) 0x10,
(byte) 0x8B, (byte) 0xBD, (byte) 0x26, (byte) 0xFA, (byte) 0x61, (byte) 0x57, (byte) 0xCC, (byte) 0x3B,
(byte) 0xA0, (byte) 0x96, (byte) 0x0D, (byte) 0xE3, (byte) 0x78, (byte) 0x4E, (byte) 0xD5, (byte) 0x22,
(byte) 0xB9, (byte) 0x8F, (byte) 0x14, (byte) 0xAC, (byte) 0x37, (byte) 0x01, (byte) 0x9A, (byte) 0x6D,
(byte) 0xF6, (byte) 0xC0, (byte) 0x5B, (byte) 0xB5, (byte) 0x2E, (byte) 0x18, (byte) 0x83, (byte) 0x74,
(byte) 0xEF, (byte) 0xD9, (byte) 0x42, (byte) 0x9E, (byte) 0x05, (byte) 0x33, (byte) 0xA8, (byte) 0x5F,
(byte) 0xC4, (byte) 0xF2, (byte) 0x69, (byte) 0x87, (byte) 0x1C, (byte) 0x2A, (byte) 0xB1, (byte) 0x46,
(byte) 0xDD, (byte) 0xEB, (byte) 0x70, (byte) 0x0B, (byte) 0x90, (byte) 0xA6, (byte) 0x3D, (byte) 0xCA,
(byte) 0x51, (byte) 0x67, (byte) 0xFC, (byte) 0x12, (byte) 0x89, (byte) 0xBF, (byte) 0x24, (byte) 0xD3,
(byte) 0x48, (byte) 0x7E, (byte) 0xE5, (byte) 0x39, (byte) 0xA2, (byte) 0x94, (byte) 0x0F, (byte) 0xF8,
(byte) 0x63, (byte) 0x55, (byte) 0xCE, (byte) 0x20, (byte) 0xBB, (byte) 0x8D, (byte) 0x16, (byte) 0xE1,
(byte) 0x7A, (byte) 0x4C, (byte) 0xD7, (byte) 0x6F, (byte) 0xF4, (byte) 0xC2, (byte) 0x59, (byte) 0xAE,
(byte) 0x35, (byte) 0x03, (byte) 0x98, (byte) 0x76, (byte) 0xED, (byte) 0xDB, (byte) 0x40, (byte) 0xB7,
(byte) 0x2C, (byte) 0x1A, (byte) 0x81, (byte) 0x5D, (byte) 0xC6, (byte) 0xF0, (byte) 0x6B, (byte) 0x9C,
(byte) 0x07, (byte) 0x31, (byte) 0xAA, (byte) 0x44, (byte) 0xDF, (byte) 0xE9, (byte) 0x72, (byte) 0x85,
(byte) 0x1E, (byte) 0x28, (byte) 0xB3, (byte) 0xC3, (byte) 0x58, (byte) 0x6E, (byte) 0xF5, (byte) 0x02,
(byte) 0x99, (byte) 0xAF, (byte) 0x34, (byte) 0xDA, (byte) 0x41, (byte) 0x77, (byte) 0xEC, (byte) 0x1B,
(byte) 0x80, (byte) 0xB6, (byte) 0x2D, (byte) 0xF1, (byte) 0x6A, (byte) 0x5C, (byte) 0xC7, (byte) 0x30,
(byte) 0xAB, (byte) 0x9D, (byte) 0x06, (byte) 0xE8, (byte) 0x73, (byte) 0x45, (byte) 0xDE, (byte) 0x29,
(byte) 0xB2, (byte) 0x84, (byte) 0x1F, (byte) 0xA7, (byte) 0x3C, (byte) 0x0A, (byte) 0x91, (byte) 0x66,
(byte) 0xFD, (byte) 0xCB, (byte) 0x50, (byte) 0xBE, (byte) 0x25, (byte) 0x13, (byte) 0x88, (byte) 0x7F,
(byte) 0xE4, (byte) 0xD2, (byte) 0x49, (byte) 0x95, (byte) 0x0E, (byte) 0x38, (byte) 0xA3, (byte) 0x54,
(byte) 0xCF, (byte) 0xF9, (byte) 0x62, (byte) 0x8C, (byte) 0x17, (byte) 0x21, (byte) 0xBA, (byte) 0x4D,
(byte) 0xD6, (byte) 0xE0, (byte) 0x7B };
/**
* CRC calculation with tab operations
*
* @param data The byte to crc check
* @param crc_init The current crc result or another start value
* @return The crc result
*/
public static byte crc8_tab(byte data, byte crcInit) {
short ci = (short) (crcInit & 0xFF);
byte crc = (byte) (CRC_TAB_8_VALUE[ci] ^ (data & 0xFF));
return crc;
}
/**
* Convert to unsigned int
*
* @param v
* @return
*/
public static int uint(byte v) {
return v & 0xFF;
}
/**
* CRC calculation
*
* @param data The byte to crc check
* @param crcInit The current crc result or another start value
* @param poly The polynom
* @return The crc result
*/
public static byte crc8(byte data, byte crcInit, byte poly) {
byte crc;
byte polynom;
int i;
crc = crcInit;
for (i = 0; i < 8; i++) {
if ((uint(crc) & 0x80) != 0) {
polynom = poly;
} else {
polynom = (byte) 0;
}
crc = (byte) ((uint(crc) & ~0x80) << 1);
if ((uint(data) & 0x80) != 0) {
crc = (byte) (uint(crc) | 1);
}
crc = (byte) (uint(crc) ^ uint(polynom));
data = (byte) (uint(data) << 1);
}
return crc;
}
/**
* Convert eBus Type BIT
*
* @param data
* @param bit
* @return The decoded value
*/
public static boolean decodeBit(byte data, short bit) {
return (data >> bit & 0x1) == 1;
}
/**
* Expands ebus-data bytes 0xAA and 0xA9 from byte data. All other bytes
* are unchanged.
*
* @param data The received byte buffer
* @param pos The position to check
* @return The new value or the unchanged byte
*/
static private byte decodeEBusData(byte[] data, int pos) {
if (data[pos - 1] == (byte) 0xA9) {
if (data[pos] == (byte) 0x00) {
return (byte) 0xA9;
} else if (data[pos] == (byte) 0x01) {
return (byte) 0xAA;
}
}
return data[pos];
}
/**
* Encodes the eBUS data to replace sync and 0x9A bytes
*
* @param data
* @return
*/
public static byte[] encodeEBusData(byte[] data) {
final ByteArrayOutputStream byteBuffer = new ByteArrayOutputStream();
try {
for (byte b : data) {
if (b == (byte) 0xAA) {
byteBuffer.write(new byte[] { (byte) 0xA9, (byte) 0x01 });
} else if (b == (byte) 0xA9) {
byteBuffer.write(new byte[] { (byte) 0xA9, (byte) 0x00 });
} else {
byteBuffer.write(b);
}
}
} catch (IOException e) {
logger.error("io error", e);
}
return byteBuffer.toByteArray();
}
/**
* Check if the address is a valid master address.
*
* @param address
* @return
*/
public static boolean isMasterAddress(byte address) {
byte addr = (byte) (address >> 4);
byte prio = (byte) (address & (byte) 0x0F);
// check if it's a broadcast
if (address != (byte) 0xFE) {
if (addr == (byte) 0x00 || addr == (byte) 0x01 || addr == (byte) 0x03 || addr == (byte) 0x07
|| addr == (byte) 0x15) {
if (prio == (byte) 0x00 || prio == (byte) 0x01 || prio == (byte) 0x03 || prio == (byte) 0x07
|| prio == (byte) 0x15) {
return true;
}
}
}
return false;
}
/**
* Processes a EBus received byte array, crc check, expand special bytes.
*
* @param data The received raw byte data
* @return A valid object or null if the data was incorrect
*/
static public EBusTelegram processEBusData(byte[] data) {
try {
ByteBuffer buffer = ByteBuffer.allocate(data.length + 10);
buffer.put(data, 0, 5);
int nnPos = 4;
int nn = data[nnPos];
byte uc_crc = 0;
// crc-check first bytes
for (int i = 0; i < 5; i++) {
byte b = data[i];
uc_crc = crc8_tab(b, uc_crc);
}
// process sender data and find data end pos.
// (may moved because expanded bytes)
if (nn > 0) {
nnPos = 0;
for (int i = 5; i < data.length; i++) {
byte b = data[i];
uc_crc = crc8_tab(b, uc_crc);
if (b != (byte) 0xA9) {
nnPos++;
buffer.put(decodeEBusData(data, i));
}
if (nnPos == nn) {
nnPos = i;
break;
}
}
}
int crcPos = nnPos + 1;
byte crc = data[crcPos];
if (uc_crc == (byte) 0xAA && crc == (byte) 0xA9 && data[++crcPos] == (byte) 0x01) {
// replace with original value
crc = (byte) 0xAA;
}
// check calculted crc with received crc
if (crc != uc_crc) {
if (logger.isTraceEnabled()) {
logger.trace("EBus telegram master-crc invalid, skip data! Data: {}");
logger.trace(" -> DATA: {}", toHexDumpString(data));
}
// invalid, return null
return null;
}
buffer.put(crc);
buffer.put(data[crcPos + 1]);
if (data[crcPos + 1] == EBusTelegram.SYN) {
if (data[1] == EBusTelegram.BROADCAST_ADDRESS) {
// Broadcast Telegram, end
return new EBusTelegram(buffer);
}
if (logger.isTraceEnabled()) {
logger.debug("No answer from slave, skip data!");
logger.trace(" -> DATA: {}", toHexDumpString(data));
}
return null;
}
if ((data[crcPos + 1] == EBusTelegram.ACK_OK || data[crcPos + 1] == EBusTelegram.ACK_FAIL)
&& data[crcPos + 2] == EBusTelegram.SYN) {
// Master-Master Telegram, add ack value and end
buffer.put(data[crcPos + 2]);
return new EBusTelegram(buffer);
}
if (data[crcPos + 1] != EBusTelegram.ACK_OK && data[crcPos + 1] != EBusTelegram.ACK_FAIL) {
// Unexpected value on this position
if (logger.isTraceEnabled()) {
logger.trace("Unexpect ACK value in eBUS telegram, skip data!");
logger.trace(" -> DATA: {}", toHexDumpString(data));
}
return null;
}
// ok, read slave answer
int nn2Pos = crcPos + 2;
byte nn2 = data[nn2Pos];
buffer.put(nn2);
uc_crc = crc8_tab(nn2, (byte) 0);
// process answer data and find data end pos.
// (may moved because expanded bytes)
if (nn2 > 0) {
for (int i = nn2Pos + 1; i < data.length - 3; i++) {
byte b = data[i];
uc_crc = crc8_tab(b, uc_crc);
if (b != (byte) 0xA9) {
buffer.put(decodeEBusData(data, i));
}
}
}
crc = data[data.length - 3];
buffer.put(data, data.length - 3, 3);
if (uc_crc == (byte) 0xAA && crc == (byte) 0xA9 && data[++crcPos] == (byte) 0x01) {
// replace with original value
crc = (byte) 0xAA;
}
// check calculted crc with received crc
if (crc != uc_crc) {
if (logger.isTraceEnabled()) {
logger.trace("eBUS telegram answer-crc invalid, skip data!");
logger.trace(" -> DATA: {}", toHexDumpString(data));
}
return null;
}
// return valid telegram
return new EBusTelegram(buffer);
} catch (IndexOutOfBoundsException e) {
// can happen, no problem
return null;
}
}
/**
* Generates a hex string representative of byte data
*
* @param data The source
* @return The hex string
*/
static public String toHexDumpString(byte data) {
return String.format("%02X", (0xFF & data));
}
/**
* Converts a hex string to byte array
*
* @param hexDumpString
* @return
*/
static public byte[] toByteArray(String hexDumpString) {
return DatatypeConverter.parseHexBinary(hexDumpString.replaceAll(" ", ""));
}
/**
* FIXME: Badly programmed
*
* @param hexDumpString
* @return
*/
static public byte toByte(String hexDumpString) {
return toByteArray(hexDumpString)[0];
}
/**
* Generates a string hex dump from a byte array
*
* @param data The source
* @return The StringBuilder with hex dump
*/
static public StringBuilder toHexDumpString(byte[] data) {
StringBuilder sb = new StringBuilder();
if (data != null && data.length > 0) {
for (int i = 0; i < data.length; i++) {
byte c = data[i];
if (i > 0) {
sb.append(' ');
}
sb.append(toHexDumpString(c));
}
}
return sb;
}
/**
* Generates a string hex dump from a ByteBuffer
*
* @param data The source
* @return The StringBuilder with hex dump
*/
static public StringBuilder toHexDumpString(ByteBuffer data) {
StringBuilder sb = new StringBuilder();
for (int i = 0; i < data.position(); i++) {
byte c = data.get(i);
if (i > 0) {
sb.append(' ');
}
sb.append(toHexDumpString(c));
}
return sb;
}
}