mirror of
https://github.com/CalcProgrammer1/OpenRGB.git
synced 2025-12-26 16:57:51 -05:00
404 lines
11 KiB
C++
404 lines
11 KiB
C++
/*-------------------------------------------------------------------*\
|
|
| RoccatVulcanKeyboardController.cpp |
|
|
| |
|
|
| Driver for Roccat Vulcan Keyboard |
|
|
| |
|
|
| Mola19 17/12/2021 |
|
|
| |
|
|
\*-------------------------------------------------------------------*/
|
|
|
|
#include "RoccatVulcanKeyboardController.h"
|
|
|
|
#include <cstring>
|
|
#include <math.h>
|
|
#include <chrono>
|
|
#include <thread>
|
|
#include <vector>
|
|
|
|
#include "LogManager.h"
|
|
|
|
RoccatVulcanKeyboardController::RoccatVulcanKeyboardController(hid_device* dev_ctrl_handle, hid_device* dev_led_handle, char *path, uint16_t pid)
|
|
{
|
|
dev_ctrl = dev_ctrl_handle;
|
|
dev_led = dev_led_handle;
|
|
location = path;
|
|
device_pid = pid;
|
|
}
|
|
|
|
RoccatVulcanKeyboardController::~RoccatVulcanKeyboardController()
|
|
{
|
|
hid_close(dev_ctrl);
|
|
hid_close(dev_led);
|
|
}
|
|
|
|
std::string RoccatVulcanKeyboardController::GetSerial()
|
|
{
|
|
wchar_t serial_string[128];
|
|
int ret = hid_get_serial_number_string(dev_ctrl, serial_string, 128);
|
|
|
|
if(ret != 0)
|
|
{
|
|
return("");
|
|
}
|
|
|
|
std::wstring return_wstring = serial_string;
|
|
std::string return_string(return_wstring.begin(), return_wstring.end());
|
|
|
|
return(return_string);
|
|
|
|
}
|
|
|
|
std::string RoccatVulcanKeyboardController::GetLocation()
|
|
{
|
|
return("HID: " + location);
|
|
}
|
|
|
|
|
|
device_info RoccatVulcanKeyboardController::InitDeviceInfo()
|
|
{
|
|
uint8_t packet_length;
|
|
uint8_t report_id;
|
|
|
|
switch(device_pid)
|
|
{
|
|
case ROCCAT_PYRO_PID:
|
|
case ROCCAT_MAGMA_PID:
|
|
case ROCCAT_MAGMA_MINI_PID:
|
|
case ROCCAT_VULCAN_PRO_PID:
|
|
packet_length = 9;
|
|
report_id = 0x09;
|
|
break;
|
|
default:
|
|
packet_length = 8;
|
|
report_id = 0x0F;
|
|
}
|
|
|
|
uint8_t* buf = new uint8_t[packet_length];
|
|
memset(buf, 0x00, packet_length);
|
|
|
|
buf[0] = report_id;
|
|
hid_get_feature_report(dev_ctrl, buf, packet_length);
|
|
|
|
char version[5];
|
|
snprintf(version, 5, "%d.%02d", buf[2] / 100, buf[2] % 100);
|
|
|
|
dev_info.version = version;
|
|
|
|
if(device_pid == ROCCAT_MAGMA_PID || device_pid == ROCCAT_MAGMA_MINI_PID)
|
|
{
|
|
/*---------------------------------------------------------*\
|
|
| This device doesn't need a layout, |
|
|
| because it doesn't have per-led lighting. |
|
|
| Taking us layout as placeholder instead |
|
|
\*---------------------------------------------------------*/
|
|
dev_info.layout_type = ROCCAT_VULCAN_LAYOUT_US;
|
|
}
|
|
else
|
|
{
|
|
dev_info.layout_type = buf[6];
|
|
}
|
|
|
|
LOG_DEBUG("[Roccat Vulcan Keyboard]: Detected layout '0x%02X'", buf[6]);
|
|
|
|
delete[] buf;
|
|
return dev_info;
|
|
}
|
|
|
|
device_info RoccatVulcanKeyboardController::GetDeviceInfo()
|
|
{
|
|
return dev_info;
|
|
}
|
|
|
|
void RoccatVulcanKeyboardController::EnableDirect(bool on_off_switch)
|
|
{
|
|
uint8_t* buf;
|
|
switch(device_pid)
|
|
{
|
|
case ROCCAT_PYRO_PID:
|
|
case ROCCAT_MAGMA_PID:
|
|
case ROCCAT_MAGMA_MINI_PID:
|
|
case ROCCAT_VULCAN_PRO_PID:
|
|
buf = new uint8_t[5] { 0x0E, 0x05, on_off_switch, 0x00, 0x00 };
|
|
hid_send_feature_report(dev_ctrl, buf, 5);
|
|
break;
|
|
default:
|
|
buf = new uint8_t[3] { 0x15, 0x00, on_off_switch };
|
|
hid_send_feature_report(dev_ctrl, buf, 3);
|
|
}
|
|
delete[] buf;
|
|
}
|
|
|
|
void RoccatVulcanKeyboardController::SendColors(std::vector<led_color> colors)
|
|
{
|
|
unsigned short packet_length;
|
|
unsigned char column_length;
|
|
unsigned char protocol_version;
|
|
|
|
switch(device_pid)
|
|
{
|
|
case ROCCAT_MAGMA_PID:
|
|
case ROCCAT_MAGMA_MINI_PID:
|
|
packet_length = 64;
|
|
column_length = 5;
|
|
protocol_version = 2;
|
|
break;
|
|
case ROCCAT_PYRO_PID:
|
|
packet_length = 378;
|
|
column_length = 1;
|
|
protocol_version = 2;
|
|
break;
|
|
case ROCCAT_VULCAN_PRO_PID:
|
|
packet_length = 384;
|
|
column_length = 12;
|
|
protocol_version = 2;
|
|
break;
|
|
default:
|
|
packet_length = 436;
|
|
column_length = 12;
|
|
protocol_version = 1;
|
|
}
|
|
|
|
unsigned char packet_num = ceil((float) packet_length / 64);
|
|
|
|
std::vector<std::vector<uint8_t>> bufs(packet_num);
|
|
|
|
for(int p = 0; p < packet_num; p++)
|
|
{
|
|
bufs[p].resize(65);
|
|
memset(&bufs[p][0], 0x00, sizeof(bufs[p][0]) * bufs[p].size());
|
|
}
|
|
|
|
if(protocol_version > 1)
|
|
{
|
|
for(unsigned int i = 0; i < packet_num; i++)
|
|
{
|
|
bufs[i][1] = 0xA1;
|
|
bufs[i][2] = i + 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
bufs[0][1] = 0xA1;
|
|
bufs[0][2] = 0x01;
|
|
}
|
|
|
|
unsigned char header_length_first = (packet_length > 255) ? 4 : 3;
|
|
|
|
if(header_length_first == 3)
|
|
{
|
|
bufs[0][3] = packet_length;
|
|
}
|
|
else
|
|
{
|
|
if(protocol_version > 1)
|
|
{
|
|
bufs[0][3] = packet_length % 256;
|
|
bufs[0][4] = packet_length / 256;
|
|
}
|
|
else
|
|
{
|
|
bufs[0][3] = packet_length / 256;
|
|
bufs[0][4] = packet_length % 256;
|
|
}
|
|
}
|
|
|
|
for(unsigned int i = 0; i < colors.size(); i++)
|
|
{
|
|
int coloumn = floor(colors[i].value / column_length);
|
|
int row = colors[i].value % column_length;
|
|
|
|
/*-----------------------------------------------------------------------*\
|
|
| This has to be split up for readability. |
|
|
| This assumes that the header for each packet besides the first |
|
|
| is either 0 bytes long (protocol v1) or as long as the first one (v2). |
|
|
| This currently covers all keyboards. |
|
|
| A solution unified solution that can handle general header length |
|
|
| independent of the first header would be desirable, but seems |
|
|
| too complicated for now. |
|
|
\*-----------------------------------------------------------------------*/
|
|
if(protocol_version == 1)
|
|
{
|
|
int offset = coloumn * 3 * column_length + row + header_length_first;
|
|
|
|
bufs[offset / 64][offset % 64 + 1] = RGBGetRValue(colors[i].color);
|
|
|
|
offset += column_length;
|
|
bufs[offset / 64][offset % 64 + 1] = RGBGetGValue(colors[i].color);
|
|
|
|
offset += column_length;
|
|
bufs[offset / 64][offset % 64 + 1] = RGBGetBValue(colors[i].color);
|
|
}
|
|
else
|
|
{
|
|
unsigned int data_length_packet = 64 - header_length_first;
|
|
|
|
int offset = coloumn * 3 * column_length + row;
|
|
|
|
bufs[offset / data_length_packet][offset % data_length_packet + header_length_first + 1] = RGBGetRValue(colors[i].color);
|
|
|
|
offset += column_length;
|
|
bufs[offset / data_length_packet][offset % data_length_packet + header_length_first + 1] = RGBGetGValue(colors[i].color);
|
|
|
|
offset += column_length;
|
|
bufs[offset / data_length_packet][offset % data_length_packet + header_length_first + 1] = RGBGetBValue(colors[i].color);
|
|
}
|
|
}
|
|
|
|
for(int p = 0; p < packet_num; p++)
|
|
{
|
|
hid_write(dev_led, &bufs[p][0], 65);
|
|
}
|
|
|
|
ClearResponses();
|
|
AwaitResponse(20);
|
|
}
|
|
|
|
void RoccatVulcanKeyboardController::SendMode(unsigned int mode, unsigned int speed, unsigned int brightness, std::vector<led_color> colors)
|
|
{
|
|
if(speed == 0) speed = ROCCAT_VULCAN_SPEED_DEFAULT;
|
|
if(brightness == 0) brightness = ROCCAT_VULCAN_BRIGHTNESS_DEFAULT;
|
|
|
|
unsigned short packet_length;
|
|
unsigned char protocol_version;
|
|
unsigned char column_length;
|
|
|
|
switch(device_pid)
|
|
{
|
|
case ROCCAT_PYRO_PID:
|
|
protocol_version = 2;
|
|
packet_length = 365;
|
|
column_length = 1;
|
|
break;
|
|
case ROCCAT_MAGMA_PID:
|
|
case ROCCAT_MAGMA_MINI_PID:
|
|
protocol_version = 2;
|
|
packet_length = 26;
|
|
column_length = 5;
|
|
break;
|
|
case ROCCAT_VULCAN_PRO_PID:
|
|
protocol_version = 2;
|
|
packet_length = 371;
|
|
column_length = 12;
|
|
break;
|
|
default:
|
|
protocol_version = 1;
|
|
packet_length = 443;
|
|
column_length = 12;
|
|
}
|
|
|
|
|
|
uint8_t* buf = new uint8_t[packet_length];
|
|
memset(buf, 0x00, packet_length);
|
|
|
|
unsigned char header_length = (packet_length > 255) ? 2 : 1;
|
|
|
|
buf[0] = (protocol_version == 1) ? 0x0D : 0x11;
|
|
|
|
if(header_length == 1)
|
|
{
|
|
buf[1] = packet_length;
|
|
}
|
|
else
|
|
{
|
|
buf[1] = packet_length % 256;
|
|
buf[2] = packet_length / 256;
|
|
}
|
|
|
|
unsigned char offset = header_length + 1;
|
|
|
|
buf[0 + offset] = 0x00;
|
|
buf[1 + offset] = mode;
|
|
buf[2 + offset] = speed;
|
|
|
|
if(protocol_version == 1)
|
|
{
|
|
buf[3 + offset] = 0x00;
|
|
buf[4 + offset] = brightness;
|
|
buf[5 + offset] = 0x00;
|
|
}
|
|
else
|
|
{
|
|
buf[3 + offset] = brightness;
|
|
buf[4 + offset] = 0x00;
|
|
buf[5 + offset] = 0x00;
|
|
}
|
|
|
|
|
|
for(unsigned int i = 0; i < colors.size(); i++)
|
|
{
|
|
int coloumn = floor(colors[i].value / column_length);
|
|
int row = colors[i].value % column_length;
|
|
|
|
int offset = coloumn * 3 * column_length + row + 9;
|
|
|
|
buf[offset] = RGBGetRValue(colors[i].color);
|
|
|
|
offset += column_length;
|
|
|
|
buf[offset] = RGBGetGValue(colors[i].color);
|
|
|
|
offset += column_length;
|
|
|
|
buf[offset] = RGBGetBValue(colors[i].color);
|
|
}
|
|
|
|
unsigned short total = 0;
|
|
for(int i = 0; i < packet_length - 2; i++) total += buf[i];
|
|
|
|
buf[packet_length - 2] = total & 0xFF;
|
|
buf[packet_length - 1] = total >> 8;
|
|
|
|
int ret = hid_send_feature_report(dev_ctrl, buf, packet_length);
|
|
|
|
delete[] buf;
|
|
}
|
|
|
|
void RoccatVulcanKeyboardController::WaitUntilReady()
|
|
{
|
|
unsigned short packet_length;
|
|
|
|
switch(device_pid)
|
|
{
|
|
case ROCCAT_PYRO_PID:
|
|
case ROCCAT_MAGMA_PID:
|
|
case ROCCAT_MAGMA_MINI_PID:
|
|
case ROCCAT_VULCAN_PRO_PID:
|
|
packet_length = 4;
|
|
break;
|
|
default:
|
|
packet_length = 3;
|
|
}
|
|
|
|
uint8_t* buf = new uint8_t[packet_length];
|
|
|
|
buf[0] = 0x04;
|
|
|
|
for(unsigned char i = 0; buf[1] != 1 && i < 100; i++)
|
|
{
|
|
if(i != 0)
|
|
{
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(25));
|
|
}
|
|
|
|
hid_get_feature_report(dev_ctrl, buf, packet_length);
|
|
}
|
|
|
|
delete[] buf;
|
|
}
|
|
|
|
void RoccatVulcanKeyboardController::AwaitResponse(int ms)
|
|
{
|
|
unsigned char usb_buf_out[65];
|
|
hid_read_timeout(dev_led, usb_buf_out, 65, ms);
|
|
}
|
|
|
|
void RoccatVulcanKeyboardController::ClearResponses()
|
|
{
|
|
int result = 1;
|
|
unsigned char usb_buf_flush[65];
|
|
while(result > 0)
|
|
{
|
|
result = hid_read_timeout(dev_led, usb_buf_flush, 65, 0);
|
|
}
|
|
}
|