1
0
mirror of https://github.com/mcgurk/Arduino-USB-HID-RetroJoystickAdapter synced 2024-12-21 23:18:56 -05:00

Update x-arcade_c64.ino

This commit is contained in:
Jarno Lehtinen 2017-05-26 11:45:35 +03:00 committed by GitHub
parent e2850f8cbe
commit dca2f2f9ad

View File

@ -5,11 +5,7 @@
#define IRQPIN 3
//#define DEBUG
//#define COMMANDO
//#define DECATHLON
#define C64
#ifdef C64
#define oUP1 5
#define oDOWN1 6
#define oLEFT1 7
@ -21,7 +17,6 @@
#define oLEFT2 14
#define oRIGHT2 16
#define oFIRE2 10
#endif
#define UP1 0x75
#define DOWN1 0x72
@ -53,158 +48,19 @@
#define L2 0x5B
#define R2 0x36
inline void translateState(uint8_t *data, uint8_t *state) {
state[0] = data[0];
state[1] = data[1];
state[2] = 127;
state[3] = 127;
if (bitRead(data[2], 2)) state[2] = 0; /* up */
if (bitRead(data[2], 3)) state[2] = 255; /* down */
if (bitRead(data[2], 0)) state[3] = 0; /* left */
if (bitRead(data[2], 1)) state[3] = 255; /* right */
}
#include "HID.h"
#if ARDUINO < 10606
#error The Joystick2 library requires Arduino IDE 1.6.6 or greater. Please update your IDE.
#endif
#if !defined(USBCON)
#error The Joystick2 library can only be used with a USB MCU (e.g. Arduino Leonardo, Arduino Micro, etc.).
#endif
#if !defined(_USING_HID)
#error "legacy HID core (non pluggable)"
#endif
#define JOYSTICK_REPORT_ID 0x04
#define JOYSTICK2_REPORT_ID 0x05
#define JOYSTICK_DATA_SIZE 3
#define JOYSTICK_STATE_SIZE 4
//================================================================================
//================================================================================
// Joystick (Gamepad)
#define HIDDESC_MACRO(REPORT_ID) \
/* Joystick # */ \
0x05, 0x01, /* USAGE_PAGE (Generic Desktop) */ \
0x09, 0x04, /* USAGE (Joystick) */ \
0xa1, 0x01, /* COLLECTION (Application) */ \
0x85, REPORT_ID, /* REPORT_ID */ \
/* 8 Buttons */ \
0x05, 0x09, /* USAGE_PAGE (Button) */ \
0x19, 0x01, /* USAGE_MINIMUM (Button 1) */ \
0x29, 0x10, /* USAGE_MAXIMUM (Button 16) */ \
0x15, 0x00, /* LOGICAL_MINIMUM (0) */ \
0x25, 0x01, /* LOGICAL_MAXIMUM (1) */ \
0x75, 0x01, /* REPORT_SIZE (1) */ \
0x95, 0x10, /* REPORT_COUNT (16) */ \
0x81, 0x02, /* INPUT (Data,Var,Abs) */ \
/* X and Y Axis */ \
0x05, 0x01, /* USAGE_PAGE (Generic Desktop) */ \
0x09, 0x01, /* USAGE (Pointer) */ \
0xA1, 0x00, /* COLLECTION (Physical) */ \
0x09, 0x30, /* USAGE (x) */ \
0x09, 0x31, /* USAGE (y) */ \
0x15, 0x00, /* LOGICAL_MINIMUM (0) */ \
0x26, 0xff, 0x00, /* LOGICAL_MAXIMUM (255) */ \
0x75, 0x08, /* REPORT_SIZE (8) */ \
0x95, 0x02, /* REPORT_COUNT (2) */ \
0x81, 0x02, /* INPUT (Data,Var,Abs) */ \
0xc0, /* END_COLLECTION */ \
0xc0 /* END_COLLECTION */
static const uint8_t hidReportDescriptor[] PROGMEM = {
HIDDESC_MACRO(JOYSTICK_REPORT_ID),
HIDDESC_MACRO(JOYSTICK2_REPORT_ID)
};
class Joystick_ {
private:
uint8_t joystickId;
uint8_t reportId;
uint8_t olddata[JOYSTICK_DATA_SIZE];
uint8_t state[JOYSTICK_STATE_SIZE];
uint8_t flag;
public:
uint8_t type;
uint8_t data[JOYSTICK_DATA_SIZE];
Joystick_(uint8_t initJoystickId, uint8_t initReportId) {
// Setup HID report structure
static bool usbSetup = false;
if (!usbSetup) {
static HIDSubDescriptor node(hidReportDescriptor, sizeof(hidReportDescriptor));
HID().AppendDescriptor(&node);
usbSetup = true;
}
// Initalize State
joystickId = initJoystickId;
reportId = initReportId;
data[0] = 0;
data[1] = 0;
data[2] = 0;
memcpy(olddata, data, JOYSTICK_DATA_SIZE);
translateState(data, state);
sendState(1);
}
void updateState() {
if (memcmp(olddata, data, JOYSTICK_DATA_SIZE)) {
memcpy(olddata, data, JOYSTICK_DATA_SIZE);
translateState(data, state);
flag = 1;
}
}
void sendState(uint8_t force = 0) {
if (flag || force) {
// HID().SendReport(Report number, array of values in same order as HID descriptor, length)
HID().SendReport(reportId, state, JOYSTICK_STATE_SIZE);
flag = 0;
}
}
void updateAndSendState() {
translateState(data, state);
HID().SendReport(reportId, state, JOYSTICK_STATE_SIZE);
}
};
Joystick_ Joystick[2] =
{
Joystick_(0, JOYSTICK_REPORT_ID),
Joystick_(1, JOYSTICK2_REPORT_ID)
};
//================================================================================
//================================================================================
uint8_t s1 = 0;
uint8_t s2 = 0;
uint8_t st1 = 0;
uint8_t st2 = 0;
uint8_t commando = 0;
PS2KeyRaw keyboard;
void setup() {
keyboard.begin(DATAPIN, IRQPIN);
Serial.begin(115200);
Serial.println( "PS2 Raw Test of PS2 Keyboard codes" );
Serial.println( "X-Arcade -> Atari -adapter" );
#ifdef C64
pinMode(oUP1, INPUT);
pinMode(oDOWN1, INPUT);
pinMode(oLEFT1, INPUT);
@ -216,7 +72,6 @@ void setup() {
pinMode(oLEFT2, INPUT);
pinMode(oRIGHT2, INPUT);
pinMode(oFIRE2, INPUT);
#endif
}
@ -236,51 +91,43 @@ void loop() {
#ifdef DEBUG
Serial.print("0x"); Serial.println(c, HEX);
#endif
j = clearData(c);
#ifdef C64
clearDataC64(c);
#endif
} else {
#ifdef DEBUG
Serial.print("0x"); Serial.println(c, HEX);
#endif
j = setData(c);
#ifdef C64
setDataC64(c);
#endif
}
#ifdef DEBUG
Serial.print(" data0 j1: 0x"); Serial.print(Joystick[0].data[0], HEX);
Serial.print(" data1 j1: 0x"); Serial.print(Joystick[0].data[1], HEX);
Serial.print(" data2 j1: 0x"); Serial.print(Joystick[0].data[2], HEX);
Serial.print(" data0 j2: 0x"); Serial.print(Joystick[1].data[0], HEX);
Serial.print(" data1 j2: 0x"); Serial.print(Joystick[1].data[1], HEX);
Serial.print(" data2 j2: 0x"); Serial.print(Joystick[1].data[2], HEX);
Serial.println();
Serial.flush();
#endif
if (s1 && s2) commando = 1;
if (st1 && st2) commando = 0;
Joystick[j].updateAndSendState();
}
}
#ifdef C64
#define SET64(p) pinMode(p, OUTPUT); break;
#define UNSET64(p) pinMode(p, INPUT); break;
inline void setDataC64(uint8_t c) {
switch (c) {
case SELECT1:
s1 = 1;
break;
case START1:
st1 = 1;
break;
case A1:
case B1:
case START1:
case SELECT1:
SET64(oFIRE1);
case X1:
case Y1:
SET64(oFIRE2);
if (commando) {
SET64(oFIRE2);
} else {
SET64(oFIRE1);
}
case C1:
SET64(oDOWN1);
case Z1:
@ -297,14 +144,22 @@ inline void setDataC64(uint8_t c) {
SET64(oLEFT1);
case RIGHT1:
SET64(oRIGHT1);
case SELECT2:
s2 = 1;
break;
case START2:
st2 = 1;
break;
case A2:
case B2:
case START2:
case SELECT2:
SET64(oFIRE2);
case X2:
case Y2:
SET64(oFIRE1);
if (commando) {
SET64(oFIRE1);
} else {
SET64(oFIRE2);
}
case C2:
SET64(oDOWN2);
case Z2:
@ -328,14 +183,22 @@ inline void setDataC64(uint8_t c) {
inline void clearDataC64(uint8_t c) {
switch (c) {
case SELECT1:
s1 = 0;
break;
case START1:
st1 = 0;
break;
case A1:
case B1:
case START1:
case SELECT1:
UNSET64(oFIRE1);
case X1:
case Y1:
UNSET64(oFIRE2);
if (commando) {
UNSET64(oFIRE2);
} else{
UNSET64(oFIRE1);
}
case C1:
UNSET64(oDOWN1);
case Z1:
@ -352,14 +215,22 @@ inline void clearDataC64(uint8_t c) {
UNSET64(oLEFT1);
case RIGHT1:
UNSET64(oRIGHT1);
case SELECT2:
s2 = 0;
break;
case START2:
st2 = 0;
break;
case A2:
case B2:
case START2:
case SELECT2:
UNSET64(oFIRE2);
case X2:
case Y2:
UNSET64(oFIRE1);
if (commando) {
UNSET64(oFIRE1);
} else {
UNSET64(oFIRE2);
}
case C2:
UNSET64(oDOWN2);
case Z2:
@ -378,190 +249,4 @@ inline void clearDataC64(uint8_t c) {
UNSET64(oRIGHT2);
}
}
#endif
//--------------------------------------------------------------------
// USB
//--------------------------------------------------------------------
#define SET(j,d,b) bitSet(Joystick[j].data[d], b); return j
#define UNSET(j,d,b) bitClear(Joystick[j].data[d], b); return j
#ifdef COMMANDO //swap two bottom buttons between joysticks
#define L1_SET SET(1, 1, 0)
#define L1_UNSET UNSET(1, 1, 0)
#define R1_SET SET(1, 1, 1)
#define R1_UNSET UNSET(1, 1, 1)
#define L2_SET SET(0, 1, 0)
#define L2_UNSET UNSET(0, 1, 0)
#define R2_SET SET(0, 1, 1)
#define R2_UNSET UNSET(0, 1, 1)
#endif
#ifdef DECATHLON //two bottom buttons left and right directions
#define L1_SET SET(0, 2, 2)
#define L1_UNSET UNSET(0, 2, 2)
#define R1_SET SET(0, 2, 3)
#define R1_UNSET UNSET(0, 2, 3)
#define L2_SET SET(1, 2, 2)
#define L2_UNSET UNSET(1, 2, 2)
#define R2_SET SET(1, 2, 3)
#define R2_UNSET UNSET(1, 2, 3)
#endif
inline uint8_t setData(uint8_t c) {
switch (c) {
case A1:
SET(0, 0, 0);
case B1:
SET(0, 0, 1);
case C1:
SET(0, 0, 2);
case X1:
SET(0, 0, 3);
case Y1:
SET(0, 0, 4);
case Z1:
SET(0, 0, 5);
case START1:
SET(0, 0, 6);
case SELECT1:
SET(0, 0, 7);
case L1:
#ifdef L1_SET
L1_SET;
#else
SET(0, 1, 0);
#endif
case R1:
#ifdef R1_SET
R1_SET;
#else
SET(0, 1, 1);
#endif
case UP1:
SET(0, 2, 0);
case DOWN1:
SET(0, 2, 1);
case LEFT1:
SET(0, 2, 2);
case RIGHT1:
SET(0, 2, 3);
case A2:
SET(1, 0, 0);
case B2:
SET(1, 0, 1);
case C2:
SET(1, 0, 2);
case X2:
SET(1, 0, 3);
case Y2:
SET(1, 0, 4);
case Z2:
SET(1, 0, 5);
case START2:
SET(1, 0, 6);
case SELECT2:
SET(1, 0, 7);
case L2:
#ifdef L2_SET
L2_SET;
#else
SET(1, 1, 0);
#endif
case R2:
#ifdef R2_SET
R2_SET;
#else
SET(1, 1, 1);
#endif
case UP2:
SET(1, 2, 0);
case DOWN2:
SET(1, 2, 1);
case LEFT2:
SET(1, 2, 2);
case RIGHT2:
SET(1, 2, 3);
}
}
inline uint8_t clearData(uint8_t c) {
switch (c) {
case A1:
UNSET(0, 0, 0);
case B1:
UNSET(0, 0, 1);
case C1:
UNSET(0, 0, 2);
case X1:
UNSET(0, 0, 3);
case Y1:
UNSET(0, 0, 4);
case Z1:
UNSET(0, 0, 5);
case START1:
UNSET(0, 0, 6);
case SELECT1:
UNSET(0, 0, 7);
case L1:
#ifdef L1_UNSET
L1_UNSET;
#else
UNSET(0, 1, 0);
#endif
case R1:
#ifdef R1_UNSET
R1_UNSET;
#else
UNSET(0, 1, 1);
#endif
case UP1:
UNSET(0, 2, 0);
case DOWN1:
UNSET(0, 2, 1);
case LEFT1:
UNSET(0, 2, 2);
case RIGHT1:
UNSET(0, 2, 3);
case A2:
UNSET(1, 0, 0);
case B2:
UNSET(1, 0, 1);
case C2:
UNSET(1, 0, 2);
case X2:
UNSET(1, 0, 3);
case Y2:
UNSET(1, 0, 4);
case Z2:
UNSET(1, 0, 5);
case START2:
UNSET(1, 0, 6);
case SELECT2:
UNSET(1, 0, 7);
case L2:
#ifdef L2_UNSET
L2_UNSET;
#else
UNSET(1, 1, 0);
#endif
case R2:
#ifdef R2_UNSET
R2_UNSET;
#else
UNSET(1, 1, 1);
#endif
case UP2:
UNSET(1, 2, 0);
case DOWN2:
UNSET(1, 2, 1);
case LEFT2:
UNSET(1, 2, 2);
case RIGHT2:
UNSET(1, 2, 3);
}
}