2008-04-11 05:43:15 -06:00
|
|
|
|
/*
|
2008-04-24 11:12:59 -06:00
|
|
|
|
** Xbox360 USB Gamepad Userspace Driver
|
2008-04-11 05:43:15 -06:00
|
|
|
|
** Copyright (C) 2008 Ingo Ruhnke <grumbel@gmx.de>
|
|
|
|
|
**
|
|
|
|
|
** 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
2008-04-10 14:04:38 -06:00
|
|
|
|
#include <signal.h>
|
2008-04-10 11:33:56 -06:00
|
|
|
|
#include <boost/format.hpp>
|
|
|
|
|
#include <usb.h>
|
|
|
|
|
#include <unistd.h>
|
2008-04-10 10:38:50 -06:00
|
|
|
|
#include <iostream>
|
2008-04-10 15:43:47 -06:00
|
|
|
|
|
|
|
|
|
#include "uinput.hpp"
|
2008-04-13 12:19:54 -06:00
|
|
|
|
#include "xboxdrv.hpp"
|
2008-04-10 10:38:50 -06:00
|
|
|
|
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
XPadDevice xpad_devices[] = {
|
2008-04-12 07:07:20 -06:00
|
|
|
|
// Evil?! Anymore info we could use to identify the devices?
|
2008-04-11 13:03:43 -06:00
|
|
|
|
// { GAMEPAD_XBOX, 0x0000, 0x0000, "Generic X-Box pad" },
|
|
|
|
|
// { GAMEPAD_XBOX, 0xffff, 0xffff, "Chinese-made Xbox Controller" },
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
|
|
|
|
// These should work
|
2008-04-11 09:47:11 -06:00
|
|
|
|
{ GAMEPAD_XBOX, 0x045e, 0x0202, "Microsoft X-Box pad v1 (US)" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x045e, 0x0285, "Microsoft X-Box pad (Japan)" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x045e, 0x0285, "Microsoft Xbox Controller S" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x045e, 0x0287, "Microsoft Xbox Controller S" },
|
2008-04-12 07:12:58 -06:00
|
|
|
|
{ GAMEPAD_XBOX, 0x045e, 0x0289, "Microsoft X-Box pad v2 (US)" }, // duplicate
|
|
|
|
|
{ GAMEPAD_XBOX, 0x045e, 0x0289, "Microsoft Xbox Controller S" }, // duplicate
|
2008-04-11 09:47:11 -06:00
|
|
|
|
{ GAMEPAD_XBOX, 0x046d, 0xca84, "Logitech Xbox Cordless Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x046d, 0xca88, "Logitech Compact Controller for Xbox" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x05fd, 0x1007, "Mad Catz Controller (unverified)" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x05fd, 0x107a, "InterAct 'PowerPad Pro' X-Box pad (Germany)" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0738, 0x4516, "Mad Catz Control Pad" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0738, 0x4522, "Mad Catz LumiCON" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0738, 0x4526, "Mad Catz Control Pad Pro" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0738, 0x4536, "Mad Catz MicroCON" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0738, 0x4556, "Mad Catz Lynx Wireless Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0c12, 0x8802, "Zeroplus Xbox Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0c12, 0x8810, "Zeroplus Xbox Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0c12, 0x9902, "HAMA VibraX - *FAULTY HARDWARE*" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0e4c, 0x1097, "Radica Gamester Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0e4c, 0x2390, "Radica Games Jtech Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0e6f, 0x0003, "Logic3 Freebird wireless Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0e6f, 0x0005, "Eclipse wireless Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0e6f, 0x0006, "Edge wireless Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0e8f, 0x0201, "SmartJoy Frag Xpad/PS2 adaptor" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0f30, 0x0202, "Joytech Advanced Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x0f30, 0x8888, "BigBen XBMiniPad Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x102c, 0xff0c, "Joytech Wireless Advanced Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX, 0x044f, 0x0f07, "Thrustmaster, Inc. Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX360, 0x045e, 0x028e, "Microsoft Xbox 360 Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX360, 0x0738, 0x4716, "Mad Catz Xbox 360 Controller" },
|
2008-04-17 12:31:50 -06:00
|
|
|
|
{ GAMEPAD_XBOX360_GUITAR, 0x1430, 0x4748, "RedOctane Guitar Hero X-plorer" },
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
|
|
|
|
// Do these work?
|
2008-04-11 09:47:11 -06:00
|
|
|
|
{ GAMEPAD_XBOX360_WIRELESS, 0x045e, 0x0291, "Microsoft Xbox 360 Wireless Controller" },
|
|
|
|
|
{ GAMEPAD_XBOX360_WIRELESS, 0x045e, 0x0719, "Microsoft Xbox 360 Wireless Controller (PC)" },
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
2008-04-11 19:49:33 -06:00
|
|
|
|
{ GAMEPAD_XBOX_MAT, 0x0738, 0x4540, "Mad Catz Beat Pad" },
|
|
|
|
|
{ GAMEPAD_XBOX_MAT, 0x0738, 0x6040, "Mad Catz Beat Pad Pro" },
|
|
|
|
|
{ GAMEPAD_XBOX_MAT, 0x0c12, 0x8809, "RedOctane Xbox Dance Pad" },
|
|
|
|
|
{ GAMEPAD_XBOX_MAT, 0x12ab, 0x8809, "Xbox DDR dancepad" },
|
|
|
|
|
{ GAMEPAD_XBOX_MAT, 0x1430, 0x8888, "TX6500+ Dance Pad (first generation)" },
|
2008-04-11 09:47:11 -06:00
|
|
|
|
};
|
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
const int xpad_devices_count = sizeof(xpad_devices)/sizeof(XPadDevice);
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
std::ostream& operator<<(std::ostream& out, const GamepadType& type)
|
|
|
|
|
{
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case GAMEPAD_XBOX360:
|
2008-04-24 11:12:59 -06:00
|
|
|
|
return out << "Xbox360";
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
|
|
|
|
case GAMEPAD_XBOX360_WIRELESS:
|
2008-04-24 11:12:59 -06:00
|
|
|
|
return out << "Xbox360 (wireless)";
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
|
|
|
|
case GAMEPAD_XBOX:
|
2008-04-24 11:12:59 -06:00
|
|
|
|
return out << "Xbox Classic";
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
|
|
|
|
case GAMEPAD_XBOX_MAT:
|
2008-04-24 11:12:59 -06:00
|
|
|
|
return out << "Xbox Dancepad";
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
2008-04-17 15:02:28 -06:00
|
|
|
|
case GAMEPAD_XBOX360_GUITAR:
|
2008-04-24 11:12:59 -06:00
|
|
|
|
return out << "Xbox360 Guitar";
|
2008-04-17 15:02:28 -06:00
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
default:
|
|
|
|
|
return out << "unknown" << std::endl;
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::ostream& operator<<(std::ostream& out, const Xbox360GuitarMsg& msg)
|
2008-04-17 12:31:50 -06:00
|
|
|
|
{
|
2008-04-24 15:43:44 -06:00
|
|
|
|
out << boost::format(" whammy:%6d tilt:%6d | up:%d down:%d left:%d right:%d | back:%d guide:%d start:%d | green:%d red:%d yellow:%d blue:%d orange:%d ")
|
2008-04-17 12:31:50 -06:00
|
|
|
|
% int(msg.whammy)
|
|
|
|
|
% int(msg.tilt)
|
|
|
|
|
% int(msg.dpad_up)
|
|
|
|
|
% int(msg.dpad_down)
|
|
|
|
|
% int(msg.dpad_left)
|
|
|
|
|
% int(msg.dpad_right)
|
2008-04-24 11:12:59 -06:00
|
|
|
|
% int(msg.back)
|
2008-04-24 15:43:44 -06:00
|
|
|
|
% int(msg.guide)
|
2008-04-17 12:31:50 -06:00
|
|
|
|
% int(msg.start)
|
|
|
|
|
% int(msg.green)
|
|
|
|
|
% int(msg.red)
|
|
|
|
|
% int(msg.yellow)
|
|
|
|
|
% int(msg.blue)
|
|
|
|
|
% int(msg.orange);
|
|
|
|
|
|
2008-04-17 15:02:28 -06:00
|
|
|
|
if (0)
|
2008-04-17 12:31:50 -06:00
|
|
|
|
out << boost::format("| dummy: %d %d %d %d %02hhx %02hhx %04hx %04hx %02x %02x")
|
|
|
|
|
% int(msg.thumb_l)
|
|
|
|
|
% int(msg.thumb_r)
|
|
|
|
|
% int(msg.rb)
|
|
|
|
|
% int(msg.dummy1)
|
|
|
|
|
|
|
|
|
|
% int(msg.lt)
|
|
|
|
|
% int(msg.rt)
|
|
|
|
|
|
|
|
|
|
% int16_t(msg.x1)
|
|
|
|
|
% int16_t(msg.y1)
|
|
|
|
|
|
|
|
|
|
% int(msg.dummy2)
|
|
|
|
|
% int(msg.dummy3);
|
|
|
|
|
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::ostream& operator<<(std::ostream& out, const Xbox360Msg& msg)
|
2008-04-11 09:47:11 -06:00
|
|
|
|
{
|
2008-04-17 15:02:28 -06:00
|
|
|
|
out << boost::format("S1:(%6d, %6d)")
|
2008-04-11 09:47:11 -06:00
|
|
|
|
% int(msg.x1) % int(msg.y1);
|
|
|
|
|
|
|
|
|
|
out << boost::format(" S2:(%6d, %6d)")
|
|
|
|
|
% int(msg.x2) % int(msg.y2);
|
|
|
|
|
|
|
|
|
|
out << boost::format(" [u:%d|d:%d|l:%d|r:%d]")
|
|
|
|
|
% int(msg.dpad_up)
|
|
|
|
|
% int(msg.dpad_down)
|
|
|
|
|
% int(msg.dpad_left)
|
|
|
|
|
% int(msg.dpad_right);
|
|
|
|
|
|
2008-04-24 11:12:59 -06:00
|
|
|
|
out << " back:" << msg.back;
|
2008-04-24 15:43:44 -06:00
|
|
|
|
out << " guide:" << msg.guide;
|
2008-04-11 11:56:18 -06:00
|
|
|
|
out << " start:" << msg.start;
|
2008-04-11 09:47:11 -06:00
|
|
|
|
|
|
|
|
|
out << " sl:" << msg.thumb_l;
|
|
|
|
|
out << " sr:" << msg.thumb_r;
|
|
|
|
|
|
|
|
|
|
out << " A:" << msg.a;
|
|
|
|
|
out << " B:" << msg.b;
|
|
|
|
|
out << " X:" << msg.x;
|
|
|
|
|
out << " Y:" << msg.y;
|
|
|
|
|
|
|
|
|
|
out << " LB:" << msg.lb;
|
|
|
|
|
out << " RB:" << msg.rb;
|
|
|
|
|
|
|
|
|
|
out << boost::format(" LT:%3d RT:%3d")
|
|
|
|
|
% int(msg.lt) % int(msg.rt);
|
|
|
|
|
|
2008-04-17 15:02:28 -06:00
|
|
|
|
if (0)
|
2008-04-17 12:31:50 -06:00
|
|
|
|
out << " Dummy: " << msg.dummy1 << " " << msg.dummy2 << " " << msg.dummy3;
|
2008-04-11 09:47:11 -06:00
|
|
|
|
|
|
|
|
|
return out;
|
|
|
|
|
}
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::ostream& operator<<(std::ostream& out, const XboxMsg& msg)
|
2008-04-11 09:47:11 -06:00
|
|
|
|
{
|
|
|
|
|
out << boost::format(" S1:(%6d, %6d) S2:(%6d, %6d) "
|
|
|
|
|
" [u:%d|d:%d|l:%d|r:%d] "
|
|
|
|
|
" start:%d back:%d "
|
|
|
|
|
" sl:%d sr:%d "
|
|
|
|
|
" A:%3d B:%3d X:%3d Y:%3d "
|
|
|
|
|
" black:%3d white:%3d "
|
|
|
|
|
" LT:%3d RT:%3d ")
|
|
|
|
|
% int(msg.x1) % int(msg.y1)
|
|
|
|
|
% int(msg.x2) % int(msg.y2)
|
|
|
|
|
|
|
|
|
|
% int(msg.dpad_up)
|
|
|
|
|
% int(msg.dpad_down)
|
|
|
|
|
% int(msg.dpad_left)
|
|
|
|
|
% int(msg.dpad_right)
|
|
|
|
|
|
|
|
|
|
% int(msg.start)
|
|
|
|
|
% int(msg.back)
|
|
|
|
|
|
|
|
|
|
% int(msg.thumb_l)
|
|
|
|
|
% int(msg.thumb_r)
|
|
|
|
|
|
|
|
|
|
% int(msg.a)
|
|
|
|
|
% int(msg.b)
|
|
|
|
|
% int(msg.x)
|
|
|
|
|
% int(msg.y)
|
|
|
|
|
|
|
|
|
|
% int(msg.black)
|
|
|
|
|
% int(msg.white)
|
|
|
|
|
|
|
|
|
|
% int(msg.lt)
|
|
|
|
|
% int(msg.rt);
|
|
|
|
|
|
|
|
|
|
return out;
|
|
|
|
|
}
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-11 13:03:43 -06:00
|
|
|
|
void list_controller()
|
2008-04-10 13:39:08 -06:00
|
|
|
|
{
|
|
|
|
|
struct usb_bus* busses = usb_get_busses();
|
|
|
|
|
|
2008-04-11 13:03:43 -06:00
|
|
|
|
int id = 0;
|
|
|
|
|
std::cout << " id | idVendor | idProduct | Name" << std::endl;
|
|
|
|
|
std::cout << "----+----------+-----------+---------------------------------" << std::endl;
|
|
|
|
|
for (struct usb_bus* bus = busses; bus; bus = bus->next)
|
|
|
|
|
{
|
|
|
|
|
for (struct usb_device* dev = bus->devices; dev; dev = dev->next)
|
|
|
|
|
{
|
|
|
|
|
for(int i = 0; i < xpad_devices_count; ++i)
|
|
|
|
|
{
|
|
|
|
|
if (dev->descriptor.idVendor == xpad_devices[i].idVendor &&
|
|
|
|
|
dev->descriptor.idProduct == xpad_devices[i].idProduct)
|
|
|
|
|
{
|
|
|
|
|
std::cout << boost::format(" %2d | 0x%04x | 0x%04x | %s")
|
|
|
|
|
% id
|
|
|
|
|
% int(xpad_devices[i].idVendor)
|
|
|
|
|
% int(xpad_devices[i].idProduct)
|
|
|
|
|
% xpad_devices[i].name
|
|
|
|
|
<< std::endl;
|
|
|
|
|
id += 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (id == 0)
|
|
|
|
|
std::cout << "\nNo controller detected" << std::endl;
|
|
|
|
|
}
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-19 22:58:20 -06:00
|
|
|
|
bool find_controller_by_path(char* busid, char* devid,struct usb_device** xbox_device)
|
|
|
|
|
{
|
|
|
|
|
struct usb_bus* busses = usb_get_busses();
|
|
|
|
|
|
|
|
|
|
for (struct usb_bus* bus = busses; bus; bus = bus->next)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp(bus->dirname, busid) == 0)
|
|
|
|
|
{
|
|
|
|
|
for (struct usb_device* dev = bus->devices; dev; dev = dev->next)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp(dev->filename, devid) == 0)
|
|
|
|
|
{
|
|
|
|
|
*xbox_device = dev;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-04-11 13:03:43 -06:00
|
|
|
|
bool find_xbox360_controller(int id, struct usb_device** xbox_device, XPadDevice** type)
|
|
|
|
|
{
|
|
|
|
|
struct usb_bus* busses = usb_get_busses();
|
|
|
|
|
|
|
|
|
|
int id_count = 0;
|
2008-04-11 11:56:18 -06:00
|
|
|
|
for (struct usb_bus* bus = busses; bus; bus = bus->next)
|
2008-04-10 13:39:08 -06:00
|
|
|
|
{
|
|
|
|
|
for (struct usb_device* dev = bus->devices; dev; dev = dev->next)
|
|
|
|
|
{
|
|
|
|
|
if (0)
|
|
|
|
|
std::cout << (boost::format("UsbDevice: idVendor: 0x%04x idProduct: 0x%04x")
|
2008-04-11 11:56:18 -06:00
|
|
|
|
% int(dev->descriptor.idProduct)
|
|
|
|
|
% int(dev->descriptor.idVendor))
|
2008-04-10 13:39:08 -06:00
|
|
|
|
<< std::endl;
|
2008-04-10 11:33:56 -06:00
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
for(int i = 0; i < xpad_devices_count; ++i)
|
|
|
|
|
{
|
|
|
|
|
if (dev->descriptor.idVendor == xpad_devices[i].idVendor &&
|
|
|
|
|
dev->descriptor.idProduct == xpad_devices[i].idProduct)
|
|
|
|
|
{
|
2008-04-11 13:03:43 -06:00
|
|
|
|
if (id_count == id)
|
|
|
|
|
{
|
|
|
|
|
*xbox_device = dev;
|
|
|
|
|
*type = &xpad_devices[i];
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
id_count += 1;
|
2008-04-12 07:12:58 -06:00
|
|
|
|
break;
|
2008-04-11 13:03:43 -06:00
|
|
|
|
}
|
2008-04-11 11:56:18 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-10 13:39:08 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
struct CommandLineOptions
|
2008-04-10 10:38:50 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
bool verbose;
|
|
|
|
|
bool rumble;
|
|
|
|
|
char led;
|
|
|
|
|
int rumble_l;
|
|
|
|
|
int rumble_r;
|
|
|
|
|
int controller_id;
|
|
|
|
|
bool instant_exit;
|
|
|
|
|
bool no_uinput;
|
2008-04-20 19:19:02 -06:00
|
|
|
|
GamepadType gamepad_type;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
char busid[4];
|
|
|
|
|
char devid[4];
|
2008-04-19 22:58:20 -06:00
|
|
|
|
uInputCfg uinput_config;
|
2008-04-20 19:19:02 -06:00
|
|
|
|
int deadzone;
|
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
CommandLineOptions() {
|
2008-04-20 19:19:02 -06:00
|
|
|
|
verbose = false;
|
|
|
|
|
rumble = false;
|
|
|
|
|
led = 0;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
rumble_l = 0;
|
|
|
|
|
rumble_r = 0;
|
|
|
|
|
controller_id = 0;
|
|
|
|
|
instant_exit = false;
|
|
|
|
|
no_uinput = false;
|
2008-04-20 19:19:02 -06:00
|
|
|
|
gamepad_type = GAMEPAD_UNKNOWN;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
busid[0] = '\0';
|
|
|
|
|
devid[0] = '\0';
|
2008-04-20 19:19:02 -06:00
|
|
|
|
deadzone = 0;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void print_command_line_help(int argc, char** argv)
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Usage: " << argv[0] << " [OPTION]..." << std::endl;
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::cout << "Xbox360 USB Gamepad Userspace Driver" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
std::cout << "General Options: " << std::endl;
|
|
|
|
|
std::cout << " -h, --help display this help and exit" << std::endl;
|
|
|
|
|
std::cout << " --help-led list possible values for the led" << std::endl;
|
|
|
|
|
std::cout << " --help-devices list supported devices" << std::endl;
|
|
|
|
|
std::cout << " -v, --verbose display controller events" << std::endl;
|
|
|
|
|
std::cout << " -i, --id N use controller number (default: 0)" << std::endl;
|
|
|
|
|
std::cout << " -L, --list-controller list available controllers" << std::endl;
|
|
|
|
|
std::cout << " -R, --test-rumble map rumbling to LT and RT (for testing only)" << std::endl;
|
|
|
|
|
std::cout << " --no-uinput do not try to start uinput event dispatching" << std::endl;
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
std::cout << "Device Options: " << std::endl;
|
|
|
|
|
std::cout << " -d, --device BUS:DEV Use device BUS:DEV, do not do any scanning" << std::endl;
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
std::cout << "Status Options: " << std::endl;
|
|
|
|
|
std::cout << " -l, --led NUM set LED status, see --list-led-values (default: 0)" << std::endl;
|
|
|
|
|
std::cout << " -r, --rumble L,R set the speed for both rumble motors [0-255] (default: 0,0)" << std::endl;
|
|
|
|
|
std::cout << " -q, --quit only set led and rumble status then quit" << std::endl;
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
std::cout << "Configuration Options: " << std::endl;
|
2008-04-20 19:19:02 -06:00
|
|
|
|
std::cout << " --deadzone INT Threshold under which axis events are ignored (default: 0)" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
std::cout << " --trigger-as-button LT and RT send button instead of axis events" << std::endl;
|
|
|
|
|
std::cout << " --trigger-as-zaxis Combine LT and RT to form a zaxis instead" << std::endl;
|
|
|
|
|
std::cout << " --dpad-as-button DPad sends button instead of axis events" << std::endl;
|
|
|
|
|
std::cout << " --type TYPE Ignore autodetection and enforce controller type\n"
|
|
|
|
|
<< " (xbox, xbox360, xbox360-wireless, xbox360-guitar)" << std::endl;
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
std::cout << "Report bugs to Ingo Ruhnke <grumbel@gmx.de>" << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void print_led_help()
|
|
|
|
|
{
|
|
|
|
|
std::cout <<
|
|
|
|
|
"Possible values for '--led VALUE' are:\n\n"
|
|
|
|
|
" 0: off\n"
|
|
|
|
|
" 1: all blinking\n"
|
|
|
|
|
" 2: 1/top-left blink, then on\n"
|
|
|
|
|
" 3: 2/top-right blink, then on\n"
|
|
|
|
|
" 4: 3/bottom-left blink, then on\n"
|
|
|
|
|
" 5: 4/bottom-right blink, then on\n"
|
|
|
|
|
" 6: 1/top-left on\n"
|
|
|
|
|
" 7: 2/top-right on\n"
|
|
|
|
|
" 8: 3/bottom-left on\n"
|
|
|
|
|
" 9: 4/bottom-right on\n"
|
|
|
|
|
" 10: rotate\n"
|
|
|
|
|
" 11: blink\n"
|
|
|
|
|
" 12: blink slower\n"
|
|
|
|
|
" 13: rotate with two lights\n"
|
|
|
|
|
" 14: blink\n"
|
|
|
|
|
" 15: blink once\n"
|
|
|
|
|
<< std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void parse_command_line(int argc, char** argv, CommandLineOptions& opts)
|
|
|
|
|
{
|
2008-04-11 05:43:15 -06:00
|
|
|
|
for(int i = 1; i < argc; ++i)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp(argv[i], "-h") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--help") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
print_command_line_help(argc, argv);
|
|
|
|
|
exit(EXIT_SUCCESS);
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "-v") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--verbose") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.verbose = true;
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
2008-04-19 22:58:20 -06:00
|
|
|
|
else if (strcmp(argv[i], "--test-rumble") == 0 ||
|
|
|
|
|
strcmp(argv[i], "-R") == 0)
|
2008-04-13 08:35:21 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.rumble = true;
|
2008-04-13 08:35:21 -06:00
|
|
|
|
}
|
2008-04-11 05:43:15 -06:00
|
|
|
|
else if (strcmp(argv[i], "-r") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--rumble") == 0)
|
|
|
|
|
{
|
2008-04-13 08:35:21 -06:00
|
|
|
|
++i;
|
|
|
|
|
if (i < argc)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (sscanf(argv[i], "%d,%d", &opts.rumble_l, &opts.rumble_r) == 2)
|
2008-04-19 22:58:20 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.rumble_l = std::max(0, std::min(255, opts.rumble_l));
|
|
|
|
|
opts.rumble_r = std::max(0, std::min(255, opts.rumble_r));
|
2008-04-19 22:58:20 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected a argument in form INT,INT" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-19 22:58:20 -06:00
|
|
|
|
}
|
2008-04-13 08:35:21 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected a argument" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-13 08:35:21 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "-q") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--quit") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.instant_exit = true;
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
2008-04-15 08:06:48 -06:00
|
|
|
|
else if (strcmp(argv[i], "--no-uinput") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.verbose = true;
|
|
|
|
|
opts.no_uinput = true;
|
2008-04-15 08:06:48 -06:00
|
|
|
|
}
|
2008-04-17 15:02:28 -06:00
|
|
|
|
else if (strcmp(argv[i], "-t") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--type") == 0)
|
|
|
|
|
{
|
|
|
|
|
++i;
|
|
|
|
|
if (i < argc)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp(argv[i], "xbox") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.gamepad_type = GAMEPAD_XBOX;
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "xbox360") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.gamepad_type = GAMEPAD_XBOX360;
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "xbox360-guitar") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.gamepad_type = GAMEPAD_XBOX360_GUITAR;
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "xbox360-wireless") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.gamepad_type = GAMEPAD_XBOX360_WIRELESS;
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "xbox-dancemat") == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.gamepad_type = GAMEPAD_XBOX_MAT;
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-19 22:58:20 -06:00
|
|
|
|
std::cout << "Error: unknown type: " << argv[i] << std::endl;
|
|
|
|
|
std::cout << "Possible types are:" << std::endl;
|
|
|
|
|
std::cout << " * xbox" << std::endl;
|
|
|
|
|
std::cout << " * xbox360" << std::endl;
|
|
|
|
|
std::cout << " * xbox360-guitar" << std::endl;
|
|
|
|
|
std::cout << " * xbox360-wireless" << std::endl;
|
|
|
|
|
std::cout << " * xbox360-dancemat" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected a argument" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-17 15:02:28 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-11 13:03:43 -06:00
|
|
|
|
else if (strcmp(argv[i], "-i") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--id") == 0)
|
|
|
|
|
{
|
|
|
|
|
++i;
|
|
|
|
|
if (i < argc)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.controller_id = atoi(argv[i]);
|
2008-04-11 13:03:43 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected a argument" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-11 13:03:43 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-11 05:43:15 -06:00
|
|
|
|
else if (strcmp(argv[i], "-l") == 0 ||
|
|
|
|
|
strcmp(argv[i], "--led") == 0)
|
|
|
|
|
{
|
|
|
|
|
++i;
|
|
|
|
|
if (i < argc)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (strcmp(argv[i], "help") == 0)
|
|
|
|
|
{
|
|
|
|
|
print_led_help();
|
|
|
|
|
exit(EXIT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
opts.led = atoi(argv[i]);
|
|
|
|
|
}
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected a argument" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-13 09:08:27 -06:00
|
|
|
|
else if (strcmp("--dpad-as-button", argv[i]) == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.uinput_config.dpad_as_button = true;
|
2008-04-13 09:08:27 -06:00
|
|
|
|
}
|
2008-04-20 19:19:02 -06:00
|
|
|
|
else if (strcmp("--deadzone", argv[i]) == 0)
|
|
|
|
|
{
|
|
|
|
|
++i;
|
|
|
|
|
if (i < argc)
|
|
|
|
|
{
|
|
|
|
|
opts.deadzone = atoi(argv[i]);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected an INT argument" << std::endl;
|
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-13 08:56:31 -06:00
|
|
|
|
else if (strcmp("--trigger-as-button", argv[i]) == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.uinput_config.trigger_as_zaxis)
|
2008-04-13 09:22:14 -06:00
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: Can't combine --trigger-as-button and --trigger-as-zaxis" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-13 09:22:14 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.uinput_config.trigger_as_button = true;
|
2008-04-13 09:22:14 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp("--trigger-as-zaxis", argv[i]) == 0)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.uinput_config.trigger_as_button)
|
2008-04-13 09:22:14 -06:00
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: Can't combine --trigger-as-button and --trigger-as-zaxis" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-13 09:22:14 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.uinput_config.trigger_as_zaxis = true;
|
2008-04-13 09:22:14 -06:00
|
|
|
|
}
|
2008-04-13 08:56:31 -06:00
|
|
|
|
}
|
2008-04-13 09:08:27 -06:00
|
|
|
|
else if (strcmp("--help-led", argv[i]) == 0)
|
2008-04-12 20:47:15 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
print_led_help();
|
|
|
|
|
exit(EXIT_SUCCESS);
|
2008-04-12 20:47:15 -06:00
|
|
|
|
}
|
2008-04-19 22:58:20 -06:00
|
|
|
|
else if (strcmp(argv[i], "--device") == 0 ||
|
|
|
|
|
strcmp(argv[i], "-d") == 0)
|
|
|
|
|
{
|
|
|
|
|
++i;
|
|
|
|
|
if (i < argc)
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (sscanf(argv[i], "%3s:%3s", opts.busid, opts.devid) == 2)
|
2008-04-19 22:58:20 -06:00
|
|
|
|
{
|
|
|
|
|
std::cout << " ***************************************" << std::endl;
|
|
|
|
|
std::cout << " *** WARNING *** WARNING *** WARNING ***" << std::endl;
|
|
|
|
|
std::cout << " ***************************************" << std::endl;
|
|
|
|
|
std::cout << "The '--device DEV' option should not be needed for normal use" << std::endl;
|
|
|
|
|
std::cout << "and might potentially be harmful when used on devices that" << std::endl;
|
|
|
|
|
std::cout << "are not a gamepad, use at your own risk and ensure that you" << std::endl;
|
|
|
|
|
std::cout << "are accessing the right device.\n" << std::endl;
|
|
|
|
|
std::cout << "If you have multiple gamepads and want to select a differnt" << std::endl;
|
|
|
|
|
std::cout << "one use the '-id N' option instead.\n" << std::endl;
|
|
|
|
|
std::cout << "Press Ctrl-c to exit and Enter to continue." << std::endl;
|
|
|
|
|
getchar();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected a argument in form BUS:DEV (i.e. 006:003)" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-19 22:58:20 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: " << argv[i-1] << " expected an argument" << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-19 22:58:20 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[i], "--list-controller") == 0 ||
|
2008-04-15 08:06:48 -06:00
|
|
|
|
strcmp(argv[i], "-L") == 0)
|
2008-04-11 13:03:43 -06:00
|
|
|
|
{
|
|
|
|
|
usb_init();
|
|
|
|
|
usb_find_busses();
|
|
|
|
|
usb_find_devices();
|
|
|
|
|
|
|
|
|
|
list_controller();
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_SUCCESS);
|
2008-04-11 13:03:43 -06:00
|
|
|
|
}
|
2008-04-13 09:08:27 -06:00
|
|
|
|
else if (strcmp(argv[i], "--help-devices") == 0)
|
2008-04-11 11:56:18 -06:00
|
|
|
|
{
|
|
|
|
|
std::cout << " idVendor | idProduct | Name" << std::endl;
|
|
|
|
|
std::cout << "----------+-----------+---------------------------------" << std::endl;
|
|
|
|
|
for(unsigned int i = 0; i < sizeof(xpad_devices)/sizeof(XPadDevice); ++i)
|
|
|
|
|
{
|
|
|
|
|
std::cout << boost::format(" 0x%04x | 0x%04x | %s")
|
|
|
|
|
% int(xpad_devices[i].idVendor)
|
|
|
|
|
% int(xpad_devices[i].idProduct)
|
|
|
|
|
% xpad_devices[i].name
|
|
|
|
|
<< std::endl;
|
|
|
|
|
}
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_SUCCESS);
|
2008-04-11 11:56:18 -06:00
|
|
|
|
}
|
2008-04-11 05:43:15 -06:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: unknown command line option: " << argv[i] << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-20 19:00:51 -06:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
|
{
|
|
|
|
|
srand(time(0));
|
|
|
|
|
|
|
|
|
|
CommandLineOptions opts;
|
|
|
|
|
|
|
|
|
|
parse_command_line(argc, argv, opts);
|
2008-04-11 05:43:15 -06:00
|
|
|
|
|
2008-04-10 11:33:56 -06:00
|
|
|
|
usb_init();
|
|
|
|
|
usb_find_busses();
|
|
|
|
|
usb_find_devices();
|
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
struct usb_device* dev = 0;
|
|
|
|
|
XPadDevice* dev_type = 0;
|
2008-04-19 22:58:20 -06:00
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.busid[0] != '\0' && opts.devid[0] != '\0')
|
2008-04-10 11:33:56 -06:00
|
|
|
|
{
|
2008-04-20 19:19:02 -06:00
|
|
|
|
if (opts.gamepad_type == GAMEPAD_UNKNOWN)
|
2008-04-19 22:58:20 -06:00
|
|
|
|
{
|
|
|
|
|
std::cout << "Error: --device BUS:DEV option must be used in combination with --type TYPE option" << std::endl;
|
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (!find_controller_by_path(opts.busid, opts.devid, &dev))
|
2008-04-19 22:58:20 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
std::cout << "Error: couldn't find device " << opts.busid << ":" << opts.devid << std::endl;
|
2008-04-19 22:58:20 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (!find_xbox360_controller(opts.controller_id, &dev, &dev_type))
|
2008-04-19 22:58:20 -06:00
|
|
|
|
{
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::cout << "No Xbox or Xbox360 controller found" << std::endl;
|
2008-04-19 22:58:20 -06:00
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!dev)
|
|
|
|
|
{
|
|
|
|
|
std::cout << "No suitable USB device found, abort" << std::endl;
|
|
|
|
|
exit(EXIT_FAILURE);
|
2008-04-10 13:39:08 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-11 13:03:43 -06:00
|
|
|
|
// Could/should fork here to hande multiple controllers at once
|
2008-04-20 19:19:02 -06:00
|
|
|
|
if (opts.gamepad_type == GAMEPAD_UNKNOWN)
|
2008-04-19 22:58:20 -06:00
|
|
|
|
{
|
|
|
|
|
assert(dev_type);
|
2008-04-20 19:00:51 -06:00
|
|
|
|
opts.gamepad_type = dev_type->type;
|
2008-04-19 22:58:20 -06:00
|
|
|
|
}
|
2008-04-17 15:02:28 -06:00
|
|
|
|
|
2008-04-19 22:58:20 -06:00
|
|
|
|
std::cout << "USB Device: " << dev->bus->dirname << ":" << dev->filename << std::endl;
|
2008-04-11 11:56:18 -06:00
|
|
|
|
std::cout << "Controller: " << boost::format("\"%s\" (idVendor: 0x%04x, idProduct: 0x%04x)")
|
2008-04-19 22:58:20 -06:00
|
|
|
|
% (dev_type ? dev_type->name : "unknown") % uint16_t(dev->descriptor.idVendor) % uint16_t(dev->descriptor.idProduct) << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
std::cout << "Controller Type: " << opts.gamepad_type << std::endl;
|
2008-04-20 19:19:02 -06:00
|
|
|
|
std::cout << "Deadzone: " << opts.deadzone << std::endl;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
std::cout << "Rumble Debug: " << (opts.rumble ? "on" : "off") << std::endl;
|
|
|
|
|
std::cout << "Rumble Speed: " << "left: " << opts.rumble_l << " right: " << opts.rumble_r << std::endl;
|
|
|
|
|
std::cout << "LED Status: " << int(opts.led) << std::endl;
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
2008-04-10 13:39:08 -06:00
|
|
|
|
struct usb_dev_handle* handle = usb_open(dev);
|
|
|
|
|
if (!handle)
|
2008-04-10 11:33:56 -06:00
|
|
|
|
{
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::cout << "Error opening Xbox360 controller" << std::endl;
|
2008-04-10 13:39:08 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-04-15 10:09:16 -06:00
|
|
|
|
if (usb_claim_interface(handle, 0) != 0) // FIXME: bInterfaceNumber shouldn't be hardcoded
|
|
|
|
|
std::cout << "Error claiming the interface: " << usb_strerror() << std::endl;
|
|
|
|
|
|
2008-04-24 11:12:59 -06:00
|
|
|
|
// Handle LED on Xbox360 Controller
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.gamepad_type == GAMEPAD_XBOX360 ||
|
|
|
|
|
opts.gamepad_type == GAMEPAD_XBOX360_GUITAR)
|
2008-04-10 13:39:08 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
char ledcmd[] = { 1, 3, opts.led };
|
2008-04-17 09:11:08 -06:00
|
|
|
|
usb_interrupt_write(handle, 2, ledcmd, 3, 0);
|
2008-04-10 13:39:08 -06:00
|
|
|
|
}
|
2008-04-10 11:33:56 -06:00
|
|
|
|
|
2008-04-11 11:56:18 -06:00
|
|
|
|
// Switch of Rumble
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.gamepad_type == GAMEPAD_XBOX360)
|
2008-04-11 05:43:15 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
char l = opts.rumble_r; // light weight
|
|
|
|
|
char b = opts.rumble_l; // big weight
|
2008-04-11 05:43:15 -06:00
|
|
|
|
char rumblecmd[] = { 0x00, 0x08, 0x00, b, l, 0x00, 0x00, 0x00 };
|
2008-04-17 09:11:08 -06:00
|
|
|
|
usb_interrupt_write(handle, 2, rumblecmd, 8, 0);
|
2008-04-11 05:43:15 -06:00
|
|
|
|
}
|
2008-04-20 19:00:51 -06:00
|
|
|
|
else if (opts.gamepad_type == GAMEPAD_XBOX)
|
2008-04-11 11:56:18 -06:00
|
|
|
|
{
|
2008-04-20 19:00:51 -06:00
|
|
|
|
char l = opts.rumble_l;
|
|
|
|
|
char b = opts.rumble_r;
|
2008-04-11 11:56:18 -06:00
|
|
|
|
char rumblecmd[] = { 0x00, 0x06, 0x00, l, 0x00, b };
|
2008-04-17 09:11:08 -06:00
|
|
|
|
usb_interrupt_write(handle, 2, rumblecmd, 6, 0);
|
2008-04-11 11:56:18 -06:00
|
|
|
|
}
|
2008-04-13 08:35:21 -06:00
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.instant_exit)
|
2008-04-13 08:43:30 -06:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
else
|
2008-04-13 08:35:21 -06:00
|
|
|
|
{
|
2008-04-15 08:06:48 -06:00
|
|
|
|
uInput* uinput = 0;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (!opts.no_uinput)
|
2008-04-15 08:06:48 -06:00
|
|
|
|
{
|
|
|
|
|
std::cout << "Starting uinput" << std::endl;
|
2008-04-20 19:19:02 -06:00
|
|
|
|
uinput = new uInput(opts.gamepad_type, opts.uinput_config);
|
2008-04-15 08:06:48 -06:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Starting without uinput" << std::endl;
|
|
|
|
|
}
|
2008-04-24 11:12:59 -06:00
|
|
|
|
std::cout << "\nYour Xbox360 controller should now be available as /dev/input/jsX and /dev/input/eventX" << std::endl;
|
2008-04-13 08:35:21 -06:00
|
|
|
|
std::cout << "Press Ctrl-c to quit" << std::endl;
|
|
|
|
|
|
|
|
|
|
bool quit = false;
|
|
|
|
|
uint8_t old_data[20];
|
|
|
|
|
memset(old_data, 0, 20);
|
|
|
|
|
while(!quit)
|
2008-04-10 11:33:56 -06:00
|
|
|
|
{
|
2008-04-13 08:35:21 -06:00
|
|
|
|
uint8_t data[20];
|
2008-04-17 09:11:08 -06:00
|
|
|
|
int ret = usb_interrupt_read(handle, 1 /*EndPoint*/, (char*)data, 20, 0 /*Timeout*/);
|
2008-04-13 08:35:21 -06:00
|
|
|
|
if (ret < 0)
|
|
|
|
|
{ // Error
|
|
|
|
|
std::cout << "USBError: " << ret << "\n" << usb_strerror() << std::endl;
|
|
|
|
|
std::cout << "Shutting down" << std::endl;
|
|
|
|
|
quit = true;
|
|
|
|
|
}
|
|
|
|
|
else if (ret == 0)
|
2008-04-11 05:43:15 -06:00
|
|
|
|
{
|
2008-04-24 11:12:59 -06:00
|
|
|
|
// happen with the Xbox360 every now and then, just
|
2008-04-13 08:35:21 -06:00
|
|
|
|
// ignore, seems harmless
|
|
|
|
|
}
|
2008-04-14 07:47:15 -06:00
|
|
|
|
#if 0
|
|
|
|
|
else if (ret == 2 && data[1] == 0x80)
|
|
|
|
|
{ // wireless connect
|
|
|
|
|
}
|
|
|
|
|
else if (ret == 2 && data[1] == 0x80)
|
|
|
|
|
{ // wireless disconnect
|
|
|
|
|
}
|
|
|
|
|
else if (ret == 29 && data[1] == 0x01)
|
|
|
|
|
{
|
|
|
|
|
xpad_process_packet(xpad, 0, (char*) ((unsigned long)xpad->idata + 4));
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2008-04-13 08:35:21 -06:00
|
|
|
|
else if (ret == 20 && data[0] == 0x00 && data[1] == 0x14)
|
2008-04-12 20:16:29 -06:00
|
|
|
|
{
|
2008-04-13 08:35:21 -06:00
|
|
|
|
if (memcmp(data, old_data, 20) == 0)
|
2008-04-11 09:47:11 -06:00
|
|
|
|
{
|
2008-04-13 08:35:21 -06:00
|
|
|
|
// Ignore the data, since nothing has changed
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
memcpy(old_data, data, 20);
|
2008-04-11 11:56:18 -06:00
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.gamepad_type == GAMEPAD_XBOX360_GUITAR)
|
2008-04-17 12:31:50 -06:00
|
|
|
|
{
|
2008-04-24 11:12:59 -06:00
|
|
|
|
Xbox360GuitarMsg& msg = (Xbox360GuitarMsg&)data;
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.verbose)
|
2008-04-17 12:31:50 -06:00
|
|
|
|
std::cout << msg << std::endl;
|
2008-04-17 21:01:04 -06:00
|
|
|
|
|
|
|
|
|
uinput->send(msg);
|
2008-04-17 12:31:50 -06:00
|
|
|
|
}
|
2008-04-20 19:00:51 -06:00
|
|
|
|
else if (opts.gamepad_type == GAMEPAD_XBOX360)
|
2008-04-12 20:16:29 -06:00
|
|
|
|
{
|
2008-04-24 11:12:59 -06:00
|
|
|
|
Xbox360Msg& msg = (Xbox360Msg&)data;
|
2008-04-12 20:16:29 -06:00
|
|
|
|
|
2008-04-20 19:19:02 -06:00
|
|
|
|
if (abs(msg.x1) < opts.deadzone)
|
|
|
|
|
msg.x1 = 0;
|
|
|
|
|
|
|
|
|
|
if (abs(msg.y1) < opts.deadzone)
|
|
|
|
|
msg.y1 = 0;
|
|
|
|
|
|
|
|
|
|
if (abs(msg.x2) < opts.deadzone)
|
|
|
|
|
msg.x2 = 0;
|
|
|
|
|
|
|
|
|
|
if (abs(msg.y2) < opts.deadzone)
|
|
|
|
|
msg.y2 = 0;
|
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.verbose)
|
2008-04-13 08:35:21 -06:00
|
|
|
|
std::cout << msg << std::endl;
|
2008-04-12 20:16:29 -06:00
|
|
|
|
|
2008-04-15 08:06:48 -06:00
|
|
|
|
if (uinput) uinput->send(msg);
|
2008-04-13 08:35:21 -06:00
|
|
|
|
|
2008-04-20 19:19:02 -06:00
|
|
|
|
if (opts.rumble)
|
2008-04-20 19:00:51 -06:00
|
|
|
|
{
|
2008-04-20 19:19:02 -06:00
|
|
|
|
char l = msg.rt;
|
|
|
|
|
char b = msg.lt;
|
|
|
|
|
char rumblecmd[] = { 0x00, 0x08, 0x00, b, l, 0x00, 0x00, 0x00 };
|
|
|
|
|
usb_interrupt_write(handle, 2, rumblecmd, 8, 0);
|
2008-04-13 08:35:21 -06:00
|
|
|
|
}
|
2008-04-12 20:16:29 -06:00
|
|
|
|
}
|
2008-04-20 19:00:51 -06:00
|
|
|
|
else if (opts.gamepad_type == GAMEPAD_XBOX)
|
2008-04-13 08:35:21 -06:00
|
|
|
|
{
|
2008-04-24 11:12:59 -06:00
|
|
|
|
XboxMsg& msg = (XboxMsg&)data;
|
2008-04-13 08:35:21 -06:00
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.verbose)
|
2008-04-13 08:35:21 -06:00
|
|
|
|
std::cout << msg << std::endl;
|
|
|
|
|
|
2008-04-15 08:06:48 -06:00
|
|
|
|
if (uinput) uinput->send(msg);
|
2008-04-13 08:35:21 -06:00
|
|
|
|
|
2008-04-20 19:00:51 -06:00
|
|
|
|
if (opts.rumble)
|
2008-04-13 08:35:21 -06:00
|
|
|
|
{
|
|
|
|
|
char l = msg.lt;
|
|
|
|
|
char b = msg.rt;
|
|
|
|
|
char rumblecmd[] = { 0x00, 0x06, 0x00, l, 0x00, b };
|
2008-04-17 09:11:08 -06:00
|
|
|
|
usb_interrupt_write(handle, 2, rumblecmd, 6, 0);
|
2008-04-13 08:35:21 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
2008-04-10 13:39:08 -06:00
|
|
|
|
{
|
2008-04-13 08:35:21 -06:00
|
|
|
|
std::cout << "Unknown data: bytes: " << ret
|
|
|
|
|
<< " Data: ";
|
|
|
|
|
|
|
|
|
|
for(int j = 0; j < ret; ++j)
|
|
|
|
|
{
|
|
|
|
|
std::cout << boost::format("0x%02x ") % int(data[j]);
|
|
|
|
|
}
|
|
|
|
|
//std::cout << "\r" << std::flush;
|
|
|
|
|
std::cout << std::endl;
|
2008-04-10 11:33:56 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-15 10:09:16 -06:00
|
|
|
|
usb_release_interface(handle, 0); // FIXME: bInterfaceNumber shouldn't be hardcoded
|
2008-04-10 13:39:08 -06:00
|
|
|
|
|
2008-04-12 07:07:20 -06:00
|
|
|
|
// Almost never reached since the user will Ctrl-c and we
|
2008-04-17 09:11:08 -06:00
|
|
|
|
// can't use sigint since we block in usb_interrupt_read()
|
2008-04-10 13:39:08 -06:00
|
|
|
|
usb_close(handle);
|
2008-04-10 11:33:56 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-10 14:04:38 -06:00
|
|
|
|
|
2008-04-10 11:33:56 -06:00
|
|
|
|
return 0;
|
2008-04-10 10:38:50 -06:00
|
|
|
|
}
|
2008-04-17 09:11:08 -06:00
|
|
|
|
|
2008-04-10 13:39:08 -06:00
|
|
|
|
/* EOF */
|