#include "application_layer.h" #include "transport_layer.h" #include "cemi_frame.h" #include "association_table_object.h" #include "apdu.h" #include "bau.h" #include "string.h" #include "bits.h" #include ApplicationLayer::ApplicationLayer(AssociationTableObject& assocTable, BusAccessUnit& bau): _assocTable(assocTable), _bau(bau) { } void ApplicationLayer::transportLayer(TransportLayer& layer) { _transportLayer = &layer; } #pragma region TL Callbacks void ApplicationLayer::dataGroupIndication(HopCountType hopType, Priority priority, uint16_t tsap, APDU& apdu) { uint16_t entries = _assocTable.entryCount(); uint8_t len = apdu.length(); uint8_t dataArray[len]; uint8_t* data = dataArray; memcpy(data, apdu.data(), len); if (len == 1) { //less than six bit are encoded in first byte *data &= 0x3f; } else { data += 1; len -= 1; } for (uint16_t i = 0; i < entries; i++) { uint16_t entry = _assocTable[i]; if (highByte(entry) == tsap) { uint16_t asap = lowByte(entry); switch (apdu.type()) { case GroupValueRead: _bau.groupValueReadIndication(asap, priority, hopType); break; case GroupValueResponse: _bau.groupValueReadAppLayerConfirm(asap, priority, hopType, data, len); break; case GroupValueWrite: _bau.groupValueWriteIndication(asap, priority, hopType, data, len); } } } } void ApplicationLayer::dataGroupConfirm(AckType ack, HopCountType hopType, Priority priority, uint16_t tsap, APDU& apdu, bool status) { switch (apdu.type()) { case GroupValueRead: _bau.groupValueReadLocalConfirm(ack, _savedAsapReadRequest, priority, hopType, status); break; case GroupValueResponse: _bau.groupValueReadResponseConfirm(ack, _savedAsapResponse, priority, hopType, apdu.data(), apdu.length() - 1, status); break; case GroupValueWrite: _bau.groupValueWriteLocalConfirm(ack, _savedAsapWriteRequest, priority, hopType, apdu.data(), apdu.length() - 1, status); break; } } void ApplicationLayer::dataBroadcastIndication(HopCountType hopType, Priority priority, uint16_t source, APDU& apdu) { uint8_t* data = apdu.data(); switch (apdu.type()) { case IndividualAddressWrite: { uint16_t newAddress; popWord(newAddress, data + 1); _bau.individualAddressWriteIndication(hopType, newAddress); break; } case IndividualAddressRead: _bau.individualAddressReadIndication(hopType); break; case IndividualAddressResponse: _bau.individualAddressReadAppLayerConfirm(hopType, apdu.frame().sourceAddress()); break; case IndividualAddressSerialNumberRead: _bau.individualAddressSerialNumberReadIndication(hopType, data + 1); break; case IndividualAddressSerialNumberResponse: { uint16_t domainAddress; popWord(domainAddress, data + 7); _bau.individualAddressSerialNumberReadAppLayerConfirm(hopType, data + 1, apdu.frame().sourceAddress(), domainAddress); break; } case IndividualAddressSerialNumberWrite: { uint16_t newAddress; popWord(newAddress, data + 7); _bau.individualAddressSerialNumberWriteIndication(hopType, data + 1, newAddress); break; } } } void ApplicationLayer::dataBroadcastConfirm(AckType ack, HopCountType hopType, Priority priority, APDU& apdu, bool status) { uint8_t* data = apdu.data(); switch (apdu.type()) { case IndividualAddressWrite: { uint16_t newAddress; popWord(newAddress, data + 1); _bau.individualAddressWriteLocalConfirm(ack, hopType, newAddress, status); break; } case IndividualAddressRead: _bau.individualAddressReadLocalConfirm(ack, hopType, status); break; case IndividualAddressResponse: _bau.individualAddressReadResponseConfirm(ack, hopType, status); break; case IndividualAddressSerialNumberRead: _bau.individualAddressSerialNumberReadLocalConfirm(ack, hopType, data + 1, status); break; case IndividualAddressSerialNumberResponse: { uint16_t domainAddress; popWord(domainAddress, data + 7); _bau.individualAddressSerialNumberReadResponseConfirm(ack, hopType, data + 1, domainAddress, status); break; } case IndividualAddressSerialNumberWrite: { uint16_t newAddress; popWord(newAddress, data + 7); _bau.individualAddressSerialNumberWriteLocalConfirm(ack, hopType, data + 1, newAddress, status); break; } } } void ApplicationLayer::dataSystemBroadcastIndication(HopCountType hopType, Priority priority, uint16_t source, APDU& apdu) { } void ApplicationLayer::dataSystemBroadcastConfirm(HopCountType hopType, Priority priority, APDU& apdu, bool status) { } void ApplicationLayer::dataIndividualIndication(HopCountType hopType, Priority priority, uint16_t tsap, APDU& apdu) { individualIndication(hopType, priority, tsap, apdu); } void ApplicationLayer::dataIndividualConfirm(AckType ack, HopCountType hopType, Priority priority, uint16_t tsap, APDU& apdu, bool status) { individualConfirm(ack, hopType, priority, tsap, apdu, status); } void ApplicationLayer::connectIndication(uint16_t tsap) { _connectedTsap = tsap; } void ApplicationLayer::connectConfirm(uint16_t destination, uint16_t tsap, bool status) { } void ApplicationLayer::disconnectIndication(uint16_t tsap) { _connectedTsap = -1; } void ApplicationLayer::disconnectConfirm(Priority priority, uint16_t tsap, bool status) { } void ApplicationLayer::dataConnectedIndication(Priority priority, uint16_t tsap, APDU& apdu) { individualIndication(NetworkLayerParameter, priority, tsap, apdu); } void ApplicationLayer::dataConnectedConfirm(uint16_t tsap) { } #pragma endregion void ApplicationLayer::groupValueReadRequest(AckType ack, uint16_t asap, Priority priority, HopCountType hopType) { _savedAsapReadRequest = asap; CemiFrame frame(1); APDU& apdu = frame.apdu(); apdu.type(GroupValueRead); int32_t value = _assocTable.translateAsap(asap); if (value < 0) return; // there is no tsap in association table for this asap uint16_t tsap = (uint16_t)value; // first to bus then to itself _transportLayer->dataGroupRequest(ack, hopType, priority, tsap, apdu); dataGroupIndication(hopType, priority, tsap, apdu); } void ApplicationLayer::groupValueReadResponse(AckType ack, uint16_t asap, Priority priority, HopCountType hopType, uint8_t * data, uint8_t dataLength) { _savedAsapResponse = asap; groupValueSend(GroupValueResponse, ack, asap, priority, hopType, data, dataLength); } void ApplicationLayer::groupValueWriteRequest(AckType ack, uint16_t asap, Priority priority, HopCountType hopType, uint8_t * data, uint8_t dataLength) { _savedAsapWriteRequest = asap; groupValueSend(GroupValueWrite, ack, asap, priority, hopType, data, dataLength); } void ApplicationLayer::individualAddressWriteRequest(AckType ack, HopCountType hopType, uint16_t newaddress) { CemiFrame frame(3); APDU& apdu = frame.apdu(); apdu.type(IndividualAddressWrite); uint8_t* apduData = apdu.data(); pushWord(newaddress, apduData + 1); _transportLayer->dataBroadcastRequest(ack, hopType, SystemPriority, apdu); } void ApplicationLayer::individualAddressReadRequest(AckType ack, HopCountType hopType) { CemiFrame frame(1); APDU& apdu = frame.apdu(); apdu.type(IndividualAddressRead); _transportLayer->dataBroadcastRequest(ack, hopType, SystemPriority, apdu); } void ApplicationLayer::individualAddressReadResponse(AckType ack, HopCountType hopType) { CemiFrame frame(1); APDU& apdu = frame.apdu(); apdu.type(IndividualAddressResponse); _transportLayer->dataBroadcastRequest(ack, hopType, SystemPriority, apdu); } void ApplicationLayer::individualAddressSerialNumberReadRequest(AckType ack, HopCountType hopType, uint8_t * serialNumber) { CemiFrame frame(7); APDU& apdu = frame.apdu(); apdu.type(IndividualAddressSerialNumberRead); uint8_t* data = apdu.data() + 1; memcpy(data, serialNumber, 6); _transportLayer->dataBroadcastRequest(ack, hopType, SystemPriority, apdu); } void ApplicationLayer::individualAddressSerialNumberReadResponse(AckType ack, HopCountType hopType, uint8_t * serialNumber, uint16_t domainAddress) { CemiFrame frame(7); APDU& apdu = frame.apdu(); apdu.type(IndividualAddressSerialNumberResponse); uint8_t* data = apdu.data() + 1; memcpy(data, serialNumber, 6); data += 6; pushWord(domainAddress, data); _transportLayer->dataBroadcastRequest(ack, hopType, SystemPriority, apdu); } void ApplicationLayer::individualAddressSerialNumberWriteRequest(AckType ack, HopCountType hopType, uint8_t * serialNumber, uint16_t newaddress) { CemiFrame frame(13); APDU& apdu = frame.apdu(); apdu.type(IndividualAddressSerialNumberWrite); uint8_t* data = apdu.data() + 1; memcpy(data, serialNumber, 6); data += 6; pushWord(newaddress, data); _transportLayer->dataBroadcastRequest(ack, hopType, SystemPriority, apdu); } void ApplicationLayer::deviceDescriptorReadRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t descriptorType) { CemiFrame frame(1); APDU& apdu = frame.apdu(); apdu.type(DeviceDescriptorRead); uint8_t* data = apdu.data(); *data |= (descriptorType & 0x3f); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::deviceDescriptorReadResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t descriptorType, uint8_t* deviceDescriptor) { uint8_t length = 0; switch (descriptorType) { case 0: length = 3; break; case 2: length = 14; break; default: length = 1; descriptorType = 0x3f; break; } CemiFrame frame(length); APDU& apdu = frame.apdu(); apdu.type(DeviceDescriptorResponse); uint8_t* data = apdu.data(); *data |= (descriptorType & 0x3f); if (length > 1) memcpy(data + 1, deviceDescriptor, length - 1); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::restartRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap) { CemiFrame frame(1); APDU& apdu = frame.apdu(); apdu.type(Restart); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::propertyValueReadRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t objectIndex, uint8_t propertyId, uint8_t numberOfElements, uint16_t startIndex) { CemiFrame frame(5); APDU& apdu = frame.apdu(); apdu.type(PropertyValueRead); uint8_t* data = apdu.data(); data += 1; data = pushByte(objectIndex, data); data = pushByte(propertyId, data); pushWord(startIndex & 0xfff, data); *data &= ((numberOfElements & 0xf) << 4); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::propertyValueReadResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t objectIndex, uint8_t propertyId, uint8_t numberOfElements, uint16_t startIndex, uint8_t* data, uint8_t length) { propertyDataSend(PropertyValueResponse, ack, priority, hopType, asap, objectIndex, propertyId, numberOfElements, startIndex, data, length); } void ApplicationLayer::propertyValueWriteRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t objectIndex, uint8_t propertyId, uint8_t numberOfElements, uint16_t startIndex, uint8_t * data, uint8_t length) { propertyDataSend(PropertyValueWrite, ack, priority, hopType, asap, objectIndex, propertyId, numberOfElements, startIndex, data, length); } void ApplicationLayer::propertyDescriptionReadRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t objectIndex, uint8_t propertyId, uint8_t propertyIndex) { CemiFrame frame(4); APDU& apdu = frame.apdu(); apdu.type(PropertyDescriptionRead); uint8_t* data = apdu.data(); data[1] = objectIndex; data[2] = propertyId; data[3] = propertyIndex; individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::propertyDescriptionReadResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t objectIndex, uint8_t propertyId, uint8_t propertyIndex, bool writeEnable, uint8_t type, uint16_t maxNumberOfElements, uint8_t access) { CemiFrame frame(8); APDU& apdu = frame.apdu(); apdu.type(PropertyDescriptionResponse); uint8_t* data = apdu.data(); data[1] = objectIndex; data[2] = propertyId; data[3] = propertyIndex; if (writeEnable) data[4] |= 0x80; data[4] |= (type & 0x3f); pushWord(maxNumberOfElements & 0xfff, data + 5); data[7] = access; individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::memoryReadRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint16_t memoryAddress) { CemiFrame frame(3); APDU& apdu = frame.apdu(); apdu.type(MemoryRead); uint8_t* data = apdu.data(); *data |= (number & 0x3f); pushWord(memoryAddress, data + 1); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::memoryReadResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint16_t memoryAddress, uint8_t * memoryData) { memorySend(MemoryResponse, ack, priority, hopType, asap, number, memoryAddress, memoryData); } void ApplicationLayer::memoryWriteRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint16_t memoryAddress, uint8_t * data) { memorySend(MemoryWrite, ack, priority, hopType, asap, number, memoryAddress, data); } void ApplicationLayer::userMemoryReadRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint32_t memoryAddress) { CemiFrame frame(4); APDU& apdu = frame.apdu(); apdu.type(UserMemoryRead); uint8_t* data = apdu.data(); data[1] |= (number & 0xf); data[1] |= ((memoryAddress >> 12) & 0xf0); pushWord(memoryAddress & 0xff, data + 2); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::userMemoryReadResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint32_t memoryAddress, uint8_t * memoryData) { userMemorySend(UserMemoryResponse, ack, priority, hopType, asap, number, memoryAddress, memoryData); } void ApplicationLayer::userMemoryWriteRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint32_t memoryAddress, uint8_t * memoryData) { userMemorySend(UserMemoryWrite, ack, priority, hopType, asap, number, memoryAddress, memoryData); } void ApplicationLayer::userManufacturerInfoReadRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap) { CemiFrame frame(1); APDU& apdu = frame.apdu(); apdu.type(UserManufacturerInfoRead); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::userManufacturerInfoReadResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t* info) { CemiFrame frame(4); APDU& apdu = frame.apdu(); apdu.type(UserMemoryRead); uint8_t* data = apdu.data(); memcpy(data + 1, info, 3); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::authorizeRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint32_t key) { CemiFrame frame(6); APDU& apdu = frame.apdu(); apdu.type(AuthorizeRequest); uint8_t* data = apdu.data(); pushInt(key, data + 2); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::authorizeResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t level) { CemiFrame frame(2); APDU& apdu = frame.apdu(); apdu.type(AuthorizeResponse); uint8_t* data = apdu.data(); data[1] = level; individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::keyWriteRequest(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t level, uint32_t key) { CemiFrame frame(6); APDU& apdu = frame.apdu(); apdu.type(KeyWrite); uint8_t* data = apdu.data(); data[1] = level; pushInt(key, data + 2); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::keyWriteResponse(AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t level) { CemiFrame frame(6); APDU& apdu = frame.apdu(); apdu.type(KeyResponse); uint8_t* data = apdu.data(); data[1] = level; individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::propertyDataSend(ApduType type, AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t objectIndex, uint8_t propertyId, uint8_t numberOfElements, uint16_t startIndex, uint8_t* data, uint8_t length) { CemiFrame frame(5 + length); APDU& apdu = frame.apdu(); apdu.type(type); uint8_t* apduData = apdu.data(); apduData += 1; apduData = pushByte(objectIndex, apduData); apduData = pushByte(propertyId, apduData); pushWord(startIndex & 0xfff, apduData); *apduData |= ((numberOfElements & 0xf) << 4); apduData += 2; if (length > 0) memcpy(apduData, data, length); if (asap == _connectedTsap) _transportLayer->dataConnectedRequest(asap, priority, apdu); else _transportLayer->dataIndividualRequest(ack, hopType, priority, asap, apdu); } void ApplicationLayer::groupValueSend(ApduType type, AckType ack, uint16_t asap, Priority priority, HopCountType hopType, uint8_t* data, uint8_t& dataLength) { CemiFrame frame(dataLength + 1); APDU& apdu = frame.apdu(); apdu.type(type); uint8_t* apdudata = apdu.data(); if (dataLength == 0) { // data size is six bit or less. So store int first byte *apdudata &= ~0x3f; *apdudata |= (*data & 0x3f); } else { memcpy(apdudata + 1, data, dataLength); } // no need to check if there is a tsap. This is a response, so the read got trough uint16_t tsap = (uint16_t)_assocTable.translateAsap(asap); _transportLayer->dataGroupRequest(ack, hopType, priority, tsap, apdu); dataGroupIndication(hopType, priority, tsap, apdu); } void ApplicationLayer::memorySend(ApduType type, AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint16_t memoryAddress, uint8_t * memoryData) { CemiFrame frame(3 + number); APDU& apdu = frame.apdu(); apdu.type(type); uint8_t* data = apdu.data(); *data |= (number & 0x3f); pushWord(memoryAddress, data + 1); if (number > 0) memcpy(data + 3, memoryData, number); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::userMemorySend(ApduType type, AckType ack, Priority priority, HopCountType hopType, uint16_t asap, uint8_t number, uint32_t memoryAddress, uint8_t * memoryData) { CemiFrame frame(4 + number); APDU& apdu = frame.apdu(); apdu.type(type); uint8_t* data = apdu.data(); data[1] |= (number & 0xf); data[1] |= ((memoryAddress >> 12) & 0xf0); pushWord(memoryAddress & 0xffff, data + 2); if (number > 0) memcpy(data + 4, memoryData, number); individualSend(ack, hopType, priority, asap, apdu); } void ApplicationLayer::individualIndication(HopCountType hopType, Priority priority, uint16_t tsap, APDU & apdu) { uint8_t* data = apdu.data(); switch (apdu.type()) { case DeviceDescriptorRead: _bau.deviceDescriptorReadIndication(priority, hopType, tsap, *data & 0x3f); break; case DeviceDescriptorResponse: _bau.deviceDescriptorReadAppLayerConfirm(priority, hopType, tsap, *data & 0x3f, data + 1); break; case Restart: if ((*data & 0x3f) == 0) _bau.restartRequestIndication(priority, hopType, tsap); break; case PropertyValueRead: { uint16_t startIndex; popWord(startIndex, data + 3); startIndex &= 0xfff; _bau.propertyValueReadIndication(priority, hopType, tsap, data[1], data[2], data[3] >> 4, startIndex); break; } case PropertyValueResponse: { uint16_t startIndex; popWord(startIndex, data + 3); startIndex &= 0xfff; _bau.propertyValueReadAppLayerConfirm(priority, hopType, tsap, data[1], data[2], data[3] >> 4, startIndex, data + 5, apdu.length() - 5); break; } case PropertyValueWrite: { uint16_t startIndex; popWord(startIndex, data + 3); startIndex &= 0xfff; _bau.propertyValueWriteIndication(priority, hopType, tsap, data[1], data[2], data[3] >> 4, startIndex, data + 5, apdu.length() - 5); break; } case PropertyDescriptionRead: _bau.propertyDescriptionReadIndication(priority, hopType, tsap, data[1], data[2], data[3]); break; case PropertyDescriptionResponse: _bau.propertyDescriptionReadAppLayerConfirm(priority, hopType, tsap, data[1], data[2], data[3], (data[4] & 0x80) > 0, data[4] & 0x3f, getWord(data + 5) & 0xfff, data[7]); break; case MemoryRead: _bau.memoryReadIndication(priority, hopType, tsap, data[0] & 0x3f, getWord(data + 1)); break; case MemoryResponse: _bau.memoryReadAppLayerConfirm(priority, hopType, tsap, data[0] & 0x3f, getWord(data + 1), data + 3); break; case MemoryWrite: _bau.memoryWriteIndication(priority, hopType, tsap, data[0] & 0x3f, getWord(data + 1), data + 3); break; case UserMemoryRead: { uint32_t address = ((data[1] & 0xf0) << 12) + (data[2] << 8) + data[3]; _bau.userMemoryReadIndication(priority, hopType, tsap, data[1] & 0xf, address); break; } case UserMemoryResponse: { uint32_t address = ((data[1] & 0xf0) << 12) + (data[2] << 8) + data[3]; _bau.userMemoryReadAppLayerConfirm(priority, hopType, tsap, data[1] & 0xf, address, data + 4); break; } case UserMemoryWrite: { uint32_t address = ((data[1] & 0xf0) << 12) + (data[2] << 8) + data[3]; _bau.userMemoryWriteIndication(priority, hopType, tsap, data[1] & 0xf, address, data + 4); break; } case UserManufacturerInfoRead: _bau.userManufacturerInfoIndication(priority, hopType, tsap); break; case UserManufacturerInfoResponse: _bau.userManufacturerInfoAppLayerConfirm(priority, hopType, tsap, data + 1); break; case AuthorizeRequest: _bau.authorizeIndication(priority, hopType, tsap, getInt(data + 2)); break; case AuthorizeResponse: _bau.authorizeAppLayerConfirm(priority, hopType, tsap, data[1]); break; case KeyWrite: _bau.keyWriteIndication(priority, hopType, tsap, data[1], getInt(data + 2)); break; case KeyResponse: _bau.keyWriteAppLayerConfirm(priority, hopType, tsap, data[1]); break; } } void ApplicationLayer::individualConfirm(AckType ack, HopCountType hopType, Priority priority, uint16_t tsap, APDU & apdu, bool status) { uint8_t* data = apdu.data(); switch (apdu.type()) { case DeviceDescriptorRead: _bau.deviceDescriptorReadLocalConfirm(ack, priority, hopType, tsap, *data & 0x3f, status); break; case DeviceDescriptorResponse: _bau.deviceDescriptorReadResponseConfirm(ack, priority, hopType, tsap, *data & 0x3f, data + 1, status); break; case Restart: _bau.restartRequestLocalConfirm(ack, priority, hopType, tsap, status); break; case PropertyValueRead: { uint16_t startIndex; popWord(startIndex, data + 3); startIndex &= 0xfff; _bau.propertyValueReadLocalConfirm(ack, priority, hopType, tsap, data[1], data[2], data[3] >> 4, startIndex, status); break; } case PropertyValueResponse: { uint16_t startIndex; popWord(startIndex, data + 3); startIndex &= 0xfff; _bau.propertyValueReadResponseConfirm(ack, priority, hopType, tsap, data[1], data[2], data[3] >> 4, startIndex, data + 5, apdu.length() - 5, status); break; } case PropertyValueWrite: { uint16_t startIndex; popWord(startIndex, data + 3); startIndex &= 0xfff; _bau.propertyValueWriteLocalConfirm(ack, priority, hopType, tsap, data[1], data[2], data[3] >> 4, startIndex, data + 5, apdu.length() - 5, status); break; } case PropertyDescriptionRead: _bau.propertyDescriptionReadLocalConfirm(ack, priority, hopType, tsap, data[1], data[2], data[3], status); break; case PropertyDescriptionResponse: _bau.propertyDescriptionReadResponseConfirm(ack, priority, hopType, tsap, data[1], data[2], data[3], (data[4] & 0x80) > 0, data[4] & 0x3f, getWord(data + 5) & 0xfff, data[7], status); break; case MemoryRead: _bau.memoryReadLocalConfirm(ack, priority, hopType, tsap, data[0] & 0x3f, getWord(data + 1), status); break; case MemoryResponse: _bau.memoryReadResponseConfirm(ack, priority, hopType, tsap, data[0] & 0x3f, getWord(data + 1), data + 3, status); break; case MemoryWrite: _bau.memoryWriteLocalConfirm(ack, priority, hopType, tsap, data[0] & 0x3f, getWord(data + 1), data + 3, status); break; case UserMemoryRead: { uint32_t address = ((data[1] & 0xf0) << 12) + (data[2] << 8) + data[3]; _bau.memoryReadLocalConfirm(ack, priority, hopType, tsap, data[1] & 0xf, address, status); break; } case UserMemoryResponse: { uint32_t address = ((data[1] & 0xf0) << 12) + (data[2] << 8) + data[3]; _bau.memoryReadResponseConfirm(ack, priority, hopType, tsap, data[1] & 0xf, address, data + 4, status); break; } case UserMemoryWrite: { uint32_t address = ((data[1] & 0xf0) << 12) + (data[2] << 8) + data[3]; _bau.memoryWriteLocalConfirm(ack, priority, hopType, tsap, data[1] & 0xf, address, data + 4, status); break; } case UserManufacturerInfoRead: _bau.userManufacturerInfoLocalConfirm(ack, priority, hopType, tsap, status); break; case UserManufacturerInfoResponse: _bau.userManufacturerInfoResponseConfirm(ack, priority, hopType, tsap, data + 1, status); break; case AuthorizeRequest: _bau.authorizeLocalConfirm(ack, priority, hopType, tsap, getInt(data + 2), status); break; case AuthorizeResponse: _bau.authorizeResponseConfirm(ack, priority, hopType, tsap, data[1], status); break; case KeyWrite: _bau.keyWriteLocalConfirm(ack, priority, hopType, tsap, data[1], getInt(data + 2), status); break; case KeyResponse: _bau.keyWriteResponseConfirm(ack, priority, hopType, tsap, data[1], status); break; } } void ApplicationLayer::individualSend(AckType ack, HopCountType hopType, Priority priority, uint16_t asap, APDU& apdu) { if (asap == _connectedTsap) _transportLayer->dataConnectedRequest(asap, priority, apdu); else _transportLayer->dataIndividualRequest(ack, hopType, priority, asap, apdu); }