Skip to content

Library to interface with Voltronic devices like the Axpert & InfiniSolar

License

Notifications You must be signed in to change notification settings

jvandervyver/libvoltronic

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

58 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

libvoltronic

Library to interface with Voltronic devices like the Axpert & InfiniSolar

License

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

Description

Devices from Voltronic are shipped with 4 possible hardware interfaces: RS232, USB, Bluetooth & RS485

All the interfaces share the same underlying communication protocol

Usage

#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);
}

Serial

#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

USB

#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

Communication protocol

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

Reserved characters

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

Bytes

The bytes being sent to the device appear to be simply ASCII in the form of a command

Multiple documents exist listing possible commands

CRC

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.

End of input character

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

Input methods

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

Simultaneous communication across multiple interfaces

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

RS232

Nothing special to mention here, synchronous protocol with the following configuration:

  • Baud 2400
  • Data bits 8
  • Stop bits 1
  • Parity None (provided by CRC16)

USB

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

Bluetooth

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

RS485

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

What does this library provide?

  • Cross OS support
  • Support for RS232, USB & Bluetooth (Perhaps RS485)
  • CRC calculation written in C
  • Common interface for all device communication methods

What does this library NOT provide?

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

Dependencies

To remove a lot of the heavy lifting, the library relies:

Building

Dependencies

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

FreeBSD

Tested on FreeBSD 10, 11, 12

Required dependencies:

su
pkg install gcc git autoconf automake libtool libiconv gmake

gmake instead of make to build

Linux

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

Windows

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

x86

sudo apt-get install gcc-mingw-w64-i686

x64

sudo apt-get install gcc-mingw-w64-x86-64

OSX

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

About

Library to interface with Voltronic devices like the Axpert & InfiniSolar

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published