mirror of
				https://github.com/hyperion-project/hyperion.ng.git
				synced 2025-03-01 10:33:28 +00:00 
			
		
		
		
	* 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;
 | 
						|
}
 |