mirror of
https://github.com/thelsing/knx.git
synced 2024-12-18 19:08:18 +01:00
459 lines
10 KiB
C++
459 lines
10 KiB
C++
#include "tpuart_data_link_layer.h"
|
|
|
|
#include "bits.h"
|
|
#include "platform.h"
|
|
#include "device_object.h"
|
|
#include "address_table_object.h"
|
|
#include "cemi_frame.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
// NCN5120
|
|
//#define NCN5120
|
|
|
|
// services Host -> Controller :
|
|
// internal commands, device specific
|
|
#define U_RESET_REQ 0x01
|
|
#define U_STATE_REQ 0x02
|
|
#define U_SET_BUSY_REQ 0x03
|
|
#define U_QUIT_BUSY_REQ 0x04
|
|
#define U_BUSMON_REQ 0x05
|
|
#define U_SET_ADDRESS_REQ 0xF1 // different on TP-UART
|
|
#define U_SET_REPETITION_REQ 0xF2
|
|
#define U_L_DATA_OFFSET_REQ 0x08 //-0x0C
|
|
#define U_SYSTEM_STATE 0x0D
|
|
#define U_STOP_MODE_REQ 0x0E
|
|
#define U_EXIT_STOP_MODE_REQ 0x0F
|
|
#define U_ACK_REQ 0x10 //-0x17
|
|
#define U_CONFIGURE_REQ 0x18
|
|
#define U_INT_REG_WR_REQ 0x28
|
|
#define U_INT_REG_RD_REQ 0x38
|
|
#define U_POLLING_STATE_REQ 0xE0
|
|
|
|
//knx transmit data commands
|
|
#define U_L_DATA_START_CONT_REQ 0x80 //-0xBF
|
|
#define U_L_DATA_END_REQ 0x47 //-0x7F
|
|
|
|
//serices to host controller
|
|
|
|
// DLL services (device is transparent)
|
|
#define L_DATA_STANDARD_IND 0x90
|
|
#define L_DATA_EXTENDED_IND 0x10
|
|
#define L_DATA_MASK 0xD3
|
|
#define L_POLL_DATA_IND 0xF0
|
|
|
|
// acknowledge services (device is transparent in bus monitor mode)
|
|
#define L_ACKN_IND 0x00
|
|
#define L_ACKN_MASK 0x33
|
|
#define L_DATA_CON 0x0B
|
|
#define L_DATA_CON_MASK 0x7F
|
|
#define SUCCESS 0x80
|
|
|
|
// control services, device specific
|
|
#define U_RESET_IND 0x03
|
|
#define U_STATE_IND 0x07
|
|
#define SLAVE_COLLISION 0x80
|
|
#define RECEIVE_ERROR 0x40
|
|
#define TRANSMIT_ERROR 0x20
|
|
#define PROTOCOL_ERROR 0x10
|
|
#define TEMPERATURE_WARNING 0x08
|
|
#define U_FRAME_STATE_IND 0x13
|
|
#define U_FRAME_STATE_MASK 0x17
|
|
#define PARITY_BIT_ERROR 0x80
|
|
#define CHECKSUM_LENGTH_ERROR 0x40
|
|
#define TIMING_ERROR 0x20
|
|
#define U_CONFIGURE_IND 0x01
|
|
#define U_CONFIGURE_MASK 0x83
|
|
#define AUTO_ACKNOWLEDGE 0x20
|
|
#define AUTO_POLLING 0x10
|
|
#define CRC_CCITT 0x80
|
|
#define FRAME_END_WITH_MARKER 0x40
|
|
#define U_FRAME_END_IND 0xCB
|
|
#define U_STOP_MODE_IND 0x2B
|
|
#define U_SYSTEM_STAT_IND 0x4B
|
|
|
|
void TpUartDataLinkLayer::resetChip()
|
|
{
|
|
uint8_t cmd = U_RESET_REQ;
|
|
_platform.writeUart(cmd);
|
|
while (true)
|
|
{
|
|
int resp = _platform.readUart();
|
|
if (resp == U_RESET_IND)
|
|
break;
|
|
}
|
|
}
|
|
|
|
void TpUartDataLinkLayer::stopChip()
|
|
{
|
|
#ifdef NCN5120
|
|
uint8_t cmd = U_STOP_MODE_REQ;
|
|
_platform.writeUart(cmd);
|
|
while (true)
|
|
{
|
|
int resp = _platform.readUart();
|
|
if (resp == U_STOP_MODE_IND)
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void printHex(const char* suffix, const uint8_t *data, size_t length);
|
|
|
|
|
|
TpUartDataLinkLayer::TpUartDataLinkLayer(DeviceObject& devObj, AddressTableObject& addrTab,
|
|
NetworkLayer& layer, Platform& platform) : DataLinkLayer(devObj, addrTab, layer, platform)
|
|
{
|
|
}
|
|
|
|
|
|
bool TpUartDataLinkLayer::sendFrame(CemiFrame& frame)
|
|
{
|
|
if (!_enabled)
|
|
return false;
|
|
|
|
uint16_t length = frame.telegramLengthtTP();
|
|
uint8_t* buffer = new uint8_t[length];
|
|
frame.fillTelegramTP(buffer);
|
|
|
|
_sendBuffer = buffer;
|
|
_sendResult = false;
|
|
_sendBufferLength = length;
|
|
sendBytes(buffer, length);
|
|
|
|
while (_sendBuffer != 0)
|
|
loop();
|
|
|
|
delete[] buffer;
|
|
return _sendResult;
|
|
}
|
|
|
|
void TpUartDataLinkLayer::loop()
|
|
{
|
|
if (!_enabled)
|
|
return;
|
|
|
|
if (_platform.uartAvailable() == 0)
|
|
return;
|
|
|
|
uint8_t firstByte = _platform.readUart();
|
|
|
|
if (checkDataInd(firstByte))
|
|
return;
|
|
|
|
if (checkDataCon(firstByte))
|
|
return;
|
|
|
|
if (checkPollDataInd(firstByte))
|
|
return;
|
|
|
|
if (checkAckNackInd(firstByte))
|
|
return;
|
|
|
|
if (checkResetInd(firstByte))
|
|
return;
|
|
|
|
if (checkStateInd(firstByte))
|
|
return;
|
|
|
|
if (checkFrameStateInd(firstByte))
|
|
return;
|
|
|
|
if (checkConfigureInd(firstByte))
|
|
return;
|
|
|
|
if (checkFrameEndInd(firstByte))
|
|
return;
|
|
|
|
if (checkStopModeInd(firstByte))
|
|
return;
|
|
|
|
if (checkSystemStatInd(firstByte))
|
|
return;
|
|
|
|
handleUnexpected(firstByte);
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkDataInd(uint8_t firstByte)
|
|
{
|
|
const size_t bufferSize = 512;
|
|
|
|
uint8_t tmp = firstByte & L_DATA_MASK;
|
|
if (tmp != L_DATA_STANDARD_IND && tmp != L_DATA_EXTENDED_IND)
|
|
return false;
|
|
|
|
int len = 0;
|
|
uint8_t buffer[bufferSize];
|
|
buffer[0] = firstByte;
|
|
|
|
uint8_t payloadLength = 0;
|
|
if (tmp == L_DATA_STANDARD_IND)
|
|
{
|
|
//convert to extended frame format
|
|
_platform.readBytesUart(buffer + 2, 5);
|
|
payloadLength = buffer[6] & 0xF;
|
|
_platform.readBytesUart(buffer + 7, payloadLength + 2); //+1 for TCPI +1 for CRC
|
|
printHex("->", buffer, 1);
|
|
printHex("->", buffer + 2, 5);
|
|
printHex("->", buffer + 7, payloadLength + 3);
|
|
buffer[1] = buffer[6] & 0xF0;
|
|
buffer[6] = payloadLength;
|
|
}
|
|
else
|
|
{
|
|
//extended frame
|
|
_platform.readBytesUart(buffer + 1, 6);
|
|
payloadLength = buffer[6];
|
|
_platform.readBytesUart(buffer + 7, payloadLength + 2); //+1 for TCPI +1 for CRC
|
|
}
|
|
len = payloadLength + 9;
|
|
|
|
printHex("=>", buffer, len);
|
|
CemiFrame frame(buffer, len);
|
|
|
|
if ((frame.addressType() == InduvidualAddress && _deviceObject.induvidualAddress() == frame.destinationAddress())
|
|
|| (frame.addressType() == GroupAddress && _groupAddressTable.contains(frame.destinationAddress())))
|
|
{
|
|
//send ack.
|
|
_platform.writeUart(U_ACK_REQ + 1);
|
|
}
|
|
else
|
|
{
|
|
// send not addressed
|
|
_platform.writeUart(U_ACK_REQ);
|
|
}
|
|
|
|
|
|
const uint8_t queueLength = 5;
|
|
static uint8_t buffers[queueLength][bufferSize];
|
|
static uint16_t bufferLengths[queueLength];
|
|
|
|
if (_sendBuffer != 0)
|
|
{
|
|
// we are trying to send a telegram queue received telegrams until we get our sent telegram
|
|
|
|
if (len == _sendBufferLength && memcmp(_sendBuffer, buffer, len) == 0)
|
|
{
|
|
//we got the send telegramm back next byte is L_Data.con byte
|
|
uint8_t confirm = _platform.readUart();
|
|
confirm &= L_DATA_CON_MASK;
|
|
_sendResult = (confirm > 0);
|
|
_sendBuffer = 0;
|
|
_sendBufferLength = 0;
|
|
|
|
return true;
|
|
}
|
|
|
|
// queue telegram, if we get more the queueLength before send succeeds
|
|
// ignore the telegram
|
|
for (int i = 0; i < queueLength; i++)
|
|
{
|
|
if (bufferLengths[i] != 0)
|
|
continue;
|
|
|
|
bufferLengths[i] = len;
|
|
memcpy(&buffers[i][0], buffer, len);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// process all previously queued telegramms first
|
|
for (int i = 0; i < queueLength; i++)
|
|
{
|
|
if (bufferLengths[i] == 0)
|
|
break;
|
|
|
|
frameBytesReceived(&buffers[i][0], bufferLengths[i]);
|
|
bufferLengths[i] = 0;
|
|
}
|
|
|
|
frameBytesReceived(buffer, len);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void TpUartDataLinkLayer::frameBytesReceived(uint8_t* buffer, uint16_t length)
|
|
{
|
|
CemiFrame frame(buffer, length);
|
|
|
|
frameRecieved(frame);
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkDataCon(uint8_t firstByte)
|
|
{
|
|
uint8_t tmp = firstByte & L_DATA_CON_MASK;
|
|
if (tmp != L_DATA_CON)
|
|
return false;
|
|
|
|
if (_sendBuffer == 0)
|
|
{
|
|
_println("got unexpected L_DATA_CON");
|
|
return true;
|
|
}
|
|
|
|
_sendResult = (firstByte & SUCCESS) > 0;
|
|
_sendBuffer = 0;
|
|
_sendBufferLength = 0;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkPollDataInd(uint8_t firstByte)
|
|
{
|
|
if (firstByte != L_POLL_DATA_IND)
|
|
return false;
|
|
|
|
// not sure if this can happen
|
|
_println("got L_POLL_DATA_IND");
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkAckNackInd(uint8_t firstByte)
|
|
{
|
|
uint8_t tmp = firstByte & L_ACKN_MASK;
|
|
if (tmp != L_ACKN_IND)
|
|
return false;
|
|
|
|
// this can only happen in bus monitor mode
|
|
_println("got L_ACKN_IND");
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkResetInd(uint8_t firstByte)
|
|
{
|
|
if (firstByte != U_RESET_IND)
|
|
return false;
|
|
|
|
_println("got U_RESET_IND");
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkStateInd(uint8_t firstByte)
|
|
{
|
|
uint8_t tmp = firstByte & U_STATE_IND;
|
|
if (tmp != U_STATE_IND)
|
|
return false;
|
|
|
|
_print("got U_STATE_IND: 0x");
|
|
_print(firstByte, HEX);
|
|
_println();
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkFrameStateInd(uint8_t firstByte)
|
|
{
|
|
uint8_t tmp = firstByte & U_FRAME_STATE_MASK;
|
|
if (tmp != U_FRAME_STATE_IND)
|
|
return false;
|
|
|
|
_print("got U_FRAME_STATE_IND: 0x");
|
|
_print(firstByte, HEX);
|
|
_println();
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkConfigureInd(uint8_t firstByte)
|
|
{
|
|
uint8_t tmp = firstByte & U_CONFIGURE_MASK;
|
|
if (tmp != U_CONFIGURE_IND)
|
|
return false;
|
|
|
|
_print("got U_CONFIGURE_IND: 0x");
|
|
_print(firstByte, HEX);
|
|
_println();
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkFrameEndInd(uint8_t firstByte)
|
|
{
|
|
if (firstByte != U_FRAME_END_IND)
|
|
return false;
|
|
|
|
_println("got U_FRAME_END_IND");
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkStopModeInd(uint8_t firstByte)
|
|
{
|
|
if (firstByte != U_STOP_MODE_IND)
|
|
return false;
|
|
|
|
_println("got U_STOP_MODE_IND");
|
|
return true;
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::checkSystemStatInd(uint8_t firstByte)
|
|
{
|
|
if (firstByte != U_SYSTEM_STAT_IND)
|
|
return false;
|
|
|
|
_print("got U_SYSTEM_STAT_IND: 0x");
|
|
while (true)
|
|
{
|
|
int tmp = _platform.readUart();
|
|
if (tmp < 0)
|
|
continue;
|
|
|
|
_print(tmp, HEX);
|
|
break;
|
|
}
|
|
_println();
|
|
return true;
|
|
}
|
|
|
|
void TpUartDataLinkLayer::handleUnexpected(uint8_t firstByte)
|
|
{
|
|
_print("got UNEXPECTED: 0x");
|
|
_print(firstByte, HEX);
|
|
_println();
|
|
}
|
|
|
|
void TpUartDataLinkLayer::enabled(bool value)
|
|
{
|
|
if (value && !_enabled)
|
|
{
|
|
_platform.setupUart();
|
|
|
|
resetChip();
|
|
_enabled = true;
|
|
return;
|
|
}
|
|
|
|
if (!value && _enabled)
|
|
{
|
|
_enabled = false;
|
|
stopChip();
|
|
_platform.closeUart();
|
|
return;
|
|
}
|
|
}
|
|
|
|
bool TpUartDataLinkLayer::enabled() const
|
|
{
|
|
return _enabled;
|
|
}
|
|
|
|
|
|
void TpUartDataLinkLayer::sendBytes(uint8_t* bytes, uint16_t length)
|
|
{
|
|
uint8_t cmd[2];
|
|
|
|
for (int i = 0; i < length; i++)
|
|
{
|
|
uint8_t idx = length / 64;
|
|
cmd[0] = U_L_DATA_OFFSET_REQ | idx;
|
|
_platform.writeUart(cmd, 1);
|
|
|
|
if (i != length - 1)
|
|
cmd[0] = U_L_DATA_START_CONT_REQ | i;
|
|
else
|
|
cmd[0] = U_L_DATA_END_REQ;
|
|
|
|
cmd[1] = bytes[i];
|
|
|
|
_platform.writeUart(cmd, 2);
|
|
}
|
|
} |