2 #include <linux/joystick.h>
3 #include <msp/core/algorithm.h>
4 #include <msp/core/systemerror.h>
5 #include <msp/fs/dir.h>
6 #include <msp/io/handle_private.h>
7 #include <msp/strings/format.h>
8 #include "gamecontroller.h"
9 #include "gamecontroller_platform.h"
16 vector<string> GameController::Private::detected_controllers;
18 GameController::GameController(unsigned index):
23 if(index>=Private::detected_controllers.size())
24 throw device_not_available(format("GameController(%d)", index));
26 JsDevice *device = new JsDevice(Private::detected_controllers[index]);
30 priv->dev->signal_data_available.connect(sigc::mem_fun(this, static_cast<void (GameController::*)()>(&GameController::tick)));
31 name = priv->dev->get_name();
35 GameController::~GameController()
41 unsigned GameController::detect()
43 Private::detected_controllers.clear();
45 FS::Path dev_input = "/dev/input";
46 vector<string> devices = FS::list_filtered(dev_input, "^js[0-9]+");
48 for(vector<string>::const_iterator i=devices.begin(); i!=devices.end(); ++i)
49 // TODO check permissions
50 Private::detected_controllers.push_back((dev_input / *i).str());
53 n_detected_controllers = Private::detected_controllers.size();
55 return Private::detected_controllers.size();
58 void GameController::use_event_dispatcher(IO::EventDispatcher *ed)
61 event_disp->remove(*priv->dev);
64 event_disp->add(*priv->dev);
67 void GameController::tick()
73 if(!first && !IO::poll(*priv->dev, IO::P_INPUT, Time::zero))
77 unsigned len = priv->dev->read(reinterpret_cast<char *>(events), sizeof(events));
79 unsigned count = len/sizeof(js_event);
80 for(unsigned i=0; i<count; ++i)
82 unsigned type = events[i].type&0x7F;
83 bool init = events[i].type&JS_EVENT_INIT;
84 if(type==JS_EVENT_AXIS)
85 set_axis_value(events[i].number, events[i].value/32768.0f, !init);
86 else if(type==JS_EVENT_BUTTON)
87 set_button_state(events[i].number, events[i].value, !init);
90 if(len<sizeof(events))
96 void GameController::tick(const Time::TimeDelta &timeout)
98 if(IO::poll(*priv->dev, IO::P_INPUT, timeout))
103 JsDevice::JsDevice(const string &fn)
106 *handle = open(fn.c_str(), O_RDONLY);
108 throw system_error(format("open(%s)", fn));
109 set_events(IO::P_INPUT);
112 JsDevice::~JsDevice()
117 string JsDevice::get_name() const
119 char buf[128] = { 0 };
120 int ret = ioctl(*handle, JSIOCGNAME(sizeof(buf)), buf);
122 throw system_error("ioctl(JSIOCGNAME)");
126 unsigned JsDevice::do_read(char *buf, unsigned size)
128 return IO::sys_read(handle, buf, size);
131 unsigned JsDevice::do_write(const char *, unsigned)
133 throw IO::invalid_access(IO::M_WRITE);