337 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			337 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  *  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 <string.h>
 | |
| #include <math.h>
 | |
| #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<Section_prop *>(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); 
 | |
| }
 |