| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- /**
- * esp-knx-ip library for KNX/IP communication on an ESP8266
- * Author: Nico Weichbrodt <envy>
- * License: MIT
- */
- #include "esp-knx-ip.h"
- /**
- * Send functions
- */
- void ESPKNXIP::send(address_t const &receiver, knx_command_type_t ct, uint8_t data_len, uint8_t *data)
- {
- if (receiver.value == 0)
- return;
- #if SEND_CHECKSUM
- uint32_t len = 6 + 2 + 8 + data_len + 1; // knx_pkt + cemi_msg + cemi_service + data + checksum
- #else
- uint32_t len = 6 + 2 + 8 + data_len; // knx_pkt + cemi_msg + cemi_service + data
- #endif
- DEBUG_PRINT(F("Creating packet with len "));
- DEBUG_PRINTLN(len)
- uint8_t buf[len];
- knx_ip_pkt_t *knx_pkt = (knx_ip_pkt_t *)buf;
- knx_pkt->header_len = 0x06;
- knx_pkt->protocol_version = 0x10;
- knx_pkt->service_type = __ntohs(KNX_ST_ROUTING_INDICATION);
- knx_pkt->total_len.len = __ntohs(len);
- cemi_msg_t *cemi_msg = (cemi_msg_t *)knx_pkt->pkt_data;
- cemi_msg->message_code = KNX_MT_L_DATA_IND;
- cemi_msg->additional_info_len = 0;
- cemi_service_t *cemi_data = &cemi_msg->data.service_information;
- cemi_data->control_1.bits.confirm = 0;
- //cemi_data->control_1.bits.ack = 1;
- cemi_data->control_1.bits.ack = 0; // ask for ACK? 0-no 1-yes
- cemi_data->control_1.bits.priority = B11;
- cemi_data->control_1.bits.system_broadcast = 0x01;
- cemi_data->control_1.bits.repeat = 0x01; // 0 = repeated telegram, 1 = not repeated telegram
- cemi_data->control_1.bits.reserved = 0;
- cemi_data->control_1.bits.frame_type = 0x01;
- cemi_data->control_2.bits.extended_frame_format = 0x00;
- cemi_data->control_2.bits.hop_count = 0x06;
- cemi_data->control_2.bits.dest_addr_type = 0x01;
- cemi_data->source = physaddr;
- cemi_data->destination = receiver;
- //cemi_data->destination.bytes.high = (area << 3) | line;
- //cemi_data->destination.bytes.low = member;
- cemi_data->data_len = data_len;
- cemi_data->pci.apci = (ct & 0x0C) >> 2;
- //cemi_data->pci.apci = KNX_COT_NCD_ACK;
- cemi_data->pci.tpci_seq_number = 0x00;
- cemi_data->pci.tpci_comm_type = KNX_COT_UDP; // Type of communication: DATA PACKAGE or CONTROL DATA
- //cemi_data->pci.tpci_comm_type = KNX_COT_NCD; // Type of communication: DATA PACKAGE or CONTROL DATA
- memcpy(cemi_data->data, data, data_len);
- //cemi_data->data[0] = (cemi_data->data[0] & 0x3F) | ((KNX_COT_NCD_ACK & 0x03) << 6);
- cemi_data->data[0] = (cemi_data->data[0] & 0x3F) | ((ct & 0x03) << 6);
- #if SEND_CHECKSUM
- // Calculate checksum, which is just XOR of all bytes
- uint8_t cs = buf[0] ^ buf[1];
- for (uint32_t i = 2; i < len - 1; ++i)
- {
- cs ^= buf[i];
- }
- buf[len - 1] = cs;
- #endif
- #ifdef ESP_KNX_DEBUG
- DEBUG_PRINT(F("Sending packet:"));
- for (int i = 0; i < len; ++i)
- {
- DEBUG_PRINT(F(" 0x"));
- DEBUG_PRINT(buf[i], 16);
- }
- DEBUG_PRINTLN(F(""));
- #endif
- udp.beginPacketMulticast(MULTICAST_IP, MULTICAST_PORT, WiFi.localIP());
- udp.write(buf, len);
- udp.endPacket();
- }
- void ESPKNXIP::send_1bit(address_t const &receiver, knx_command_type_t ct, uint8_t bit)
- {
- uint8_t buf[] = {(uint8_t)(bit & 0b00000001)};
- send(receiver, ct, 1, buf);
- }
- void ESPKNXIP::send_2bit(address_t const &receiver, knx_command_type_t ct, uint8_t twobit)
- {
- uint8_t buf[] = {(uint8_t)(twobit & 0b00000011)};
- send(receiver, ct, 1, buf);
- }
- void ESPKNXIP::send_4bit(address_t const &receiver, knx_command_type_t ct, uint8_t fourbit)
- {
- uint8_t buf[] = {(uint8_t)(fourbit & 0b00001111)};
- send(receiver, ct, 1, buf);
- }
- void ESPKNXIP::send_1byte_int(address_t const &receiver, knx_command_type_t ct, int8_t val)
- {
- uint8_t buf[] = {0x00, (uint8_t)val};
- send(receiver, ct, 2, buf);
- }
- void ESPKNXIP::send_1byte_uint(address_t const &receiver, knx_command_type_t ct, uint8_t val)
- {
- uint8_t buf[] = {0x00, val};
- send(receiver, ct, 2, buf);
- }
- void ESPKNXIP::send_2byte_int(address_t const &receiver, knx_command_type_t ct, int16_t val)
- {
- uint8_t buf[] = {0x00, (uint8_t)(val >> 8), (uint8_t)(val & 0x00FF)};
- send(receiver, ct, 3, buf);
- }
- void ESPKNXIP::send_2byte_uint(address_t const &receiver, knx_command_type_t ct, uint16_t val)
- {
- uint8_t buf[] = {0x00, (uint8_t)(val >> 8), (uint8_t)(val & 0x00FF)};
- send(receiver, ct, 3, buf);
- }
- void ESPKNXIP::send_2byte_float(address_t const &receiver, knx_command_type_t ct, float val)
- {
- float v = val * 100.0f;
- int e = 0;
- for (; v < -2048.0f; v /= 2)
- ++e;
- for (; v > 2047.0f; v /= 2)
- ++e;
- long m = (long)round(v) & 0x7FF;
- short msb = (short) (e << 3 | m >> 8);
- if (val < 0.0f)
- msb |= 0x80;
- uint8_t buf[] = {0x00, (uint8_t)msb, (uint8_t)m};
- send(receiver, ct, 3, buf);
- }
- void ESPKNXIP::send_3byte_time(address_t const &receiver, knx_command_type_t ct, uint8_t weekday, uint8_t hours, uint8_t minutes, uint8_t seconds)
- {
- weekday <<= 5;
- uint8_t buf[] = {0x00, (uint8_t)(((weekday << 5) & 0xE0) | (hours & 0x1F)), (uint8_t)(minutes & 0x3F), (uint8_t)(seconds & 0x3F)};
- send(receiver, ct, 4, buf);
- }
- void ESPKNXIP::send_3byte_date(address_t const &receiver, knx_command_type_t ct, uint8_t day, uint8_t month, uint8_t year)
- {
- uint8_t buf[] = {0x00, (uint8_t)(day & 0x1F), (uint8_t)(month & 0x0F), year};
- send(receiver, ct, 4, buf);
- }
- void ESPKNXIP::send_3byte_color(address_t const &receiver, knx_command_type_t ct, uint8_t red, uint8_t green, uint8_t blue)
- {
- uint8_t buf[] = {0x00, red, green, blue};
- send(receiver, ct, 4, buf);
- }
- void ESPKNXIP::send_4byte_int(address_t const &receiver, knx_command_type_t ct, int32_t val)
- {
- uint8_t buf[] = {0x00,
- (uint8_t)((val & 0xFF000000) >> 24),
- (uint8_t)((val & 0x00FF0000) >> 16),
- (uint8_t)((val & 0x0000FF00) >> 8),
- (uint8_t)((val & 0x000000FF) >> 0)};
- send(receiver, ct, 5, buf);
- }
- void ESPKNXIP::send_4byte_uint(address_t const &receiver, knx_command_type_t ct, uint32_t val)
- {
- uint8_t buf[] = {0x00,
- (uint8_t)((val & 0xFF000000) >> 24),
- (uint8_t)((val & 0x00FF0000) >> 16),
- (uint8_t)((val & 0x0000FF00) >> 8),
- (uint8_t)((val & 0x000000FF) >> 0)};
- send(receiver, ct, 5, buf);
- }
- void ESPKNXIP::send_4byte_float(address_t const &receiver, knx_command_type_t ct, float val)
- {
- uint8_t buf[] = {0x00, ((uint8_t *)&val)[3], ((uint8_t *)&val)[2], ((uint8_t *)&val)[1], ((uint8_t *)&val)[0]};
- send(receiver, ct, 5, buf);
- }
|