2016-01-06 20:40:48 +11:00
|
|
|
|
|
|
|
// Local-Hyperion includes
|
|
|
|
#include "LedDeviceUdp.h"
|
2016-08-14 10:46:44 +02:00
|
|
|
|
|
|
|
#include <fstream>
|
2016-01-06 20:40:48 +11:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
struct addrinfo hints, *servinfo, *p;
|
|
|
|
//char udpbuffer[1024];
|
|
|
|
int sockfd;
|
|
|
|
int ledprotocol;
|
2016-05-22 20:56:44 +10:00
|
|
|
unsigned leds_per_pkt;
|
2016-01-06 20:40:48 +11:00
|
|
|
int update_number;
|
|
|
|
int fragment_number;
|
|
|
|
|
2016-08-14 10:46:44 +02:00
|
|
|
LedDeviceUdp::LedDeviceUdp(const std::string& output, const unsigned protocol, const unsigned maxPacket)
|
|
|
|
: LedDevice()
|
2016-01-06 20:40:48 +11:00
|
|
|
{
|
2016-05-26 23:44:27 +02:00
|
|
|
std::string hostname;
|
|
|
|
std::string port;
|
|
|
|
ledprotocol = protocol;
|
|
|
|
leds_per_pkt = ((maxPacket-4)/3);
|
|
|
|
if (leds_per_pkt <= 0) {
|
|
|
|
leds_per_pkt = 200;
|
|
|
|
}
|
2016-01-06 20:40:48 +11:00
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
int got_colon=0;
|
|
|
|
for (unsigned int i=0; i<output.length(); i++) {
|
|
|
|
if (output[i] == ':') {
|
|
|
|
got_colon++;
|
|
|
|
} else if (got_colon == 0) {
|
|
|
|
hostname+=output[i];
|
|
|
|
} else {
|
|
|
|
port+=output[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(got_colon==1);
|
|
|
|
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
memset(&hints, 0, sizeof hints);
|
|
|
|
hints.ai_family = AF_UNSPEC;
|
|
|
|
hints.ai_socktype = SOCK_DGRAM;
|
|
|
|
|
|
|
|
if ((rv = getaddrinfo(hostname.c_str() , port.c_str(), &hints, &servinfo)) != 0) {
|
2016-06-25 22:08:17 +02:00
|
|
|
Debug(_log, "getaddrinfo: %s", gai_strerror(rv));
|
2016-05-26 23:44:27 +02:00
|
|
|
assert(rv==0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// loop through all the results and make a socket
|
|
|
|
for(p = servinfo; p != NULL; p = p->ai_next) {
|
|
|
|
if ((sockfd = socket(p->ai_family, p->ai_socktype,
|
|
|
|
p->ai_protocol)) == -1) {
|
2016-06-30 00:49:05 +10:00
|
|
|
Error(_log,"talker: socket %s", strerror(errno));
|
|
|
|
// perror("talker: socket");
|
2016-05-26 23:44:27 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p == NULL) {
|
2016-06-25 22:08:17 +02:00
|
|
|
Error(_log,"talker: failed to create socket");
|
2016-05-26 23:44:27 +02:00
|
|
|
assert(p!=NULL);
|
2016-01-06 20:40:48 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LedDeviceUdp::~LedDeviceUdp()
|
|
|
|
{
|
|
|
|
// empty
|
|
|
|
}
|
|
|
|
|
|
|
|
int LedDeviceUdp::write(const std::vector<ColorRgb> & ledValues)
|
|
|
|
{
|
2016-08-14 10:46:44 +02:00
|
|
|
_ledCount = ledValues.size();
|
2016-01-06 20:40:48 +11:00
|
|
|
|
|
|
|
char udpbuffer[4096];
|
|
|
|
int udpPtr=0;
|
|
|
|
|
|
|
|
update_number++;
|
|
|
|
update_number &= 0xf;
|
|
|
|
|
2016-08-14 10:46:44 +02:00
|
|
|
if (ledprotocol == 0)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
int i=0;
|
|
|
|
for (const ColorRgb& color : ledValues)
|
|
|
|
{
|
2016-08-14 10:46:44 +02:00
|
|
|
if (i<4090)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
udpbuffer[i++] = color.red;
|
|
|
|
udpbuffer[i++] = color.green;
|
|
|
|
udpbuffer[i++] = color.blue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sendto(sockfd, udpbuffer, i, 0, p->ai_addr, p->ai_addrlen);
|
|
|
|
}
|
2016-08-14 10:46:44 +02:00
|
|
|
if (ledprotocol == 1)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
#define MAXLEDperFRAG 450
|
2016-08-14 10:46:44 +02:00
|
|
|
for (int frag=0; frag<4; frag++)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
udpPtr=0;
|
|
|
|
udpbuffer[udpPtr++] = 0;
|
|
|
|
udpbuffer[udpPtr++] = 0;
|
|
|
|
udpbuffer[udpPtr++] = (frag*MAXLEDperFRAG)/256; // high byte
|
|
|
|
udpbuffer[udpPtr++] = (frag*MAXLEDperFRAG)%256; // low byte
|
|
|
|
int ct=0;
|
2016-08-14 10:46:44 +02:00
|
|
|
for (int this_led = frag*300; ((this_led<_ledCount) && (ct++<MAXLEDperFRAG)); this_led++)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
const ColorRgb& color = ledValues[this_led];
|
2016-08-14 10:46:44 +02:00
|
|
|
if (udpPtr<4090)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
udpbuffer[udpPtr++] = color.red;
|
|
|
|
udpbuffer[udpPtr++] = color.green;
|
|
|
|
udpbuffer[udpPtr++] = color.blue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (udpPtr > 7)
|
2016-08-14 10:46:44 +02:00
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
sendto(sockfd, udpbuffer, udpPtr, 0, p->ai_addr, p->ai_addrlen);
|
2016-08-14 10:46:44 +02:00
|
|
|
}
|
2016-01-06 20:40:48 +11:00
|
|
|
}
|
|
|
|
}
|
2016-08-14 10:46:44 +02:00
|
|
|
if (ledprotocol == 2)
|
|
|
|
{
|
2016-01-06 20:40:48 +11:00
|
|
|
udpPtr = 0;
|
|
|
|
unsigned int ledCtr = 0;
|
|
|
|
fragment_number = 0;
|
|
|
|
udpbuffer[udpPtr++] = update_number & 0xf;
|
|
|
|
udpbuffer[udpPtr++] = fragment_number++;
|
|
|
|
udpbuffer[udpPtr++] = ledCtr/256; // high byte
|
|
|
|
udpbuffer[udpPtr++] = ledCtr%256; // low byte
|
|
|
|
|
|
|
|
for (const ColorRgb& color : ledValues)
|
|
|
|
{
|
|
|
|
if (udpPtr<4090) {
|
|
|
|
udpbuffer[udpPtr++] = color.red;
|
|
|
|
udpbuffer[udpPtr++] = color.green;
|
|
|
|
udpbuffer[udpPtr++] = color.blue;
|
|
|
|
}
|
|
|
|
ledCtr++;
|
|
|
|
if ( (ledCtr % leds_per_pkt == 0) || (ledCtr == ledValues.size()) ) {
|
|
|
|
sendto(sockfd, udpbuffer, udpPtr, 0, p->ai_addr, p->ai_addrlen);
|
|
|
|
memset(udpbuffer, 0, sizeof udpbuffer);
|
|
|
|
udpPtr = 0;
|
|
|
|
udpbuffer[udpPtr++] = update_number & 0xf;
|
|
|
|
udpbuffer[udpPtr++] = fragment_number++;
|
|
|
|
udpbuffer[udpPtr++] = ledCtr/256; // high byte
|
|
|
|
udpbuffer[udpPtr++] = ledCtr%256; // low byte
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-05-15 18:39:26 +02:00
|
|
|
|
2016-08-14 10:46:44 +02:00
|
|
|
if (ledprotocol == 3)
|
|
|
|
{
|
2016-05-15 18:39:26 +02:00
|
|
|
udpPtr = 0;
|
|
|
|
unsigned int ledCtr = 0;
|
|
|
|
unsigned int fragments = 1;
|
|
|
|
unsigned int datasize = ledValues.size() * 3;
|
|
|
|
if (ledValues.size() > leds_per_pkt) {
|
|
|
|
fragments = (ledValues.size() / leds_per_pkt) + 1;
|
|
|
|
}
|
|
|
|
fragment_number = 1;
|
|
|
|
udpbuffer[udpPtr++] = 0x9C;
|
|
|
|
udpbuffer[udpPtr++] = 0xDA;
|
|
|
|
udpbuffer[udpPtr++] = datasize/256; // high byte
|
|
|
|
udpbuffer[udpPtr++] = datasize%256; // low byte
|
|
|
|
udpbuffer[udpPtr++] = fragment_number++;
|
|
|
|
udpbuffer[udpPtr++] = fragments;
|
|
|
|
|
|
|
|
for (const ColorRgb& color : ledValues)
|
|
|
|
{
|
2016-08-14 10:46:44 +02:00
|
|
|
if (udpPtr<4090)
|
|
|
|
{
|
2016-05-15 18:39:26 +02:00
|
|
|
udpbuffer[udpPtr++] = color.red;
|
|
|
|
udpbuffer[udpPtr++] = color.green;
|
|
|
|
udpbuffer[udpPtr++] = color.blue;
|
|
|
|
}
|
|
|
|
ledCtr++;
|
2016-08-14 10:46:44 +02:00
|
|
|
if ( (ledCtr % leds_per_pkt == 0) || (ledCtr == ledValues.size()) )
|
|
|
|
{
|
2016-05-15 18:39:26 +02:00
|
|
|
udpbuffer[udpPtr++] = 0x36;
|
|
|
|
sendto(sockfd, udpbuffer, udpPtr, 0, p->ai_addr, p->ai_addrlen);
|
|
|
|
memset(udpbuffer, 0, sizeof udpbuffer);
|
|
|
|
udpPtr = 0;
|
|
|
|
udpbuffer[udpPtr++] = 0x9C;
|
|
|
|
udpbuffer[udpPtr++] = 0xDA;
|
|
|
|
udpbuffer[udpPtr++] = datasize/256; // high byte
|
|
|
|
udpbuffer[udpPtr++] = datasize%256; // low byte
|
|
|
|
udpbuffer[udpPtr++] = fragment_number++;
|
|
|
|
udpbuffer[udpPtr++] = fragments;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-06 20:40:48 +11:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int LedDeviceUdp::switchOff()
|
|
|
|
{
|
2016-08-14 10:46:44 +02:00
|
|
|
// return write(std::vector<ColorRgb>(_ledCount, ColorRgb{0,0,0}));
|
2016-01-06 20:40:48 +11:00
|
|
|
return 0;
|
|
|
|
}
|