2017-12-30 20:12:01 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2017 Shaun Feakes - All rights reserved
|
|
|
|
*
|
|
|
|
* You may use redistribute and/or modify this code under the terms of
|
|
|
|
* the GNU General Public License version 2 as published by the
|
|
|
|
* Free Software Foundation. For the terms of this license,
|
|
|
|
* see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* You are free to use this software under the terms of the GNU General
|
|
|
|
* Public License, but WITHOUT ANY WARRANTY; without even the implied
|
|
|
|
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
* See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* https://github.com/sfeakes/aqualinkd
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <termios.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
2018-03-14 17:29:52 +00:00
|
|
|
#include <string.h>
|
2018-08-02 00:47:55 +00:00
|
|
|
#include <sys/ioctl.h>
|
2017-12-30 20:12:01 +00:00
|
|
|
|
|
|
|
#include "aq_serial.h"
|
|
|
|
#include "utils.h"
|
|
|
|
|
|
|
|
//#define BLOCKING_MODE
|
|
|
|
|
|
|
|
static struct termios _oldtio;
|
|
|
|
|
2018-04-18 00:33:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void log_packet(char *init_str, unsigned char* packet, int length)
|
2017-12-30 20:12:01 +00:00
|
|
|
{
|
2018-07-29 21:01:30 +00:00
|
|
|
if ( getLogLevel() < LOG_DEBUG_SERIAL) {
|
2018-08-02 00:47:55 +00:00
|
|
|
//logMessage(LOG_INFO, "Send '0x%02hhx'|'0x%02hhx' to controller\n", packet[5] ,packet[6]);
|
2018-04-18 00:33:17 +00:00
|
|
|
return;
|
2018-07-29 21:01:30 +00:00
|
|
|
}
|
2018-04-18 00:33:17 +00:00
|
|
|
|
|
|
|
int cnt;
|
|
|
|
int i;
|
|
|
|
char buff[MAXLEN];
|
|
|
|
|
|
|
|
cnt = sprintf(buff, "%s", init_str);
|
|
|
|
cnt += sprintf(buff+cnt, " | HEX: ");
|
|
|
|
//printHex(packet_buffer, packet_length);
|
|
|
|
for (i=0;i<length;i++)
|
|
|
|
cnt += sprintf(buff+cnt, "0x%02hhx|",packet[i]);
|
|
|
|
|
|
|
|
cnt += sprintf(buff+cnt, "\n");
|
2018-06-06 00:18:44 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, buff);
|
2018-04-18 00:33:17 +00:00
|
|
|
/*
|
2017-12-30 20:12:01 +00:00
|
|
|
int i;
|
|
|
|
char temp_string[64];
|
|
|
|
char message_buffer[MAXLEN];
|
|
|
|
|
2018-04-18 00:33:17 +00:00
|
|
|
sprintf(temp_string, "Send 0x%02hhx|", packet[0]);
|
2017-12-30 20:12:01 +00:00
|
|
|
strcpy(message_buffer, temp_string);
|
|
|
|
|
|
|
|
for (i = 1; i < length; i++) {
|
2018-04-18 00:33:17 +00:00
|
|
|
sprintf(temp_string, "0x%02hhx|", packet[i]);
|
2017-12-30 20:12:01 +00:00
|
|
|
strcat(message_buffer, temp_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
strcat(message_buffer, "\n");
|
|
|
|
logMessage(LOG_DEBUG, message_buffer);
|
2018-04-18 00:33:17 +00:00
|
|
|
*/
|
2017-12-30 20:12:01 +00:00
|
|
|
}
|
|
|
|
|
2018-07-29 21:01:30 +00:00
|
|
|
const char* get_packet_type(unsigned char* packet , int length)
|
2018-03-10 20:35:22 +00:00
|
|
|
{
|
2018-03-10 23:07:49 +00:00
|
|
|
static char buf[15];
|
|
|
|
|
2018-03-10 20:35:22 +00:00
|
|
|
if (length <= 0 )
|
|
|
|
return "";
|
2017-12-30 20:12:01 +00:00
|
|
|
|
2018-03-10 20:35:22 +00:00
|
|
|
switch (packet[PKT_CMD]) {
|
|
|
|
case CMD_ACK:
|
|
|
|
return "Ack";
|
|
|
|
break;
|
|
|
|
case CMD_STATUS:
|
|
|
|
return "Status";
|
|
|
|
break;
|
|
|
|
case CMD_MSG:
|
|
|
|
case CMD_MSG_LONG:
|
|
|
|
return "Message";
|
|
|
|
break;
|
|
|
|
case CMD_PROBE:
|
|
|
|
return "Probe";
|
|
|
|
break;
|
2018-06-06 00:18:44 +00:00
|
|
|
case CMD_GETID:
|
|
|
|
return "GetID";
|
|
|
|
break;
|
|
|
|
case CMD_PERCENT:
|
|
|
|
return "AR %%";
|
|
|
|
break;
|
|
|
|
case CMD_PPM:
|
|
|
|
return "AR PPM";
|
|
|
|
break;
|
2018-03-10 20:35:22 +00:00
|
|
|
default:
|
2018-04-18 00:33:17 +00:00
|
|
|
sprintf(buf, "Unknown '0x%02hhx'", packet[PKT_CMD]);
|
2018-03-10 23:07:49 +00:00
|
|
|
return buf;
|
2018-03-10 20:35:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-12-30 20:12:01 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
Open and Initialize the serial communications port to the Aqualink RS8 device.
|
|
|
|
Arg is tty or port designation string
|
|
|
|
returns the file descriptor
|
|
|
|
*/
|
|
|
|
int init_serial_port(char* tty)
|
|
|
|
{
|
|
|
|
long BAUD = B9600;
|
|
|
|
long DATABITS = CS8;
|
|
|
|
long STOPBITS = 0;
|
|
|
|
long PARITYON = 0;
|
|
|
|
long PARITY = 0;
|
|
|
|
|
|
|
|
struct termios newtio; //place for old and new port settings for serial port
|
|
|
|
|
|
|
|
//int fd = open(tty, O_RDWR | O_NOCTTY | O_NONBLOCK);
|
|
|
|
int fd = open(tty, O_RDWR | O_NOCTTY | O_NONBLOCK | O_NDELAY);
|
|
|
|
if (fd < 0) {
|
|
|
|
logMessage(LOG_ERR, "Unable to open port: %s\n", tty);
|
|
|
|
return -1;
|
|
|
|
}
|
2018-03-07 01:02:47 +00:00
|
|
|
|
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Openeded serial port %s\n",tty);
|
2017-12-30 20:12:01 +00:00
|
|
|
|
2018-08-02 00:47:55 +00:00
|
|
|
// Need to change this to flock, but that depends on good exit.
|
|
|
|
if ( ioctl(fd, TIOCEXCL) != 0) {
|
|
|
|
displayLastSystemError("Locking serial port.");
|
|
|
|
}
|
|
|
|
|
2017-12-30 20:12:01 +00:00
|
|
|
#ifdef BLOCKING_MODE
|
|
|
|
fcntl(fd, F_SETFL, 0);
|
|
|
|
newtio.c_cc[VMIN]= 1;
|
|
|
|
newtio.c_cc[VTIME]= 0;
|
2018-03-07 01:02:47 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Set serial port %s to blocking mode\n",tty);
|
2017-12-30 20:12:01 +00:00
|
|
|
#else
|
|
|
|
int flags = fcntl(fd, F_GETFL, 0);
|
|
|
|
fcntl(fd, F_SETFL, flags | O_NONBLOCK | O_NDELAY);
|
|
|
|
newtio.c_cc[VMIN]= 0;
|
|
|
|
newtio.c_cc[VTIME]= 1;
|
2018-03-07 01:02:47 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Set serial port %s to non blocking mode\n",tty);
|
2017-12-30 20:12:01 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
tcgetattr(fd, &_oldtio); // save current port settings
|
|
|
|
// set new port settings for canonical input processing
|
|
|
|
newtio.c_cflag = BAUD | DATABITS | STOPBITS | PARITYON | PARITY | CLOCAL | CREAD;
|
|
|
|
newtio.c_iflag = IGNPAR;
|
|
|
|
newtio.c_lflag = 0; // ICANON;
|
|
|
|
newtio.c_oflag = 0;
|
|
|
|
|
|
|
|
tcflush(fd, TCIFLUSH);
|
|
|
|
tcsetattr(fd, TCSANOW, &newtio);
|
|
|
|
|
2018-03-07 01:02:47 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Set serial port %s io attributes\n",tty);
|
|
|
|
|
2017-12-30 20:12:01 +00:00
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* close tty port */
|
|
|
|
void close_serial_port(int fd)
|
|
|
|
{
|
|
|
|
tcsetattr(fd, TCSANOW, &_oldtio);
|
|
|
|
close(fd);
|
2018-03-07 01:02:47 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Closed serial port\n");
|
2017-12-30 20:12:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Generate and return checksum of packet.
|
|
|
|
int generate_checksum(unsigned char* packet, int length)
|
|
|
|
{
|
|
|
|
int i, sum, n;
|
|
|
|
|
|
|
|
n = length - 3;
|
|
|
|
sum = 0;
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
sum += (int) packet[i];
|
|
|
|
return(sum & 0x0ff);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Send an ack packet to the Aqualink RS8 master device.
|
|
|
|
// fd: the file descriptor of the serial port connected to the device
|
|
|
|
// command: the command byte to send to the master device, NUL if no command
|
|
|
|
//
|
|
|
|
// NUL = '\x00'
|
|
|
|
// DLE = '\x10'
|
|
|
|
// STX = '\x02'
|
|
|
|
// ETX = '\x03'
|
|
|
|
//
|
|
|
|
// masterAddr = '\x00' # address of Aqualink controller
|
|
|
|
//
|
|
|
|
//msg = DLE+STX+dest+cmd+args
|
|
|
|
//msg = msg+self.checksum(msg)+DLE+ETX
|
|
|
|
// DLE+STX+DEST+CMD+ARGS+CHECKSUM+DLE+ETX
|
|
|
|
|
|
|
|
|
|
|
|
void print_hex(char *pk, int length)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
for (i=0;i<length;i++)
|
|
|
|
{
|
|
|
|
printf("0x%02hhx|",pk[i]);
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_cmd()
|
|
|
|
{
|
|
|
|
const int length = 11;
|
|
|
|
unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, CMD_ACK, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
//send_cmd(fd, CMD_ACK, command);
|
|
|
|
|
|
|
|
print_hex((char *)ackPacket, length);
|
|
|
|
|
|
|
|
ackPacket[7] = generate_checksum(ackPacket, length-1);
|
|
|
|
print_hex((char *)ackPacket, length);
|
|
|
|
|
|
|
|
ackPacket[6] = 0x02;
|
|
|
|
ackPacket[7] = generate_checksum(ackPacket, length-1);
|
|
|
|
print_hex((char *)ackPacket, length);
|
|
|
|
}
|
|
|
|
|
2018-04-18 00:33:17 +00:00
|
|
|
void send_test_cmd(int fd, unsigned char destination, unsigned char b1, unsigned char b2, unsigned char b3)
|
|
|
|
{
|
|
|
|
const int length = 11;
|
|
|
|
unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, CMD_ACK, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
//unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, NUL, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
|
|
|
|
// Update the packet and checksum if command argument is not NUL.
|
|
|
|
ackPacket[3] = destination;
|
|
|
|
ackPacket[4] = b1;
|
|
|
|
ackPacket[5] = b2;
|
|
|
|
ackPacket[6] = b3;
|
|
|
|
ackPacket[7] = generate_checksum(ackPacket, length-1);
|
|
|
|
|
2018-06-06 00:18:44 +00:00
|
|
|
#ifdef BLOCKING_MODE
|
|
|
|
write(fd, ackPacket, length);
|
|
|
|
#else
|
|
|
|
int nwrite, i;
|
|
|
|
for (i=0; i<length; i += nwrite) {
|
|
|
|
nwrite = write(fd, ackPacket + i, length - i);
|
|
|
|
if (nwrite < 0)
|
|
|
|
logMessage(LOG_ERR, "write to serial port failed\n");
|
|
|
|
}
|
|
|
|
//logMessage(LOG_DEBUG_SERIAL, "Send %d bytes to serial\n",length);
|
|
|
|
//tcdrain(fd);
|
|
|
|
//logMessage(LOG_DEBUG, "Send '0x%02hhx' to '0x%02hhx'\n", command, destination);
|
|
|
|
#endif
|
|
|
|
|
2018-04-18 00:33:17 +00:00
|
|
|
log_packet("Sent ", ackPacket, length);
|
|
|
|
|
2018-06-06 00:18:44 +00:00
|
|
|
}
|
|
|
|
void send_command(int fd, unsigned char destination, unsigned char b1, unsigned char b2, unsigned char b3)
|
|
|
|
{
|
|
|
|
const int length = 11;
|
|
|
|
unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, CMD_ACK, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
//unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, NUL, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
|
|
|
|
// Update the packet and checksum if command argument is not NUL.
|
|
|
|
ackPacket[3] = destination;
|
|
|
|
ackPacket[4] = b1;
|
|
|
|
ackPacket[5] = b2;
|
|
|
|
ackPacket[6] = b3;
|
|
|
|
ackPacket[7] = generate_checksum(ackPacket, length-1);
|
|
|
|
|
2018-04-18 00:33:17 +00:00
|
|
|
#ifdef BLOCKING_MODE
|
|
|
|
write(fd, ackPacket, length);
|
|
|
|
#else
|
|
|
|
int nwrite, i;
|
|
|
|
for (i=0; i<length; i += nwrite) {
|
|
|
|
nwrite = write(fd, ackPacket + i, length - i);
|
|
|
|
if (nwrite < 0)
|
|
|
|
logMessage(LOG_ERR, "write to serial port failed\n");
|
|
|
|
}
|
|
|
|
//logMessage(LOG_DEBUG_SERIAL, "Send %d bytes to serial\n",length);
|
|
|
|
//tcdrain(fd);
|
|
|
|
//logMessage(LOG_DEBUG, "Send '0x%02hhx' to '0x%02hhx'\n", command, destination);
|
|
|
|
#endif
|
2018-06-06 00:18:44 +00:00
|
|
|
|
|
|
|
if ( getLogLevel() >= LOG_DEBUG_SERIAL) {
|
|
|
|
char buf[30];
|
2018-07-29 21:01:30 +00:00
|
|
|
sprintf(buf, "Sent %8.8s ", get_packet_type(ackPacket+1 , length));
|
2018-06-06 00:18:44 +00:00
|
|
|
log_packet(buf, ackPacket, length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void send_messaged(int fd, unsigned char destination, char *message)
|
|
|
|
{
|
|
|
|
const int length = 24;
|
|
|
|
int i;
|
|
|
|
//unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, CMD_ACK, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
unsigned char msgPacket[] = { DLE,STX,DEV_MASTER,CMD_MSG,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,NUL,DLE,ETX };
|
|
|
|
//unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, NUL, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
|
|
|
|
// Update the packet and checksum if command argument is not NUL.
|
|
|
|
msgPacket[2] = destination;
|
|
|
|
for (i=0; i < strlen(message) && i < AQ_MSGLEN; i++)
|
|
|
|
msgPacket[4+i] = message[i];
|
|
|
|
|
|
|
|
msgPacket[length-3] = generate_checksum(msgPacket, length-1);
|
|
|
|
|
|
|
|
#ifdef BLOCKING_MODE
|
|
|
|
write(fd, msgPacket, length);
|
|
|
|
#else
|
|
|
|
int nwrite;
|
|
|
|
for (i=0; i<length; i += nwrite) {
|
|
|
|
nwrite = write(fd, msgPacket + i, length - i);
|
|
|
|
if (nwrite < 0)
|
|
|
|
logMessage(LOG_ERR, "write to serial port failed\n");
|
|
|
|
}
|
|
|
|
//logMessage(LOG_DEBUG_SERIAL, "Send %d bytes to serial\n",length);
|
|
|
|
//tcdrain(fd);
|
|
|
|
//logMessage(LOG_DEBUG, "Send '0x%02hhx' to '0x%02hhx'\n", command, destination);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
log_packet("Sent ", msgPacket, length);
|
2018-04-18 00:33:17 +00:00
|
|
|
}
|
2017-12-30 20:12:01 +00:00
|
|
|
|
2018-08-02 00:47:55 +00:00
|
|
|
void _send_ack(int fd, unsigned char ack_type, unsigned char command);
|
|
|
|
|
|
|
|
void send_extended_ack(int fd, unsigned char ack_type, unsigned char command)
|
|
|
|
{
|
|
|
|
// ack_typ should only be ACK_NORMAL, ACK_SCREEN_BUSY, ACK_SCREEN_BUSY_DISPLAY
|
|
|
|
_send_ack(fd, ack_type, command);
|
|
|
|
}
|
|
|
|
|
2017-12-30 20:12:01 +00:00
|
|
|
void send_ack(int fd, unsigned char command)
|
2018-08-02 00:47:55 +00:00
|
|
|
{
|
|
|
|
_send_ack(fd, ACK_NORMAL, command);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _send_ack(int fd, unsigned char ack_type, unsigned char command)
|
2017-12-30 20:12:01 +00:00
|
|
|
{
|
|
|
|
const int length = 11;
|
2018-08-02 00:47:55 +00:00
|
|
|
// Default null ack with checksum generated, don't mess with it, just over right
|
|
|
|
unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, CMD_ACK, NUL, NUL, 0x13, DLE, ETX, NUL };
|
2017-12-30 20:12:01 +00:00
|
|
|
//unsigned char ackPacket[] = { NUL, DLE, STX, DEV_MASTER, NUL, NUL, NUL, 0x13, DLE, ETX, NUL };
|
|
|
|
|
|
|
|
// Update the packet and checksum if command argument is not NUL.
|
2018-08-02 00:47:55 +00:00
|
|
|
if(command != NUL || ack_type != NUL) {
|
2018-07-31 23:21:03 +00:00
|
|
|
//ackPacket[5] = 0x00 normal, 0x03 some pause, 0x01 some pause ending (0x01 = Screen Busy (also return from logn message))
|
2018-08-02 00:47:55 +00:00
|
|
|
ackPacket[5] = ack_type;
|
2017-12-30 20:12:01 +00:00
|
|
|
ackPacket[6] = command;
|
|
|
|
ackPacket[7] = generate_checksum(ackPacket, length-1);
|
|
|
|
|
|
|
|
// NULL out the command byte if it is the same. Difference implies that
|
|
|
|
// a new command has come in, and is awaiting processing.
|
|
|
|
/*
|
|
|
|
if(aqualink_cmd == command) {
|
|
|
|
aqualink_cmd = NUL;
|
|
|
|
}
|
|
|
|
*/
|
2018-06-06 00:18:44 +00:00
|
|
|
|
2017-12-30 20:12:01 +00:00
|
|
|
// In debug mode, log the packet to the private log file.
|
|
|
|
//log_packet(ackPacket, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send the packet to the master device.
|
|
|
|
//write(fd, ackPacket, length);
|
2018-04-18 00:33:17 +00:00
|
|
|
//logMessage(LOG_DEBUG, "Send '0x%02hhx' to controller\n", command);
|
2018-03-14 17:29:52 +00:00
|
|
|
|
2017-12-30 20:12:01 +00:00
|
|
|
#ifdef BLOCKING_MODE
|
|
|
|
write(fd, ackPacket, length);
|
|
|
|
#else
|
|
|
|
int nwrite, i;
|
|
|
|
for (i=0; i<length; i += nwrite) {
|
|
|
|
nwrite = write(fd, ackPacket + i, length - i);
|
|
|
|
if (nwrite < 0)
|
|
|
|
logMessage(LOG_ERR, "write to serial port failed\n");
|
|
|
|
}
|
2018-03-05 23:52:42 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Send %d bytes to serial\n",length);
|
2017-12-30 20:12:01 +00:00
|
|
|
//tcdrain(fd);
|
|
|
|
#endif
|
|
|
|
|
2018-06-06 00:18:44 +00:00
|
|
|
log_packet("Sent ", ackPacket, length);
|
2017-12-30 20:12:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Reads the bytes of the next incoming packet, and
|
|
|
|
// returns when a good packet is available in packet
|
|
|
|
// fd: the file descriptor to read the bytes from
|
|
|
|
// packet: the unsigned char buffer to store the bytes in
|
|
|
|
// returns the length of the packet
|
|
|
|
int get_packet(int fd, unsigned char* packet)
|
|
|
|
{
|
|
|
|
unsigned char byte;
|
|
|
|
int bytesRead;
|
|
|
|
int index = 0;
|
|
|
|
int endOfPacket = FALSE;
|
|
|
|
int packetStarted = FALSE;
|
|
|
|
int foundDLE = FALSE;
|
|
|
|
bool started = FALSE;
|
2018-06-06 00:18:44 +00:00
|
|
|
int retry=0;
|
2017-12-30 20:12:01 +00:00
|
|
|
|
|
|
|
while (!endOfPacket) {
|
|
|
|
//printf("Read loop %d\n",++i);
|
|
|
|
bytesRead = read(fd, &byte, 1);
|
|
|
|
|
|
|
|
if (bytesRead < 0 && errno == EAGAIN && started == FALSE) {
|
|
|
|
// We just have nothing to read
|
|
|
|
return 0;
|
|
|
|
} else if (bytesRead < 0 && errno == EAGAIN) {
|
|
|
|
// If we are in the middle of reading a packet, keep going
|
2018-06-06 00:18:44 +00:00
|
|
|
if (retry > 10)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
retry++;
|
2017-12-30 20:12:01 +00:00
|
|
|
delay(10);
|
|
|
|
} else if (bytesRead == 1) {
|
|
|
|
started = TRUE;
|
|
|
|
//if (bytesRead == 1) {
|
|
|
|
if (byte == DLE) {
|
|
|
|
// Found a DLE byte. Set the flag, and record the byte.
|
|
|
|
foundDLE = TRUE;
|
|
|
|
packet[index] = byte;
|
|
|
|
}
|
|
|
|
else if (byte == STX && foundDLE == TRUE) {
|
|
|
|
// Found the DLE STX byte sequence. Start of packet detected.
|
|
|
|
// Reset the DLE flag, and record the byte.
|
|
|
|
foundDLE = FALSE;
|
|
|
|
packetStarted = TRUE;
|
|
|
|
packet[index] = byte;
|
|
|
|
}
|
|
|
|
else if (byte == NUL && foundDLE == TRUE) {
|
|
|
|
// Found the DLE NUL byte sequence. Detected a delimited data byte.
|
|
|
|
// Reset the DLE flag, and decrement the packet index to offset the
|
|
|
|
// index increment at the end of the loop. The delimiter, [NUL], byte
|
|
|
|
// is not recorded.
|
|
|
|
foundDLE = FALSE;
|
|
|
|
//trimmed = true;
|
|
|
|
index--;
|
|
|
|
}
|
|
|
|
else if (byte == ETX && foundDLE == TRUE) {
|
|
|
|
// Found the DLE ETX byte sequence. End of packet detected.
|
|
|
|
// Reset the DLE flag, set the end of packet flag, and record
|
|
|
|
// the byte.
|
|
|
|
foundDLE = FALSE;
|
|
|
|
packetStarted = FALSE;
|
|
|
|
endOfPacket = TRUE;
|
|
|
|
packet[index] = byte;
|
|
|
|
}
|
|
|
|
else if (packetStarted == TRUE) {
|
|
|
|
// Found a data byte. Reset the DLE flag just in case it is set
|
|
|
|
// to prevent anomalous detections, and record the byte.
|
|
|
|
foundDLE = FALSE;
|
|
|
|
packet[index] = byte;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Found an extraneous byte. Probably a NUL between packets.
|
|
|
|
// Ignore it, and decrement the packet index to offset the
|
|
|
|
// index increment at the end of the loop.
|
|
|
|
index--;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finished processing the byte. Increment the packet index for the
|
|
|
|
// next byte.
|
|
|
|
index++;
|
|
|
|
|
|
|
|
// Break out of the loop if we exceed maximum packet
|
|
|
|
// length.
|
|
|
|
if (index >= AQ_MAXPKTLEN) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(bytesRead < 0) {
|
|
|
|
// Got a read error. Wait one millisecond for the next byte to
|
|
|
|
// arrive.
|
|
|
|
logMessage(LOG_WARNING, "Read error: %d - %s\n", errno, strerror(errno));
|
|
|
|
if(errno == 9) {
|
|
|
|
// Bad file descriptor. Port has been disconnected for some reason.
|
|
|
|
// Return a -1.
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
delay(100);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-16 17:18:15 +00:00
|
|
|
if (generate_checksum(packet, index) != packet[index-3]){
|
|
|
|
logMessage(LOG_WARNING, "Serial read bad checksum, ignoring\n");
|
2018-04-18 00:33:17 +00:00
|
|
|
log_packet("Bad packet ", packet, index);
|
2018-03-16 17:18:15 +00:00
|
|
|
return 0;
|
|
|
|
} else if (index < AQ_MINPKTLEN) {
|
2018-04-18 00:33:17 +00:00
|
|
|
logMessage(LOG_WARNING, "Serial read too small\n");
|
|
|
|
log_packet("Bad packet ", packet, index);
|
2018-03-16 17:18:15 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-03-05 23:52:42 +00:00
|
|
|
logMessage(LOG_DEBUG_SERIAL, "Serial read %d bytes\n",index);
|
2017-12-30 20:12:01 +00:00
|
|
|
// Return the packet length.
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|