/* * Copyright (C) 2002-2019 The DOSBox Team * * 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 2 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, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include #include #include "dosbox.h" #include "inout.h" #include "setup.h" #include "joystick.h" #include "pic.h" #include "support.h" //TODO: higher axis can't be mapped. Find out why again //Set to true, to enable automated switching back to square on circle mode if the inputs are outside the cirle. #define SUPPORT_MAP_AUTO 0 #define RANGE 64 #define TIMEOUT 10 #define OHMS 120000/2 #define JOY_S_CONSTANT 0.0000242 #define S_PER_OHM 0.000000011 struct JoyStick { enum {JOYMAP_SQUARE,JOYMAP_CIRCLE,JOYMAP_INBETWEEN} mapstate; bool enabled; float xpos, ypos; //position as set by SDL. double xtick, ytick; Bitu xcount, ycount; bool button[2]; int deadzone; //Deadzone (value between 0 and 100) interpreted as percentage. bool transformed; //Whether xpos,ypos have been converted to xfinal and yfinal. Cleared when new xpos orypos have been set float xfinal, yfinal; //position returned to the game for stick 0. void clip() { if (xfinal > 1.0) xfinal = 1.0; else if (xfinal < -1.0) xfinal = -1.0; if (yfinal > 1.0) yfinal = 1.0; else if (yfinal < -1.0) yfinal = -1.0; } void fake_digital() { if (xpos > 0.5f) xfinal = 1.0f; else if (xpos < -0.5f) xfinal = -1.0f; else xfinal = 0.0f; if (ypos > 0.5f) yfinal = 1.0f; else if (ypos < -0.5f) yfinal = -1.0f; else yfinal = 0.0f; } void transform_circular(){ float r = sqrtf(xpos * xpos + ypos * ypos); if (r == 0.0) {xfinal = xpos; yfinal = ypos; return;} float deadzone_f = deadzone / 100.0f; float s = 1.0f - deadzone_f; if (r < deadzone_f) { xfinal = yfinal = 0.0f; return; } float deadzonescale = (r - deadzone_f) / s; //r if deadzone=0; float xa = fabsf(xpos); float ya = fabsf(ypos); float maxpos = (ya>xa?ya:xa); xfinal = xpos * deadzonescale/maxpos; yfinal = ypos * deadzonescale/maxpos; } void transform_square() { float deadzone_f = deadzone / 100.0f; float s = 1.0f - deadzone_f; if (xpos > deadzone_f) { xfinal = (xpos - deadzone_f)/ s; } else if ( xpos < -deadzone_f) { xfinal = (xpos + deadzone_f) / s; } else xfinal = 0.0f; if (ypos > deadzone_f) { yfinal = (ypos - deadzone_f)/ s; } else if ( ypos < - deadzone_f) { yfinal = (ypos + deadzone_f) / s; } else yfinal = 0.0f; } #if SUPPORT_MAP_AUTO void transform_inbetween(){ //First transform to a circle and crop the values to -1.0 -> 1.0 //then keep on doing this in future calls until it is safe to switch square mapping // safe = 0.95 as ratio for both axis, or in deadzone transform_circular(); clip(); float xrate = xpos / xfinal; float yrate = ypos / yfinal; if (xrate > 0.95 && yrate > 0.95) { mapstate = JOYMAP_SQUARE; //TODO misschien xfinal=xpos... //LOG_MSG("switched to square %f %f",xrate,yrate); } } #endif void transform_input(){ if (transformed) return; transformed = true; if (deadzone == 100) fake_digital(); else { if (mapstate == JOYMAP_SQUARE) transform_square(); else if (mapstate == JOYMAP_CIRCLE) transform_circular(); #if SUPPORT_MAP_AUTO if (mapstate == JOYMAP_INBETWEEN) transform_inbetween(); //No else here #endif clip(); } } }; JoystickType joytype; static JoyStick stick[2]; static Bitu last_write = 0; static bool write_active = false; static bool swap34 = false; bool button_wrapping_enabled = true; extern bool autofire; //sdl_mapper.cpp static Bitu read_p201(Bitu port,Bitu iolen) { /* Reset Joystick to 0 after TIMEOUT ms */ if(write_active && ((PIC_Ticks - last_write) > TIMEOUT)) { write_active = false; stick[0].xcount = 0; stick[1].xcount = 0; stick[0].ycount = 0; stick[1].ycount = 0; // LOG_MSG("reset by time %d %d",PIC_Ticks,last_write); } /** Format of the byte to be returned: ** | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | ** +-------------------------------+ ** | | | | | | | | ** Joystick B, Button 2 ---+ | | | | | | +--- Joystick A, X Axis ** Joystick B, Button 1 -------+ | | | | +------- Joystick A, Y Axis ** Joystick A, Button 2 -----------+ | | +----------- Joystick B, X Axis ** Joystick A, Button 1 ---------------+ +--------------- Joystick B, Y Axis **/ Bit8u ret=0xff; if (stick[0].enabled) { if (stick[0].xcount) stick[0].xcount--; else ret&=~1; if (stick[0].ycount) stick[0].ycount--; else ret&=~2; if (stick[0].button[0]) ret&=~16; if (stick[0].button[1]) ret&=~32; } if (stick[1].enabled) { if (stick[1].xcount) stick[1].xcount--; else ret&=~4; if (stick[1].ycount) stick[1].ycount--; else ret&=~8; if (stick[1].button[0]) ret&=~64; if (stick[1].button[1]) ret&=~128; } return ret; } static Bitu read_p201_timed(Bitu port,Bitu iolen) { Bit8u ret=0xff; double currentTick = PIC_FullIndex(); if( stick[0].enabled ){ if( stick[0].xtick < currentTick ) ret &=~1; if( stick[0].ytick < currentTick ) ret &=~2; } if( stick[1].enabled ){ if( stick[1].xtick < currentTick ) ret &=~4; if( stick[1].ytick < currentTick ) ret &=~8; } if (stick[0].enabled) { if (stick[0].button[0]) ret&=~16; if (stick[0].button[1]) ret&=~32; } if (stick[1].enabled) { if (stick[1].button[0]) ret&=~64; if (stick[1].button[1]) ret&=~128; } return ret; } static void write_p201(Bitu port,Bitu val,Bitu iolen) { /* Store writetime index */ write_active = true; last_write = PIC_Ticks; if (stick[0].enabled) { stick[0].transform_input(); stick[0].xcount=(Bitu)((stick[0].xfinal*RANGE)+RANGE); stick[0].ycount=(Bitu)((stick[0].yfinal*RANGE)+RANGE); } if (stick[1].enabled) { stick[1].xcount=(Bitu)(((swap34? stick[1].ypos : stick[1].xpos)*RANGE)+RANGE); stick[1].ycount=(Bitu)(((swap34? stick[1].xpos : stick[1].ypos)*RANGE)+RANGE); } } static void write_p201_timed(Bitu port,Bitu val,Bitu iolen) { // Axes take time = 24.2 microseconds + ( 0.011 microsecons/ohm * resistance ) // to reset to 0 // Pre-calculate the time at which each axis hits 0 here double currentTick = PIC_FullIndex(); if (stick[0].enabled) { stick[0].transform_input(); stick[0].xtick = currentTick + 1000.0*( JOY_S_CONSTANT + S_PER_OHM * (double)(((stick[0].xfinal+1.0)* OHMS)) ); stick[0].ytick = currentTick + 1000.0*( JOY_S_CONSTANT + S_PER_OHM * (double)(((stick[0].yfinal+1.0)* OHMS)) ); } if (stick[1].enabled) { stick[1].xtick = currentTick + 1000.0*( JOY_S_CONSTANT + S_PER_OHM * (double)((swap34? stick[1].ypos : stick[1].xpos)+1.0) * OHMS); stick[1].ytick = currentTick + 1000.0*( JOY_S_CONSTANT + S_PER_OHM * (double)((swap34? stick[1].xpos : stick[1].ypos)+1.0) * OHMS); } } void JOYSTICK_Enable(Bitu which,bool enabled) { if (which<2) stick[which].enabled = enabled; } void JOYSTICK_Button(Bitu which,Bitu num,bool pressed) { if ((which<2) && (num<2)) stick[which].button[num] = pressed; } void JOYSTICK_Move_X(Bitu which,float x) { if(which > 2) return; if (stick[which].xpos == x) return; stick[which].xpos = x; stick[which].transformed = false; // if( which == 0 || joytype != JOY_FCS) // stick[which].applied_conversion; //todo } void JOYSTICK_Move_Y(Bitu which,float y) { if(which > 2) return; if (stick[which].ypos == y) return; stick[which].ypos = y; stick[which].transformed = false; } bool JOYSTICK_IsEnabled(Bitu which) { if (which<2) return stick[which].enabled; return false; } bool JOYSTICK_GetButton(Bitu which, Bitu num) { if ((which<2) && (num<2)) return stick[which].button[num]; return false; } float JOYSTICK_GetMove_X(Bitu which) { if (which > 1) return 0.0f; if (which == 0) { stick[0].transform_input(); return stick[0].xfinal;} return stick[1].xpos; } float JOYSTICK_GetMove_Y(Bitu which) { if (which > 1) return 0.0f; if (which == 0) { stick[0].transform_input(); return stick[0].yfinal;} return stick[1].ypos; } class JOYSTICK:public Module_base{ private: IO_ReadHandleObject ReadHandler; IO_WriteHandleObject WriteHandler; public: JOYSTICK(Section* configuration):Module_base(configuration){ Section_prop * section = static_cast(configuration); const char * type = section->Get_string("joysticktype"); if (!strcasecmp(type,"none")) joytype = JOY_NONE; else if (!strcasecmp(type,"false")) joytype = JOY_NONE; else if (!strcasecmp(type,"auto")) joytype = JOY_AUTO; else if (!strcasecmp(type,"2axis")) joytype = JOY_2AXIS; else if (!strcasecmp(type,"4axis")) joytype = JOY_4AXIS; else if (!strcasecmp(type,"4axis_2")) joytype = JOY_4AXIS_2; else if (!strcasecmp(type,"fcs")) joytype = JOY_FCS; else if (!strcasecmp(type,"ch")) joytype = JOY_CH; else joytype = JOY_AUTO; bool timed = section->Get_bool("timed"); if (timed) { ReadHandler.Install(0x201,read_p201_timed,IO_MB); WriteHandler.Install(0x201,write_p201_timed,IO_MB); } else { ReadHandler.Install(0x201,read_p201,IO_MB); WriteHandler.Install(0x201,write_p201,IO_MB); } autofire = section->Get_bool("autofire"); swap34 = section->Get_bool("swap34"); button_wrapping_enabled = section->Get_bool("buttonwrap"); stick[0].xtick = stick[0].ytick = stick[1].xtick = stick[1].ytick = PIC_FullIndex(); stick[0].xpos = stick[0].ypos = stick[1].xpos = stick[1].ypos = 0.0f; stick[0].transformed = false; stick[0].mapstate = JoyStick::JOYMAP_SQUARE; bool circ = section->Get_bool("circularinput"); if (circ) stick[0].mapstate = JoyStick::JOYMAP_CIRCLE; stick[0].deadzone = section->Get_int("deadzone"); } }; static JOYSTICK* test; void JOYSTICK_Destroy(Section* sec) { delete test; } void JOYSTICK_Init(Section* sec) { test = new JOYSTICK(sec); sec->AddDestroyFunction(&JOYSTICK_Destroy,true); }