diff options
Diffstat (limited to 'drivers/input/rmtctl/wmt-rmtctl.c')
-rw-r--r-- | drivers/input/rmtctl/wmt-rmtctl.c | 1631 |
1 files changed, 1631 insertions, 0 deletions
diff --git a/drivers/input/rmtctl/wmt-rmtctl.c b/drivers/input/rmtctl/wmt-rmtctl.c new file mode 100644 index 00000000..69182ffd --- /dev/null +++ b/drivers/input/rmtctl/wmt-rmtctl.c @@ -0,0 +1,1631 @@ +/*++ + * linux/drivers/input/rmtctl/wmt-rmtctl.c + * WonderMedia input remote control driver + * + * Copyright c 2010 WonderMedia Technologies, Inc. + * + * 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, see <http://www.gnu.org/licenses/>. + * + * WonderMedia Technologies, Inc. + * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C +--*/ + +#include <linux/module.h> +#include <linux/interrupt.h> +#include <linux/device.h> +#include <linux/version.h> +#include <linux/input.h> +#include <linux/platform_device.h> +#include <asm/irq.h> +#include <mach/hardware.h> +#include <mach/wmt_pmc.h> +#include <linux/delay.h> +#include <linux/firmware.h> +#include <linux/workqueue.h> +#include <linux/slab.h> +#include <linux/gpio.h> + +#include "wmt-rmtctl.h" + +enum CIR_CODEC_TYPE { + CIR_CODEC_NEC = 0, + CIR_CODEC_TOSHIBA = 1, + CIR_CODEC_PHILIPS_RC6 = 2, + CIR_CODEC_MAX, +}; + +struct cir_param { + enum CIR_CODEC_TYPE codec; + unsigned int param[7]; + unsigned int repeat_timeout; +}; + +struct cir_vendor_info { + char *vendor_name; + enum CIR_CODEC_TYPE codec; + + unsigned int vendor_code; + unsigned int wakeup_code; + unsigned int key_codes[258]; // usb-keyboard standard +}; + +#define RMTCTL_DEBUG 0 +#define REL_DELTA 20 +#define WMT_RMTCTL_VENDOR_ENV "wmt.io.rmtctl.vendorcode" +struct rmtctl_led{ + int gpio; + int active; + int on; + int enable; + struct timer_list timer; +}; + +struct rmtctl_rel { + int on; + unsigned int code; + struct timer_list timer; + struct input_dev *input; +}; + +struct rmtctl_priv { + enum CIR_CODEC_TYPE codec; + int vendor_index; + int table_index; + unsigned int saved_vcode; + unsigned int vendor_code; + unsigned int scan_code; + + struct input_dev *idev; + struct timer_list timer; + struct delayed_work delaywork; + + struct rmtctl_led led;//led control + + struct rmtctl_rel rel_dev;//remote cursor removing +}; + +static struct cir_param rmtctl_params[] = { + // NEC : |9ms carrier wave| + |4.5ms interval| + [CIR_CODEC_NEC] = { + .codec = CIR_CODEC_NEC, + .param = { 0x10a, 0x8e, 0x42, 0x55, 0x9, 0x13, 0x13 }, + .repeat_timeout = 17965000, + }, + + // TOSHIBA : |4.5ms carrier wave| + |4.5ms interval| + [CIR_CODEC_TOSHIBA] = { + .codec = CIR_CODEC_TOSHIBA, + .param = { 0x8e, 0x8e, 0x42, 0x55, 0x9, 0x13, 0x13 }, + .repeat_timeout = 17965000, + }, + + [CIR_CODEC_PHILIPS_RC6] = { + .codec = CIR_CODEC_PHILIPS_RC6, + .param = { 0x7, 0x15, 0x23, 0x31, 0x40, 0x4C, 0x5B }, + .repeat_timeout = 17965000, + }, +}; + +static struct cir_vendor_info rmtctl_vendors[] = { + +//Add by Flourise Kevin for FuRuiYing IR keypad 2014.12.07 Start + { + .vendor_name = "FRY-01", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00FF, + .wakeup_code = 0x0F, + .key_codes = { + //No.1 Line + [0x0F] = KEY_POWER, /* power down */ + [0x52] = KEY_HOME, /* Home */ + [0x10] = KEY_MENU, /* menu */ + [0x58] = KEY_MUTE, /* mute */ + + //No.2 Line + [0x5E] = KEY_KATAKANA, /* cursor */ + [0x4D] = KEY_UP, /* up */ + [0x11] = KEY_PREVIOUSSONG, /*previous song*/ + [0x5C] = KEY_VOLUMEUP, /*volume Up*/ + + //No.3 Line + [0x57] = KEY_LEFT, /*left*/ + [0x5B] = KEY_ENTER, /* OK */ + [0x5F] = KEY_RIGHT, /*Right*/ + [0x54] = KEY_VOLUMEDOWN, /*volume down*/ + + //No.4 Line + [0x56] = KEY_TAB, /*table key*/ + [0x5A] = KEY_DOWN, /*Down*/ + [0x12] = KEY_NEXTSONG, /*next song*/ + [0x53] = KEY_BACK, /* back */ + + //No.5 Line + [0x17] = KEY_1, + [0x1B] = KEY_2, + [0x1F] = KEY_3, + [0x16] = KEY_4, + + //No.6 Line + [0x1A] = KEY_5, + [0x1E] = KEY_6, + [0x15] = KEY_7, + [0x19] = KEY_8, + + //No.7 Line + [0x1D] = KEY_9, + [0x18] = KEY_0, + [0x14] = KEY_DOT, /*dot //small point*/ + [0x1c] = KEY_BACKSPACE, /* backspace */ + + //No.8 Line + [0x13] = KEY_PAUSE, /*vod */ + [0x50] = KEY_PLAY, /*live */ + [0x59] = KEY_PAUSE, /*pause */ + [0x55] = KEY_PLAY, /*play */ + + + }, + }, + //Add by Flourise Kevin for FuRuiYing IR keypad 2014.12.07 End + + + //Add by Flourise Kevin for FuRuiYing IR keypad 2014.11.29 Start + { + .vendor_name = "Ly-01", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00FF, + .wakeup_code = 0x18, + .key_codes = { + [0x18] = KEY_POWER, /* power down */ + [0xF2] = KEY_SWITCHVIDEOMODE, /*Cycle between available vide outputs (Monitor/LCD/TV-out/etc) */ + + [0x54] = KEY_1, + [0x48] = KEY_2, + [0x07] = KEY_3, + [0x50] = KEY_4, + [0x12] = KEY_5, + [0x11] = KEY_6, + [0x4c] = KEY_7, + [0x0e] = KEY_8, + [0x0d] = KEY_9, +// [0x41] = KEY_SW_INPUT_TYPE, /*switch input type*/ + [0x0c] = KEY_0, + [0x4b] = KEY_BACKSPACE, /* backspace */ + + [0x06] = KEY_HOME, /* Home */ + [0x40] = KEY_BACK, /* back */ + + [0x46] = KEY_UP, /* up */ + + [0x47] = KEY_LEFT, /*left*/ + [0x55] = KEY_ENTER, /* OK */ + [0x15] = KEY_RIGHT, /*Right*/ + + [0x16] = KEY_DOWN, /*Down*/ + + [0x04] = KEY_MENU, /* menu */ + [0x17] = KEY_KATAKANA, /* cursor */ + + + + [0x14] = KEY_VOLUMEDOWN, /*volume down*/ + [0x08] = KEY_MUTE, /* mute */ + [0x10] = KEY_VOLUMEUP, /*volume Up*/ + + +// [0x56] = KEY_TV, /*start TV*/ +// [0x57] = KEY_QQ, /*start QQ*/ + [0x1f] = KEY_PREVIOUSSONG, /*previous song*/ + [0x5b] = KEY_NEXTSONG, /*next song*/ + }, + }, + //Add by Flourise Kevin for FuRuiYing IR keypad 2014.11.29 End + + + + // SRC1804 0 + { + .vendor_name = "SRC1804", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x02fd, + .wakeup_code = 0x57, + .key_codes = { + [0x57] = KEY_POWER, /* power down */ + [0x56] = KEY_VOLUMEDOWN, /* vol- */ + [0x14] = KEY_VOLUMEUP, /* vol+ */ + [0x53] = KEY_HOME, /* home */ + [0x11] = KEY_MENU, /* menu */ + [0x10] = KEY_BACK, /* back */ + [0x4b] = KEY_ZOOMOUT, /* zoom out */ + [0x08] = KEY_ZOOMIN, /* zoom in */ + [0x0d] = KEY_UP, /* up */ + [0x4e] = KEY_LEFT, /* left */ + [0x19] = KEY_ENTER, /* OK */ + [0x0c] = KEY_RIGHT, /* right */ + [0x4f] = KEY_DOWN, /* down */ + [0x09] = KEY_PAGEUP, /* page up */ + [0x47] = KEY_PREVIOUSSONG, /* rewind */ + [0x05] = KEY_PAGEDOWN, /* page down */ + [0x04] = KEY_NEXTSONG /* forward */ + }, + }, + + // IH8950 1 + { + .vendor_name = "IH8950", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x0909, + .wakeup_code = 0xdc, + .key_codes = { + [0xdc] = KEY_POWER, /* power down */ + [0x81] = KEY_VOLUMEDOWN, /* vol- */ + [0x80] = KEY_VOLUMEUP, /* vol+ */ + [0x82] = KEY_HOME, /* home */ + [0xc5] = KEY_BACK, /* back */ + [0xca] = KEY_UP, /* up */ + [0x99] = KEY_LEFT, /* left */ + [0xce] = KEY_ENTER, /* OK */ + [0xc1] = KEY_RIGHT, /* right */ + [0xd2] = KEY_DOWN, /* down */ + [0x9c] = KEY_MUTE, + [0x95] = KEY_PLAYPAUSE, + [0x88] = KEY_MENU, + }, + }, + + // sunday 2 + { + .vendor_name = "sunday", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x02fd, + .wakeup_code = 0x1a, + .key_codes = { + [0x1a] = KEY_POWER, /* power down */ + [0x16] = KEY_VOLUMEDOWN, /* vol- */ + [0x44] = KEY_VOLUMEUP, /* vol+ */ + [0x59] = KEY_HOME, /* home */ + [0x1b] = KEY_BACK, /* back */ + [0x06] = KEY_UP, /* up */ + [0x5d] = KEY_LEFT, /* left */ + [0x1e] = KEY_ENTER, /* OK */ + [0x5c] = KEY_RIGHT, /* right */ + [0x1f] = KEY_DOWN, /* down */ + [0x55] = KEY_PLAYPAUSE, + [0x54] = KEY_REWIND, + [0x17] = KEY_FASTFORWARD, + [0x58] = KEY_AGAIN, /* recent app */ + }, + }, + + /* F1 - F12 */ + // KT-9211 3 + { + .vendor_name = "KT-9211", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x02fd, + .wakeup_code = 0x57, + .key_codes = { + [0x57] = KEY_POWER, /* power down */ + [0x56] = KEY_VOLUMEDOWN, /* volume- */ + [0x15] = KEY_MUTE, /* mute */ + [0x14] = KEY_VOLUMEUP, /* volume+ */ + [0x0d] = KEY_UP, /* up */ + [0x4e] = KEY_LEFT, /* left */ + [0x19] = KEY_ENTER, /* OK */ + [0x0c] = KEY_RIGHT, /* right */ + [0x4f] = KEY_DOWN, /* down */ + [0x53] = KEY_HOME, /* home */ + [0x09] = KEY_F5, /* my photo in default */ + [0x11] = KEY_F12, /* setting apk in default */ + [0x47] = KEY_F3, /* My Music in default */ + [0x10] = KEY_BACK, /* Back */ + [0x08] = KEY_F4, /* my video in default */ + [0x17] = KEY_F1, /* web browser in default */ + + //Following items are configured manually. + [0x05] = KEY_F10, /* file browser in default */ + [0x04] = KEY_F2, /* camera in default */ + [0x4b] = KEY_F11, /* calendar in default */ + [0x16] = KEY_SCREEN, /* calculator in default */ + [0x18] = KEY_F9, /* recorder in default */ + }, + }, + + // KT-8830 4 + { + .vendor_name = "KT-8830", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x866b, + .wakeup_code = 0x1c, + .key_codes = { + [0x1c] = KEY_POWER, + [0x14] = KEY_MUTE, + [0x09] = KEY_1, + [0x1d] = KEY_2, + [0x1f] = KEY_3, + [0x0d] = KEY_4, + [0x19] = KEY_5, + [0x1b] = KEY_6, + [0x11] = KEY_7, + [0x15] = KEY_8, + [0x17] = KEY_9, + [0x12] = KEY_0, + [0x16] = KEY_VOLUMEDOWN, + [0x04] = KEY_VOLUMEUP, + [0x40] = KEY_HOME, + [0x4c] = KEY_BACK, + [0x00] = KEY_F12, // setup + [0x13] = KEY_MENU, + [0x03] = KEY_UP, + [0x0e] = KEY_LEFT, + [0x1a] = KEY_RIGHT, + [0x02] = KEY_DOWN, + [0x07] = KEY_ENTER, + [0x47] = KEY_F4, // video + [0x10] = KEY_F3, // music + [0x0f] = KEY_FASTFORWARD, + [0x43] = KEY_REWIND, + [0x18] = KEY_F1, // browser + [0x0b] = KEY_YEN, // multi-functions + [0x4e] = KEY_PLAYPAUSE, + }, + }, + + // Haier-OTT 5 + { + .vendor_name = "Haier-OTT", + .codec = CIR_CODEC_NEC, + .vendor_code = 0xb34c, + .wakeup_code = 0xdc, + .key_codes = { + [0xdc] = KEY_POWER, + [0x9c] = KEY_MUTE, + [0xca] = KEY_UP, + [0x99] = KEY_LEFT, + [0xc1] = KEY_RIGHT, + [0xd2] = KEY_DOWN, + [0xce] = KEY_ENTER, + [0x80] = KEY_VOLUMEDOWN, + [0x81] = KEY_VOLUMEUP, + [0xc5] = KEY_HOME, + [0x95] = KEY_BACK, + [0x88] = KEY_MENU, + [0x82] = KEY_F12, /* setting apk in default */ + }, + }, + + // GPRC-11933 6 + { + .vendor_name = "GPRC-11933", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x4040, + .wakeup_code = 0x4D, + .key_codes = { + [0x4D] = KEY_POWER, /* power down */ + [0x53] = KEY_PLAYPAUSE, + [0x5B] = KEY_F3, /* My Music in default */ + [0x57] = KEY_F1, /* web browser in default */ + [0x54] = KEY_F10, /* file browser in default */ + + [0x17] = KEY_VOLUMEDOWN, /* volume- */ + [0x43] = KEY_MUTE, /* mute */ + [0x18] = KEY_VOLUMEUP, /* volume+ */ + [0x1F] = KEY_PREVIOUSSONG, /* rewind */ + [0x1E] = KEY_NEXTSONG, /* forward */ + + [0x0B] = KEY_UP, /* up */ + [0x10] = KEY_LEFT, /* left */ + [0x0D] = KEY_ENTER, /* OK */ + [0x11] = KEY_RIGHT, /* right */ + [0x0E] = KEY_DOWN, /* down */ + [0x1A] = KEY_HOME, /* home */ + [0x42] = KEY_BACK, /* Back */ + + [0x45] = KEY_F12, /* setting apk in default */ + [0x47] = KEY_KATAKANA, /* multi-functions */ + [0x01] = KEY_1, + [0x02] = KEY_2, + [0x03] = KEY_3, + [0x04] = KEY_4, + [0x05] = KEY_5, + [0x06] = KEY_6, + [0x07] = KEY_7, + [0x08] = KEY_8, + [0x09] = KEY_9, + [0x00] = KEY_0, + [0x44] = KEY_MENU, + [0x0C] = KEY_BACKSPACE, + }, + }, + + // TS-Y118 7 + { + .vendor_name = "TS-Y118", + .codec = CIR_CODEC_NEC, + .vendor_code = 0xb34c, + .wakeup_code = 0xdc, + .key_codes = { + [0xdc] = KEY_POWER, + [0x9c] = KEY_MUTE, + [0x8d] = KEY_F12, // apk-settings + [0x88] = KEY_HOME, + [0xca] = KEY_UP, + [0xd2] = KEY_DOWN, + [0x99] = KEY_LEFT, + [0xc1] = KEY_RIGHT, + [0xce] = KEY_ENTER, + [0x95] = KEY_PLAYPAUSE, + [0xc5] = KEY_BACK, + [0x80] = KEY_VOLUMEUP, + [0x81] = KEY_VOLUMEDOWN, + [0xdd] = KEY_PAGEUP, + [0x8c] = KEY_PAGEDOWN, + [0x92] = KEY_1, + [0x93] = KEY_2, + [0xcc] = KEY_3, + [0x8e] = KEY_4, + [0x8f] = KEY_5, + [0xc8] = KEY_6, + [0x8a] = KEY_7, + [0x8b] = KEY_8, + [0xc4] = KEY_9, + [0x87] = KEY_0, + }, + }, + + // Hisense 8 + { + .vendor_name = "Hisense", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00ff, + .wakeup_code = 0x14, + .key_codes = { + [0x14] = KEY_POWER, + [0x1c] = KEY_MUTE, + [0x40] = KEY_PAGEUP, + [0x44] = KEY_PAGEDOWN, + [0x0b] = KEY_VOLUMEUP, + [0x58] = KEY_VOLUMEDOWN, + [0x01] = KEY_MENU, + [0x03] = KEY_UP, + [0x02] = KEY_DOWN, + [0x0e] = KEY_LEFT, + [0x1a] = KEY_RIGHT, + [0x07] = KEY_ENTER, + [0x48] = KEY_HOME, + [0x5c] = KEY_BACK, + [0x09] = KEY_1, + [0x1d] = KEY_2, + [0x1f] = KEY_3, + [0x0d] = KEY_4, + [0x19] = KEY_5, + [0x1b] = KEY_6, + [0x11] = KEY_7, + [0x15] = KEY_8, + [0x17] = KEY_9, + [0x12] = KEY_0, + [0x06] = KEY_DOT, + [0x16] = KEY_DELETE, + [0x0c] = KEY_ZOOMIN, + [0x4c] = KEY_F11, + [0x13] = KEY_YEN, + }, + }, + + // Mountain 9 + { + .vendor_name = "TV BOX Mountain", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00df, + .wakeup_code = 0x1c, + .key_codes = { + [0x1c] = KEY_POWER, /* power down */ + [0x08] = KEY_MUTE, /* volume mute */ + [0x1a] = KEY_UP, /* up */ + [0x47] = KEY_LEFT, /* left */ + [0x06] = KEY_ENTER, /* OK */ + [0x07] = KEY_RIGHT, /* right */ + [0x48] = KEY_DOWN, /* down */ + [0x4f] = KEY_VOLUMEDOWN, /* vol- */ + [0x4b] = KEY_VOLUMEUP, /* vol+ */ + [0x0a] = KEY_BACK, /* back */ + [0x03] = KEY_HOME, /* home */ + [0x42] = KEY_KATAKANA, /* TV */ + [0x55] = KEY_MENU, /* menu */ + }, + }, + + // GPRC-11933 10 + { + .vendor_name = "GPRC-11933A", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00FF, + .wakeup_code = 0x57, + .key_codes = { + [0x57] = KEY_POWER, /* power down */ + [0x5B] = KEY_MUTE, /* mute */ + + [0x16] = KEY_F3, /* My Musicin default */ + [0x5A] = KEY_F1, /* web browser in default */ + [0x52] = KEY_PLAYPAUSE, + [0x50] = KEY_KATAKANA, /* cursor */ + + [0x0F] = KEY_PREVIOUSSONG, /* rewind */ + [0x4C] = KEY_NEXTSONG, /* forward */ + [0x58] = KEY_VOLUMEDOWN, /* volume- */ + [0x1B] = KEY_VOLUMEUP, /* volume+ */ + + [0x4F] = KEY_F12, /* setting apk in default */ + [0x1A] = KEY_MENU, + + [0x43] = KEY_UP, /* up */ + + [0x06] = KEY_LEFT, /* left */ + [0x02] = KEY_ENTER, /* OK */ + [0x0E] = KEY_RIGHT, /* right */ + + [0x0A] = KEY_DOWN, /* down */ + + [0x4E] = KEY_HOME, /* home */ + [0x4D] = KEY_BACK, /* back */ + + [0x10] = KEY_1, + [0x11] = KEY_2, + [0x12] = KEY_3, + [0x13] = KEY_4, + [0x14] = KEY_5, + [0x15] = KEY_6, + [0x17] = KEY_7, + [0x18] = KEY_8, + [0x19] = KEY_9, + [0x1D] = KEY_0, + + [0x1C] = KEY_F1, + [0x1E] = KEY_BACKSPACE, + }, + }, + + //China Telecom White 11 + { + .vendor_name = "TS-Y118A", + .codec = CIR_CODEC_NEC, + .vendor_code = 0xb34c, + .wakeup_code = 0xdc, + .key_codes = { + [0xdc] = KEY_POWER, + + [0x98] =KEY_AUDIO, + [0x9c] = KEY_MUTE, + + [0x8d] = KEY_SETUP, + [0xd6] = KEY_LAST, + + [0xcd] = KEY_RED, + [0x91] = KEY_GREEN, + [0x83] = KEY_YELLOW, + [0xc3] = KEY_BLUE, + + [0x88] = KEY_HOMEPAGE, + [0xca] = KEY_UP, + [0x82] = KEY_HOME, + + [0x99] = KEY_LEFT, + [0xce] = KEY_SELECT, + [0xc1] = KEY_RIGHT, + + [0x95] = KEY_PLAYPAUSE, + [0xd2] = KEY_DOWN, + [0xc5] = KEY_BACK, + + [0x80] = KEY_VOLUMEUP, + [0x81] = KEY_VOLUMEDOWN, + + [0xdd] = KEY_PAGEUP, + [0x8c] = KEY_PAGEDOWN, + + [0x85] = KEY_CHANNELUP, + [0x86] = KEY_CHANNELDOWN, + + [0x92] = KEY_1, + [0x93] = KEY_2, + [0xcc] = KEY_3, + [0x8e] = KEY_4, + [0x8f] = KEY_5, + [0xc8] = KEY_6, + [0x8a] = KEY_7, + [0x8b] = KEY_8, + [0xc4] = KEY_9, + [0x87] = KEY_0, + + [0xda] = KEY_NUMERIC_STAR, + [0xd0] = KEY_NUMERIC_POUND, + }, + }, + + //China Telecom Black 12 + { + .vendor_name = "TS-Y118B", + .codec = CIR_CODEC_NEC, + .vendor_code = 0xb24d, + .wakeup_code = 0xdc, + .key_codes = { + [0xdc] = KEY_POWER, + + [0x98] =KEY_AUDIO, + [0x9c] = KEY_MUTE, + + [0x8d] = KEY_SETUP, + [0xd6] = KEY_LAST, + + [0xcd] = KEY_RED, + [0x91] = KEY_GREEN, + [0x83] = KEY_YELLOW, + [0xc3] = KEY_BLUE, + + [0x88] = KEY_HOMEPAGE, + [0xca] = KEY_UP, + [0x82] = KEY_HOME, + + [0x99] = KEY_LEFT, + [0xce] = KEY_SELECT, + [0xc1] = KEY_RIGHT, + + [0x95] = KEY_PLAYPAUSE, + [0xd2] = KEY_DOWN, + [0xc5] = KEY_BACK, + + [0x80] = KEY_VOLUMEUP, + [0x81] = KEY_VOLUMEDOWN, + + [0xdd] = KEY_PAGEUP, + [0x8c] = KEY_PAGEDOWN, + + [0x85] = KEY_CHANNELUP, + [0x86] = KEY_CHANNELDOWN, + + [0x92] = KEY_1, + [0x93] = KEY_2, + [0xcc] = KEY_3, + [0x8e] = KEY_4, + [0x8f] = KEY_5, + [0xc8] = KEY_6, + [0x8a] = KEY_7, + [0x8b] = KEY_8, + [0xc4] = KEY_9, + [0x87] = KEY_0, + + [0xda] = KEY_NUMERIC_STAR, + [0xd0] = KEY_NUMERIC_POUND, + }, + }, + + // Jensen 13 + { + .vendor_name = "Jensen", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00ff, + .wakeup_code = 0x46, + .key_codes = { + [0x08] = KEY_RESERVED, + + [0x5a] = KEY_VOLUMEUP, + [0x4a] = KEY_VOLUMEDOWN, + + [0x19] = KEY_UP, + [0x1c] = KEY_DOWN, + [0x0c] = KEY_LEFT, + [0x5e] = KEY_RIGHT, + [0x18] = KEY_ENTER, + + [0x0d] = KEY_BACK, + [0x45] = KEY_1, + [0x46] = KEY_POWER, + [0x47] = KEY_3, + [0x44] = KEY_4, + [0x40] = KEY_F1, + [0x43] = KEY_6, + [0x07] = KEY_7, + [0x15] = KEY_BACK, + [0x09] = KEY_9, + [0x16] = KEY_0, + + [0x42] = KEY_PREVIOUSSONG, /* rewind */ + [0x52] = KEY_NEXTSONG, /* forward */ + + }, + }, + // Foxconn 14 + { + .vendor_name = "CIR-9F", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x009f, + .wakeup_code = 0x57, + .key_codes = { + [0x57] = KEY_POWER, /* power */ + [0x5d] = KEY_VOLUMEDOWN, /* vol- */ + [0xff] = KEY_VOLUMEUP, /* vol+ */ + [0x47] = KEY_HOME, /* home */ + [0x16] = KEY_MENU, /* menu */ + [0x4f] = KEY_BACK, /* back */ + [0x43] = KEY_UP, /* up */ + [0x06] = KEY_LEFT, /* left */ + [0x02] = KEY_ENTER, /* PLAYPAUSE */ + //[0x02] = KEY_PLAYPAUSE, /* PLAYPAUSE */ + [0x0e] = KEY_RIGHT, /* right */ + [0x0a] = KEY_DOWN, /* down */ + [0x0b] = KEY_REWIND, /* rewind data is 0x36*/ + [0x0f] = KEY_FASTFORWARD, /* forward */ + + [0x5b] = KEY_ENTER /* OK - notify */ + }, + }, + + // Philips 15 + { + .vendor_name = "PHILIPS", + .codec = CIR_CODEC_PHILIPS_RC6, + .vendor_code = 60, + .wakeup_code = 12, + .key_codes = { + [12] = KEY_POWER, + [146] = KEY_HOME, + [84] = KEY_MENU, + [92] = KEY_ENTER, + [88] = KEY_UP, + [89] = KEY_DOWN, + [90] = KEY_LEFT, + [91] = KEY_RIGHT, + [83] = BTN_DEAD, + [131] = KEY_BACK, + }, + }, + + //Ronsheng 16 + { + .vendor_name = "Ronsheng", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x00FF, + .wakeup_code = 0x18, + .key_codes = { + [0x18] = KEY_POWER, /* power down */ + + [0x56] = KEY_F3, /* My Musicin default */ + [0x57] = KEY_F1, /* web browser in default */ + + [0x1f] = KEY_PREVIOUSSONG, /* rewind */ + [0x5b] = KEY_NEXTSONG, /* forward */ + + [0x14] = KEY_VOLUMEDOWN, /* volume- */ + [0x08] = KEY_MUTE, /* mute */ + [0x10] = KEY_VOLUMEUP, /* volume+ */ + + [0x17] = KEY_KATAKANA, /* cursor */ + [0x04] = KEY_MENU, + + [0x46] = KEY_UP, /* up */ + [0x47] = KEY_LEFT, /* left */ + [0x55] = KEY_ENTER, /* OK */ + [0x15] = KEY_RIGHT, /* right */ + [0x16] = KEY_DOWN, /* down */ + + [0x06] = KEY_HOME, /* home */ + [0x40] = KEY_BACK, /* back */ + + [0x54] = KEY_1, + [0x48] = KEY_2, + [0x07] = KEY_3, + [0x50] = KEY_4, + [0x12] = KEY_5, + [0x11] = KEY_6, + [0x4c] = KEY_7, + [0x0e] = KEY_8, + [0x0d] = KEY_9, + [0x0c] = KEY_0, + [0x41] = KEY_RESERVED, + [0x4b] = KEY_BACKSPACE, + + }, + }, + + // GPRC-11933 17 + { + .vendor_name = "GPRC-11933", + .codec = CIR_CODEC_NEC, + .vendor_code = 0x4040, + .wakeup_code = 0x4D, + .key_codes = { + [0x4D] = KEY_POWER, /* power down */ + [0x53] = KEY_F1, + [0x5B] = KEY_F2, + [0x57] = KEY_F3, + [0x54] = KEY_F4, + + [0x17] = KEY_VOLUMEDOWN, /* volume- */ + [0x43] = KEY_MUTE, /* mute */ + [0x18] = KEY_VOLUMEUP, /* volume+ */ + + [0x1F] = KEY_F12, /* setting */ + [0x1E] = KEY_F5, + + [0x0B] = KEY_UP, /* up */ + [0x10] = KEY_LEFT, /* left */ + [0x0D] = KEY_ENTER, /* OK */ + [0x11] = KEY_RIGHT, /* right */ + [0x0E] = KEY_DOWN, /* down */ + [0x1A] = KEY_HOME, /* home */ + [0x47] = KEY_BACK, /* Back */ + + [0x45] = KEY_MENU, /* MENU */ + [0x42] = KEY_KATAKANA, /* multi-functions */ + [0x01] = KEY_1, + [0x02] = KEY_2, + [0x03] = KEY_3, + [0x04] = KEY_4, + [0x05] = KEY_5, + [0x06] = KEY_6, + [0x07] = KEY_7, + [0x08] = KEY_8, + [0x09] = KEY_9, + [0x00] = KEY_0, + [0x44] = KEY_AGAIN, + [0x0C] = KEY_BACKSPACE, + }, + }, + + +}; + +extern int wmt_getsyspara(char *varname, unsigned char *varval, int *varlen); +extern int wmt_setsyspara(char *varname, char *varval); + +static int rmtctl_report_rel(struct rmtctl_rel *rel_dev,const unsigned int code ) +{ + switch (code) { + case KEY_UP: + input_report_rel(rel_dev->input, REL_X, 0); + input_report_rel(rel_dev->input, REL_Y, -REL_DELTA); + input_sync(rel_dev->input); + break; + case KEY_DOWN: + input_report_rel(rel_dev->input, REL_X, 0); + input_report_rel(rel_dev->input, REL_Y, REL_DELTA); + input_sync(rel_dev->input); + break; + case KEY_LEFT: + input_report_rel(rel_dev->input, REL_X, -REL_DELTA); + input_report_rel(rel_dev->input, REL_Y, 0); + input_sync(rel_dev->input); + break; + case KEY_RIGHT: + input_report_rel(rel_dev->input, REL_X, REL_DELTA); + input_report_rel(rel_dev->input, REL_Y, 0); + input_sync(rel_dev->input); + break; + default: + break; + } + return 0; +} + +static void rmtctl_report_event(unsigned int *code, int repeat) +{ + static unsigned int last_code = KEY_RESERVED; + int ret; + struct rmtctl_priv *priv = container_of(code, struct rmtctl_priv, scan_code); + + ret = del_timer(&priv->timer); + + if (!repeat) { + // new code coming + if (ret == 1) { + // del_timer() of active timer returns 1 means that active timer has been stoped by new key, + // so report last key up event + if (RMTCTL_DEBUG) + printk("[%d] up reported caused by new key[%d]\n", last_code, *code); + + if (priv->rel_dev.on) { + switch (*code) { + case KEY_UP: + case KEY_DOWN: + case KEY_LEFT: + case KEY_RIGHT: + del_timer(&priv->rel_dev.timer); //stop report mouse + break; + case KEY_ENTER: + input_report_key(priv->rel_dev.input, BTN_LEFT, 0); + input_sync(priv->rel_dev.input); + del_timer(&priv->rel_dev.timer); //stop report mouse + break; + default: + input_report_key(priv->idev, last_code, 0); + input_sync(priv->idev); + break; + } + } + else { + input_report_key(priv->idev, last_code, 0); + input_sync(priv->idev); + } + + if (*code == KEY_KATAKANA) { + if (priv->rel_dev.on == 0) { + input_report_rel(priv->rel_dev.input, REL_X, 1); + input_report_rel(priv->rel_dev.input, REL_Y, 1); + input_sync(priv->rel_dev.input); + priv->rel_dev.on = 1; + } + else + priv->rel_dev.on = 0; + } + + } + + if (RMTCTL_DEBUG) + printk("[%d] down\n", *code); + + // report key down event, mod_timer() to report key up event later for measure key repeat. + + if (priv->rel_dev.on) { + switch (*code) { + case KEY_UP: + case KEY_DOWN: + case KEY_LEFT: + case KEY_RIGHT: + rmtctl_report_rel(&priv->rel_dev,*code); + break; + case KEY_ENTER: + input_report_key(priv->rel_dev.input, BTN_LEFT, 1); + input_sync(priv->rel_dev.input); + break; + default: + input_event(priv->idev, EV_KEY, *code, 1); + input_sync(priv->idev); + break; + } + } + else { + input_event(priv->idev, EV_KEY, *code, 1); + input_sync(priv->idev); + } + + if(priv->led.enable){ + priv->led.on = 0; + gpio_direction_output(priv->led.gpio, !priv->led.active); + mod_timer(&priv->led.timer, jiffies+HZ/20); + } + + priv->timer.data = (unsigned long)code; + mod_timer(&priv->timer, jiffies + 250*HZ/1000); + } + else { + // report key up event after report repeat event according to usb-keyboard standard + priv->timer.data = (unsigned long)code; + mod_timer(&priv->timer, jiffies + 250*HZ/1000); + + // detect 'repeat' flag, report repeat event + if (*code != KEY_POWER && *code != KEY_END) { + if (RMTCTL_DEBUG) + printk("[%d] repeat\n", *code); + + if (priv->rel_dev.on) { + switch (*code) { + case KEY_UP: + case KEY_DOWN: + case KEY_LEFT: + case KEY_RIGHT: + case KEY_ENTER: + priv->rel_dev.code = *code; + mod_timer(&priv->rel_dev.timer, jiffies+HZ/100); //ready to repeat report mouse event + break; + default: + input_event(priv->idev, EV_KEY, *code, 2); + input_sync(priv->idev); + break; + } + } + else { + input_event(priv->idev, EV_KEY, *code, 2); + input_sync(priv->idev); + } + } + } + + last_code = *code; +} + +/** no hw repeat detect, so we measure repeat timeout event by timer */ +static void rmtctl_report_event_without_hwrepeat(unsigned int *code) +{ + static unsigned int last_code = KEY_RESERVED; + static unsigned long last_jiffies = 0; + int repeat = (jiffies_to_msecs(jiffies - last_jiffies) <250 && *code==last_code) ? 1 : 0; + + last_code = *code; + last_jiffies = jiffies; + + rmtctl_report_event(code, repeat); +} + +static void rmtctl_timer_handler(unsigned long data) +{ + struct rmtctl_priv *priv = container_of((void *)data, struct rmtctl_priv, scan_code); + unsigned int code = *(unsigned int *)data; + + // report key up event + if (RMTCTL_DEBUG) + printk("[%d] up reported by timer\n", code); + + if (priv->rel_dev.on) { + switch (code) { + case KEY_UP: + case KEY_DOWN: + case KEY_LEFT: + case KEY_RIGHT: + del_timer(&priv->rel_dev.timer); + break; + case KEY_ENTER: + input_report_key(priv->rel_dev.input, BTN_LEFT, 0); + input_sync(priv->rel_dev.input); + break; + default: + input_report_key(priv->idev, code, 0); + input_sync(priv->idev); + break; + } + } + else { + input_report_key(priv->idev, code, 0); + input_sync(priv->idev); + } + + if (code == KEY_KATAKANA) { + if (priv->rel_dev.on == 0) { + input_report_rel(priv->rel_dev.input, REL_X, 1); + input_report_rel(priv->rel_dev.input, REL_Y, 1); + input_sync(priv->rel_dev.input); + priv->rel_dev.on = 1; + } + else + priv->rel_dev.on = 0; + } + + if(priv->led.enable){ + del_timer_sync(&priv->led.timer); + gpio_direction_output(priv->led.gpio,priv->led.active); + } +} + +static irqreturn_t rmtctl_interrupt(int irq, void *dev_id) +{ + unsigned int status, ir_data, ir_code,vendor, repeat; + unsigned char *key; + int i; + + struct rmtctl_priv *priv = (struct rmtctl_priv *)dev_id; + + /* get IR status. */ + status = REG32_VAL(IRSTS); + + /* check 'IR received data' flag. */ + if ((status & 0x1) == 0x0) { + printk("IR IRQ was triggered without data received. (0x%x)\n", + status); + return IRQ_NONE; + } + + /* read IR data. */ + ir_data = REG32_VAL(IRDATA(0)) ; + key = (char *) &ir_data; + + /* clear INT status*/ + REG32_VAL(IRSTS)=0x1 ; + + if (RMTCTL_DEBUG){ + printk("ir_data = 0x%08x, status = 0x%x \n", ir_data, status); + } + + if(priv->codec == CIR_CODEC_PHILIPS_RC6){ + vendor = key[1]; + ir_code = key[0]; + //trailer = key[2] & 0x01; + } + else{//NEC + /* get vendor ID. */ + vendor = (key[0] << 8) | (key[1]); + + /* check if key is valid. Key[3] is XORed t o key[2]. */ + if (key[2] & key[3]) { + printk("Invalid IR key received. (0x%x, 0x%x)\n", key[2], key[3]); + return IRQ_NONE; + } + + /* keycode mapping. */ + ir_code = key[2]; + } + + if ( priv->vendor_index >= 0 ) { + if (vendor == rmtctl_vendors[priv->vendor_index].vendor_code && + rmtctl_vendors[priv->vendor_index].codec == priv->codec) { + priv->table_index = priv->vendor_index; + priv->scan_code = rmtctl_vendors[priv->vendor_index].key_codes[ir_code]; + } + else{ + if(vendor != priv->vendor_code){ + for (i = 0; i < ARRAY_SIZE(rmtctl_vendors); i++) { + if (vendor == rmtctl_vendors[i].vendor_code && + rmtctl_vendors[i].codec == priv->codec) { + priv->table_index = i; + priv->scan_code = rmtctl_vendors[i].key_codes[ir_code]; + break; + } + } + + if(i==ARRAY_SIZE(rmtctl_vendors)) + return IRQ_HANDLED; + } + else{ + priv->scan_code = rmtctl_vendors[priv->table_index].key_codes[ir_code]; + } + } + } + else { + if(vendor != priv->vendor_code){ + for (i = 0; i < ARRAY_SIZE(rmtctl_vendors); i++) { + if (vendor == rmtctl_vendors[i].vendor_code && + rmtctl_vendors[i].codec == priv->codec) { + priv->table_index = i; + priv->scan_code = rmtctl_vendors[i].key_codes[ir_code]; + break; + } + } + + if(i==ARRAY_SIZE(rmtctl_vendors)) + return IRQ_HANDLED; + } + else{ + priv->scan_code = rmtctl_vendors[priv->table_index].key_codes[ir_code]; + } + } + + //switch to a new remote controller shoud reset cursor mode + if (priv->vendor_code != vendor) + priv->rel_dev.on = 0; + + + if ((status & 0x2) || (priv->scan_code == KEY_RESERVED)) { + /* ignore repeated or reserved keys. */ + } + else if (priv->codec == CIR_CODEC_NEC) { + /* check 'IR code repeat' flag. */ + repeat = status & 0x10; + rmtctl_report_event(&priv->scan_code, repeat); + } + else if(priv->codec == CIR_CODEC_PHILIPS_RC6){ + //repeat = !(trailer^key[2]); + //trailer = key[2]; + //printk("RCV philips rc6,repeat=%d\n",repeat); + + rmtctl_report_event_without_hwrepeat(&priv->scan_code); + } + else { + rmtctl_report_event_without_hwrepeat(&priv->scan_code); + } + + if (priv->vendor_code != vendor) { + priv->vendor_code = vendor; + /* save new vendor code to env + *1.remote controller configed index is active, no need to save it's vendor code + *2.current vendor code is saved, no need to save again + */ + if(priv->table_index != priv->vendor_index && vendor != priv->saved_vcode){ + priv->saved_vcode = vendor; + //schedule_delayed_work(&priv->delaywork, msecs_to_jiffies(20)); + } + } + + return IRQ_HANDLED; +} + +static void rmtctl_hw_suspend(unsigned long data) +{ + unsigned int wakeup_code; + int i; + struct rmtctl_priv *priv = (struct rmtctl_priv *)data; + + if (priv->vendor_index >= 0 && priv->vendor_index == priv->table_index) { + i = priv->vendor_index; + } + else { + for (i = 0; i < ARRAY_SIZE(rmtctl_vendors); i++) + if (priv->vendor_code == rmtctl_vendors[i].vendor_code && + priv->codec == rmtctl_vendors[i].codec) + break; + + if (i == ARRAY_SIZE(rmtctl_vendors)) + goto out; + } + + if(priv->codec == CIR_CODEC_PHILIPS_RC6){ + wakeup_code = + ((unsigned char)(rmtctl_vendors[i].vendor_code & 0xff) << 8) | + (unsigned char)(rmtctl_vendors[i].wakeup_code) ; + REG32_VAL(WAKEUP_CMD2(0)) = 0x10000| wakeup_code; + }else{ + wakeup_code = + ((unsigned char)(~rmtctl_vendors[i].wakeup_code) << 24) | + ((unsigned char)(rmtctl_vendors[i].wakeup_code) << 16 ) | + ((unsigned char)(rmtctl_vendors[i].vendor_code & 0xff) << 8) | + ((unsigned char)((rmtctl_vendors[i].vendor_code) >> 8) << 0 ); + + } + REG32_VAL(WAKEUP_CMD1(0)) = wakeup_code; + REG32_VAL(WAKEUP_CMD1(1)) = 0x0; + REG32_VAL(WAKEUP_CMD1(2)) = 0x0; + REG32_VAL(WAKEUP_CMD1(3)) = 0x0; + REG32_VAL(WAKEUP_CMD1(4)) = 0x0; + +out: + REG32_VAL(WAKEUP_CTRL) = 0x101; +} + +static void rmtctl_hw_init(unsigned long data) +{ + unsigned int st; + int i, retries = 0; + struct rmtctl_priv *priv = (struct rmtctl_priv *)data; + + /*setting shared pin*/ + GPIO_CTRL_GP62_WAKEUP_SUS_BYTE_VAL &= 0xFFFD; + PULL_EN_GP62_WAKEUP_SUS_BYTE_VAL |= 0x02; + PULL_CTRL_GP62_WAKEUP_SUS_BYTE_VAL |= 0x02; + + /* turn off CIR SW reset. */ + REG32_VAL(IRSWRST) = 1; + REG32_VAL(IRSWRST) = 0; + + for (i = 0; i < ARRAY_SIZE(rmtctl_params[priv->codec].param); i++) + REG32_VAL(PARAMETER(i)) = rmtctl_params[priv->codec].param[i]; + + if (priv->codec == CIR_CODEC_NEC || priv->codec == CIR_CODEC_TOSHIBA) { + printk("NEC\n"); + REG32_VAL(NEC_REPEAT_TIME_OUT_CTRL) = 0x1; + REG32_VAL(NEC_REPEAT_TIME_OUT_COUNT) = rmtctl_params[priv->codec].repeat_timeout; + REG32_VAL(IRCTL) = 0X100; //NEC repeat key + } + else if(priv->codec == CIR_CODEC_PHILIPS_RC6){ + printk("PHILIPS RC6\n"); + REG32_VAL(IRCTL) =0x40; //PHILIPS RC6 + REG32_VAL(INT_MASK_COUNT) = 40*1000000*1/4; + //REG32_VAL(IRCTL_2) =0xFE;//Enable PHILIPS RC6 mode 0,Mask mode1-7 ? + }else{ + REG32_VAL(IRCTL) = 0; //NEC repeat key + REG32_VAL(INT_MASK_COUNT) = 28*1000000*1/4; //0x47868C0/4;//count for 1 sec 0x47868C0 + } + + REG32_VAL(IRCTL) |= (0x1<<25); + REG32_VAL(INT_MASK_CTRL) = 0x1; + + /* IR_EN */ + REG32_VAL(IRCTL) |= 0x1; + while(retries++ < 100 && !(REG32_VAL(IRCTL)&0x01)){ + REG32_VAL(IRCTL) |= 0x1; + udelay(5); + } + + /* read CIR status to clear IR interrupt. */ + st = REG32_VAL(IRSTS); +} + +static void rmtctl_refresh_vendorcode(struct work_struct *work) +{ + unsigned char data[64]; + struct rmtctl_priv *priv = container_of(work, struct rmtctl_priv, delaywork.work); + sprintf(data, "0x%x", priv->vendor_code); + wmt_setsyspara(WMT_RMTCTL_VENDOR_ENV, data); +} + +static void rel_handler(unsigned long data) +{ + struct rmtctl_rel *rel = (struct rmtctl_rel *)data; + struct rmtctl_priv *priv = container_of(rel, struct rmtctl_priv, rel_dev); + + rmtctl_report_rel(&priv->rel_dev, rel->code); + mod_timer(&rel->timer, jiffies+HZ/15); // report rate 15pps +} + +static void rmtctl_rel_init(struct rmtctl_rel *rel) +{ + struct input_dev *input; + + input = input_allocate_device(); + input->name = "rmtctl-rel"; + + /* register the device as mouse */ + set_bit(EV_REL, input->evbit); + set_bit(REL_X, input->relbit); + set_bit(REL_Y, input->relbit); + + /* register device's buttons and keys */ + set_bit(EV_KEY, input->evbit); + set_bit(BTN_LEFT, input->keybit); + set_bit(BTN_RIGHT, input->keybit); + + input_register_device(input); + rel->input = input; + + init_timer(&rel->timer); + rel->timer.data = (unsigned long)rel; + rel->timer.function = rel_handler; +} + +void led_handler(unsigned long data) +{ + struct rmtctl_led *led = (struct rmtctl_led*)data; + + gpio_direction_output(led->gpio, led->on++%2); + mod_timer(&led->timer, jiffies+HZ/20); + return; +} + +static void rmtctl_led_init(struct rmtctl_led *led) +{ + init_timer(&led->timer); + led->timer.data = (unsigned long)led; + led->timer.function = led_handler; + + gpio_direction_output(led->gpio, led->active); + + return; +} + +static int rmtctl_probe(struct platform_device *dev) +{ + int i,ivendor, ikeycode; + char buf[64]; + int varlen = sizeof(buf); + struct rmtctl_priv *priv; + + priv = kzalloc(sizeof(struct rmtctl_priv), GFP_KERNEL); + if (priv == NULL) + return -ENOMEM; + platform_set_drvdata(dev, priv); + + // get codec type of cir device + if (wmt_getsyspara("wmt.io.rmtctl", buf, &varlen) == 0) { + sscanf(buf, "%d", &priv->codec); + if (priv->codec < CIR_CODEC_NEC || priv->codec >= CIR_CODEC_MAX){ + kfree(priv); + return -EINVAL; + } + } + else { + kfree(priv); + return -EINVAL; + } + + // using assigned cir device? + if (wmt_getsyspara("wmt.io.rmtctl.index", buf, &varlen) == 0) { + sscanf(buf, "%d", &priv->vendor_index); + if (priv->vendor_index >= ARRAY_SIZE(rmtctl_vendors) || + rmtctl_vendors[priv->vendor_index].codec != priv->codec) + priv->vendor_index = -1; + + priv->table_index = priv->vendor_index; + } + else + priv->vendor_index = -1; + + // get last vendor code saved in uboot environment +#if 0 + if (wmt_getsyspara(WMT_RMTCTL_VENDOR_ENV, buf, &varlen) == 0){ + sscanf(buf, "0x%x", &priv->vendor_code); + priv->saved_vcode = priv->vendor_code; + for (i = 0; i < ARRAY_SIZE(rmtctl_vendors); i++) { + if (priv->vendor_code == rmtctl_vendors[i].vendor_code && + rmtctl_vendors[i].codec == priv->codec) { + priv->table_index = i; + break; + } + } + }else +#endif + priv->vendor_code = 0xffff; + + if (wmt_getsyspara("wmt.io.rmtctl.led", buf, &varlen) == 0) { + sscanf(buf, "%d:%d", &priv->led.gpio, &priv->led.active); + priv->led.enable = 1; + } + + /* register an input device. */ + if ((priv->idev = input_allocate_device()) == NULL) + return -ENOMEM; + + set_bit(EV_KEY, priv->idev->evbit); + + for (ivendor = 0; ivendor < ARRAY_SIZE(rmtctl_vendors); ivendor++) { + if (priv->codec == rmtctl_vendors[ivendor].codec) { + for (ikeycode = 0; ikeycode < ARRAY_SIZE(rmtctl_vendors[ivendor].key_codes); ikeycode++) { + if (rmtctl_vendors[ivendor].key_codes[ikeycode]) { + set_bit(rmtctl_vendors[ivendor].key_codes[ikeycode], priv->idev->keybit); + } + } + } + } + + priv->idev->name = "rmtctl"; + priv->idev->phys = "rmtctl"; + input_register_device(priv->idev); + rmtctl_rel_init(&priv->rel_dev); + if(priv->led.enable){ + gpio_request(priv->led.gpio,"rmtctl-led"); + rmtctl_led_init(&priv->led); + } + INIT_DELAYED_WORK(&priv->delaywork, rmtctl_refresh_vendorcode); + + init_timer(&priv->timer); + priv->timer.data = (unsigned long)priv->idev; + priv->timer.function = rmtctl_timer_handler; + + /* Register an ISR */ + request_irq(IRQ_CIR, rmtctl_interrupt, IRQF_SHARED, "rmtctl", priv); + + /* Initial H/W */ + rmtctl_hw_init((unsigned long)priv); + + if (RMTCTL_DEBUG) + printk("WonderMedia rmtctl driver v0.98 initialized: ok\n"); + + return 0; +} + +static int rmtctl_remove(struct platform_device *dev) +{ + struct rmtctl_priv *priv = platform_get_drvdata(dev); + + if (RMTCTL_DEBUG) + printk("rmtctl_remove\n"); + + del_timer_sync(&priv->timer); + cancel_delayed_work_sync(&priv->delaywork); + del_timer_sync(&priv->rel_dev.timer); + input_unregister_device(priv->rel_dev.input); + input_free_device(priv->rel_dev.input); + if(priv->led.enable){ + del_timer_sync(&priv->led.timer); + gpio_direction_output(priv->led.gpio, priv->led.active); + gpio_free(priv->led.gpio); + } + + free_irq(IRQ_CIR, priv); + input_unregister_device(priv->idev); + input_free_device(priv->idev); + + kfree(priv); + + return 0; +} + +#ifdef CONFIG_PM +static int rmtctl_suspend(struct platform_device *dev, pm_message_t state) +{ + struct rmtctl_priv *priv = platform_get_drvdata(dev); + + del_timer_sync(&priv->timer); + cancel_delayed_work_sync(&priv->delaywork); + del_timer_sync(&priv->rel_dev.timer); + if(priv->led.enable){ + priv->led.on = 0; + del_timer(&priv->led.timer); + gpio_direction_output(priv->led.gpio, priv->led.active); + } + + /* Nothing to suspend? */ + rmtctl_hw_init((unsigned long)priv); + rmtctl_hw_suspend((unsigned long)priv); + + disable_irq(IRQ_CIR); + + /* Enable rmt wakeup */ + PMWE_VAL |= (1 << WKS_CIR); + + return 0; +} + +static int rmtctl_resume(struct platform_device *dev) +{ + volatile unsigned int regval; + int i =0 ; + struct rmtctl_priv *priv = platform_get_drvdata(dev); + + /* Initial H/W */ + REG32_VAL(WAKEUP_CTRL) &=~ BIT0; + + for (i=0;i<10;i++) + { + regval = REG32_VAL(WAKEUP_STS) ; + + if (regval & BIT0){ + REG32_VAL(WAKEUP_STS) |= BIT4; + + }else{ + break; + } + msleep_interruptible(5); + } + + regval = REG32_VAL(WAKEUP_STS) ; + if (regval & BIT0) + printk("CIR resume NG WAKEUP_STS 0x%08x \n",regval); + + rmtctl_hw_init((unsigned long)priv); + if(priv->led.enable) + rmtctl_led_init(&priv->led); + enable_irq(IRQ_CIR); + return 0; +} +#else +#define rmtctl_suspend NULL +#define rmtctl_resume NULL +#endif + +static struct platform_driver rmtctl_driver = { + .driver.name = "wmt-rmtctl", + //.bus = &platform_bus_type, + //.probe = rmtctl_probe, + .remove = rmtctl_remove, + .suspend = rmtctl_suspend, + .resume = rmtctl_resume +}; + +static void rmtctl_release(struct device *dev) +{ + /* Nothing to release? */ +} + +static u64 rmtctl_dmamask = 0xffffffff; + +static struct platform_device rmtctl_device = { + .name = "wmt-rmtctl", + .id = 0, + .dev = { + .release = rmtctl_release, + .dma_mask = &rmtctl_dmamask, + }, + .num_resources = 0, + .resource = NULL, +}; + +static int __init rmtctl_init(void) +{ + int ret; + + if (platform_device_register(&rmtctl_device)) + return -1; + ret = platform_driver_probe(&rmtctl_driver, rmtctl_probe); + + return ret; +} + +static void __exit rmtctl_exit(void) +{ + platform_driver_unregister(&rmtctl_driver); + platform_device_unregister(&rmtctl_device); +} + +module_init(rmtctl_init); +module_exit(rmtctl_exit); + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("WonderMedia Technologies, Inc."); +MODULE_DESCRIPTION("WMT [Remoter] driver"); |