mirror of
https://github.com/jtgans/g13gui.git
synced 2025-06-20 08:23:50 -04:00
This preps us for proper system installs to distributions, finding depending libraries, and allowing us to build the foundation for the GUI and support tooling next. The plan is to run g13d as a system daemon managed by systemd. We'll use the system-wide input group to control access to the daemon. - Add a CMakeLists for etc so we can install the udev rules. - Move the src dir to g13d to disambiguate a bit. - Update the toplevel Makefile so we can still build the old way (for now).
584 lines
14 KiB
C++
584 lines
14 KiB
C++
#ifndef __G13_H__
|
|
#define __G13_H__
|
|
|
|
|
|
#include "helper.h"
|
|
|
|
#include <boost/log/trivial.hpp>
|
|
|
|
#include <libusb-1.0/libusb.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <fstream>
|
|
#include <linux/uinput.h>
|
|
#include <fcntl.h>
|
|
|
|
// *************************************************************************
|
|
|
|
namespace G13 {
|
|
|
|
#define G13_LOG( level, message ) BOOST_LOG_TRIVIAL( level ) << message
|
|
#define G13_OUT( message ) BOOST_LOG_TRIVIAL( info ) << message
|
|
|
|
const size_t G13_INTERFACE = 0;
|
|
const size_t G13_KEY_ENDPOINT = 1;
|
|
const size_t G13_LCD_ENDPOINT = 2;
|
|
const size_t G13_KEY_READ_TIMEOUT = 0;
|
|
const size_t G13_VENDOR_ID = 0x046d;
|
|
const size_t G13_PRODUCT_ID = 0xc21c;
|
|
const size_t G13_REPORT_SIZE = 8;
|
|
const size_t G13_LCD_BUFFER_SIZE = 0x3c0;
|
|
const size_t G13_NUM_KEYS = 40;
|
|
|
|
|
|
const size_t G13_LCD_COLUMNS = 160;
|
|
const size_t G13_LCD_ROWS = 48;
|
|
const size_t G13_LCD_BYTES_PER_ROW = G13_LCD_COLUMNS/8;
|
|
const size_t G13_LCD_BUF_SIZE = G13_LCD_ROWS * G13_LCD_BYTES_PER_ROW;
|
|
const size_t G13_LCD_TEXT_CHEIGHT = 8;
|
|
const size_t G13_LCD_TEXT_ROWS = 160 / G13_LCD_TEXT_CHEIGHT;
|
|
|
|
enum stick_mode_t { STICK_ABSOLUTE, STICK_RELATIVE, STICK_KEYS, STICK_CALCENTER, STICK_CALBOUNDS, STICK_CALNORTH };
|
|
|
|
typedef int LINUX_KEY_VALUE;
|
|
const LINUX_KEY_VALUE BAD_KEY_VALUE = -1;
|
|
|
|
typedef int G13_KEY_INDEX;
|
|
|
|
// *************************************************************************
|
|
|
|
using Helper::repr;
|
|
using Helper::find_or_throw;
|
|
|
|
// *************************************************************************
|
|
|
|
class G13_Action;
|
|
class G13_Stick;
|
|
class G13_LCD;
|
|
class G13_Profile;
|
|
class G13_Device;
|
|
class G13_Manager;
|
|
|
|
class G13_CommandException : public std::exception {
|
|
public:
|
|
G13_CommandException( const std::string &reason ) : _reason(reason) {}
|
|
virtual ~G13_CommandException() throw () {}
|
|
virtual const char *what() const throw () { return _reason.c_str(); }
|
|
|
|
std::string _reason;
|
|
};
|
|
|
|
// *************************************************************************
|
|
|
|
/*! holds potential actions which can be bound to G13 activity
|
|
*
|
|
*/
|
|
class G13_Action {
|
|
public:
|
|
G13_Action( G13_Device & keypad ) : _keypad(keypad) {}
|
|
virtual ~G13_Action();
|
|
|
|
virtual void act( G13_Device &, bool is_down ) = 0;
|
|
virtual void dump( std::ostream & ) const = 0;
|
|
|
|
void act( bool is_down ) { act( keypad(), is_down ); }
|
|
|
|
G13_Device & keypad() { return _keypad; }
|
|
const G13_Device & keypad() const { return _keypad; }
|
|
|
|
G13_Manager &manager();
|
|
const G13_Manager &manager() const;
|
|
|
|
private:
|
|
G13_Device & _keypad;
|
|
};
|
|
|
|
|
|
/*!
|
|
* action to send one or more keystrokes
|
|
*/
|
|
class G13_Action_Keys : public G13_Action {
|
|
public:
|
|
G13_Action_Keys( G13_Device & keypad, const std::string &keys );
|
|
virtual ~G13_Action_Keys();
|
|
|
|
virtual void act( G13_Device &, bool is_down );
|
|
virtual void dump( std::ostream & ) const;
|
|
|
|
std::vector<LINUX_KEY_VALUE> _keys;
|
|
};
|
|
|
|
/*!
|
|
* action to send a string to the output pipe
|
|
*/
|
|
class G13_Action_PipeOut : public G13_Action {
|
|
public:
|
|
G13_Action_PipeOut( G13_Device & keypad, const std::string &out );
|
|
virtual ~G13_Action_PipeOut();
|
|
|
|
virtual void act( G13_Device &, bool is_down );
|
|
virtual void dump( std::ostream & ) const;
|
|
|
|
std::string _out;
|
|
};
|
|
|
|
/*!
|
|
* action to send a command to the g13
|
|
*/
|
|
class G13_Action_Command : public G13_Action {
|
|
public:
|
|
G13_Action_Command( G13_Device & keypad, const std::string &cmd );
|
|
virtual ~G13_Action_Command();
|
|
|
|
virtual void act( G13_Device &, bool is_down );
|
|
virtual void dump( std::ostream & ) const;
|
|
|
|
std::string _cmd;
|
|
};
|
|
|
|
|
|
typedef boost::shared_ptr<G13_Action> G13_ActionPtr;
|
|
|
|
// *************************************************************************
|
|
template <class PARENT_T>
|
|
class G13_Actionable {
|
|
public:
|
|
G13_Actionable( PARENT_T &parent_arg, const std::string &name ) :
|
|
_parent_ptr(&parent_arg), _name(name)
|
|
{}
|
|
virtual ~G13_Actionable() { _parent_ptr = 0; }
|
|
|
|
G13_ActionPtr action() const { return _action; }
|
|
const std::string & name() const { return _name; }
|
|
PARENT_T & parent() { return *_parent_ptr; }
|
|
const PARENT_T & parent() const { return *_parent_ptr; }
|
|
G13_Manager & manager() { return _parent_ptr->manager(); }
|
|
const G13_Manager & manager() const { return _parent_ptr->manager(); }
|
|
|
|
virtual void set_action( const G13_ActionPtr &action ) {
|
|
_action = action;
|
|
}
|
|
|
|
protected:
|
|
|
|
std::string _name;
|
|
G13_ActionPtr _action;
|
|
|
|
private:
|
|
PARENT_T *_parent_ptr;
|
|
};
|
|
|
|
// *************************************************************************
|
|
/*! manages the bindings for a G13 key
|
|
*
|
|
*/
|
|
class G13_Key : public G13_Actionable<G13_Profile> {
|
|
public:
|
|
|
|
|
|
void dump( std::ostream &o ) const;
|
|
G13_KEY_INDEX index() const { return _index.index; }
|
|
|
|
void parse_key( unsigned char *byte, G13_Device *g13 );
|
|
|
|
protected:
|
|
|
|
|
|
struct KeyIndex {
|
|
KeyIndex( int key ) :
|
|
index(key),
|
|
offset( key / 8 ),
|
|
mask( 1 << (key % 8) )
|
|
{}
|
|
|
|
int index;
|
|
unsigned char offset;
|
|
unsigned char mask;
|
|
};
|
|
|
|
// G13_Profile is the only class able to instantiate G13_Keys
|
|
friend class G13_Profile;
|
|
|
|
G13_Key( G13_Profile & mode, const std::string &name, int index ) : G13_Actionable<G13_Profile>( mode, name ),
|
|
_index(index),
|
|
_should_parse(true)
|
|
{
|
|
}
|
|
|
|
G13_Key( G13_Profile & mode, const G13_Key &key ) : G13_Actionable<G13_Profile>( mode, key.name() ),
|
|
_index(key._index),
|
|
_should_parse(key._should_parse)
|
|
{
|
|
set_action(key.action());
|
|
}
|
|
|
|
|
|
KeyIndex _index;
|
|
bool _should_parse;
|
|
};
|
|
|
|
/*!
|
|
* Represents a set of configured key mappings
|
|
*
|
|
* This allows a keypad to have multiple configured
|
|
* profiles and switch between them easily
|
|
*/
|
|
class G13_Profile {
|
|
public:
|
|
G13_Profile(G13_Device &keypad, const std::string &name_arg ) : _keypad(keypad), _name(name_arg) {
|
|
_init_keys();
|
|
}
|
|
G13_Profile(const G13_Profile &other, const std::string &name_arg ) : _keypad(other._keypad), _name(name_arg), _keys(other._keys)
|
|
{
|
|
}
|
|
|
|
|
|
// search key by G13 keyname
|
|
G13_Key * find_key( const std::string &keyname );
|
|
|
|
void dump( std::ostream &o ) const;
|
|
|
|
void parse_keys( unsigned char *buf );
|
|
const std::string &name() const { return _name; }
|
|
|
|
const G13_Manager &manager() const;
|
|
|
|
protected:
|
|
G13_Device &_keypad;
|
|
std::vector<G13_Key> _keys;
|
|
std::string _name;
|
|
|
|
void _init_keys();
|
|
};
|
|
|
|
typedef boost::shared_ptr<G13_Profile> ProfilePtr;
|
|
|
|
class G13_FontChar {
|
|
public:
|
|
static const int CHAR_BUF_SIZE = 8;
|
|
enum FONT_FLAGS { FF_ROTATE= 0x01 };
|
|
|
|
G13_FontChar() {
|
|
memset(bits_regular, 0, CHAR_BUF_SIZE);
|
|
memset(bits_inverted, 0, CHAR_BUF_SIZE);
|
|
}
|
|
void set_character( unsigned char *data, int width, unsigned flags );
|
|
unsigned char bits_regular[CHAR_BUF_SIZE];
|
|
unsigned char bits_inverted[CHAR_BUF_SIZE];
|
|
};
|
|
|
|
class G13_Font {
|
|
public:
|
|
G13_Font();
|
|
G13_Font( const std::string &name, unsigned int width = 8 );
|
|
|
|
|
|
void set_character( unsigned int c, unsigned char *data );
|
|
|
|
template < class ARRAY_T, class FLAGST >
|
|
void install_font( ARRAY_T &data, FLAGST flags, int first = 0 );
|
|
|
|
const std::string &name() const { return _name; }
|
|
unsigned int width() const { return _width; }
|
|
|
|
const G13_FontChar &char_data( unsigned int x ) { return _chars[x]; }
|
|
protected:
|
|
std::string _name;
|
|
unsigned int _width;
|
|
|
|
G13_FontChar _chars[256];
|
|
|
|
//unsigned char font_basic[256][8];
|
|
//unsigned char font_inverted[256][8];
|
|
};
|
|
typedef boost::shared_ptr<G13_Font> FontPtr;
|
|
|
|
class G13_LCD {
|
|
public:
|
|
|
|
|
|
G13_LCD( G13_Device &keypad );
|
|
|
|
G13_Device &_keypad;
|
|
unsigned char image_buf[G13_LCD_BUF_SIZE+8];
|
|
unsigned cursor_row;
|
|
unsigned cursor_col;
|
|
int text_mode;
|
|
|
|
void image(unsigned char *data, int size);
|
|
void image_send() {
|
|
image( image_buf, G13_LCD_BUF_SIZE );
|
|
}
|
|
|
|
void image_test( int x, int y );
|
|
void image_clear() {
|
|
memset( image_buf, 0, G13_LCD_BUF_SIZE );
|
|
}
|
|
|
|
unsigned image_byte_offset( unsigned row, unsigned col ) {
|
|
return col + (row /8 ) * G13_LCD_BYTES_PER_ROW * 8;
|
|
}
|
|
|
|
void image_setpixel( unsigned row, unsigned col );
|
|
void image_clearpixel( unsigned row, unsigned col );
|
|
|
|
|
|
|
|
void write_char( char c, int row = -1, int col = -1);
|
|
void write_string( const char *str );
|
|
void write_pos(int row, int col );
|
|
|
|
};
|
|
using Helper::repr;
|
|
|
|
typedef Helper::Coord<int> G13_StickCoord;
|
|
typedef Helper::Bounds<int> G13_StickBounds;
|
|
typedef Helper::Coord<double> G13_ZoneCoord;
|
|
typedef Helper::Bounds<double> G13_ZoneBounds;
|
|
|
|
// *************************************************************************
|
|
|
|
class G13_StickZone : public G13_Actionable<G13_Stick> {
|
|
public:
|
|
|
|
G13_StickZone( G13_Stick &, const std::string &name, const G13_ZoneBounds &, G13_ActionPtr = 0 );
|
|
|
|
bool operator == ( const G13_StickZone &other ) const { return _name == other._name; }
|
|
|
|
void dump( std::ostream & ) const;
|
|
|
|
void parse_key( unsigned char *byte, G13_Device *g13);
|
|
void test( const G13_ZoneCoord &loc );
|
|
void set_bounds( const G13_ZoneBounds &bounds ) { _bounds = bounds; }
|
|
|
|
|
|
protected:
|
|
|
|
bool _active;
|
|
|
|
G13_ZoneBounds _bounds;
|
|
|
|
};
|
|
|
|
typedef boost::shared_ptr< G13_StickZone> G13_StickZonePtr;
|
|
|
|
// *************************************************************************
|
|
|
|
class G13_Stick {
|
|
public:
|
|
G13_Stick( G13_Device &keypad );
|
|
|
|
void parse_joystick(unsigned char *buf);
|
|
|
|
void set_mode( stick_mode_t );
|
|
G13_StickZone * zone( const std::string &, bool create=false );
|
|
void remove_zone( const G13_StickZone &zone );
|
|
|
|
const std::vector<G13_StickZone> & zones() const { return _zones; }
|
|
|
|
void dump( std::ostream & ) const;
|
|
|
|
protected:
|
|
|
|
void _recalc_calibrated();
|
|
|
|
G13_Device &_keypad;
|
|
std::vector<G13_StickZone> _zones;
|
|
|
|
G13_StickBounds _bounds;
|
|
G13_StickCoord _center_pos;
|
|
G13_StickCoord _north_pos;
|
|
|
|
G13_StickCoord _current_pos;
|
|
|
|
stick_mode_t _stick_mode;
|
|
|
|
};
|
|
|
|
// *************************************************************************
|
|
|
|
class G13_Device {
|
|
public:
|
|
|
|
G13_Device( G13_Manager &manager, libusb_device_handle *handle, int id );
|
|
|
|
|
|
G13_Manager &manager() { return _manager; }
|
|
const G13_Manager &manager() const { return _manager; }
|
|
|
|
G13_LCD &lcd() { return _lcd; }
|
|
const G13_LCD &lcd() const { return _lcd; }
|
|
G13_Stick &stick() { return _stick; }
|
|
const G13_Stick &stick() const { return _stick; }
|
|
|
|
FontPtr switch_to_font( const std::string &name );
|
|
void switch_to_profile( const std::string &name );
|
|
ProfilePtr profile( const std::string &name );
|
|
|
|
void dump(std::ostream &, int detail = 0 );
|
|
void command(char const *str);
|
|
|
|
void read_commands();
|
|
void read_config_file( const std::string &filename );
|
|
|
|
int read_keys();
|
|
void parse_joystick(unsigned char *buf);
|
|
|
|
G13_ActionPtr make_action( const std::string & );
|
|
|
|
void set_key_color( int red, int green, int blue );
|
|
void set_mode_leds( int leds );
|
|
|
|
|
|
void send_event( int type, int code, int val );
|
|
void write_output_pipe( const std::string &out );
|
|
|
|
void write_lcd( unsigned char *data, size_t size );
|
|
|
|
bool is_set(int key) ;
|
|
bool update(int key, bool v) ;
|
|
|
|
// used by G13_Manager
|
|
void cleanup();
|
|
void register_context(libusb_context *ctx);
|
|
void write_lcd_file( const std::string &filename);
|
|
|
|
G13_Font ¤t_font() { return *_current_font; }
|
|
G13_Profile ¤t_profile() { return *_current_profile; }
|
|
|
|
int id_within_manager() const { return _id_within_manager; }
|
|
|
|
typedef boost::function<void ( const char * )> COMMAND_FUNCTION;
|
|
typedef std::map<std::string, COMMAND_FUNCTION> CommandFunctionTable;
|
|
|
|
protected:
|
|
|
|
void _init_fonts();
|
|
void init_lcd();
|
|
void _init_commands();
|
|
|
|
|
|
//typedef void (COMMAND_FUNCTION)( G13_Device*, const char *, const char * );
|
|
CommandFunctionTable _command_table;
|
|
|
|
struct timeval _event_time;
|
|
struct input_event _event;
|
|
|
|
int _id_within_manager;
|
|
libusb_device_handle *handle;
|
|
libusb_context *ctx;
|
|
|
|
int _uinput_fid;
|
|
|
|
int _input_pipe_fid;
|
|
std::string _input_pipe_name;
|
|
int _output_pipe_fid;
|
|
std::string _output_pipe_name;
|
|
|
|
std::map<std::string,FontPtr> _fonts;
|
|
FontPtr _current_font;
|
|
std::map<std::string,ProfilePtr> _profiles;
|
|
ProfilePtr _current_profile;
|
|
|
|
G13_Manager &_manager;
|
|
G13_LCD _lcd;
|
|
G13_Stick _stick;
|
|
|
|
|
|
bool keys[G13_NUM_KEYS];
|
|
};
|
|
|
|
// *************************************************************************
|
|
|
|
|
|
/*!
|
|
* top level class, holds what would otherwise be in global variables
|
|
*/
|
|
|
|
class G13_Manager {
|
|
public:
|
|
G13_Manager();
|
|
|
|
G13_KEY_INDEX find_g13_key_value( const std::string &keyname ) const;
|
|
std::string find_g13_key_name( G13_KEY_INDEX ) const;
|
|
|
|
LINUX_KEY_VALUE find_input_key_value( const std::string &keyname ) const;
|
|
std::string find_input_key_name( LINUX_KEY_VALUE ) const;
|
|
|
|
void set_logo( const std::string &fn ) { logo_filename = fn; }
|
|
int run();
|
|
|
|
std::string string_config_value( const std::string &name ) const;
|
|
void set_string_config_value( const std::string &name, const std::string &val );
|
|
|
|
std::string make_pipe_name( G13_Device *d, bool is_input );
|
|
|
|
void set_log_level( ::boost::log::trivial::severity_level lvl );
|
|
void set_log_level( const std::string & );
|
|
|
|
protected:
|
|
|
|
void init_keynames();
|
|
void display_keys();
|
|
void discover_g13s(libusb_device **devs, ssize_t count, std::vector<G13_Device*>& g13s);
|
|
void cleanup();
|
|
|
|
|
|
|
|
std::string logo_filename;
|
|
libusb_device **devs;
|
|
|
|
libusb_context *ctx;
|
|
std::vector<G13_Device*> g13s;
|
|
|
|
|
|
std::map<G13_KEY_INDEX,std::string> g13_key_to_name;
|
|
std::map<std::string,G13_KEY_INDEX> g13_name_to_key;
|
|
std::map<LINUX_KEY_VALUE,std::string> input_key_to_name;
|
|
std::map<std::string,LINUX_KEY_VALUE> input_name_to_key;
|
|
|
|
std::map<std::string, std::string> _string_config_values;
|
|
|
|
static bool running;
|
|
static void set_stop(int);
|
|
};
|
|
|
|
// *************************************************************************
|
|
|
|
// inlines
|
|
|
|
inline G13_Manager &G13_Action::manager() {
|
|
return _keypad.manager();
|
|
}
|
|
|
|
inline const G13_Manager &G13_Action::manager() const{
|
|
return _keypad.manager();
|
|
}
|
|
|
|
inline bool G13_Device::is_set(int key)
|
|
{
|
|
return keys[key];
|
|
}
|
|
|
|
inline bool G13_Device::update(int key, bool v) {
|
|
bool old = keys[key];
|
|
keys[key] = v;
|
|
return old != v;
|
|
}
|
|
|
|
inline const G13_Manager &G13_Profile::manager() const
|
|
{
|
|
return _keypad.manager();
|
|
}
|
|
|
|
// *************************************************************************
|
|
|
|
|
|
} // namespace G13
|
|
|
|
#endif // __G13_H__
|