/* * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * Copyright (c) 2004 Arnaud Patard * iPAQ H1940 touchscreen support * * ChangeLog * * 2004-09-05: Herbert Pƶtzl * - added clock (de-)allocation code * * 2005-03-06: Arnaud Patard * - h1940_ -> s3c24xx (this driver is now also used on the n30 * machines :P) * - Debug messages are now enabled with the config option * TOUCHSCREEN_S3C_DEBUG * - Changed the way the value are read * - Input subsystem should now work * - Use ioremap and readl/writel * * 2005-03-23: Arnaud Patard * - Make use of some undocumented features of the touchscreen * controller * */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include //#define USE_12BITS_ADC //#define USE_10BITS_ADC #define LEVEL0_MIN 2980 #define LEVEL0_MAX 3384 #define LEVEL1_MIN 3385 #define LEVEL1_MAX 3484 #define LEVEL2_MIN 3485 #define LEVEL2_MAX 3684 #define LEVEL3_MIN 3685 #define LEVEL3_MAX 4080 #define BATTERY_FULL 4081 #define ADC_READ_LEVEL_VAULE _IOR('S', 0x01, unsigned long) #define ADC_READ_INPUT_SOURCE_STATUS _IOR('S', 0x02, unsigned long) #define ADC_SET_PM_WKUP_SOURCE _IOW('S', 0x03, unsigned long) #define ADC_READ_RAW_VAULE _IOR('S', 0x04, unsigned long) #define ADC_POWER_OFF _IO ('S', 0x05) #define ADC_READ_INPUT_POWER_STATUS _IOR('S', 0x06, unsigned long) #define ADC_READ_CHARGING_STATUS _IOR('S', 0x07, unsigned long) #define ADC_SET_PM_FUNCTION_POWER _IOW('S', 0x08, unsigned long) #ifdef CONFIG_PM_CPU_MODE #define ADC_SET_CPU_PM_MODE _IOW('S', 0x09, unsigned long) #endif #define ADC_READ_WAKEUP_SOURCE _IOR('S', 0x0A, unsigned long) #define ADC_READ_POWER_FAIL_STATUS _IOR('S', 0x0B, unsigned long) //#define ADC_INPUT_PIN _IOW('S', 0x0c, unsigned long) /*Qisda Qube 20091103 for Smart card power control*/ #define ADC_SMARTCARD_PWR _IOR('S', 0x8F, unsigned long) /*Qisda Qube 20091103 for Smart card power control*/ #define NON_ZERO_VALUE 0xFF #define INPUT_SOURCE_USB 0 #define INPUT_SOURCE_CHARGER 1 #define INPUT_SOURCE_NOTHING 2 unsigned int wkup_srce = NON_ZERO_VALUE; #ifdef CONFIG_PM_CPU_MODE unsigned char pm_cpu_mode = 0; #endif #define ADC_PM_DEBUG_PORT (0x00000003) #define ADC_PM_WIFI (0x0000000C) #define ADC_PM_AUDIO (0x00000030) #define ADC_PM_SD (0x000000C0) #define ADC_PM_TCON (0x00000300) #define ADC_PM_USB (0x00000C00) #define ADC_PM_TOUCH (0x0000F000) #define ADC_PM_SMART_CARD (0x00030000) #define ADC_PM_3G_MODULE (0x000C0000) #define ADC_PM_I2C (0x00300000) extern int IsWallCharger(void); /* Qisda, ShiYong Lin, 2009/10/27, Modify for exception in IsWallCharger function {*/ extern int charging_source; /* } Qisda, ShiYong Lin, 2009/10/27, Modify for exception in IsWallCharger function */ #define ADC_MINOR 131 #if defined(CONFIG_CPU_S3C2443) #define ADC_ENV(x) (S3C2410_ADCCON_PRSCVL(49)) #else #define ADC_ENV(x) (S3C2410_ADCCON_PRSCVL(49) | S3C2410_ADCCON_SELMUX(x)) #endif struct s3c_adc_mach_info { int delay; int presc; int resol_bit; }; /* ADC default configuration */ struct s3c_adc_mach_info s3c_adc_cfg __initdata = { .delay = 10000, .presc = 49, //#if defined(CONFIG_CPU_S3C2450) || defined(CONFIG_CPU_S3C2416) || defined(CONFIG_CPU_S3C6410) //.resol_bit = 12, //#else /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status {*/ .resol_bit = 12, /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status }*/ //#endif }; static struct clk *adc_clock; static void __iomem *base_addr; static DEFINE_MUTEX(adc_mutex); static unsigned long data_for_ADCCON; static unsigned long data_for_ADCTSC; static int adc_port = 0; static void s3c_adc_save_SFR_on_ADC(void) { data_for_ADCCON = readl(base_addr+S3C2410_ADCCON); data_for_ADCTSC = readl(base_addr+S3C2410_ADCTSC); } static void s3c_adc_restore_SFR_on_ADC(void) { writel(data_for_ADCCON, base_addr+S3C2410_ADCCON); writel(data_for_ADCTSC, base_addr+S3C2410_ADCTSC); } static int s3c_adc_open(struct inode *inode, struct file *file) { return 0; } /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status {*/ #ifdef USE_10BITS_ADC static ssize_t s3c_adc_read(struct file *file, char __user * buffer, size_t size, loff_t * pos) { unsigned long data0; unsigned long data1; int adc_value = 0; // printk(KERN_INFO " s3c_adc_read() entered\n"); mutex_lock(&adc_mutex); s3c_adc_save_SFR_on_ADC(); writel(0x0, base_addr+S3C2410_ADCCON); writel(0x58, base_addr+S3C2410_ADCTSC); udelay(10); writel(ADC_ENV(adc_port)|S3C2410_ADCCON_STDBM, base_addr+S3C2410_ADCCON); #if defined(CONFIG_CPU_S3C2443) writel(S3C2410_ADCCON_SELMUX(adc_port), base_addr+S3C2410_ADCMUX); #endif udelay(10); writel(S3C2410_ADCCON_PRSCEN|ADC_ENV(adc_port)|S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON); do{ data0 = readl(base_addr+S3C2410_ADCCON); } while(!(data0 & S3C2410_ADCCON_ECFLG)); data1 = readl(base_addr+S3C2410_ADCDAT0); adc_value = data1 & S3C2410_ADCDAT0_XPDATA_MASK; s3c_adc_restore_SFR_on_ADC(); mutex_unlock(&adc_mutex); // printk(KERN_INFO " Converted Value: %03d\n", adc_value); if (copy_to_user(buffer, &adc_value, sizeof(unsigned int))) { return -EFAULT; } return sizeof(unsigned int); } #else unsigned long s3c_adc_read(void) { unsigned long data0; unsigned long data1; int adc_value = 0; // printk("s3c_adc_read entered\n"); mutex_lock(&adc_mutex); s3c_adc_save_SFR_on_ADC(); writel(0x0, base_addr+S3C2410_ADCCON); writel(0x58, base_addr+S3C2410_ADCTSC); udelay(10); writel(ADC_ENV(adc_port)|S3C2410_ADCCON_STDBM, base_addr+S3C2410_ADCCON); #if defined(CONFIG_CPU_S3C2443) writel(S3C2410_ADCCON_SELMUX(adc_port), base_addr+S3C2410_ADCMUX); #else writel(0, base_addr+0x18); #endif udelay(10); writel(S3C2410_ADCCON_PRSCEN|ADC_ENV(adc_port)|S3C2410_ADCCON_ENABLE_START|S3C2450_ADCCON_RESSEL_12BIT, base_addr+S3C2410_ADCCON); do{ data0 = readl(base_addr+S3C2410_ADCCON); } while(!(data0 & S3C2410_ADCCON_ECFLG)); data1 = readl(base_addr+S3C2410_ADCDAT0); adc_value = data1 & S3C_ADCDAT1_YPDATA_MASK_12BIT; s3c_adc_restore_SFR_on_ADC(); mutex_unlock(&adc_mutex); // printk("Converted Value: %05d\n", adc_value); return adc_value; } #endif /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status }*/ static int s3c_adc_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg) { int i = 0; int average_adc_value = 0; int data_mask = 0; unsigned int ret = -ENOIOCTLCMD; unsigned long level = NON_ZERO_VALUE; unsigned long input_source = NON_ZERO_VALUE; switch (cmd) { /*Qisda Qube 20091103 for Smart card power control*/ case ADC_SMARTCARD_PWR: if(arg==0) { printk("Enable smard card power\n"); s3c2410_gpio_cfgpin(S3C2410_GPC3, S3C2410_GPC3_OUTP); s3c2410_gpio_setpin(S3C2410_GPC3,1); s3c2410_gpio_cfgpin(S3C2410_GPH2, S3C2410_GPH2_TXD0); s3c2410_gpio_cfgpin(S3C2410_GPH3, S3C2410_GPH3_RXD0); } else if(arg==3) { printk("Disable smard card power\n"); s3c2410_gpio_cfgpin(S3C2410_GPC3, S3C2410_GPC3_OUTP); s3c2410_gpio_setpin(S3C2410_GPC3,0); s3c2410_gpio_cfgpin(S3C2410_GPH2, S3C2410_GPH2_OUTP); s3c2410_gpio_setpin(S3C2410_GPH2,0); s3c2410_gpio_cfgpin(S3C2410_GPH3, S3C2410_GPH3_OUTP); s3c2410_gpio_setpin(S3C2410_GPH3,0); } return 0; /*Qisda Qube 20091103 for Smart card power control*/ /* case ADC_INPUT_PIN: adc_port = (unsigned int) arg; if (adc_port >= 4) printk(" %d is already reserved for TouchScreen\n", adc_port); return 0; */ case ADC_READ_INPUT_SOURCE_STATUS: //Read the level data from GPG1 //CHG_STA: 0 => no andy insertion source // 1 => There is an insertion existing in GPG /* Qisda, ShiYong Lin, 2009/10/27, Modify for exception in IsWallCharger function {*/ input_source = charging_source; put_user(input_source, (unsigned long __user *)arg); ret = 0; /* } Qisda, ShiYong Lin, 2009/10/27, Modify for exception in IsWallCharger function */ break; case ADC_SET_PM_WKUP_SOURCE: wkup_srce = (unsigned long) arg; printk("Wakeup parameter = %d\n", wkup_srce); /* Qisda, ShiYong Lin, 2009/09/28, Add the sleep event message when sleep {*/ #ifdef QISDA s3c_keypad_pm_sleep_message_to_ap(1); #endif /* } Qisda, ShiYong Lin, 2009/09/28, Add the sleep event message when sleep */ ret = 0; break; case ADC_READ_LEVEL_VAULE: for(i=0; i<20 ;i++) { udelay(1000); average_adc_value += s3c_adc_read(); } average_adc_value = average_adc_value/20; if (average_adc_value <= LEVEL0_MAX) { level = 0; } else if (average_adc_value < LEVEL1_MAX && average_adc_value >= LEVEL1_MIN) { level = 1; } else if (average_adc_value < LEVEL2_MAX && average_adc_value >= LEVEL2_MIN) { level = 2; } else if(average_adc_value < LEVEL3_MAX && average_adc_value >= LEVEL3_MIN) { level = 3; } else if(average_adc_value >= BATTERY_FULL) { level = 4; } // printk("level = %d, ADC data = %d\n", (int)level, (int)average_adc_value); put_user(level, (unsigned long __user *)arg); ret = 0; break; case ADC_READ_RAW_VAULE: for(i=0; i<20 ;i++) { udelay(1000); average_adc_value += s3c_adc_read(); } average_adc_value = average_adc_value / 20; printk("ADC data = %d\n", (int)average_adc_value); put_user(average_adc_value, (unsigned long __user *)arg); ret = 0; break; case ADC_POWER_OFF: printk("power off"); s3c2410_gpio_cfgpin(S3C2410_GPG0, S3C2410_GPG0_OUTP); s3c2410_gpio_setpin(S3C2410_GPG0, 0); ret = 0; break; #ifdef CONFIG_QISDA_AS090B00_EVT1 case ADC_READ_INPUT_POWER_STATUS: data_mask = readl(S3C2410_GPHDAT); data_mask &= 0x1<<5; printk("GPH5 = %d\n", data_mask); ret = put_user(data_mask, (unsigned long __user *)arg); break; #endif #if defined(CONFIG_QISDA_AS090B00_EVT1_1) || defined (CONFIG_QISDA_QD090B00_EVT1) || defined(CONFIG_QISDA_QD060N00_DVT1_1) case ADC_READ_CHARGING_STATUS: data_mask = readl(S3C2410_GPFDAT); data_mask &= 0x1<<6; printk("GPF6 = %d\n", data_mask); ret = put_user(data_mask, (unsigned long __user *)arg); break; #endif case ADC_SET_PM_FUNCTION_POWER: printk("ADC_SET_PM_FUNCTION_POWER\n"); #if defined(CONFIG_QISDA_AS090B00_EVT1_1) || defined(CONFIG_QISDA_AS090B00_EVT1) || defined(CONFIG_QISDA_QD060N00_DVT1_1) || defined(CONFIG_QISDA_QD090B00_EVT1) if((arg & ADC_PM_DEBUG_PORT)==0){ printk("ADC_PM_DEBUG_PORT, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPB6, S3C2410_GPB6_OUTP); s3c2410_gpio_setpin(S3C2410_GPB6, 0); s3c2410_gpio_cfgpin(S3C2410_GPH0, S3C2410_GPH0_OUTP); s3c2410_gpio_setpin(S3C2410_GPH0, 0); s3c2410_gpio_cfgpin(S3C2410_GPH1, S3C2410_GPH1_OUTP); s3c2410_gpio_setpin(S3C2410_GPH1, 0); } else if((arg & ADC_PM_DEBUG_PORT)==1){ printk("ADC_PM_DEBUG_PORT, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPB6, S3C2410_GPB6_OUTP); s3c2410_gpio_setpin(S3C2410_GPB6, 0); s3c2410_gpio_cfgpin(S3C2410_GPH0, S3C2410_GPH0_nCTS0); s3c2410_gpio_cfgpin(S3C2410_GPH1, S3C2410_GPH1_nRTS0); } else if((arg & ADC_PM_DEBUG_PORT)==2){ printk("ADC_PM_DEBUG_PORT, 2 do special test\n"); } else{ printk("ADC_PM_DEBUG_PORT, 3 do nothing\n"); } //WIFI if(((arg & ADC_PM_WIFI) >> 2)==0){ printk("ADC_PM_WIFI, 0\n"); /* s3c2410_gpio_cfgpin(S3C2410_GPH12, S3C2410_GPH12_OUTP); s3c2410_gpio_setpin(S3C2410_GPH12, 0); writel(((readl(S3C2416_GPKCON) & ~(1<<13)) | (1<<12)), S3C2416_GPKCON); writel(((readl(S3C2416_GPKDAT) | 0<<6)), S3C2416_GPKDAT); writel(((readl(S3C2416_GPKCON) & ~(1<<11)) | (1<<10)), S3C2416_GPKCON); writel(((readl(S3C2416_GPKDAT) | 0<<5)), S3C2416_GPKDAT); //Signal s3c2410_gpio_cfgpin(S3C2410_GPE11, S3C2410_GPE11_OUTP); s3c2410_gpio_setpin(S3C2410_GPE11, 0); s3c2410_gpio_cfgpin(S3C2410_GPE12, S3C2410_GPE12_OUTP); s3c2410_gpio_setpin(S3C2410_GPE12, 0); s3c2410_gpio_cfgpin(S3C2410_GPE13, S3C2410_GPE13_OUTP); s3c2410_gpio_setpin(S3C2410_GPE13, 0); s3c2410_gpio_cfgpin(S3C2410_GPF3, S3C2410_GPF3_OUTP); s3c2410_gpio_setpin(S3C2410_GPF3, 0); s3c2410_gpio_cfgpin(S3C2410_GPG4, S3C2410_GPG4_OUTP); s3c2410_gpio_setpin(S3C2410_GPG4, 0); s3c2410_gpio_cfgpin(S3C2410_GPD8, S3C2410_GPD8_OUTP); s3c2410_gpio_setpin(S3C2410_GPD8, 0); s3c2410_gpio_cfgpin(S3C2410_GPL13, S3C2410_GPL13_OUTP); s3c2410_gpio_setpin(S3C2410_GPL13, 0); */ s3c2410_gpio_cfgpin(S3C2410_GPD8, S3C2410_GPD8_OUTP); s3c2410_gpio_setpin(S3C2410_GPD8, 0); msleep(100); s3c2410_gpio_cfgpin(S3C2410_GPE11, S3C2410_GPE11_OUTP); s3c2410_gpio_setpin(S3C2410_GPE11, 0); s3c2410_gpio_cfgpin(S3C2410_GPE12, S3C2410_GPE12_OUTP); s3c2410_gpio_setpin(S3C2410_GPE12, 0); s3c2410_gpio_cfgpin(S3C2410_GPE13, S3C2410_GPE13_OUTP); s3c2410_gpio_setpin(S3C2410_GPE13, 0); s3c2410_gpio_cfgpin(S3C2410_GPF3, S3C2410_GPF3_OUTP); s3c2410_gpio_setpin(S3C2410_GPF3, 0); s3c2410_gpio_cfgpin(S3C2410_GPG4, S3C2410_GPG4_OUTP); s3c2410_gpio_setpin(S3C2410_GPG4, 0); writel(((readl(S3C2410_GPHCON) & ~(1<<25)) | (1<<24)), S3C2410_GPHCON); /* GPH12, 3.3V */ writel((readl(S3C2410_GPHDAT) & ~(1<<12)), S3C2410_GPHDAT); writel(((readl(S3C2410_GPHUP) & ~(1<<25)) | (1<<24)), S3C2410_GPHUP); msleep(100); s3c2410_gpio_cfgpin(S3C2410_GPL13, S3C2410_GPL13_OUTP); s3c2410_gpio_setpin(S3C2410_GPL13, 0); writel(((readl(S3C2416_GPKCON) & ~(1<<13)) | (1<<12)), S3C2416_GPKCON); writel((readl(S3C2416_GPKDAT) & ~(1<<6)), S3C2416_GPKDAT); msleep(100); writel(((readl(S3C2416_GPKCON) & ~(1<<11)) | (1<<10)), S3C2416_GPKCON); writel((readl(S3C2416_GPKDAT) & ~(1<<5)), S3C2416_GPKDAT); msleep(100); } else if(((arg & ADC_PM_WIFI) >> 2)==1){ printk("ADC_PM_WIFI, 1\n"); // Powering on wifi is via the script file /* //power s3c2410_gpio_cfgpin(S3C2410_GPH12, S3C2410_GPH12_OUTP); s3c2410_gpio_setpin(S3C2410_GPH12, 1); writel(((readl(S3C2416_GPKCON) & ~(1<<13)) | (1<<12)), S3C2416_GPKCON); writel(((readl(S3C2416_GPKDAT) | 1<<6)), S3C2416_GPKDAT); writel(((readl(S3C2416_GPKCON) & ~(1<<11)) | (1<<10)), S3C2416_GPKCON); writel(((readl(S3C2416_GPKDAT) | 1<<5)), S3C2416_GPKDAT); //Signal s3c2410_gpio_cfgpin(S3C2410_GPE11, S3C2410_GPE11_SPIMISO0); s3c2410_gpio_cfgpin(S3C2410_GPE12, S3C2410_GPE12_SPIMOSI0); s3c2410_gpio_cfgpin(S3C2410_GPE13, S3C2410_GPE13_SPICLK0); s3c2410_gpio_cfgpin(S3C2410_GPF3, S3C2410_GPF3_EINT3); s3c2410_gpio_cfgpin(S3C2410_GPG4, S3C2410_GPG4_EINT12); s3c2410_gpio_cfgpin(S3C2410_GPD8, S3C2410_GPD8_OUTP); s3c2410_gpio_setpin(S3C2410_GPD8, 1); s3c2410_gpio_cfgpin(S3C2410_GPL13, S3C2410_GPL13_OUTP); s3c2410_gpio_setpin(S3C2410_GPL13, 1); */ /* s3c2410_gpio_setpin(S3C2410_GPH12, 1); msleep(100); s3c2410_gpio_setpin(S3C2410_GPIONO(S3C2416_GPIO_BANKK, 5), 1); msleep(100); s3c2410_gpio_setpin(S3C2410_GPIONO(S3C2416_GPIO_BANKK, 6), 1); msleep(100); s3c2410_gpio_setpin(S3C2410_GPD8, 1); msleep(100); s3c2410_gpio_setpin(S3C2410_GPD8, 0); msleep(100); s3c2410_gpio_setpin(S3C2410_GPD8, 1); msleep(100); */ } else if(((arg & ADC_PM_WIFI) >> 2)==2){ printk("ADC_PM_WIFI, 2 do special test\n"); } else{ printk("ADC_PM_WIFI, 3 do nothing\n"); } //Audio ok if(((arg & ADC_PM_AUDIO) >> 4)==0){ printk("ADC_PM_AUDIO, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPB10, S3C2410_GPB10_OUTP); s3c2410_gpio_setpin(S3C2410_GPB10, 0); s3c2410_gpio_cfgpin(S3C2410_GPE0, S3C2410_GPE0_OUTP); s3c2410_gpio_setpin(S3C2410_GPE0, 0); s3c2410_gpio_cfgpin(S3C2410_GPE1, S3C2410_GPE1_OUTP); s3c2410_gpio_setpin(S3C2410_GPE1, 0); s3c2410_gpio_cfgpin(S3C2410_GPE2, S3C2410_GPE2_OUTP); s3c2410_gpio_setpin(S3C2410_GPE2, 0); s3c2410_gpio_cfgpin(S3C2410_GPE3, S3C2410_GPE3_OUTP); s3c2410_gpio_setpin(S3C2410_GPE3, 0); s3c2410_gpio_cfgpin(S3C2410_GPE4, S3C2410_GPE4_OUTP); s3c2410_gpio_setpin(S3C2410_GPE4, 0); s3c2410_gpio_cfgpin(S3C2410_GPF4, S3C2410_GPF4_OUTP); s3c2410_gpio_setpin(S3C2410_GPF4, 0); #ifdef CONFIG_QISDA_AS090B00_EVT1 s3c2410_gpio_cfgpin(S3C2410_GPG3, S3C2410_GPG3_OUTP); s3c2410_gpio_setpin(S3C2410_GPG3, 0); #endif #if defined(CONFIG_QISDA_AS090B00_EVT1_1)|| defined (CONFIG_QISDA_QD090B00_EVT1) || defined(CONFIG_QISDA_QD060N00_DVT1_1) s3c2410_gpio_cfgpin(S3C2410_GPA1, S3C2410_GPA1_OUT); //s02 ????S3C2410_GPA1_OUT s3c2410_gpio_setpin(S3C2410_GPA1, 0); #endif } else if(((arg & ADC_PM_AUDIO) >> 4)==1){ printk("ADC_PM_AUDIO, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPB10, S3C2410_GPB10_OUTP); s3c2410_gpio_setpin(S3C2410_GPB10, 1); s3c2410_gpio_cfgpin(S3C2410_GPE0, S3C2410_GPE0_I2SLRCK); s3c2410_gpio_cfgpin(S3C2410_GPE1, S3C2410_GPE1_I2SSCLK); s3c2410_gpio_cfgpin(S3C2410_GPE2, S3C2410_GPE2_CDCLK); s3c2410_gpio_cfgpin(S3C2410_GPE3, S3C2410_GPE3_I2SSDI); s3c2410_gpio_cfgpin(S3C2410_GPE4, S3C2410_GPE4_I2SSDO); s3c2410_gpio_cfgpin(S3C2410_GPF4, S3C2410_GPF4_EINT4); s3c2410_gpio_pullup(S3C2410_GPF4, 2); #ifdef CONFIG_QISDA_AS090B00_EVT1 s3c2410_gpio_cfgpin(S3C2410_GPG3, S3C2410_GPG3_OUTP); //s01 s3c2410_gpio_setpin(S3C2410_GPG3, 1); #endif #if defined(CONFIG_QISDA_AS090B00_EVT1_1) || defined (CONFIG_QISDA_QD090B00_EVT1) || defined(CONFIG_QISDA_QD060N00_DVT1_1) s3c2410_gpio_cfgpin(S3C2410_GPA1, S3C2410_GPA1_OUT); //s02 ????S3C2410_GPA1_OUT s3c2410_gpio_setpin(S3C2410_GPA1, 1); #endif } else if(((arg & ADC_PM_AUDIO) >> 4)==2){ printk("ADC_PM_AUDIO, 2 do special test\n"); } else{ printk("ADC_PM_AUDIO, 3 do nothing\n"); } //SD ok if(((arg & ADC_PM_SD) >> 6)==0){ printk("ADC_PM_SD, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPG2, S3C2410_GPG2_OUTP); s3c2410_gpio_setpin(S3C2410_GPG2, 0); s3c2410_gpio_cfgpin(S3C2410_GPE5, S3C2410_GPE5_OUTP); s3c2410_gpio_setpin(S3C2410_GPE5, 0); s3c2410_gpio_cfgpin(S3C2410_GPE6, S3C2410_GPE6_OUTP); s3c2410_gpio_setpin(S3C2410_GPE6, 0); s3c2410_gpio_cfgpin(S3C2410_GPE7, S3C2410_GPE7_OUTP); s3c2410_gpio_setpin(S3C2410_GPE7, 0); s3c2410_gpio_cfgpin(S3C2410_GPE8, S3C2410_GPE8_OUTP); s3c2410_gpio_setpin(S3C2410_GPE8, 0); s3c2410_gpio_cfgpin(S3C2410_GPE9, S3C2410_GPE9_OUTP); s3c2410_gpio_setpin(S3C2410_GPE9, 0); s3c2410_gpio_cfgpin(S3C2410_GPE10, S3C2410_GPE10_OUTP); s3c2410_gpio_setpin(S3C2410_GPE10, 0); s3c2410_gpio_cfgpin(S3C2410_GPF1, S3C2410_GPF1_OUTP); s3c2410_gpio_setpin(S3C2410_GPF1, 0); } else if(((arg & ADC_PM_SD) >> 6)==1){ printk("ADC_PM_SD, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPG2, S3C2410_GPG2_OUTP); s3c2410_gpio_setpin(S3C2410_GPG2, 1); s3c2410_gpio_cfgpin(S3C2410_GPE5, S3C2450_GPE5_SD0_CLK); s3c2410_gpio_cfgpin(S3C2410_GPE6, S3C2450_GPE6_SD0_CMD); s3c2410_gpio_cfgpin(S3C2410_GPE7, S3C2450_GPE7_SD0_DAT0); s3c2410_gpio_cfgpin(S3C2410_GPE8, S3C2450_GPE8_SD0_DAT1); s3c2410_gpio_cfgpin(S3C2410_GPE9, S3C2450_GPE9_SD0_DAT2); s3c2410_gpio_cfgpin(S3C2410_GPE10, S3C2450_GPE10_SD0_DAT3); s3c2410_gpio_cfgpin(S3C2410_GPF1, S3C2410_GPF1_EINT1); s3c2410_gpio_setpin(S3C2410_GPF1, 0); } else if(((arg & ADC_PM_SD) >> 6)==2){ printk("ADC_PM_SD, 2 do special test\n"); } else{ printk("ADC_PM_SD, 3 do nothing\n"); } //TCON ivan interface ok if(((arg & ADC_PM_TCON) >> 8)==0){ //power and signal printk("ADC_PM_TCON, 0\n"); //AUO T-CON Standby Epaper_Enter_Standby_Mode(1); //AUO T-CON Sleep Epaper_Enter_Sleep_Mode(1); msleep(250); //Shutdown T-CON Power Epaper_Power(0); //Shutdown i80 of s3c EPaper_CloseLcdPort(); msleep(1000); } else if(((arg & ADC_PM_TCON) >> 8)==1){ //power and signal printk("ADC_PM_TCON, 1\n"); Epaper_SetLcdPort(); msleep(100); //sleep for RST_N pull high } else if(((arg & ADC_PM_TCON) >> 8)==2){ printk("ADC_PM_TCON, 2 do special test\n"); } else{ printk("ADC_PM_TCON, 3 do nothing\n"); } //USB ok if(((arg & ADC_PM_USB) >> 10)==0){ printk("ADC_PM_USB, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPD13, S3C2410_GPD13_OUTP); s3c2410_gpio_setpin(S3C2410_GPD13, 0); /* usb power enbale */ s3c2410_gpio_cfgpin(S3C2443_GPH14, S3C2443_GPH14_OUTP); s3c2410_gpio_setpin(S3C2443_GPH14, 0); } else if(((arg & ADC_PM_USB) >> 10)==1){ printk("ADC_PM_USB, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPD13, S3C2410_GPD13_OUTP); s3c2410_gpio_setpin(S3C2410_GPD13, 1); /* usb power enbale */ s3c2410_gpio_cfgpin(S3C2443_GPH14, S3C2443_GPH14_OUTP); s3c2410_gpio_setpin(S3C2443_GPH14, 1); } else if(((arg & ADC_PM_USB) >> 10)==2){ printk("ADC_PM_USB, 2 do special test\n"); } else{ printk("ADC_PM_USB, 3 do nothing\n"); } //Touch GPIO ok, 12~15 if(((arg & ADC_PM_TOUCH) >> 12)==0){ printk("ADC_PM_TOUCH, 0\n"); // s3c2410_gpio_cfgpin(S3C2410_GPB3, S3C2410_GPB3_OUTP); // s3c2410_gpio_setpin(S3C2410_GPB3, 0); s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_OUTP); s3c2410_gpio_setpin(S3C2410_GPF2, 0); s3c2410_gpio_cfgpin(S3C2410_GPF5, S3C2410_GPF5_OUTP); s3c2410_gpio_setpin(S3C2410_GPF5, 0); } else if(((arg & ADC_PM_TOUCH) >> 12)==1){ //power on printk("ADC_PM_TOUCH, 1\n"); // s3c2410_gpio_cfgpin(S3C2410_GPB3, S3C2410_GPB3_OUTP); // s3c2410_gpio_setpin(S3C2410_GPB3, 1); s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_INP); s3c2410_gpio_cfgpin(S3C2410_GPF5, S3C2410_GPF5_OUTP); s3c2410_gpio_setpin(S3C2410_GPF5, 1); } else if(((arg & ADC_PM_TOUCH) >> 12)==2){ //power on () + touch active mode (touch ioctl) printk("ADC_PM_TOUCH, 2 Touch GPIO on and Enter Active\n"); s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_INP); s3c2410_gpio_cfgpin(S3C2410_GPF5, S3C2410_GPF5_OUTP); s3c2410_gpio_setpin(S3C2410_GPF5, 1); } else if(((arg & ADC_PM_TOUCH) >> 12)==3){ //power on + touch sleep mode (via touch ioctl) printk("ADC_PM_TOUCH, 3 Touch GPIO on and Enter Sleep\n"); s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_INP); s3c2410_gpio_cfgpin(S3C2410_GPF5, S3C2410_GPF5_OUTP); s3c2410_gpio_setpin(S3C2410_GPF5, 1); } else if(((arg & ADC_PM_TOUCH) >> 12)==4){ //power on + touch deep sleep mode (via touch ioctl) printk("ADC_PM_TOUCH, 4 Touch GPIO on and Enter Deep sleep\n"); s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_INP); s3c2410_gpio_cfgpin(S3C2410_GPF5, S3C2410_GPF5_OUTP); s3c2410_gpio_setpin(S3C2410_GPF5, 1); } else{ printk("ADC_PM_TOUCH, 5~15 do nothing\n"); } //SMART_CARD ok if(((arg & ADC_PM_SMART_CARD) >> 16)==0){ printk("ADC_PM_SMART_CARD, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPC3, S3C2410_GPC3_OUTP); s3c2410_gpio_setpin(S3C2410_GPC3, 0); s3c2410_gpio_cfgpin(S3C2410_GPH2, S3C2410_GPH2_OUTP); s3c2410_gpio_setpin(S3C2410_GPH2, 0); s3c2410_gpio_cfgpin(S3C2410_GPH3, S3C2410_GPH3_OUTP); s3c2410_gpio_setpin(S3C2410_GPH3, 0); } else if(((arg & ADC_PM_SMART_CARD) >> 16)==1){ printk("ADC_PM_SMART_CARD, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPC3, S3C2410_GPC3_OUTP); s3c2410_gpio_setpin(S3C2410_GPC3, 1); s3c2410_gpio_cfgpin(S3C2410_GPH2, S3C2410_GPH2_TXD0); s3c2410_gpio_cfgpin(S3C2410_GPH3, S3C2410_GPH3_RXD0); } else if(((arg & ADC_PM_SMART_CARD) >> 16)==2){ printk("ADC_PM_SMART_CARD, 2 do special test\n"); } else{ printk("ADC_PM_SMART_CARD, 3 do nothing\n"); } //SMART_CARD ok if(((arg & ADC_PM_3G_MODULE) >> 18)==0){ printk("ADC_PM_3G_MODULE, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPG3, S3C2410_GPG3_OUTP); s3c2410_gpio_setpin(S3C2410_GPG3, 0); s3c2410_gpio_cfgpin(S3C2410_GPH8, S3C2410_GPH8_OUTP); s3c2410_gpio_setpin(S3C2410_GPH8, 0); s3c2410_gpio_cfgpin(S3C2410_GPH9, S3C2410_GPH9_OUTP); s3c2410_gpio_setpin(S3C2410_GPH9, 0); s3c2410_gpio_cfgpin(S3C2410_GPH11, S3C2410_GPH11_OUTP); s3c2410_gpio_setpin(S3C2410_GPH11, 0); } else if(((arg & ADC_PM_3G_MODULE) >> 18)==1){ printk("ADC_PM_3G_MODULE, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPG3, S3C2410_GPG3_OUTP); s3c2410_gpio_setpin(S3C2410_GPG3, 1); s3c2410_gpio_cfgpin(S3C2410_GPH8, S3C2410_GPH8_OUTP); s3c2410_gpio_setpin(S3C2410_GPH8, 1); // s3c2410_gpio_cfgpin(S3C2410_GPH9, S3C2410_GPH9_OUTP); // s3c2410_gpio_setpin(S3C2410_GPH9, 0); s3c2410_gpio_cfgpin(S3C2410_GPH11, S3C2410_GPH11_OUTP); s3c2410_gpio_setpin(S3C2410_GPH11, 1); } else if(((arg & ADC_PM_3G_MODULE) >> 18)==2){ printk("ADC_PM_3G_MODULE, 2 do special test\n"); } else{ printk("ADC_PM_3G_MODULE, 3 do nothing\n"); } //ADC_PM_I2C ok if(((arg & ADC_PM_I2C) >> 20)==0){ printk("ADC_PM_I2C, 0\n"); s3c2410_gpio_cfgpin(S3C2410_GPE14, S3C2410_GPE14_OUTP); s3c2410_gpio_setpin(S3C2410_GPE14, 0); // IICSCL s3c2410_gpio_cfgpin(S3C2410_GPE15, S3C2410_GPE15_OUTP); s3c2410_gpio_setpin(S3C2410_GPE15, 0); // IICSDA } else if(((arg & ADC_PM_I2C) >> 20)==1){ printk("ADC_PM_I2C, 1\n"); s3c2410_gpio_cfgpin(S3C2410_GPE15, S3C2410_GPE15_IICSDA); s3c2410_gpio_cfgpin(S3C2410_GPE14, S3C2410_GPE14_IICSCL); } else if(((arg & ADC_PM_I2C) >> 20)==2){ printk("ADC_PM_I2C, 2 do special test\n"); } else{ printk("ADC_PM_I2C, 3 do nothing\n"); } #endif ret = 0; break; #ifdef CONFIG_PM_CPU_MODE case ADC_SET_CPU_PM_MODE: // 0 ADC_CPU_DEVICE_SLEEP: // 1 ADC_CPU_STOP: // 2 ADC_CPU_DEEP_STOP: // 3 ADC_CPU_IDLE: // 4 ADC_CPU_SLEEP: pm_cpu_mode = (unsigned char) arg; // s3c2410_sleep_enter(); printk("cpu_pm_mode parameter = %d\n", pm_cpu_mode); ret = 0; break; #endif case ADC_READ_WAKEUP_SOURCE: data_mask = readl(S3C2443_WKUPSTAT); printk("Wakeup source =0x%08lx\n", data_mask); ret = 0; break; case ADC_READ_POWER_FAIL_STATUS: data_mask = readl(S3C2410_GPGDAT) & (1<<5); printk("ADC_READ_POWER_FAIL_STATUS GPG5 = %d\n", data_mask); ret = put_user(data_mask, (unsigned long __user *)arg); break; default: ret = -ENOIOCTLCMD; break; } return ret; } static struct file_operations s3c_adc_fops = { .owner = THIS_MODULE, .read = s3c_adc_read, .open = s3c_adc_open, .ioctl = s3c_adc_ioctl, }; static struct miscdevice s3c_adc_miscdev = { .minor = ADC_MINOR, .name = "adc", .fops = &s3c_adc_fops, }; /* * The functions for inserting/removing us as a module. */ static int __init s3c_adc_probe(struct platform_device *pdev) { int ret; printk(KERN_INFO "s3c_adc_probe\n"); adc_clock = clk_get(NULL, "adc"); if (!adc_clock) { printk(KERN_ERR "failed to get adc clock source\n"); return -ENOENT; } clk_enable(adc_clock); base_addr=ioremap(S3C24XX_PA_ADC, 0x20); if (base_addr == NULL) { printk(KERN_ERR "Failed to remap register block\n"); return -ENOMEM; } if ((s3c_adc_cfg.presc&0xff) > 0) writel(S3C2410_ADCCON_PRSCEN | S3C2410_ADCCON_PRSCVL(s3c_adc_cfg.presc&0xFF), base_addr+S3C2410_ADCCON); else writel(0, base_addr+S3C2410_ADCCON); /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status {*/ #ifdef ADC_TEST //open for debug tmp = readl(base_addr+S3C2410_ADCCON); printk("tmp = %04X!\n", tmp); tmp = tmp | 0x0080; printk("tmp = %04X!\n", tmp); writel(tmp, base_addr+S3C2410_ADCCON); #endif /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status }*/ /* Initialise registers */ if ((s3c_adc_cfg.delay&0xffff) > 0) writel(s3c_adc_cfg.delay & 0xffff, base_addr+S3C2410_ADCDLY); ret = misc_register(&s3c_adc_miscdev); if (ret) { printk (KERN_ERR "cannot register miscdev on minor=%d (%d)\n", ADC_MINOR, ret); return ret; } printk(KERN_INFO "S3C ADC driver successfully probed !\n"); return 0; } static int s3c_adc_remove(struct device *dev) { printk(KERN_INFO "s3c_adc_remove() of TS called !\n"); return 0; } #define s3c_adc_suspend NULL #define s3c_adc_resume NULL static struct platform_driver s3c_adc_driver = { .probe = s3c_adc_probe, .remove = s3c_adc_remove, .suspend = s3c_adc_suspend, .resume = s3c_adc_resume, .driver = { .owner = THIS_MODULE, .name = "s3c2410-adc", }, }; static char banner[] __initdata = KERN_INFO "S3C ADC driver, (c) 2007 Samsung Electronics\n"; int __init s3c_adc_init(void) { int ireturn = 0; int read_pin = -99; int i = 0; printk(banner); ireturn = platform_driver_register(&s3c_adc_driver); #ifdef CONFIG_QISDA_AS090B00_EVT1 /* Qisda, ShiYong Lin, 2009/09/12, Indication for /PGOOD pin {*/ s3c2410_gpio_cfgpin(S3C2410_GPH5, 0); s3c2410_gpio_pullup(S3C2410_GPH5, 2); /* Qisda, ShiYong Lin, 2009/09/12, Indication for /PGOOD pin }*/ #endif /* Qisda, ShiYong Lin, 2009/10/27, Modify for exception in IsWallCharger function {*/ /* Qisda, ShiYong Lin, 2009/09/29, Set En1 and En2 {*/ for(i=0;i<10;++i) { read_pin = readl(S3C2410_GPGDAT) & (1<<1); if(read_pin != 0){ if(IsWallCharger()==0){ charging_source = 0; //EN2 s3c2410_gpio_cfgpin(S3C2410_GPB0, S3C2410_GPB0_OUTP); s3c2410_gpio_setpin(S3C2410_GPB0, 0); //EN1 s3c2410_gpio_cfgpin(S3C2410_GPB4, S3C2410_GPB4_OUTP); s3c2410_gpio_setpin(S3C2410_GPB4, 1); } else { charging_source = 1; //EN2 s3c2410_gpio_cfgpin(S3C2410_GPB0, S3C2410_GPB0_OUTP); s3c2410_gpio_setpin(S3C2410_GPB0, 1); //EN1 s3c2410_gpio_cfgpin(S3C2410_GPB4, S3C2410_GPB4_OUTP); s3c2410_gpio_setpin(S3C2410_GPB4, 0); } } else{ charging_source = 2; //EN2 s3c2410_gpio_cfgpin(S3C2410_GPB0, S3C2410_GPB0_OUTP); s3c2410_gpio_setpin(S3C2410_GPB0, 0); //EN1 s3c2410_gpio_cfgpin(S3C2410_GPB4, S3C2410_GPB4_OUTP); s3c2410_gpio_setpin(S3C2410_GPB4, 1); } } /*}, Qisda, ShiYong Lin, 2009/09/29, Set En1 and En2 */ /*}, Qisda, ShiYong Lin, 2009/10/27, Modify for exception in IsWallCharger function */ /* Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status {*/ #ifdef ADC_TEST { int i=0; for(i=0; i<10 ;i++) { udelay(1000); s3c_adc_read(); } printk("Read ADC times = %d\n",i); } #else s3c_adc_read(); #endif /*}, Qisda, ShiYong Lin, 2009/07/18, ADC and Battery Status */ return ireturn; } void __exit s3c_adc_exit(void) { platform_driver_unregister(&s3c_adc_driver); } module_init(s3c_adc_init); module_exit(s3c_adc_exit); MODULE_AUTHOR("alinuxguy@samsung.com>"); MODULE_DESCRIPTION("s3c adc driver"); MODULE_LICENSE("GPL");