162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (c) 1999-2001 Vojtech Pavlik 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Based on the work of: 662306a36Sopenharmony_ci * Andree Borrmann Mats Sjövall 762306a36Sopenharmony_ci */ 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci/* 1062306a36Sopenharmony_ci * Atari, Amstrad, Commodore, Amiga, Sega, etc. joystick driver for Linux 1162306a36Sopenharmony_ci */ 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#include <linux/kernel.h> 1462306a36Sopenharmony_ci#include <linux/module.h> 1562306a36Sopenharmony_ci#include <linux/delay.h> 1662306a36Sopenharmony_ci#include <linux/init.h> 1762306a36Sopenharmony_ci#include <linux/parport.h> 1862306a36Sopenharmony_ci#include <linux/input.h> 1962306a36Sopenharmony_ci#include <linux/mutex.h> 2062306a36Sopenharmony_ci#include <linux/slab.h> 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ciMODULE_AUTHOR("Vojtech Pavlik <vojtech@ucw.cz>"); 2362306a36Sopenharmony_ciMODULE_DESCRIPTION("Atari, Amstrad, Commodore, Amiga, Sega, etc. joystick driver"); 2462306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_cistruct db9_config { 2762306a36Sopenharmony_ci int args[2]; 2862306a36Sopenharmony_ci unsigned int nargs; 2962306a36Sopenharmony_ci}; 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_ci#define DB9_MAX_PORTS 3 3262306a36Sopenharmony_cistatic struct db9_config db9_cfg[DB9_MAX_PORTS]; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cimodule_param_array_named(dev, db9_cfg[0].args, int, &db9_cfg[0].nargs, 0); 3562306a36Sopenharmony_ciMODULE_PARM_DESC(dev, "Describes first attached device (<parport#>,<type>)"); 3662306a36Sopenharmony_cimodule_param_array_named(dev2, db9_cfg[1].args, int, &db9_cfg[1].nargs, 0); 3762306a36Sopenharmony_ciMODULE_PARM_DESC(dev2, "Describes second attached device (<parport#>,<type>)"); 3862306a36Sopenharmony_cimodule_param_array_named(dev3, db9_cfg[2].args, int, &db9_cfg[2].nargs, 0); 3962306a36Sopenharmony_ciMODULE_PARM_DESC(dev3, "Describes third attached device (<parport#>,<type>)"); 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_ci#define DB9_ARG_PARPORT 0 4262306a36Sopenharmony_ci#define DB9_ARG_MODE 1 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci#define DB9_MULTI_STICK 0x01 4562306a36Sopenharmony_ci#define DB9_MULTI2_STICK 0x02 4662306a36Sopenharmony_ci#define DB9_GENESIS_PAD 0x03 4762306a36Sopenharmony_ci#define DB9_GENESIS5_PAD 0x05 4862306a36Sopenharmony_ci#define DB9_GENESIS6_PAD 0x06 4962306a36Sopenharmony_ci#define DB9_SATURN_PAD 0x07 5062306a36Sopenharmony_ci#define DB9_MULTI_0802 0x08 5162306a36Sopenharmony_ci#define DB9_MULTI_0802_2 0x09 5262306a36Sopenharmony_ci#define DB9_CD32_PAD 0x0A 5362306a36Sopenharmony_ci#define DB9_SATURN_DPP 0x0B 5462306a36Sopenharmony_ci#define DB9_SATURN_DPP_2 0x0C 5562306a36Sopenharmony_ci#define DB9_MAX_PAD 0x0D 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci#define DB9_UP 0x01 5862306a36Sopenharmony_ci#define DB9_DOWN 0x02 5962306a36Sopenharmony_ci#define DB9_LEFT 0x04 6062306a36Sopenharmony_ci#define DB9_RIGHT 0x08 6162306a36Sopenharmony_ci#define DB9_FIRE1 0x10 6262306a36Sopenharmony_ci#define DB9_FIRE2 0x20 6362306a36Sopenharmony_ci#define DB9_FIRE3 0x40 6462306a36Sopenharmony_ci#define DB9_FIRE4 0x80 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci#define DB9_NORMAL 0x0a 6762306a36Sopenharmony_ci#define DB9_NOSELECT 0x08 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci#define DB9_GENESIS6_DELAY 14 7062306a36Sopenharmony_ci#define DB9_REFRESH_TIME HZ/100 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci#define DB9_MAX_DEVICES 2 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_cistruct db9_mode_data { 7562306a36Sopenharmony_ci const char *name; 7662306a36Sopenharmony_ci const short *buttons; 7762306a36Sopenharmony_ci int n_buttons; 7862306a36Sopenharmony_ci int n_pads; 7962306a36Sopenharmony_ci int n_axis; 8062306a36Sopenharmony_ci int bidirectional; 8162306a36Sopenharmony_ci int reverse; 8262306a36Sopenharmony_ci}; 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_cistruct db9 { 8562306a36Sopenharmony_ci struct input_dev *dev[DB9_MAX_DEVICES]; 8662306a36Sopenharmony_ci struct timer_list timer; 8762306a36Sopenharmony_ci struct pardevice *pd; 8862306a36Sopenharmony_ci int mode; 8962306a36Sopenharmony_ci int used; 9062306a36Sopenharmony_ci int parportno; 9162306a36Sopenharmony_ci struct mutex mutex; 9262306a36Sopenharmony_ci char phys[DB9_MAX_DEVICES][32]; 9362306a36Sopenharmony_ci}; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_cistatic struct db9 *db9_base[3]; 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_cistatic const short db9_multi_btn[] = { BTN_TRIGGER, BTN_THUMB }; 9862306a36Sopenharmony_cistatic const short db9_genesis_btn[] = { BTN_START, BTN_A, BTN_B, BTN_C, BTN_X, BTN_Y, BTN_Z, BTN_MODE }; 9962306a36Sopenharmony_cistatic const short db9_cd32_btn[] = { BTN_A, BTN_B, BTN_C, BTN_X, BTN_Y, BTN_Z, BTN_TL, BTN_TR, BTN_START }; 10062306a36Sopenharmony_cistatic const short db9_abs[] = { ABS_X, ABS_Y, ABS_RX, ABS_RY, ABS_RZ, ABS_Z, ABS_HAT0X, ABS_HAT0Y, ABS_HAT1X, ABS_HAT1Y }; 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_cistatic const struct db9_mode_data db9_modes[] = { 10362306a36Sopenharmony_ci { NULL, NULL, 0, 0, 0, 0, 0 }, 10462306a36Sopenharmony_ci { "Multisystem joystick", db9_multi_btn, 1, 1, 2, 1, 1 }, 10562306a36Sopenharmony_ci { "Multisystem joystick (2 fire)", db9_multi_btn, 2, 1, 2, 1, 1 }, 10662306a36Sopenharmony_ci { "Genesis pad", db9_genesis_btn, 4, 1, 2, 1, 1 }, 10762306a36Sopenharmony_ci { NULL, NULL, 0, 0, 0, 0, 0 }, 10862306a36Sopenharmony_ci { "Genesis 5 pad", db9_genesis_btn, 6, 1, 2, 1, 1 }, 10962306a36Sopenharmony_ci { "Genesis 6 pad", db9_genesis_btn, 8, 1, 2, 1, 1 }, 11062306a36Sopenharmony_ci { "Saturn pad", db9_cd32_btn, 9, 6, 7, 0, 1 }, 11162306a36Sopenharmony_ci { "Multisystem (0.8.0.2) joystick", db9_multi_btn, 1, 1, 2, 1, 1 }, 11262306a36Sopenharmony_ci { "Multisystem (0.8.0.2-dual) joystick", db9_multi_btn, 1, 2, 2, 1, 1 }, 11362306a36Sopenharmony_ci { "Amiga CD-32 pad", db9_cd32_btn, 7, 1, 2, 1, 1 }, 11462306a36Sopenharmony_ci { "Saturn dpp", db9_cd32_btn, 9, 6, 7, 0, 0 }, 11562306a36Sopenharmony_ci { "Saturn dpp dual", db9_cd32_btn, 9, 12, 7, 0, 0 }, 11662306a36Sopenharmony_ci}; 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ci/* 11962306a36Sopenharmony_ci * Saturn controllers 12062306a36Sopenharmony_ci */ 12162306a36Sopenharmony_ci#define DB9_SATURN_DELAY 300 12262306a36Sopenharmony_cistatic const int db9_saturn_byte[] = { 1, 1, 1, 2, 2, 2, 2, 2, 1 }; 12362306a36Sopenharmony_cistatic const unsigned char db9_saturn_mask[] = { 0x04, 0x01, 0x02, 0x40, 0x20, 0x10, 0x08, 0x80, 0x08 }; 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci/* 12662306a36Sopenharmony_ci * db9_saturn_write_sub() writes 2 bit data. 12762306a36Sopenharmony_ci */ 12862306a36Sopenharmony_cistatic void db9_saturn_write_sub(struct parport *port, int type, unsigned char data, int powered, int pwr_sub) 12962306a36Sopenharmony_ci{ 13062306a36Sopenharmony_ci unsigned char c; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci switch (type) { 13362306a36Sopenharmony_ci case 1: /* DPP1 */ 13462306a36Sopenharmony_ci c = 0x80 | 0x30 | (powered ? 0x08 : 0) | (pwr_sub ? 0x04 : 0) | data; 13562306a36Sopenharmony_ci parport_write_data(port, c); 13662306a36Sopenharmony_ci break; 13762306a36Sopenharmony_ci case 2: /* DPP2 */ 13862306a36Sopenharmony_ci c = 0x40 | data << 4 | (powered ? 0x08 : 0) | (pwr_sub ? 0x04 : 0) | 0x03; 13962306a36Sopenharmony_ci parport_write_data(port, c); 14062306a36Sopenharmony_ci break; 14162306a36Sopenharmony_ci case 0: /* DB9 */ 14262306a36Sopenharmony_ci c = ((((data & 2) ? 2 : 0) | ((data & 1) ? 4 : 0)) ^ 0x02) | !powered; 14362306a36Sopenharmony_ci parport_write_control(port, c); 14462306a36Sopenharmony_ci break; 14562306a36Sopenharmony_ci } 14662306a36Sopenharmony_ci} 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci/* 14962306a36Sopenharmony_ci * gc_saturn_read_sub() reads 4 bit data. 15062306a36Sopenharmony_ci */ 15162306a36Sopenharmony_cistatic unsigned char db9_saturn_read_sub(struct parport *port, int type) 15262306a36Sopenharmony_ci{ 15362306a36Sopenharmony_ci unsigned char data; 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci if (type) { 15662306a36Sopenharmony_ci /* DPP */ 15762306a36Sopenharmony_ci data = parport_read_status(port) ^ 0x80; 15862306a36Sopenharmony_ci return (data & 0x80 ? 1 : 0) | (data & 0x40 ? 2 : 0) 15962306a36Sopenharmony_ci | (data & 0x20 ? 4 : 0) | (data & 0x10 ? 8 : 0); 16062306a36Sopenharmony_ci } else { 16162306a36Sopenharmony_ci /* DB9 */ 16262306a36Sopenharmony_ci data = parport_read_data(port) & 0x0f; 16362306a36Sopenharmony_ci return (data & 0x8 ? 1 : 0) | (data & 0x4 ? 2 : 0) 16462306a36Sopenharmony_ci | (data & 0x2 ? 4 : 0) | (data & 0x1 ? 8 : 0); 16562306a36Sopenharmony_ci } 16662306a36Sopenharmony_ci} 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci/* 16962306a36Sopenharmony_ci * db9_saturn_read_analog() sends clock and reads 8 bit data. 17062306a36Sopenharmony_ci */ 17162306a36Sopenharmony_cistatic unsigned char db9_saturn_read_analog(struct parport *port, int type, int powered) 17262306a36Sopenharmony_ci{ 17362306a36Sopenharmony_ci unsigned char data; 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 0, powered, 0); 17662306a36Sopenharmony_ci udelay(DB9_SATURN_DELAY); 17762306a36Sopenharmony_ci data = db9_saturn_read_sub(port, type) << 4; 17862306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 2, powered, 0); 17962306a36Sopenharmony_ci udelay(DB9_SATURN_DELAY); 18062306a36Sopenharmony_ci data |= db9_saturn_read_sub(port, type); 18162306a36Sopenharmony_ci return data; 18262306a36Sopenharmony_ci} 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci/* 18562306a36Sopenharmony_ci * db9_saturn_read_packet() reads whole saturn packet at connector 18662306a36Sopenharmony_ci * and returns device identifier code. 18762306a36Sopenharmony_ci */ 18862306a36Sopenharmony_cistatic unsigned char db9_saturn_read_packet(struct parport *port, unsigned char *data, int type, int powered) 18962306a36Sopenharmony_ci{ 19062306a36Sopenharmony_ci int i, j; 19162306a36Sopenharmony_ci unsigned char tmp; 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 3, powered, 0); 19462306a36Sopenharmony_ci data[0] = db9_saturn_read_sub(port, type); 19562306a36Sopenharmony_ci switch (data[0] & 0x0f) { 19662306a36Sopenharmony_ci case 0xf: 19762306a36Sopenharmony_ci /* 1111 no pad */ 19862306a36Sopenharmony_ci return data[0] = 0xff; 19962306a36Sopenharmony_ci case 0x4: case 0x4 | 0x8: 20062306a36Sopenharmony_ci /* ?100 : digital controller */ 20162306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 0, powered, 1); 20262306a36Sopenharmony_ci data[2] = db9_saturn_read_sub(port, type) << 4; 20362306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 2, powered, 1); 20462306a36Sopenharmony_ci data[1] = db9_saturn_read_sub(port, type) << 4; 20562306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 1, powered, 1); 20662306a36Sopenharmony_ci data[1] |= db9_saturn_read_sub(port, type); 20762306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 3, powered, 1); 20862306a36Sopenharmony_ci /* data[2] |= db9_saturn_read_sub(port, type); */ 20962306a36Sopenharmony_ci data[2] |= data[0]; 21062306a36Sopenharmony_ci return data[0] = 0x02; 21162306a36Sopenharmony_ci case 0x1: 21262306a36Sopenharmony_ci /* 0001 : analog controller or multitap */ 21362306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 2, powered, 0); 21462306a36Sopenharmony_ci udelay(DB9_SATURN_DELAY); 21562306a36Sopenharmony_ci data[0] = db9_saturn_read_analog(port, type, powered); 21662306a36Sopenharmony_ci if (data[0] != 0x41) { 21762306a36Sopenharmony_ci /* read analog controller */ 21862306a36Sopenharmony_ci for (i = 0; i < (data[0] & 0x0f); i++) 21962306a36Sopenharmony_ci data[i + 1] = db9_saturn_read_analog(port, type, powered); 22062306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 3, powered, 0); 22162306a36Sopenharmony_ci return data[0]; 22262306a36Sopenharmony_ci } else { 22362306a36Sopenharmony_ci /* read multitap */ 22462306a36Sopenharmony_ci if (db9_saturn_read_analog(port, type, powered) != 0x60) 22562306a36Sopenharmony_ci return data[0] = 0xff; 22662306a36Sopenharmony_ci for (i = 0; i < 60; i += 10) { 22762306a36Sopenharmony_ci data[i] = db9_saturn_read_analog(port, type, powered); 22862306a36Sopenharmony_ci if (data[i] != 0xff) 22962306a36Sopenharmony_ci /* read each pad */ 23062306a36Sopenharmony_ci for (j = 0; j < (data[i] & 0x0f); j++) 23162306a36Sopenharmony_ci data[i + j + 1] = db9_saturn_read_analog(port, type, powered); 23262306a36Sopenharmony_ci } 23362306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 3, powered, 0); 23462306a36Sopenharmony_ci return 0x41; 23562306a36Sopenharmony_ci } 23662306a36Sopenharmony_ci case 0x0: 23762306a36Sopenharmony_ci /* 0000 : mouse */ 23862306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 2, powered, 0); 23962306a36Sopenharmony_ci udelay(DB9_SATURN_DELAY); 24062306a36Sopenharmony_ci tmp = db9_saturn_read_analog(port, type, powered); 24162306a36Sopenharmony_ci if (tmp == 0xff) { 24262306a36Sopenharmony_ci for (i = 0; i < 3; i++) 24362306a36Sopenharmony_ci data[i + 1] = db9_saturn_read_analog(port, type, powered); 24462306a36Sopenharmony_ci db9_saturn_write_sub(port, type, 3, powered, 0); 24562306a36Sopenharmony_ci return data[0] = 0xe3; 24662306a36Sopenharmony_ci } 24762306a36Sopenharmony_ci fallthrough; 24862306a36Sopenharmony_ci default: 24962306a36Sopenharmony_ci return data[0]; 25062306a36Sopenharmony_ci } 25162306a36Sopenharmony_ci} 25262306a36Sopenharmony_ci 25362306a36Sopenharmony_ci/* 25462306a36Sopenharmony_ci * db9_saturn_report() analyzes packet and reports. 25562306a36Sopenharmony_ci */ 25662306a36Sopenharmony_cistatic int db9_saturn_report(unsigned char id, unsigned char data[60], struct input_dev *devs[], int n, int max_pads) 25762306a36Sopenharmony_ci{ 25862306a36Sopenharmony_ci struct input_dev *dev; 25962306a36Sopenharmony_ci int tmp, i, j; 26062306a36Sopenharmony_ci 26162306a36Sopenharmony_ci tmp = (id == 0x41) ? 60 : 10; 26262306a36Sopenharmony_ci for (j = 0; j < tmp && n < max_pads; j += 10, n++) { 26362306a36Sopenharmony_ci dev = devs[n]; 26462306a36Sopenharmony_ci switch (data[j]) { 26562306a36Sopenharmony_ci case 0x16: /* multi controller (analog 4 axis) */ 26662306a36Sopenharmony_ci input_report_abs(dev, db9_abs[5], data[j + 6]); 26762306a36Sopenharmony_ci fallthrough; 26862306a36Sopenharmony_ci case 0x15: /* mission stick (analog 3 axis) */ 26962306a36Sopenharmony_ci input_report_abs(dev, db9_abs[3], data[j + 4]); 27062306a36Sopenharmony_ci input_report_abs(dev, db9_abs[4], data[j + 5]); 27162306a36Sopenharmony_ci fallthrough; 27262306a36Sopenharmony_ci case 0x13: /* racing controller (analog 1 axis) */ 27362306a36Sopenharmony_ci input_report_abs(dev, db9_abs[2], data[j + 3]); 27462306a36Sopenharmony_ci fallthrough; 27562306a36Sopenharmony_ci case 0x34: /* saturn keyboard (udlr ZXC ASD QE Esc) */ 27662306a36Sopenharmony_ci case 0x02: /* digital pad (digital 2 axis + buttons) */ 27762306a36Sopenharmony_ci input_report_abs(dev, db9_abs[0], !(data[j + 1] & 128) - !(data[j + 1] & 64)); 27862306a36Sopenharmony_ci input_report_abs(dev, db9_abs[1], !(data[j + 1] & 32) - !(data[j + 1] & 16)); 27962306a36Sopenharmony_ci for (i = 0; i < 9; i++) 28062306a36Sopenharmony_ci input_report_key(dev, db9_cd32_btn[i], ~data[j + db9_saturn_byte[i]] & db9_saturn_mask[i]); 28162306a36Sopenharmony_ci break; 28262306a36Sopenharmony_ci case 0x19: /* mission stick x2 (analog 6 axis + buttons) */ 28362306a36Sopenharmony_ci input_report_abs(dev, db9_abs[0], !(data[j + 1] & 128) - !(data[j + 1] & 64)); 28462306a36Sopenharmony_ci input_report_abs(dev, db9_abs[1], !(data[j + 1] & 32) - !(data[j + 1] & 16)); 28562306a36Sopenharmony_ci for (i = 0; i < 9; i++) 28662306a36Sopenharmony_ci input_report_key(dev, db9_cd32_btn[i], ~data[j + db9_saturn_byte[i]] & db9_saturn_mask[i]); 28762306a36Sopenharmony_ci input_report_abs(dev, db9_abs[2], data[j + 3]); 28862306a36Sopenharmony_ci input_report_abs(dev, db9_abs[3], data[j + 4]); 28962306a36Sopenharmony_ci input_report_abs(dev, db9_abs[4], data[j + 5]); 29062306a36Sopenharmony_ci /* 29162306a36Sopenharmony_ci input_report_abs(dev, db9_abs[8], (data[j + 6] & 128 ? 0 : 1) - (data[j + 6] & 64 ? 0 : 1)); 29262306a36Sopenharmony_ci input_report_abs(dev, db9_abs[9], (data[j + 6] & 32 ? 0 : 1) - (data[j + 6] & 16 ? 0 : 1)); 29362306a36Sopenharmony_ci */ 29462306a36Sopenharmony_ci input_report_abs(dev, db9_abs[6], data[j + 7]); 29562306a36Sopenharmony_ci input_report_abs(dev, db9_abs[7], data[j + 8]); 29662306a36Sopenharmony_ci input_report_abs(dev, db9_abs[5], data[j + 9]); 29762306a36Sopenharmony_ci break; 29862306a36Sopenharmony_ci case 0xd3: /* sankyo ff (analog 1 axis + stop btn) */ 29962306a36Sopenharmony_ci input_report_key(dev, BTN_A, data[j + 3] & 0x80); 30062306a36Sopenharmony_ci input_report_abs(dev, db9_abs[2], data[j + 3] & 0x7f); 30162306a36Sopenharmony_ci break; 30262306a36Sopenharmony_ci case 0xe3: /* shuttle mouse (analog 2 axis + buttons. signed value) */ 30362306a36Sopenharmony_ci input_report_key(dev, BTN_START, data[j + 1] & 0x08); 30462306a36Sopenharmony_ci input_report_key(dev, BTN_A, data[j + 1] & 0x04); 30562306a36Sopenharmony_ci input_report_key(dev, BTN_C, data[j + 1] & 0x02); 30662306a36Sopenharmony_ci input_report_key(dev, BTN_B, data[j + 1] & 0x01); 30762306a36Sopenharmony_ci input_report_abs(dev, db9_abs[2], data[j + 2] ^ 0x80); 30862306a36Sopenharmony_ci input_report_abs(dev, db9_abs[3], (0xff-(data[j + 3] ^ 0x80))+1); /* */ 30962306a36Sopenharmony_ci break; 31062306a36Sopenharmony_ci case 0xff: 31162306a36Sopenharmony_ci default: /* no pad */ 31262306a36Sopenharmony_ci input_report_abs(dev, db9_abs[0], 0); 31362306a36Sopenharmony_ci input_report_abs(dev, db9_abs[1], 0); 31462306a36Sopenharmony_ci for (i = 0; i < 9; i++) 31562306a36Sopenharmony_ci input_report_key(dev, db9_cd32_btn[i], 0); 31662306a36Sopenharmony_ci break; 31762306a36Sopenharmony_ci } 31862306a36Sopenharmony_ci } 31962306a36Sopenharmony_ci return n; 32062306a36Sopenharmony_ci} 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_cistatic int db9_saturn(int mode, struct parport *port, struct input_dev *devs[]) 32362306a36Sopenharmony_ci{ 32462306a36Sopenharmony_ci unsigned char id, data[60]; 32562306a36Sopenharmony_ci int type, n, max_pads; 32662306a36Sopenharmony_ci int tmp, i; 32762306a36Sopenharmony_ci 32862306a36Sopenharmony_ci switch (mode) { 32962306a36Sopenharmony_ci case DB9_SATURN_PAD: 33062306a36Sopenharmony_ci type = 0; 33162306a36Sopenharmony_ci n = 1; 33262306a36Sopenharmony_ci break; 33362306a36Sopenharmony_ci case DB9_SATURN_DPP: 33462306a36Sopenharmony_ci type = 1; 33562306a36Sopenharmony_ci n = 1; 33662306a36Sopenharmony_ci break; 33762306a36Sopenharmony_ci case DB9_SATURN_DPP_2: 33862306a36Sopenharmony_ci type = 1; 33962306a36Sopenharmony_ci n = 2; 34062306a36Sopenharmony_ci break; 34162306a36Sopenharmony_ci default: 34262306a36Sopenharmony_ci return -1; 34362306a36Sopenharmony_ci } 34462306a36Sopenharmony_ci max_pads = min(db9_modes[mode].n_pads, DB9_MAX_DEVICES); 34562306a36Sopenharmony_ci for (tmp = 0, i = 0; i < n; i++) { 34662306a36Sopenharmony_ci id = db9_saturn_read_packet(port, data, type + i, 1); 34762306a36Sopenharmony_ci tmp = db9_saturn_report(id, data, devs, tmp, max_pads); 34862306a36Sopenharmony_ci } 34962306a36Sopenharmony_ci return 0; 35062306a36Sopenharmony_ci} 35162306a36Sopenharmony_ci 35262306a36Sopenharmony_cistatic void db9_timer(struct timer_list *t) 35362306a36Sopenharmony_ci{ 35462306a36Sopenharmony_ci struct db9 *db9 = from_timer(db9, t, timer); 35562306a36Sopenharmony_ci struct parport *port = db9->pd->port; 35662306a36Sopenharmony_ci struct input_dev *dev = db9->dev[0]; 35762306a36Sopenharmony_ci struct input_dev *dev2 = db9->dev[1]; 35862306a36Sopenharmony_ci int data, i; 35962306a36Sopenharmony_ci 36062306a36Sopenharmony_ci switch (db9->mode) { 36162306a36Sopenharmony_ci case DB9_MULTI_0802_2: 36262306a36Sopenharmony_ci 36362306a36Sopenharmony_ci data = parport_read_data(port) >> 3; 36462306a36Sopenharmony_ci 36562306a36Sopenharmony_ci input_report_abs(dev2, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 36662306a36Sopenharmony_ci input_report_abs(dev2, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 36762306a36Sopenharmony_ci input_report_key(dev2, BTN_TRIGGER, ~data & DB9_FIRE1); 36862306a36Sopenharmony_ci fallthrough; 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci case DB9_MULTI_0802: 37162306a36Sopenharmony_ci 37262306a36Sopenharmony_ci data = parport_read_status(port) >> 3; 37362306a36Sopenharmony_ci 37462306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 37562306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 37662306a36Sopenharmony_ci input_report_key(dev, BTN_TRIGGER, data & DB9_FIRE1); 37762306a36Sopenharmony_ci break; 37862306a36Sopenharmony_ci 37962306a36Sopenharmony_ci case DB9_MULTI_STICK: 38062306a36Sopenharmony_ci 38162306a36Sopenharmony_ci data = parport_read_data(port); 38262306a36Sopenharmony_ci 38362306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 38462306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 38562306a36Sopenharmony_ci input_report_key(dev, BTN_TRIGGER, ~data & DB9_FIRE1); 38662306a36Sopenharmony_ci break; 38762306a36Sopenharmony_ci 38862306a36Sopenharmony_ci case DB9_MULTI2_STICK: 38962306a36Sopenharmony_ci 39062306a36Sopenharmony_ci data = parport_read_data(port); 39162306a36Sopenharmony_ci 39262306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 39362306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 39462306a36Sopenharmony_ci input_report_key(dev, BTN_TRIGGER, ~data & DB9_FIRE1); 39562306a36Sopenharmony_ci input_report_key(dev, BTN_THUMB, ~data & DB9_FIRE2); 39662306a36Sopenharmony_ci break; 39762306a36Sopenharmony_ci 39862306a36Sopenharmony_ci case DB9_GENESIS_PAD: 39962306a36Sopenharmony_ci 40062306a36Sopenharmony_ci parport_write_control(port, DB9_NOSELECT); 40162306a36Sopenharmony_ci data = parport_read_data(port); 40262306a36Sopenharmony_ci 40362306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 40462306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 40562306a36Sopenharmony_ci input_report_key(dev, BTN_B, ~data & DB9_FIRE1); 40662306a36Sopenharmony_ci input_report_key(dev, BTN_C, ~data & DB9_FIRE2); 40762306a36Sopenharmony_ci 40862306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 40962306a36Sopenharmony_ci data = parport_read_data(port); 41062306a36Sopenharmony_ci 41162306a36Sopenharmony_ci input_report_key(dev, BTN_A, ~data & DB9_FIRE1); 41262306a36Sopenharmony_ci input_report_key(dev, BTN_START, ~data & DB9_FIRE2); 41362306a36Sopenharmony_ci break; 41462306a36Sopenharmony_ci 41562306a36Sopenharmony_ci case DB9_GENESIS5_PAD: 41662306a36Sopenharmony_ci 41762306a36Sopenharmony_ci parport_write_control(port, DB9_NOSELECT); 41862306a36Sopenharmony_ci data = parport_read_data(port); 41962306a36Sopenharmony_ci 42062306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 42162306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 42262306a36Sopenharmony_ci input_report_key(dev, BTN_B, ~data & DB9_FIRE1); 42362306a36Sopenharmony_ci input_report_key(dev, BTN_C, ~data & DB9_FIRE2); 42462306a36Sopenharmony_ci 42562306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 42662306a36Sopenharmony_ci data = parport_read_data(port); 42762306a36Sopenharmony_ci 42862306a36Sopenharmony_ci input_report_key(dev, BTN_A, ~data & DB9_FIRE1); 42962306a36Sopenharmony_ci input_report_key(dev, BTN_X, ~data & DB9_FIRE2); 43062306a36Sopenharmony_ci input_report_key(dev, BTN_Y, ~data & DB9_LEFT); 43162306a36Sopenharmony_ci input_report_key(dev, BTN_START, ~data & DB9_RIGHT); 43262306a36Sopenharmony_ci break; 43362306a36Sopenharmony_ci 43462306a36Sopenharmony_ci case DB9_GENESIS6_PAD: 43562306a36Sopenharmony_ci 43662306a36Sopenharmony_ci parport_write_control(port, DB9_NOSELECT); /* 1 */ 43762306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 43862306a36Sopenharmony_ci data = parport_read_data(port); 43962306a36Sopenharmony_ci 44062306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 44162306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 44262306a36Sopenharmony_ci input_report_key(dev, BTN_B, ~data & DB9_FIRE1); 44362306a36Sopenharmony_ci input_report_key(dev, BTN_C, ~data & DB9_FIRE2); 44462306a36Sopenharmony_ci 44562306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 44662306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 44762306a36Sopenharmony_ci data = parport_read_data(port); 44862306a36Sopenharmony_ci 44962306a36Sopenharmony_ci input_report_key(dev, BTN_A, ~data & DB9_FIRE1); 45062306a36Sopenharmony_ci input_report_key(dev, BTN_START, ~data & DB9_FIRE2); 45162306a36Sopenharmony_ci 45262306a36Sopenharmony_ci parport_write_control(port, DB9_NOSELECT); /* 2 */ 45362306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 45462306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 45562306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 45662306a36Sopenharmony_ci parport_write_control(port, DB9_NOSELECT); /* 3 */ 45762306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 45862306a36Sopenharmony_ci data=parport_read_data(port); 45962306a36Sopenharmony_ci 46062306a36Sopenharmony_ci input_report_key(dev, BTN_X, ~data & DB9_LEFT); 46162306a36Sopenharmony_ci input_report_key(dev, BTN_Y, ~data & DB9_DOWN); 46262306a36Sopenharmony_ci input_report_key(dev, BTN_Z, ~data & DB9_UP); 46362306a36Sopenharmony_ci input_report_key(dev, BTN_MODE, ~data & DB9_RIGHT); 46462306a36Sopenharmony_ci 46562306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 46662306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 46762306a36Sopenharmony_ci parport_write_control(port, DB9_NOSELECT); /* 4 */ 46862306a36Sopenharmony_ci udelay(DB9_GENESIS6_DELAY); 46962306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 47062306a36Sopenharmony_ci break; 47162306a36Sopenharmony_ci 47262306a36Sopenharmony_ci case DB9_SATURN_PAD: 47362306a36Sopenharmony_ci case DB9_SATURN_DPP: 47462306a36Sopenharmony_ci case DB9_SATURN_DPP_2: 47562306a36Sopenharmony_ci 47662306a36Sopenharmony_ci db9_saturn(db9->mode, port, db9->dev); 47762306a36Sopenharmony_ci break; 47862306a36Sopenharmony_ci 47962306a36Sopenharmony_ci case DB9_CD32_PAD: 48062306a36Sopenharmony_ci 48162306a36Sopenharmony_ci data = parport_read_data(port); 48262306a36Sopenharmony_ci 48362306a36Sopenharmony_ci input_report_abs(dev, ABS_X, (data & DB9_RIGHT ? 0 : 1) - (data & DB9_LEFT ? 0 : 1)); 48462306a36Sopenharmony_ci input_report_abs(dev, ABS_Y, (data & DB9_DOWN ? 0 : 1) - (data & DB9_UP ? 0 : 1)); 48562306a36Sopenharmony_ci 48662306a36Sopenharmony_ci parport_write_control(port, 0x0a); 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_ci for (i = 0; i < 7; i++) { 48962306a36Sopenharmony_ci data = parport_read_data(port); 49062306a36Sopenharmony_ci parport_write_control(port, 0x02); 49162306a36Sopenharmony_ci parport_write_control(port, 0x0a); 49262306a36Sopenharmony_ci input_report_key(dev, db9_cd32_btn[i], ~data & DB9_FIRE2); 49362306a36Sopenharmony_ci } 49462306a36Sopenharmony_ci 49562306a36Sopenharmony_ci parport_write_control(port, 0x00); 49662306a36Sopenharmony_ci break; 49762306a36Sopenharmony_ci } 49862306a36Sopenharmony_ci 49962306a36Sopenharmony_ci input_sync(dev); 50062306a36Sopenharmony_ci 50162306a36Sopenharmony_ci mod_timer(&db9->timer, jiffies + DB9_REFRESH_TIME); 50262306a36Sopenharmony_ci} 50362306a36Sopenharmony_ci 50462306a36Sopenharmony_cistatic int db9_open(struct input_dev *dev) 50562306a36Sopenharmony_ci{ 50662306a36Sopenharmony_ci struct db9 *db9 = input_get_drvdata(dev); 50762306a36Sopenharmony_ci struct parport *port = db9->pd->port; 50862306a36Sopenharmony_ci int err; 50962306a36Sopenharmony_ci 51062306a36Sopenharmony_ci err = mutex_lock_interruptible(&db9->mutex); 51162306a36Sopenharmony_ci if (err) 51262306a36Sopenharmony_ci return err; 51362306a36Sopenharmony_ci 51462306a36Sopenharmony_ci if (!db9->used++) { 51562306a36Sopenharmony_ci parport_claim(db9->pd); 51662306a36Sopenharmony_ci parport_write_data(port, 0xff); 51762306a36Sopenharmony_ci if (db9_modes[db9->mode].reverse) { 51862306a36Sopenharmony_ci parport_data_reverse(port); 51962306a36Sopenharmony_ci parport_write_control(port, DB9_NORMAL); 52062306a36Sopenharmony_ci } 52162306a36Sopenharmony_ci mod_timer(&db9->timer, jiffies + DB9_REFRESH_TIME); 52262306a36Sopenharmony_ci } 52362306a36Sopenharmony_ci 52462306a36Sopenharmony_ci mutex_unlock(&db9->mutex); 52562306a36Sopenharmony_ci return 0; 52662306a36Sopenharmony_ci} 52762306a36Sopenharmony_ci 52862306a36Sopenharmony_cistatic void db9_close(struct input_dev *dev) 52962306a36Sopenharmony_ci{ 53062306a36Sopenharmony_ci struct db9 *db9 = input_get_drvdata(dev); 53162306a36Sopenharmony_ci struct parport *port = db9->pd->port; 53262306a36Sopenharmony_ci 53362306a36Sopenharmony_ci mutex_lock(&db9->mutex); 53462306a36Sopenharmony_ci if (!--db9->used) { 53562306a36Sopenharmony_ci del_timer_sync(&db9->timer); 53662306a36Sopenharmony_ci parport_write_control(port, 0x00); 53762306a36Sopenharmony_ci parport_data_forward(port); 53862306a36Sopenharmony_ci parport_release(db9->pd); 53962306a36Sopenharmony_ci } 54062306a36Sopenharmony_ci mutex_unlock(&db9->mutex); 54162306a36Sopenharmony_ci} 54262306a36Sopenharmony_ci 54362306a36Sopenharmony_cistatic void db9_attach(struct parport *pp) 54462306a36Sopenharmony_ci{ 54562306a36Sopenharmony_ci struct db9 *db9; 54662306a36Sopenharmony_ci const struct db9_mode_data *db9_mode; 54762306a36Sopenharmony_ci struct pardevice *pd; 54862306a36Sopenharmony_ci struct input_dev *input_dev; 54962306a36Sopenharmony_ci int i, j, port_idx; 55062306a36Sopenharmony_ci int mode; 55162306a36Sopenharmony_ci struct pardev_cb db9_parport_cb; 55262306a36Sopenharmony_ci 55362306a36Sopenharmony_ci for (port_idx = 0; port_idx < DB9_MAX_PORTS; port_idx++) { 55462306a36Sopenharmony_ci if (db9_cfg[port_idx].nargs == 0 || 55562306a36Sopenharmony_ci db9_cfg[port_idx].args[DB9_ARG_PARPORT] < 0) 55662306a36Sopenharmony_ci continue; 55762306a36Sopenharmony_ci 55862306a36Sopenharmony_ci if (db9_cfg[port_idx].args[DB9_ARG_PARPORT] == pp->number) 55962306a36Sopenharmony_ci break; 56062306a36Sopenharmony_ci } 56162306a36Sopenharmony_ci 56262306a36Sopenharmony_ci if (port_idx == DB9_MAX_PORTS) { 56362306a36Sopenharmony_ci pr_debug("Not using parport%d.\n", pp->number); 56462306a36Sopenharmony_ci return; 56562306a36Sopenharmony_ci } 56662306a36Sopenharmony_ci 56762306a36Sopenharmony_ci mode = db9_cfg[port_idx].args[DB9_ARG_MODE]; 56862306a36Sopenharmony_ci 56962306a36Sopenharmony_ci if (mode < 1 || mode >= DB9_MAX_PAD || !db9_modes[mode].n_buttons) { 57062306a36Sopenharmony_ci printk(KERN_ERR "db9.c: Bad device type %d\n", mode); 57162306a36Sopenharmony_ci return; 57262306a36Sopenharmony_ci } 57362306a36Sopenharmony_ci 57462306a36Sopenharmony_ci db9_mode = &db9_modes[mode]; 57562306a36Sopenharmony_ci 57662306a36Sopenharmony_ci if (db9_mode->bidirectional && !(pp->modes & PARPORT_MODE_TRISTATE)) { 57762306a36Sopenharmony_ci printk(KERN_ERR "db9.c: specified parport is not bidirectional\n"); 57862306a36Sopenharmony_ci return; 57962306a36Sopenharmony_ci } 58062306a36Sopenharmony_ci 58162306a36Sopenharmony_ci memset(&db9_parport_cb, 0, sizeof(db9_parport_cb)); 58262306a36Sopenharmony_ci db9_parport_cb.flags = PARPORT_FLAG_EXCL; 58362306a36Sopenharmony_ci 58462306a36Sopenharmony_ci pd = parport_register_dev_model(pp, "db9", &db9_parport_cb, port_idx); 58562306a36Sopenharmony_ci if (!pd) { 58662306a36Sopenharmony_ci printk(KERN_ERR "db9.c: parport busy already - lp.o loaded?\n"); 58762306a36Sopenharmony_ci return; 58862306a36Sopenharmony_ci } 58962306a36Sopenharmony_ci 59062306a36Sopenharmony_ci db9 = kzalloc(sizeof(struct db9), GFP_KERNEL); 59162306a36Sopenharmony_ci if (!db9) 59262306a36Sopenharmony_ci goto err_unreg_pardev; 59362306a36Sopenharmony_ci 59462306a36Sopenharmony_ci mutex_init(&db9->mutex); 59562306a36Sopenharmony_ci db9->pd = pd; 59662306a36Sopenharmony_ci db9->mode = mode; 59762306a36Sopenharmony_ci db9->parportno = pp->number; 59862306a36Sopenharmony_ci timer_setup(&db9->timer, db9_timer, 0); 59962306a36Sopenharmony_ci 60062306a36Sopenharmony_ci for (i = 0; i < (min(db9_mode->n_pads, DB9_MAX_DEVICES)); i++) { 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_ci db9->dev[i] = input_dev = input_allocate_device(); 60362306a36Sopenharmony_ci if (!input_dev) { 60462306a36Sopenharmony_ci printk(KERN_ERR "db9.c: Not enough memory for input device\n"); 60562306a36Sopenharmony_ci goto err_unreg_devs; 60662306a36Sopenharmony_ci } 60762306a36Sopenharmony_ci 60862306a36Sopenharmony_ci snprintf(db9->phys[i], sizeof(db9->phys[i]), 60962306a36Sopenharmony_ci "%s/input%d", db9->pd->port->name, i); 61062306a36Sopenharmony_ci 61162306a36Sopenharmony_ci input_dev->name = db9_mode->name; 61262306a36Sopenharmony_ci input_dev->phys = db9->phys[i]; 61362306a36Sopenharmony_ci input_dev->id.bustype = BUS_PARPORT; 61462306a36Sopenharmony_ci input_dev->id.vendor = 0x0002; 61562306a36Sopenharmony_ci input_dev->id.product = mode; 61662306a36Sopenharmony_ci input_dev->id.version = 0x0100; 61762306a36Sopenharmony_ci 61862306a36Sopenharmony_ci input_set_drvdata(input_dev, db9); 61962306a36Sopenharmony_ci 62062306a36Sopenharmony_ci input_dev->open = db9_open; 62162306a36Sopenharmony_ci input_dev->close = db9_close; 62262306a36Sopenharmony_ci 62362306a36Sopenharmony_ci input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS); 62462306a36Sopenharmony_ci for (j = 0; j < db9_mode->n_buttons; j++) 62562306a36Sopenharmony_ci set_bit(db9_mode->buttons[j], input_dev->keybit); 62662306a36Sopenharmony_ci for (j = 0; j < db9_mode->n_axis; j++) { 62762306a36Sopenharmony_ci if (j < 2) 62862306a36Sopenharmony_ci input_set_abs_params(input_dev, db9_abs[j], -1, 1, 0, 0); 62962306a36Sopenharmony_ci else 63062306a36Sopenharmony_ci input_set_abs_params(input_dev, db9_abs[j], 1, 255, 0, 0); 63162306a36Sopenharmony_ci } 63262306a36Sopenharmony_ci 63362306a36Sopenharmony_ci if (input_register_device(input_dev)) 63462306a36Sopenharmony_ci goto err_free_dev; 63562306a36Sopenharmony_ci } 63662306a36Sopenharmony_ci 63762306a36Sopenharmony_ci db9_base[port_idx] = db9; 63862306a36Sopenharmony_ci return; 63962306a36Sopenharmony_ci 64062306a36Sopenharmony_ci err_free_dev: 64162306a36Sopenharmony_ci input_free_device(db9->dev[i]); 64262306a36Sopenharmony_ci err_unreg_devs: 64362306a36Sopenharmony_ci while (--i >= 0) 64462306a36Sopenharmony_ci input_unregister_device(db9->dev[i]); 64562306a36Sopenharmony_ci kfree(db9); 64662306a36Sopenharmony_ci err_unreg_pardev: 64762306a36Sopenharmony_ci parport_unregister_device(pd); 64862306a36Sopenharmony_ci} 64962306a36Sopenharmony_ci 65062306a36Sopenharmony_cistatic void db9_detach(struct parport *port) 65162306a36Sopenharmony_ci{ 65262306a36Sopenharmony_ci int i; 65362306a36Sopenharmony_ci struct db9 *db9; 65462306a36Sopenharmony_ci 65562306a36Sopenharmony_ci for (i = 0; i < DB9_MAX_PORTS; i++) { 65662306a36Sopenharmony_ci if (db9_base[i] && db9_base[i]->parportno == port->number) 65762306a36Sopenharmony_ci break; 65862306a36Sopenharmony_ci } 65962306a36Sopenharmony_ci 66062306a36Sopenharmony_ci if (i == DB9_MAX_PORTS) 66162306a36Sopenharmony_ci return; 66262306a36Sopenharmony_ci 66362306a36Sopenharmony_ci db9 = db9_base[i]; 66462306a36Sopenharmony_ci db9_base[i] = NULL; 66562306a36Sopenharmony_ci 66662306a36Sopenharmony_ci for (i = 0; i < min(db9_modes[db9->mode].n_pads, DB9_MAX_DEVICES); i++) 66762306a36Sopenharmony_ci input_unregister_device(db9->dev[i]); 66862306a36Sopenharmony_ci parport_unregister_device(db9->pd); 66962306a36Sopenharmony_ci kfree(db9); 67062306a36Sopenharmony_ci} 67162306a36Sopenharmony_ci 67262306a36Sopenharmony_cistatic struct parport_driver db9_parport_driver = { 67362306a36Sopenharmony_ci .name = "db9", 67462306a36Sopenharmony_ci .match_port = db9_attach, 67562306a36Sopenharmony_ci .detach = db9_detach, 67662306a36Sopenharmony_ci .devmodel = true, 67762306a36Sopenharmony_ci}; 67862306a36Sopenharmony_ci 67962306a36Sopenharmony_cistatic int __init db9_init(void) 68062306a36Sopenharmony_ci{ 68162306a36Sopenharmony_ci int i; 68262306a36Sopenharmony_ci int have_dev = 0; 68362306a36Sopenharmony_ci 68462306a36Sopenharmony_ci for (i = 0; i < DB9_MAX_PORTS; i++) { 68562306a36Sopenharmony_ci if (db9_cfg[i].nargs == 0 || db9_cfg[i].args[DB9_ARG_PARPORT] < 0) 68662306a36Sopenharmony_ci continue; 68762306a36Sopenharmony_ci 68862306a36Sopenharmony_ci if (db9_cfg[i].nargs < 2) { 68962306a36Sopenharmony_ci printk(KERN_ERR "db9.c: Device type must be specified.\n"); 69062306a36Sopenharmony_ci return -EINVAL; 69162306a36Sopenharmony_ci } 69262306a36Sopenharmony_ci 69362306a36Sopenharmony_ci have_dev = 1; 69462306a36Sopenharmony_ci } 69562306a36Sopenharmony_ci 69662306a36Sopenharmony_ci if (!have_dev) 69762306a36Sopenharmony_ci return -ENODEV; 69862306a36Sopenharmony_ci 69962306a36Sopenharmony_ci return parport_register_driver(&db9_parport_driver); 70062306a36Sopenharmony_ci} 70162306a36Sopenharmony_ci 70262306a36Sopenharmony_cistatic void __exit db9_exit(void) 70362306a36Sopenharmony_ci{ 70462306a36Sopenharmony_ci parport_unregister_driver(&db9_parport_driver); 70562306a36Sopenharmony_ci} 70662306a36Sopenharmony_ci 70762306a36Sopenharmony_cimodule_init(db9_init); 70862306a36Sopenharmony_cimodule_exit(db9_exit); 709