mirror of
https://github.com/mcgurk/Arduino-USB-HID-RetroJoystickAdapter
synced 2025-01-08 20:28:27 -05:00
269 lines
6.7 KiB
C++
269 lines
6.7 KiB
C++
#define USBHOST
|
|
|
|
#ifdef USBHOST
|
|
#include <hidboot.h>
|
|
#include <usbhub.h>
|
|
#endif
|
|
|
|
#define POTSENSE 4 //ICP1 (Arduino Pro Micro: pin4, Arduino Uno: pin8)
|
|
|
|
#define POTX 9 ///< X-line, also OC1A
|
|
#define POTY 10 ///< Y-line, also OC1B
|
|
#define LBTN 5 ///< Joystick FIRE switch
|
|
#define RBTN 6 ///< Joystick UP switch
|
|
|
|
//#define DEBUG
|
|
|
|
int16_t dx=0;
|
|
int16_t dy=0;
|
|
uint8_t buttons=0;
|
|
|
|
uint8_t update = 0;
|
|
|
|
#ifdef USBHOST
|
|
class MouseRptParser : public MouseReportParser {
|
|
protected:
|
|
void OnMouseMove(MOUSEINFO *mi);
|
|
void OnLeftButtonUp(MOUSEINFO *mi);
|
|
void OnLeftButtonDown(MOUSEINFO *mi);
|
|
void OnRightButtonUp(MOUSEINFO *mi);
|
|
void OnRightButtonDown(MOUSEINFO *mi);
|
|
void OnMiddleButtonUp(MOUSEINFO *mi);
|
|
void OnMiddleButtonDown(MOUSEINFO *mi);
|
|
};
|
|
void MouseRptParser::OnMouseMove(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.print("dx=");
|
|
Serial.print(mi->dX, DEC);
|
|
Serial.print(" dy=");
|
|
Serial.println(mi->dY, DEC);
|
|
#endif
|
|
dx=mi->dX;
|
|
dy=mi->dY;
|
|
update = 1;
|
|
};
|
|
void MouseRptParser::OnLeftButtonUp(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.println("L Butt Up");
|
|
#endif
|
|
buttons &= ~1;
|
|
update = 1;
|
|
};
|
|
void MouseRptParser::OnLeftButtonDown(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.println("L Butt Dn");
|
|
#endif
|
|
buttons |= 1;
|
|
update = 1;
|
|
};
|
|
void MouseRptParser::OnRightButtonUp(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.println("R Butt Up");
|
|
#endif
|
|
buttons &= ~2;
|
|
update = 1;
|
|
};
|
|
void MouseRptParser::OnRightButtonDown(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.println("R Butt Dn");
|
|
#endif
|
|
buttons |= 2;
|
|
update = 1;
|
|
};
|
|
void MouseRptParser::OnMiddleButtonUp(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.println("M Butt Up");
|
|
#endif
|
|
};
|
|
void MouseRptParser::OnMiddleButtonDown(MOUSEINFO *mi) {
|
|
#ifdef DEBUG
|
|
Serial.println("M Butt Dn");
|
|
#endif
|
|
};
|
|
|
|
USB Usb;
|
|
USBHub Hub(&Usb);
|
|
HIDBoot<USB_HID_PROTOCOL_MOUSE> HidMouse(&Usb);
|
|
|
|
MouseRptParser Prs;
|
|
#endif
|
|
|
|
static uint8_t potmouse_xcounter; ///< x axis counter
|
|
static uint8_t potmouse_ycounter; ///< y axis counter
|
|
|
|
static volatile uint16_t ocr1a_load; ///< precalculated OCR1A value (YPOT)
|
|
static volatile uint16_t ocr1b_load; ///< precalculated OCR1B value (XPOT)
|
|
|
|
void setup() {
|
|
#ifdef DEBUG
|
|
Serial.begin(115200);
|
|
delay(200);
|
|
Serial.println("Start");
|
|
Serial.flush();
|
|
#endif
|
|
|
|
#ifdef USBHOST
|
|
if (Usb.Init() == -1) {
|
|
#ifdef DEBUG
|
|
Serial.println("OSC did not start.");
|
|
#endif
|
|
}
|
|
delay(200);
|
|
HidMouse.SetReportParser(0, &Prs);
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
Serial.flush();
|
|
delay(200);
|
|
#endif
|
|
|
|
pinMode(LBTN, INPUT); pinMode(RBTN, INPUT);
|
|
|
|
pinMode(POTX, OUTPUT); pinMode(POTY, OUTPUT);
|
|
digitalWrite(POTX, HIGH); digitalWrite(POTY, HIGH);
|
|
pinMode(POTSENSE, INPUT); // pullup off, hi-biased by OC1A
|
|
|
|
potmouse_movt(0,0,0);
|
|
|
|
startTimers();
|
|
|
|
#ifndef USBHOST
|
|
TIMSK0 = 0;
|
|
#endif
|
|
|
|
}
|
|
|
|
void loop() {
|
|
#ifdef USBHOST
|
|
Usb.Task();
|
|
if (update) {
|
|
potmouse_movt(dx, dy, buttons);
|
|
update = 0;
|
|
}
|
|
delayMicroseconds(200);
|
|
#endif
|
|
|
|
#ifndef USBHOST
|
|
potmouse_movt(0, 0, buttons);
|
|
#endif
|
|
|
|
}
|
|
|
|
volatile uint8_t counter = 0;
|
|
volatile uint8_t upd = 0;
|
|
|
|
void potmouse_movt(int16_t dx, int16_t dy, uint8_t button) {
|
|
uint16_t a, b;
|
|
|
|
#ifndef USBHOST
|
|
if (upd) {
|
|
potmouse_xcounter++;
|
|
potmouse_ycounter++;
|
|
upd = 0;
|
|
}
|
|
#endif
|
|
|
|
potmouse_xcounter = (potmouse_xcounter + (dx/2)) & 0177; // modulo 128
|
|
potmouse_ycounter = (potmouse_ycounter - (dy/2)) & 0177;
|
|
|
|
//for testing
|
|
//potmouse_xcounter = (millis()>>6) & 077; // modulo 64
|
|
//potmouse_ycounter = (millis()>>6) & 077;
|
|
|
|
(button & 001) ? pinMode(LBTN, OUTPUT) : pinMode(LBTN, INPUT);
|
|
(button & 002) ? pinMode(RBTN, OUTPUT) : pinMode(RBTN, INPUT);
|
|
|
|
// scale should be 2x here, but for this particular chip, 66 counts work better where
|
|
// 64 counds should be. so 66/64=100/96 and times two
|
|
//a = 320*2 + ((uint32_t)potmouse_xcounter)*200/fix;
|
|
//b = 320*2 + ((uint32_t)potmouse_ycounter)*200/fix;
|
|
//a = 320*200/fix + potmouse_xcounter*2;
|
|
//b = 320*200/fix + potmouse_ycounter*2;
|
|
a = 320*2 + potmouse_xcounter*2;
|
|
b = 320*2 + potmouse_ycounter*2;
|
|
|
|
ocr1a_load = a;
|
|
ocr1b_load = b;
|
|
}
|
|
|
|
|
|
inline void startTimers() {
|
|
#ifdef DEBUG
|
|
Serial.println("startTimers"); Serial.flush();
|
|
#endif
|
|
cli();
|
|
|
|
// Prepare TIMER1
|
|
//TCCR1A = 0;
|
|
|
|
// ICIE1: Timer/Counter Input Capture Interrupt Enable, ISR(TIMER1_CAPT_vect)
|
|
// TOIE1: Timer/Counter Overflow Interrupt Enable
|
|
TIMSK1 = _BV(ICIE1); // ICIE1: Timer/Counter1, Input Capture Interrupt Enable
|
|
|
|
// Start timer1, Input Capture setup
|
|
// ICNC1: Input Capture Noise Canceller (Bit 7 of register TCCR1B)
|
|
// ICES1: Input Capture Edge Select (Bit 6 of register TCCR1B) 0 = FALLING, 1 = RISING
|
|
// CS12, CS11, CS10: Set prescaler (CS11 TIMER1: F_CPU/8)
|
|
TCCR1B = _BV(ICNC1) | _BV(CS11);
|
|
//TCCR1B = _BV(CS11);
|
|
|
|
TIFR1 = 0xff; // Clear all pending TIMER1 interrupt flags
|
|
|
|
sei();
|
|
}
|
|
|
|
|
|
|
|
ISR(TIMER1_CAPT_vect) {
|
|
// Now we little after start of SID reading process
|
|
// SID trigger pulse timer value is in ICR1
|
|
|
|
uint16_t a = ICR1;
|
|
|
|
#ifdef DEBUG
|
|
Serial.println("TIMER1_CAPT_vect:");
|
|
#endif
|
|
|
|
#ifndef USBHOST
|
|
counter++;
|
|
counter &= 63;
|
|
if (counter == 0) upd = 1;
|
|
#endif
|
|
|
|
// clear OC1A/OC1B (9 and 10 to LOW):
|
|
// 1. set output compare to clear OC1A/OC1B ("10" in table 37 on page 97)
|
|
TCCR1A = _BV(COM1A1) | _BV(COM1B1); // Clear OC1A / OC1B on Compare Match (Set output to low level).
|
|
// 2. force output compare to make it happen (doesn't raise interrupts)
|
|
TCCR1C |= _BV(FOC1A) | _BV(FOC1B); // FOC1A / FOC1B Force Output Compare A and B (that are in register TCCR1C)
|
|
|
|
// OCIE1A: Timer/Counter Output Compare Match Interrupt Enable A, ISR(TIMER1_COMPA_vect) // disable ICIE1, Input Capture Interrupt
|
|
TIMSK1 = _BV(OCIE1A);
|
|
|
|
// init the output compare values
|
|
OCR1A = ocr1a_load + a;
|
|
OCR1B = ocr1b_load + a;
|
|
|
|
// Set OC1A/OC1B on Compare Match (Set output to high level)
|
|
// WGM13:0 = 00, normal mode: count from BOTTOM to MAX
|
|
TCCR1A = _BV(COM1A1) | _BV(COM1A0) | _BV(COM1B1) | _BV(COM1B0); // Set OC1A / OC1B on Compare Match (Set output to high level).
|
|
|
|
#ifdef DEBUG
|
|
Serial.print(c); Serial.print(" "); Serial.print(a); Serial.print(" "); Serial.println(b);
|
|
Serial.flush();
|
|
#endif
|
|
|
|
TIFR1 = 0xff; //clear all timer1 interrupt flags
|
|
}
|
|
|
|
ISR(TIMER1_COMPA_vect) {
|
|
// now potx are sent. we don't know if poty is still in progress.
|
|
// POTX is HIGH from OC1A TIMER1 compare match. POTY is ?.
|
|
|
|
#ifdef DEBUG
|
|
Serial.println("TIMER1_COMPA_vect"); Serial.flush();
|
|
#endif
|
|
|
|
TIMSK1 = _BV(ICIE1); // ICIE1: Timer/Counter1, Input Capture Interrupt Enable // disable TIMER1 interrupts (Compare Match Interrupt A)
|
|
TIFR1 = 0xff; //clear all timer1 interrupt flags
|
|
}
|