mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
464de381a0
* Update CMakeLists.txt * Update Hyperion.h * Update LedDevice.h * Update LedDeviceFactory.h * Update Hyperion.cpp * Update LedString.cpp * Update JsonClientConnection.cpp * Update LedDeviceAdalight.cpp * Update LedDeviceAdalight.h * Update LedDeviceAPA102.cpp * Update LedDeviceAdalightApa102.h * Update LedDeviceAdalightApa102.cpp * Update LedDeviceAPA102.h * Update LedDeviceAtmo.cpp * Update LedDeviceAtmo.h * Update LedDeviceAtmoOrb.cpp * Update LedDeviceAtmoOrb.h * Update LedDeviceDMX.cpp * Update LedDeviceDMX.h * Update LedDeviceFactory.cpp * Update LedDeviceFadeCandy.cpp * Update LedDeviceFadeCandy.h * Update LedDeviceFile.cpp * Update LedDeviceFile.h * Update LedDeviceHyperionUsbasp.cpp * Update LedDeviceHyperionUsbasp.h * Update LedDeviceLightpack.cpp * Update LedDeviceLightpack.h * Update LedDeviceLpd6803.cpp * Update LedDeviceLpd6803.h * Update LedDeviceLpd8806.cpp * Update LedDeviceLpd8806.h * Update LedDeviceMultiLightpack.cpp * Update LedDeviceMultiLightpack.h * Update LedDeviceP9813.cpp * Update LedDeviceP9813.h * Update LedDevicePaintpack.cpp * Update LedDevicePaintpack.h * Update LedDevicePhilipsHue.cpp * Update LedDevicePhilipsHue.h * Update LedDevicePiBlaster.cpp * Update LedDevicePiBlaster.h * Update LedDeviceRawHID.cpp * Update LedDeviceRawHID.h * Update LedDeviceSedu.cpp * Update LedDeviceSedu.h * Update LedDeviceSk6812SPI.cpp * Update LedDeviceSk6812SPI.h * Update LedDeviceTinkerforge.cpp * Update LedDeviceTinkerforge.h * Update LedDeviceTpm2.cpp * Update LedDeviceTpm2.h * Update LedDeviceTpm2net.cpp * Update LedDeviceTpm2net.h * Update LedDeviceUdpE131.cpp * Update LedDeviceUdpE131.h * Update LedDeviceUdpH801.cpp * Update LedDeviceUdpH801.h * Update LedDeviceUdpRaw.cpp * Update LedDeviceUdpRaw.h * Update LedDeviceWs2801.cpp * Update LedDeviceWs2801.h * Update LedDeviceWS2812b.cpp * Update LedDeviceWS2812b.h * Update LedDeviceWs2812SPI.cpp * Update LedDeviceWs2812SPI.h * Update LedDeviceWS281x.cpp * Update LedDeviceWS281x.h * Update ProviderHID.cpp * Update ProviderHID.h * Update ProviderRs232.cpp * Update ProviderRs232.h * Update ProviderSpi.cpp * Update ProviderSpi.h * Update ProviderUdp.cpp * Update ProviderUdp.h * Update LedDevice.cpp * Update CMakeLists.txt * Update hyperiond.cpp * Update hyperiond.h * Update TestSpi.cpp * Delete AUTHORS * Delete CMakeLists.txt * Delete LICENSE * Delete json_batchallocator.h * Delete json_internalarray.inl * Delete json_internalmap.inl * Delete json_reader.cpp * Delete json_tool.h * Delete json_value.cpp * Delete json_valueiterator.inl * Delete json_writer.cpp * Delete sconscript * Delete autolink.h * Delete config.h * Delete features.h * Delete forwards.h * Delete json.h * Delete reader.h * Delete value.h * Delete writer.h
163 lines
3.7 KiB
C++
163 lines
3.7 KiB
C++
|
|
// STL includes
|
|
#include <cstring>
|
|
#include <iostream>
|
|
|
|
// Qt includes
|
|
#include <QTimer>
|
|
|
|
// Local Hyperion includes
|
|
#include "ProviderHID.h"
|
|
|
|
ProviderHID::ProviderHID()
|
|
: _useFeature(false)
|
|
, _deviceHandle(nullptr)
|
|
, _blockedForDelay(false)
|
|
{
|
|
}
|
|
|
|
ProviderHID::~ProviderHID()
|
|
{
|
|
if (_deviceHandle != nullptr)
|
|
{
|
|
hid_close(_deviceHandle);
|
|
_deviceHandle = nullptr;
|
|
}
|
|
|
|
hid_exit();
|
|
}
|
|
|
|
bool ProviderHID::init(const QJsonObject &deviceConfig)
|
|
{
|
|
_delayAfterConnect_ms = deviceConfig["delayAfterConnect"].toInt(0);
|
|
auto VendorIdString = deviceConfig["VID"].toString("0x2341").toStdString();
|
|
auto ProductIdString = deviceConfig["PID"].toString("0x8036").toStdString();
|
|
|
|
// Convert HEX values to integer
|
|
_VendorId = std::stoul(VendorIdString, nullptr, 16);
|
|
_ProductId = std::stoul(ProductIdString, nullptr, 16);
|
|
|
|
return true;
|
|
}
|
|
|
|
int ProviderHID::open()
|
|
{
|
|
// Initialize the usb context
|
|
int error = hid_init();
|
|
if (error != 0)
|
|
{
|
|
Error(_log, "Error while initializing the hidapi context");
|
|
return -1;
|
|
}
|
|
Debug(_log,"Hidapi initialized");
|
|
|
|
// Open the device
|
|
Info(_log, "Opening device: VID %04hx PID %04hx\n", _VendorId, _ProductId);
|
|
_deviceHandle = hid_open(_VendorId, _ProductId, nullptr);
|
|
|
|
if (_deviceHandle == nullptr)
|
|
{
|
|
// Failed to open the device
|
|
Error(_log,"Failed to open HID device. Maybe your PID/VID setting is wrong? Make sure to add a udev rule/use sudo.");
|
|
|
|
// http://www.signal11.us/oss/hidapi/
|
|
/*
|
|
std::cout << "Showing a list of all available HID devices:" << std::endl;
|
|
auto devs = hid_enumerate(0x00, 0x00);
|
|
auto cur_dev = devs;
|
|
while (cur_dev) {
|
|
printf("Device Found\n type: %04hx %04hx\n path: %s\n serial_number: %ls",
|
|
cur_dev->vendor_id, cur_dev->product_id, cur_dev->path, cur_dev->serial_number);
|
|
printf("\n");
|
|
printf(" Manufacturer: %ls\n", cur_dev->manufacturer_string);
|
|
printf(" Product: %ls\n", cur_dev->product_string);
|
|
printf("\n");
|
|
cur_dev = cur_dev->next;
|
|
}
|
|
hid_free_enumeration(devs);
|
|
*/
|
|
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
Info(_log,"Opened HID device successful");
|
|
}
|
|
|
|
// Wait after device got opened if enabled
|
|
if (_delayAfterConnect_ms > 0)
|
|
{
|
|
_blockedForDelay = true;
|
|
QTimer::singleShot(_delayAfterConnect_ms, this, SLOT(unblockAfterDelay()));
|
|
Debug(_log, "Device blocked for %d ms", _delayAfterConnect_ms);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int ProviderHID::writeBytes(const unsigned size, const uint8_t * data)
|
|
{
|
|
if (_blockedForDelay) {
|
|
return 0;
|
|
}
|
|
|
|
if (_deviceHandle == nullptr)
|
|
{
|
|
// try to reopen
|
|
auto status = open();
|
|
if(status < 0){
|
|
// Try again in 3 seconds
|
|
int delay_ms = 3000;
|
|
_blockedForDelay = true;
|
|
QTimer::singleShot(delay_ms, this, SLOT(unblockAfterDelay()));
|
|
Debug(_log,"Device blocked for %d ms", delay_ms);
|
|
}
|
|
// Return here, to not write led data if the device should be blocked after connect
|
|
return status;
|
|
}
|
|
|
|
// Prepend report ID to the buffer
|
|
uint8_t ledData[size + 1];
|
|
ledData[0] = 0; // Report ID
|
|
memcpy(ledData + 1, data, size_t(size));
|
|
|
|
// Send data via feature or out report
|
|
int ret;
|
|
if(_useFeature){
|
|
ret = hid_send_feature_report(_deviceHandle, ledData, size + 1);
|
|
}
|
|
else{
|
|
ret = hid_write(_deviceHandle, ledData, size + 1);
|
|
}
|
|
|
|
// Handle first error
|
|
if(ret < 0){
|
|
Error(_log,"Failed to write to HID device.");
|
|
|
|
// Try again
|
|
if(_useFeature){
|
|
ret = hid_send_feature_report(_deviceHandle, ledData, size + 1);
|
|
}
|
|
else{
|
|
ret = hid_write(_deviceHandle, ledData, size + 1);
|
|
}
|
|
|
|
// Writing failed again, device might have disconnected
|
|
if(ret < 0){
|
|
Error(_log,"Failed to write to HID device.");
|
|
|
|
hid_close(_deviceHandle);
|
|
_deviceHandle = nullptr;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void ProviderHID::unblockAfterDelay()
|
|
{
|
|
Debug(_log,"Device unblocked");
|
|
_blockedForDelay = false;
|
|
}
|