--- linux-2.6.22.5/arch/arm/common/gta01_pm_gsm.c.orig 2007-09-27 11:23:20.000000000 -0500 +++ linux-2.6.22.5/arch/arm/common/gta01_pm_gsm.c 2007-09-27 21:09:00.000000000 -0500 @@ -23,6 +23,9 @@ #include <asm/arch/gta01.h> #include <asm/arch/gta02.h> +#include <linux/serial_core.h> +void s3c24xx_set_flow_control(struct uart_port *port, int fc_on); + struct gta01pm_priv { int gpio_ngsm_en; struct console *con; @@ -49,6 +52,23 @@ static ssize_t gsm_read(struct device *dev, struct device_attribute *attr, char *buf) { + struct uart_driver *udrive = NULL; + struct uart_state *ustate = NULL; + struct uart_port *uport = NULL; + + if (gta01_gsm.con) { + udrive = gta01_gsm.con->data; + if (udrive) { + ustate = udrive->state; + if (ustate) { + uport = ustate->port; + } + } + } + printk("gsm: gsm_read of \"%s\"\n", attr->attr.name); + printk("gsm: con=%p udrive=%p ustate=%p uport=%p\n", + gta01_gsm.con, udrive, ustate, uport); + if (!strcmp(attr->attr.name, "power_on")) { if (s3c2410_gpio_getpin(GTA01_GPIO_MODEM_ON)) goto out_1; @@ -58,6 +78,24 @@ } else if (!strcmp(attr->attr.name, "download")) { if (s3c2410_gpio_getpin(GTA01_GPIO_MODEM_DNLOAD)) goto out_1; + } else if (!strcmp(attr->attr.name, "flowcontrol")) { + if (uport) { + if (uport->unused[2] & 0x2) { + if (uport->unused[2] & 0x1) { + printk("gsm: flow control allowed and on\n"); + goto out_1; + } else { + printk("gsm: flow control allowed and not on\n"); + } + } else { + if (uport->unused[2] & 0x1) + printk("gsm: flow control not allowed, " + "but is pending\n"); + else + printk("gsm: flow control not allowed, " + "not pending\n"); + } + } } return strlcpy(buf, "0\n", 3); @@ -68,10 +106,26 @@ static ssize_t gsm_write(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { + struct uart_driver *udrive = NULL; + struct uart_state *ustate = NULL; + struct uart_port *uport = NULL; + static int was_on = 0; unsigned long on = simple_strtoul(buf, NULL, 10); + if (gta01_gsm.con) { + udrive = gta01_gsm.con->data; + if (udrive) { + ustate = udrive->state; + if (ustate) { + uport = ustate->port; + } + } + } + printk("gsm: con=%p udrive=%p ustate=%p uport=%p\n", + gta01_gsm.con, udrive, ustate, uport); + if (!strcmp(attr->attr.name, "power_on")) { - if (on) { + if (on && !was_on) { if (gta01_gsm.con) { dev_info(dev, "powering up GSM, thus " "disconnecting serial console\n"); @@ -83,7 +137,24 @@ s3c2410_gpio_setpin(gta01_gsm.gpio_ngsm_en, 0); s3c2410_gpio_setpin(GTA01_GPIO_MODEM_ON, 1); - } else { + + if (uport) { + /* set any pending flow-control mode */ + uport->unused[2] |= 0x2; + s3c24xx_set_flow_control(uport, + (uport->unused[2] & 0x1)); + dev_info(dev, "flow control allowed\n"); + } + + was_on = 1; + + } else if (!on && was_on) { + if (uport) { + uport->unused[2] &= ~0x2; + s3c24xx_set_flow_control(uport, 0); + dev_info(dev, "flow control not allowed\n"); + } + s3c2410_gpio_setpin(GTA01_GPIO_MODEM_ON, 0); if (gta01_gsm.gpio_ngsm_en) @@ -95,6 +166,8 @@ dev_info(dev, "powered down GSM, thus enabling " "serial console\n"); } + + was_on = 0; } } else if (!strcmp(attr->attr.name, "reset")) { s3c2410_gpio_setpin(GTA01_GPIO_MODEM_RST, on); @@ -105,6 +178,7 @@ return count; } +static DEVICE_ATTR(flowcontrol, 0644, gsm_read, gsm_write); static DEVICE_ATTR(power_on, 0644, gsm_read, gsm_write); static DEVICE_ATTR(reset, 0644, gsm_read, gsm_write); static DEVICE_ATTR(download, 0644, gsm_read, gsm_write); @@ -136,6 +210,7 @@ #endif static struct attribute *gta01_gsm_sysfs_entries[] = { + &dev_attr_flowcontrol.attr, &dev_attr_power_on.attr, &dev_attr_reset.attr, NULL, --- linux-2.6.22.5/drivers/serial/s3c2410.c.orig 2007-09-27 11:23:27.000000000 -0500 +++ linux-2.6.22.5/drivers/serial/s3c2410.c 2007-09-27 21:07:21.000000000 -0500 @@ -80,6 +80,8 @@ #include <asm/arch/regs-serial.h> #include <asm/arch/regs-gpio.h> +#include <asm/mach-types.h> +#include <asm/arch/gta01.h> /* structures */ @@ -729,6 +731,17 @@ return best->quot; } +/* This routine is called whenever the gta01 modem/console switches */ +void s3c24xx_set_flow_control(struct uart_port *port, int fc_on) +{ + unsigned int umcon; + if (machine_is_neo1973_gta01() && port) { + umcon = (fc_on) ? S3C2410_UMCOM_AFC : 0; + wr_regl(port, S3C2410_UMCON, umcon); + } +} +EXPORT_SYMBOL(s3c24xx_set_flow_control); + static void s3c24xx_serial_set_termios(struct uart_port *port, struct ktermios *termios, struct ktermios *old) @@ -803,6 +816,23 @@ umcon = (termios->c_cflag & CRTSCTS) ? S3C2410_UMCOM_AFC : 0; + /* + * Custom handling of flow control on hwport 0 for the GTA01: + * Save the desired state for flow control, but if the port + * is being used as a console, then do not actually enable + * flow control unless the flag permiting us to do so is set. + */ + if (machine_is_neo1973_gta01() && (cfg->hwport == 0)) { + if (umcon) + port->unused[2] |= 0x1; + else + port->unused[2] &= ~0x1; + if (port->cons && (port->cons->index >= 0)) { + if (!(port->unused[2] & 0x2)) + umcon = 0; + } + } + if (termios->c_cflag & PARENB) { if (termios->c_cflag & PARODD) ulcon |= S3C2410_LCON_PODD;