319 lines
9.4 KiB
C++
319 lines
9.4 KiB
C++
/*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
* MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
//main class
|
|
#ifndef MOUSE_M908
|
|
#define MOUSE_M908
|
|
|
|
#include <libusb.h>
|
|
#include <map>
|
|
#include <array>
|
|
#include <string>
|
|
#include <algorithm>
|
|
#include <exception>
|
|
#include <regex>
|
|
#include <fstream>
|
|
#include <sstream>
|
|
#include <utility>
|
|
#include <iostream>
|
|
|
|
/**
|
|
* The main class representing the M908 mouse.
|
|
* This class has member functions to open, close and apply settings to the mouse.
|
|
*
|
|
* There are 3 main types of functions:
|
|
* - set_*: setters
|
|
* - get_*: getters
|
|
* - write_*: write the settings to the mouse
|
|
*
|
|
* Therefore the general procedure when changing settings on the mouse is as follows:
|
|
* 1. open_mouse() or open_mouse_bus_device()
|
|
* 2. set_* (this step only changes the internal state of the class)
|
|
* 3. write_* (this step sends the internal state of the class to the mouse)
|
|
* 4. close_mouse()
|
|
*/
|
|
class mouse_m908{
|
|
|
|
public:
|
|
|
|
// constructor
|
|
mouse_m908();
|
|
|
|
|
|
|
|
// enums
|
|
/// The available profiles
|
|
enum m908_profile{
|
|
profile_1 = 0,
|
|
profile_2 = 1,
|
|
profile_3 = 2,
|
|
profile_4 = 3,
|
|
profile_5 = 4,
|
|
};
|
|
|
|
/// The available led modes
|
|
enum m908_lightmode{
|
|
lightmode_breathing,
|
|
lightmode_rainbow,
|
|
lightmode_static,
|
|
lightmode_wave,
|
|
lightmode_alternating,
|
|
lightmode_reactive,
|
|
lightmode_flashing,
|
|
lightmode_off,
|
|
};
|
|
|
|
/// The available USB report rates (polling rates)
|
|
enum m908_report_rate{
|
|
r_125Hz,
|
|
r_250Hz,
|
|
r_500Hz,
|
|
r_1000Hz
|
|
};
|
|
|
|
|
|
|
|
//setter functions
|
|
/** Set the currently active profile
|
|
*/
|
|
int set_profile( m908_profile profile );
|
|
|
|
/** Set the scrollspeed for the specified profile
|
|
* \see _scrollspeed_min
|
|
* \see _scrollspeed_max
|
|
* \return 0 if successful, 1 if out of bounds
|
|
*/
|
|
int set_scrollspeed( m908_profile profile, uint8_t speed );
|
|
|
|
/** Set the led mode for the specified profile
|
|
* \see m908_lightmode
|
|
* \return 0 if successful
|
|
*/
|
|
int set_lightmode( m908_profile profile, m908_lightmode lightmode );
|
|
|
|
/** Set the led color for the specified profile
|
|
* \param color color as {r, g, b}
|
|
* \return 0 if successful
|
|
*/
|
|
int set_color( m908_profile profile, std::array<uint8_t, 3> color );
|
|
|
|
/** Set the led brightness for the specified profile
|
|
* \see _brightness_min
|
|
* \see _brightness_max
|
|
* \return 0 if successful, 1 if out of bounds
|
|
*/
|
|
int set_brightness( m908_profile profile, uint8_t brightness );
|
|
|
|
/** Set the led animation speed for the specified profile
|
|
* \see _speed_min
|
|
* \see _speed_max
|
|
* \return 0 if successful, 1 if out of bounds
|
|
*/
|
|
int set_speed( m908_profile profile, uint8_t speed );
|
|
|
|
/** Enables/Disables a dpi level for the specified profile
|
|
* \see _level_min
|
|
* \see _level_max
|
|
* \return 0 if successful, 1 if out of bounds
|
|
*/
|
|
int set_dpi_enable( m908_profile profile, int level, bool enabled );
|
|
|
|
/** Set the value of a dpi level for the specified profile
|
|
* \see _dpi_min
|
|
* \see _dpi_max
|
|
* \see _level_min
|
|
* \see _level_max
|
|
* \return 0 if successful, 1 if out of bounds
|
|
*/
|
|
int set_dpi( m908_profile profile, int level, uint8_t dpi );
|
|
|
|
/** Set a mapping for a button for the specified profile
|
|
* \param mapping 4 bytes for the usb data packets
|
|
* \return 0 if successful
|
|
*/
|
|
int set_key_mapping( m908_profile profile, int key, std::array<uint8_t, 4> mapping );
|
|
|
|
/** Set a mapping for a button for the specified profile
|
|
* \param mapping button name (see keymap.md)
|
|
* \return 0 if successful, 1 if mapping is invalid
|
|
*/
|
|
int set_key_mapping( m908_profile profile, int key, std::string mapping );
|
|
|
|
/** Set the USB poll rate for the specified profile
|
|
* \see m908_report_rate
|
|
* \return 0 if successful
|
|
*/
|
|
int set_report_rate( m908_profile profile, m908_report_rate report_rate );
|
|
|
|
/** Load the macro from the specified file into the specified slot
|
|
* \param macro_number macro slot (1-15)
|
|
* \return 0 if successful
|
|
*/
|
|
int set_macro( int macro_number, std::string file );
|
|
|
|
/** Set how many times the specified macro should be repeated
|
|
* \param macro_number macro slot (1-15)
|
|
* \return 0 if successful
|
|
*/
|
|
int set_macro_repeat( int macro_number, uint8_t repeat );
|
|
|
|
/** Set whether to try to detach the kernel driver when opening the mouse
|
|
*/
|
|
int set_detach_kernel_driver( bool detach_kernel_driver );
|
|
|
|
|
|
|
|
//getter functions
|
|
m908_profile get_profile();
|
|
uint8_t get_scrollspeed( m908_profile profile );
|
|
m908_lightmode get_lightmode( m908_profile profile );
|
|
void get_color( m908_profile profile, std::array<uint8_t, 3> &color );
|
|
uint8_t get_brightness( m908_profile profile );
|
|
uint8_t get_speed( m908_profile profile );
|
|
bool get_dpi_enable( m908_profile profile, int level );
|
|
uint8_t get_dpi( m908_profile profile, int level );
|
|
m908_report_rate get_report_rate( m908_profile profile );
|
|
uint8_t get_macro_repeat( int macro_number );
|
|
|
|
|
|
|
|
//writer functions (apply settings to mouse)
|
|
/** Write the currently active profile to the mouse
|
|
* \return 0 if successful
|
|
*/
|
|
int write_profile();
|
|
|
|
/** Write the settings (leds, button mapping, dpi, etc.) to the mouse
|
|
* \return 0 if successful
|
|
*/
|
|
int write_settings();
|
|
|
|
/** Write a macro to the mouse
|
|
* \return 0 if successful
|
|
*/
|
|
int write_macro( int macro_number );
|
|
|
|
/** Write the number of repeats for a macro to the mouse
|
|
* \return 0 if successful
|
|
*/
|
|
int write_macro_repeat( int macro_number );
|
|
|
|
|
|
|
|
//helper functions
|
|
/** Init libusb and open the mouse by its USB VID and PID
|
|
* \return 0 if successful
|
|
*/
|
|
int open_mouse();
|
|
|
|
/** Init libusb and open the mouse by the USB bus and device adress
|
|
* \return 0 if successful
|
|
*/
|
|
int open_mouse_bus_device( uint8_t bus, uint8_t device );
|
|
|
|
/** Close the mouse and libusb
|
|
* \return 0 if successful (always at the moment)
|
|
*/
|
|
int close_mouse();
|
|
|
|
|
|
//reader functions (get settings from the mouse)
|
|
/**
|
|
* Read the settings and print the raw data
|
|
*/
|
|
int dump_settings();
|
|
|
|
private:
|
|
|
|
// whether to detach kernel driver
|
|
bool _detach_kernel_driver = true;
|
|
|
|
//usb device vars
|
|
static const uint16_t _mouse_vid;
|
|
static const uint16_t _mouse_pid;
|
|
libusb_device_handle* _handle;
|
|
bool _detached_driver_0 = false;
|
|
bool _detached_driver_1 = false;
|
|
bool _detached_driver_2 = false;
|
|
|
|
//setting min and max values
|
|
static const uint8_t _scrollspeed_min, _scrollspeed_max;
|
|
static const uint8_t _brightness_min, _brightness_max;
|
|
static const uint8_t _speed_min, _speed_max;
|
|
static const uint8_t _level_min, _level_max;
|
|
static const uint8_t _dpi_min, _dpi_max;
|
|
|
|
//setting vars
|
|
m908_profile _profile;
|
|
std::array<uint8_t, 5> _scrollspeeds;
|
|
std::array<m908_lightmode, 5> _lightmodes;
|
|
std::array<std::array<uint8_t, 3>, 5> _colors;
|
|
std::array<uint8_t, 5> _brightness_levels;
|
|
std::array<uint8_t, 5> _speed_levels;
|
|
std::array<std::array<bool, 5>, 5> _dpi_enabled;
|
|
std::array<std::array<uint8_t, 5>, 5> _dpi_levels;
|
|
std::array<std::array<std::array<uint8_t, 4>, 20>, 5> _keymap_data;
|
|
std::array<m908_report_rate, 5> _report_rates;
|
|
std::array<std::array<uint8_t, 256>, 15> _macro_data;
|
|
std::array<uint8_t, 15> _macro_repeat;
|
|
|
|
//mapping of button names to values
|
|
/// Values/keycodes of mouse buttons and special button functions
|
|
static std::map< std::string, std::array<uint8_t, 3> > _keycodes;
|
|
/// Values of keyboard modifiers
|
|
static const std::map< std::string, uint8_t > _keyboard_modifier_values;
|
|
/// Values/keycodes of keyboard keys
|
|
static std::map< std::string, uint8_t > _keyboard_key_values;
|
|
|
|
//usb data packets
|
|
/// Used for changing the active profile
|
|
static uint8_t _data_profile[6][16];
|
|
/// Used for sending the settings, part 1/3
|
|
static uint8_t _data_settings_1[15][16];
|
|
/// Used for sending the settings, part 2/3
|
|
static uint8_t _data_settings_2[64];
|
|
/// Used for sending the settings, part 3/3
|
|
static uint8_t _data_settings_3[140][16];
|
|
/// Used for sending a macro, part 1/3
|
|
static uint8_t _data_macros_1[16];
|
|
/// Used for sending a macro, part 2/3
|
|
static uint8_t _data_macros_2[256];
|
|
/// Used for sending a macro, part 3/3
|
|
static uint8_t _data_macros_3[16];
|
|
/// Lookup table for used when specifying which slot to send a macro to
|
|
static uint8_t _data_macros_codes[15][2];
|
|
/// Used to send the number repeats for a macro
|
|
static uint8_t _data_macros_repeat[16];
|
|
/// Used to read the settings, part 1/3
|
|
static uint8_t _data_read_1[9][16];
|
|
/// Used to read the settings, part 2/3
|
|
static uint8_t _data_read_2[85][64];
|
|
/// Used to read the settings, part 3/3
|
|
static uint8_t _data_read_3[101][16];
|
|
};
|
|
|
|
/* the files are now compiled individually
|
|
#include "data.cpp"
|
|
#include "constructor.cpp"
|
|
#include "helpers.cpp"
|
|
#include "getters.cpp"
|
|
#include "setters.cpp"
|
|
#include "writers.cpp"
|
|
*/
|
|
|
|
#endif
|