diff options
author | Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com> | 2011-07-29 11:35:50 +0800 |
---|---|---|
committer | Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com> | 2011-08-01 16:33:09 +0800 |
commit | c71a77ab878c3d9d14e91ea4fa5bd4ff8fe1c421 (patch) | |
tree | baa9253468338c34d06d8ebdfc56aa1e7abae4ba /drivers | |
parent | 3f59bab47cc40bb0b6076143cc2e6b96be396771 (diff) | |
download | barebox-c71a77ab878c3d9d14e91ea4fa5bd4ff8fe1c421.tar.gz barebox-c71a77ab878c3d9d14e91ea4fa5bd4ff8fe1c421.tar.xz |
ns16550: switch to resource
use generic read/write depending on the memory size
if no reg_read/write defined
Signed-off-by: Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/serial/serial_ns16550.c | 126 |
1 files changed, 89 insertions, 37 deletions
diff --git a/drivers/serial/serial_ns16550.c b/drivers/serial/serial_ns16550.c index 1dea7369e2..4ed2671ad8 100644 --- a/drivers/serial/serial_ns16550.c +++ b/drivers/serial/serial_ns16550.c @@ -49,6 +49,70 @@ /*********** Private Functions **********************************/ /** + * @brief read register + * + * @param[in] cdev pointer to console device + * @param[in] offset + * + * @return value + */ +static uint32_t ns16550_read(struct console_device *cdev, uint32_t off) +{ + struct device_d *dev = cdev->dev; + struct NS16550_plat *plat = (struct NS16550_plat *)dev->platform_data; + int width = dev->resource[0].flags & IORESOURCE_MEM_TYPE_MASK; + + off <<= plat->shift; + + if (plat->reg_read) + return plat->reg_read((unsigned long)dev->priv, off); + + switch (width) { + case IORESOURCE_MEM_8BIT: + return readb(dev->priv + off); + case IORESOURCE_MEM_16BIT: + return readw(dev->priv + off); + case IORESOURCE_MEM_32BIT: + return readl(dev->priv + off); + } + return -1; +} + +/** + * @brief write register + * + * @param[in] cdev pointer to console device + * @param[in] offset + * @param[in] val + */ +static void ns16550_write(struct console_device *cdev, uint32_t val, + uint32_t off) +{ + struct device_d *dev = cdev->dev; + struct NS16550_plat *plat = (struct NS16550_plat *)dev->platform_data; + int width = dev->resource[0].flags & IORESOURCE_MEM_TYPE_MASK; + + off <<= plat->shift; + + if (plat->reg_write) { + plat->reg_write(val, (unsigned long)dev->priv, off); + return; + } + + switch (width) { + case IORESOURCE_MEM_8BIT: + writeb(val & 0xff, dev->priv + off); + break; + case IORESOURCE_MEM_16BIT: + writew(val & 0xffff, dev->priv + off); + break; + case IORESOURCE_MEM_32BIT: + writel(val, dev->priv + off); + break; + } +} + +/** * @brief Compute the divisor for a baud rate * * @param[in] cdev pointer to console device @@ -74,27 +138,24 @@ static unsigned int ns16550_calc_divisor(struct console_device *cdev, */ static void ns16550_serial_init_port(struct console_device *cdev) { - struct NS16550_plat *plat = (struct NS16550_plat *) - cdev->dev->platform_data; - unsigned long base = cdev->dev->map_base; unsigned int baud_divisor; /* Setup the serial port with the defaults first */ baud_divisor = ns16550_calc_divisor(cdev, CONFIG_BAUDRATE); /* initializing the device for the first time */ - plat->reg_write(0x00, base, ier); + ns16550_write(cdev, 0x00, ier); #ifdef CONFIG_DRIVER_SERIAL_NS16550_OMAP_EXTENSIONS - plat->reg_write(0x07, base, mdr1); /* Disable */ + ns16550_write(cdev, 0x07, mdr1); /* Disable */ #endif - plat->reg_write(LCR_BKSE | LCRVAL, base, lcr); - plat->reg_write(baud_divisor & 0xFF, base, dll); - plat->reg_write((baud_divisor >> 8) & 0xff, base, dlm); - plat->reg_write(LCRVAL, base, lcr); - plat->reg_write(MCRVAL, base, mcr); - plat->reg_write(FCRVAL, base, fcr); + ns16550_write(cdev, LCR_BKSE | LCRVAL, lcr); + ns16550_write(cdev, baud_divisor & 0xFF, dll); + ns16550_write(cdev, (baud_divisor >> 8) & 0xff, dlm); + ns16550_write(cdev, LCRVAL, lcr); + ns16550_write(cdev, MCRVAL, mcr); + ns16550_write(cdev, FCRVAL, fcr); #ifdef CONFIG_DRIVER_SERIAL_NS16550_OMAP_EXTENSIONS - plat->reg_write(0x00, base, mdr1); + ns16550_write(cdev, 0x00, mdr1); #endif } @@ -108,12 +169,9 @@ static void ns16550_serial_init_port(struct console_device *cdev) */ static void ns16550_putc(struct console_device *cdev, char c) { - struct NS16550_plat *plat = (struct NS16550_plat *) - cdev->dev->platform_data; - unsigned long base = cdev->dev->map_base; /* Loop Doing Nothing */ - while ((plat->reg_read(base, lsr) & LSR_THRE) == 0) ; - plat->reg_write(c, base, thr); + while ((ns16550_read(cdev, lsr) & LSR_THRE) == 0) ; + ns16550_write(cdev, c, thr); } /** @@ -125,12 +183,9 @@ static void ns16550_putc(struct console_device *cdev, char c) */ static int ns16550_getc(struct console_device *cdev) { - struct NS16550_plat *plat = (struct NS16550_plat *) - cdev->dev->platform_data; - unsigned long base = cdev->dev->map_base; /* Loop Doing Nothing */ - while ((plat->reg_read(base, lsr) & LSR_DR) == 0) ; - return plat->reg_read(base, rbr); + while ((ns16550_read(cdev, lsr) & LSR_DR) == 0) ; + return ns16550_read(cdev, rbr); } /** @@ -142,10 +197,7 @@ static int ns16550_getc(struct console_device *cdev) */ static int ns16550_tstc(struct console_device *cdev) { - struct NS16550_plat *plat = (struct NS16550_plat *) - cdev->dev->platform_data; - unsigned long base = cdev->dev->map_base; - return ((plat->reg_read(base, lsr) & LSR_DR) != 0); + return ((ns16550_read(cdev, lsr) & LSR_DR) != 0); } /** @@ -158,17 +210,15 @@ static int ns16550_tstc(struct console_device *cdev) */ static int ns16550_setbaudrate(struct console_device *cdev, int baud_rate) { - struct NS16550_plat *plat = (struct NS16550_plat *) - cdev->dev->platform_data; - unsigned long base = cdev->dev->map_base; unsigned int baud_divisor = ns16550_calc_divisor(cdev, baud_rate); - plat->reg_write(0x00, base, ier); - plat->reg_write(LCR_BKSE, base, lcr); - plat->reg_write(baud_divisor & 0xff, base, dll); - plat->reg_write((baud_divisor >> 8) & 0xff, base, dlm); - plat->reg_write(LCRVAL, base, lcr); - plat->reg_write(MCRVAL, base, mcr); - plat->reg_write(FCRVAL, base, fcr); + + ns16550_write(cdev, 0x00, ier); + ns16550_write(cdev, LCR_BKSE, lcr); + ns16550_write(cdev, baud_divisor & 0xff, dll); + ns16550_write(cdev, (baud_divisor >> 8) & 0xff, dlm); + ns16550_write(cdev, LCRVAL, lcr); + ns16550_write(cdev, MCRVAL, mcr); + ns16550_write(cdev, FCRVAL, fcr); return 0; } @@ -189,8 +239,10 @@ static int ns16550_probe(struct device_d *dev) /* we do expect platform specific data */ if (plat == NULL) return -EINVAL; - if ((plat->reg_read == NULL) || (plat->reg_write == NULL)) + if (!(dev->resource[0].flags & IORESOURCE_MEM_TYPE_MASK) && + ((plat->reg_read == NULL) || (plat->reg_write == NULL))) return -EINVAL; + dev->priv = dev_request_mem_region(dev, 0); cdev = xzalloc(sizeof(*cdev)); |