Files
MARTe2-isttok/DataSources/PSUCommunicator/testSerial/psuSendReceive.cpp
2025-12-06 09:01:46 +00:00

262 lines
8.0 KiB
C++

// C library headers
// setserial /dev/ttyS0 spd_cust
// setserial /dev/ttyS0 divisor 16
// stty -F /dev/ttyS0 921600
#include <cstdint>
#include <cstdlib>
#include <stdio.h>
#include <string.h>
// Linux headers
#include <errno.h> // Error integer and strerror() function
#include <fcntl.h> // Contains file controls like O_RDWR
#include <iostream>
#include <termios.h> // Contains POSIX terminal control definitions
#include <unistd.h> // write(), read(), close()
#include "PSUMessages.h"
uint8_t packet[2];
uint8_t packetR[4];
// int sourceFd;
bool isStarted;
void InterpretMessage();
int openSerialPort(const char *portname) {
int fd = open(portname, O_RDWR | O_NOCTTY | O_NDELAY);
if (fd < 0) {
std::cerr << "Error opening port: " << portname << std::endl;
return -1;
}
return fd;
}
bool configureSerialPort(int fd, int baudRate) {
// Create new termios struct, we call it 'tty' for convention
struct termios tty;
// Read in existing settings, and handle any error
if (tcgetattr(fd, &tty) != 0) {
printf("Error %i from tcgetattr: %s\n", errno, strerror(errno));
return false;
}
// tty.c_cflag &= ~PARENB; // Clear parity bit, disabling parity (most common)
tty.c_cflag |= PARENB;
tty.c_cflag |= PARODD;
tty.c_cflag |= CSTOPB;
// tty.c_cflag &= ~CSTOPB; // Clear stop field, only one stop bit used in
// communication (most common)
tty.c_cflag &= ~CSIZE; // Clear all bits that set the data size
tty.c_cflag |= CS8; // 8 bits per byte (most common)
tty.c_cflag &=
~CRTSCTS; // Disable RTS/CTS hardware flow control (most common)
tty.c_cflag |=
CREAD | CLOCAL; // Turn on READ & ignore ctrl lines (CLOCAL = 1)
// http://unixwiz.net/techtips/termios-vmin-vtime.html
// ICANON bit is turned off, a "raw mode" is selected
tty.c_lflag &= ~ICANON;
tty.c_lflag &= ~ECHO; // Disable echo
tty.c_lflag &= ~ECHOE; // Disable erasure
tty.c_lflag &= ~ECHONL; // Disable new-line echo
tty.c_lflag &= ~ISIG; // Disable interpretation of INTR, QUIT and SUSP
tty.c_iflag &= ~(IXON | IXOFF | IXANY); // Turn off s/w flow ctrl
tty.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR |
ICRNL); // Disable any special handling of received bytes
// write(fd, packet, sizeof(packet));
// printf("Sent : %lu bytes\n", sizeof(packet));
tty.c_oflag &= ~OPOST; // Prevent special interpretation of output bytes (e.g.
// newline chars)
tty.c_oflag &=
~ONLCR; // Prevent conversion of newline to carriage return/line feed
// tty.c_oflag &= ~OXTABS; // Prevent conversion of tabs to spaces (NOT
// PRESENT ON LINUX) tty.c_oflag &= ~ONOEOT; // Prevent removal of C-d chars
// (0x004) in output (NOT PRESENT ON LINUX)
tty.c_cc[VTIME] = 1; // Wait for up to 1s (10 deciseconds), returning as soon
// as any data is received.
tty.c_cc[VMIN] = 0;
// tty.c_cc[VMIN] = 2;
// Error reading: Resource temporarily unavailable%
// Set in/out baud rate
cfsetispeed(&tty, baudRate);
cfsetospeed(&tty, baudRate);
// Save tty settings, also checking for error
if (tcsetattr(fd, TCSANOW, &tty) != 0) {
printf("Error %i from tcsetattr: %s\n", errno, strerror(errno));
return false;
}
return true;
}
bool DecodeCurrentPacket() { // float &current, unsigned char packet1, unsigned
// char packet2){
// Validate packets
unsigned char validation = (packetR[1] & 0xF0) ^ ((packetR[0] & 0x1E) << 3);
if (validation != 0xF0) {
printf("WrongMessagesReceived\n");
return false;
}
short pointOfCurrent = (short)((((unsigned short)packetR[0] & 0x00E0) >> 5) |
(((unsigned short)packetR[1] & 0x00FE) << 2));
printf("pointOfCurrenti = %d\n", pointOfCurrent);
return true;
}
bool CreateCurrentPacket(unsigned int current) {
bool ok = true;
// Zero current 510
// Calculate the point in the scale of the current
int16_t pointOfCurrent = current;
// Saturate current
if (pointOfCurrent < FA_SCALE_MIN)
pointOfCurrent = FA_SCALE_MIN;
if (pointOfCurrent > FA_SCALE_MAX)
pointOfCurrent = FA_SCALE_MAX;
// Build packets
uint16_t pc = (uint16_t)pointOfCurrent;
uint16_t nc = ~pc;
packet[0] = (uint8_t)(0x0000 | ((nc & 0x03C0) >> 5) | ((pc & 0x0007) << 5));
packet[1] = (uint8_t)(0x0001 | ((pc & 0x03F8) >> 2));
/*
nc = (uint16_t)packet[1];
nc <<= 8;
nc &= 0xFF00;
nc |= packet[0];
*/
printf("Current: %d, Packet 0x%02X 0x%02X\n", current, packet[0], packet[1]);
return EXIT_SUCCESS;
}
int read_msg(int fd) {
// Allocate memory for read buffer, set size according to your needs
// char read_buf[4];
// Normally you wouldn't do this memset() call, but since we will just receive
// call printf() easily.
memset(&packetR, '\0', sizeof(packetR));
// wait 1 ms
usleep(1000);
// Read bytes. The behaviour of read() (e.g. does it block?,
// how long does it block for?) depends on the configuration
// settings above, specifically VMIN and VTIME
int num_bytes = read(fd, &packetR, sizeof(packetR));
// n is the number of bytes read. n may be 0 if no bytes were received, and
// can also be -1 to signal an error.
if (num_bytes < 0) {
printf("Error reading: %s", strerror(errno));
return EXIT_FAILURE;
}
// Here we assume we received ASCII data, but you might be sending raw bytes
// (in that case, don't try and print it to the screen like this!)
printf("Read %i bytes. \t", num_bytes);
for (int i = 0; i < num_bytes; i++) {
unsigned int val = packetR[i] & 0xFF;
printf("%i: 0x%02X \t", i, val);
// printf("%i: %u \n", i, val);
}
printf("\n");
if (num_bytes == 2)
InterpretMessage();
return EXIT_SUCCESS;
}
void InterpretMessage() {
// Operation Started message
if ((packetR[0] == FA_STARTED_MESSAGE_1) &&
(packetR[1] == FA_STARTED_MESSAGE_2)) {
printf("isStarted\n");
isStarted = true;
return;
}
// Operation Stopped message
if ((packetR[0] == FA_STOPPED_MESSAGE_1) &&
(packetR[1] == FA_STOPPED_MESSAGE_2)) {
printf("isStopped\n");
return;
}
// Stop Error message
if ((packetR[0] == FA_STOP_ERROR_MESSAGE_1) &&
(packetR[1] == FA_STOP_ERROR_MESSAGE_2)) {
printf("Stop Error message\n");
return;
}
// Communication Error message
if ((packetR[0] == FA_COMM_ERROR_MESSAGE_1) &&
(packetR[1] == FA_COMM_ERROR_MESSAGE_2)) {
printf("Communication Error message\n");
return;
}
// printf("Other message\n");
DecodeCurrentPacket();
}
int main() {
// Open the serial port. Change device path as needed (currently set to an
// standard FTDI USB-UART cable type device)
// sourceFd = open("/dev/ttyS2", O_RDWR);
int sourceFd = openSerialPort("/dev/ttyS2");
const speed_t baud_rate = B921600;
configureSerialPort(sourceFd, B921600);
// B115200; // B576000; // B460800; OK// B230400; // B9600;B921600;
// const speed_t baud_rate = B115200; // B9600;B921600;
// clear input port
read_msg(sourceFd);
isStarted = false;
// Write to serial port
// unsigned char msg[] = {FA_STARTOP_MESSAGE_1, 0xFF};
packet[0] = FA_STARTOP_MESSAGE_1;
packet[1] = FA_STARTOP_MESSAGE_2;
write(sourceFd, packet, sizeof(packet));
printf("Sent Start: %lu bytes\n", sizeof(packet));
read_msg(sourceFd);
if (isStarted) {
usleep(1000);
for (int i = 0; i < 10; i++) {
CreateCurrentPacket(500 + i);
write(sourceFd, packet, sizeof(packet));
usleep(10);
// printf("Sent : %lu bytes\n", sizeof(packet));
read_msg(sourceFd);
// Halting the execution for 100000 Microseconds (0.1 seconds)
usleep(10000);
}
}
packet[0] = FA_STOPOP_MESSAGE_1;
packet[1] = FA_STOPOP_MESSAGE_2;
write(sourceFd, packet, sizeof(packet));
printf("Sent Stop: %lu bytes\n", sizeof(packet));
// read_msg();
usleep(1000);
read_msg(sourceFd);
read_msg(sourceFd);
close(sourceFd);
return 0; // success
}
/**
void repeatData(int sourceFd, int destFd) {
char buffer[256];
while (true) {
int bytesRead = read(sourceFd, buffer, sizeof(buffer));
if (bytesRead > 0) {
write(destFd, buffer, bytesRead);
}
}
}
*/