Library to interface with Voltronic devices like the Axpert & InfiniSolar
This entire library is licenced under GNU GENERAL PUBLIC LICENSE v3
That means if you want to use this library you can, provided your source code is also also open sourced.
I will not take lightly any use of this implementation in ANY closed source software
Devices from Voltronic are shipped with 4 possible hardware interfaces: RS232, USB, Bluetooth & RS485
All the interfaces share the same underlying communication protocol
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
int main() {
const char* command = "QPI";
voltronic_dev_t dev = // See examples for Serial & USB below
// Query the device
char buffer[128];
int result = voltronic_dev_execute(
dev, // Pointer to the underlying device
0, // Options; VOLTRONIC_EXECUTE_DEFAULT_OPTIONS, DISABLE_WRITE_VOLTRONIC_CRC, DISABLE_PARSE_VOLTRONIC_CRC, DISABLE_VERIFY_VOLTRONIC_CRC
command, // Command to execute on the inverter
strlen(command), // Length of the command
buffer, // Buffer to store the response
sizeof(buffer), // Maximum length of the buffer
1000 // Timeout milliseconds to wait before giving up on a response
);
if (result > 0) {
printf("Success on command %s, got %s\n", command, buffer);
} else {
printf("Failed to execute %s\n", command);
}
// Close the connection to the device
voltronic_dev_close(dev);
}
#include "voltronic_dev_serial.h"
int main() {
// Create a serial port dev
voltronic_dev_t dev = voltronic_serial_create(
"/dev/tty.usbserial", // Port
2400, // Baud rate; Integer > 0
DATA_BITS_EIGHT, // Data bits; FIVE, SIX, SEVEN, EIGHT
STOP_BITS_ONE, // Stop Bits; ONE, ONE_AND_ONE_HALF, TWO
SERIAL_PARITY_NONE // Parity; NONE, EVEN, ODD, MARK, SPACE
);
if (dev == 0) {
printf("Could not open serial port device\n");
return 1;
}
// See usage example above
}
See serial port test code for a complete example
#include "voltronic_dev_usb.h"
int main() {
// Create a USB dev
voltronic_dev_t dev = voltronic_usb_create(
0x0665, // USB Vendor ID
0x5161 // USB Product ID
);
if (dev == 0) {
printf("Could not open USB device");
return 1;
}
// See usage example above
}
See usb test code for a complete example
The communication protocol consists of the following format:
Overall the protocol has the following format:
{bytes}{CRC16}{end of input character}
- bytes the actual bytes being sent to the device, generally speaking this is the "command"
- CRC16 common CRC protocol with many implementations online
- end of input character character signaling the end of input
These characters are reserved
\r
(0x0d) End of input character(
(0x28) Seems to indicate start of input\n
(0x0a) No material importance but still reserved
The bytes being sent to the device appear to be simply ASCII in the form of a command
Multiple documents exist listing possible commands
The CRC is used is the CRC16 XMODEM variation
Background
Multiple methods exist to generate CRC.
CRC16 as the name implies contains 16 bits or 2 bytes of data.
It is commonly written as hexadecimal for readability reason, ie> 0x17AD
Two hexadecimal character represent a single byte so given the example above.
0x
part simply indicates hexadecimal
0x17
is the first byte
0xAD
is the second byte
Exception
The Reserved characters are not allowed in the CRC. It appears the device simply expects them to be incremented by 1
So 0x28
becomes 0x29
, 0x0d
becomes 0x0e
, etc.
The \r
character signals to the device end of input
Regardless of what the device received up to that point \r
signals to the device end of current input
Once this character is received all input up to that point is taken as the command to the device
Devices from Voltronic are shipped with 4 possible hardware interfaces: RS232, USB, Bluetooth & RS485
All input methods share the same bandwidth & latency. Although it would appear at surface that USB should be faster, no measureable difference exists to device response time and symbol rate.
USB is also an asynchronous protocol and as such could be influenced by other factors slowing it down further
During testing it was found that simultaneous communication across USB & RS232 for example would result in device lockup. The device keeps operating, but the device will no longer respond to input or produce output.
As such it is adviced to pick an interface and use it exclusively
Nothing special to mention here, synchronous protocol with the following configuration:
- Baud 2400
- Data bits 8
- Stop bits 1
- Parity None (provided by CRC16)
The device makes use of an HID interface. In Linux the device is presented as a HIDRaw device
It is not a USB->Serial
So in Linux for example:
Ruby:
fd = File.open('/dev/hidraw0', IO::RDWR|IO::NONBLOCK) # May need root, or make the file 666 using udev rules
fd.binmode
fd.sync = true
fd.write("QPI\xBE\xAC\r") # Will write QPI => Returns 6
fd.gets("\r") #=> "(PI30\x9A\v\r"
Python:
import os, sys
fd = open("/dev/hidraw0", os.O_RDWR|os.O_NONBLOCK)
os.write(fd, "QPI\xBE\xAC\r")
os.read(fd, 512)
Avoiding the need for root
Make use of udev to specify more broad access:
# may require root
touch /etc/udev/rules.d/15-voltronic.rules
echo 'ATTRS{idVendor}=="0665", ATTRS{idProduct}=="5161", SUBSYSTEMS=="usb", ACTION=="add", MODE="0666", SYMLINK+="hidVoltronic"' > /etc/udev/rules.d/15-voltronic.rules
When the device is connected it will present in /dev/hidVoltronic
.
Note that if multiple devices are to be connected to the same machine, an additional udev parameter should be specified such as the device serial number to with different symlink names
Newer generation Axpert devices feature Bluetooth
No testing has been completed on these devices but Bluetooth simply operates exactly like RS232 and therefore there is no reason to believe it would be otherwise
Newer generation Axpert devices feature RS485 support
No testing has been completed on these devices but there is no reason to believe the underlying protocol has changed at all
- Cross OS support
- Support for RS232, USB & Bluetooth (Perhaps RS485)
- CRC calculation written in C
- Common interface for all device communication methods
Specific commands as given above (as examples). ie QPI
, etc.
The aim of this library is to act as a driver that allows interfacing with Voltronic devices through a simple API. What a command does, and how the data the device returns should be parsed, is a higher level concern.
Much of the community is able to easily understand the commands, how to parse them and their effect. The biggest barier to this author appears to be how to actually communicate with the device.
The communication concern requires a fair amount of knowledge of CRC, how to work with bytes and operating system API understanding ie> termios, DCB, tc.
Especially USB & USB HID is complex and has significantly complex APIs that are different across EVERY operating system. Meaning Windows, Linux, OSX and *BSD (excl. OSX) all implement different APIs to communicate with HID devices. Even using a library like HIDAPI requires understanding the underlying fundamentals
To remove a lot of the heavy lifting, the library relies:
Install depedencies: Each operating system will have a different list of prerequisites before the dependencies can be built.
See a more detailed list below
Build libserialport:
git clone git://sigrok.org/libserialport lib/libserialport/
cd lib/libserialport/
./autogen.sh
./configure
make
make install # Requires sudo or run as su
Build libhidapi:
git clone https://github.com/signal11/hidapi.git lib/libhidapi/
cd lib/libhidapi/
./bootstrap
./configure
make
make install # Requires sudo or su
Tested on FreeBSD 10, 11, 12
Required dependencies:
su
pkg install gcc git autoconf automake libtool libiconv gmake
gmake instead of make to build
Required dependencies:
Ubuntu:
sudo apt-get clean
sudo apt-get update
sudo apt-get install gcc git autoconf automake libtool pkg-config libudev-dev libusb-1.0-0-dev
Amazon Linux:
sudo yum clean all
sudo yum install gcc git autoconf automake libtool pkg-config libudev-devel libusb1-devel
make to build
Built using Ubuntu and miniGW. Building natively in Windows is a challenge for a better software developer than this author.
Using Ubuntu:
sudo apt-get clean
sudo apt-get update
sudo apt-get install gcc git autoconf automake libtool pkg-config libudev-dev libusb-1.0-0-dev
sudo apt-get install gcc-mingw-w64-i686
sudo apt-get install gcc-mingw-w64-x86-64
The library was developed & tested on OSX High Sierra but the system is setup as a dev machine. As such the complete list of dependencies have long since been forgotten.
At the very least:
- Homebrew
- gcc
- git
make to build