Re: [PATCH v4][bfin-sport-uart] fit blackfin uart over sport driver into common uart inftrastructure

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Sorry, forget to remove
Signed-off-by: Michael Frysinger <michael.frysinger@xxxxxxxxxx>
here

Sonic

On Fri, Sep 18, 2009 at 5:08 PM, sonic zhang <sonic.adi@xxxxxxxxx> wrote:
> Fit blackfin uart over sport driver into common uart inftrastructure.
>
> 1. Enable sport uart driver to change uart baud, data bit, stop bit at
> runtime. Bind the index of uart device nodes to physical index of sports.
>
> 2. Move most platform data into arch specific board files.
>
> 3. Console is registered in sport uart driver as well.
>
> 4. Remove 500 us block waiting in sport tx stop code by putting a dummy
> data into tx fifo to make sure the sport tx stops when all bytes are
> shifted out except for the dummy data.
>
> 5. clean up a bit and fix up coding style.
>
> Signed-off-by: Michael Frysinger <michael.frysinger@xxxxxxxxxx>
> Signed-off-by: Sonic Zhang <sonic.zhang@xxxxxxxxxx>
> ---
>  drivers/serial/Kconfig           |   24 ++
>  drivers/serial/bfin_sport_uart.c |  757 ++++++++++++++++++++++++++------------
>  drivers/serial/bfin_sport_uart.h |   38 +-
>  3 files changed, 568 insertions(+), 251 deletions(-)
>
> diff --git a/drivers/serial/Kconfig b/drivers/serial/Kconfig
> index 03422ce..45f63a8 100644
> --- a/drivers/serial/Kconfig
> +++ b/drivers/serial/Kconfig
> @@ -1451,6 +1451,30 @@ config SPORT_BAUD_RATE
>        default 19200 if (SERIAL_SPORT_BAUD_RATE_19200)
>        default 9600 if (SERIAL_SPORT_BAUD_RATE_9600)
>
> +config SERIAL_BFIN_SPORT0_UART
> +       bool "Enable UART over SPORT0"
> +       depends on SERIAL_BFIN_SPORT
> +       help
> +         Enable UART over SPORT0
> +
> +config SERIAL_BFIN_SPORT1_UART
> +       bool "Enable UART over SPORT1"
> +       depends on SERIAL_BFIN_SPORT
> +       help
> +         Enable UART over SPORT1
> +
> +config SERIAL_BFIN_SPORT2_UART
> +       bool "Enable UART over SPORT2"
> +       depends on SERIAL_BFIN_SPORT && (BF54x || BF538 || BF539)
> +       help
> +         Enable UART over SPORT2
> +
> +config SERIAL_BFIN_SPORT3_UART
> +       bool "Enable UART over SPORT3"
> +       depends on SERIAL_BFIN_SPORT && (BF54x || BF538 || BF539)
> +       help
> +         Enable UART over SPORT3
> +
>  config SERIAL_TIMBERDALE
>        tristate "Support for timberdale UART"
>        depends on MFD_TIMBERDALE
> diff --git a/drivers/serial/bfin_sport_uart.c b/drivers/serial/bfin_sport_uart.c
> index c108b1a..3bbc00d 100644
> --- a/drivers/serial/bfin_sport_uart.c
> +++ b/drivers/serial/bfin_sport_uart.c
> @@ -1,27 +1,11 @@
>  /*
> - * File:       linux/drivers/serial/bfin_sport_uart.c
> + * Blackfin On-Chip Sport Emulated UART Driver
>  *
> - * Based on:   drivers/serial/bfin_5xx.c by Aubrey Li.
> - * Author:     Roy Huang <roy.huang@xxxxxxxxxx>
> + * Copyright 2006-2009 Analog Devices Inc.
>  *
> - * Created:    Nov 22, 2006
> - * Copyright:  (c) 2006-2007 Analog Devices Inc.
> - * Description: this driver enable SPORTs on Blackfin emulate UART.
> + * Enter bugs at http://blackfin.uclinux.org/
>  *
> - * 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 the file COPYING, or write
> - * to the Free Software Foundation, Inc.,
> - * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
> + * Licensed under the GPL-2 or later.
>  */
>
>  /*
> @@ -29,39 +13,18 @@
>  * http://www.analog.com/UploadedFiles/Application_Notes/399447663EE191.pdf
>  * This application note describe how to implement a UART on a Sharc DSP,
>  * but this driver is implemented on Blackfin Processor.
> + * Transmit Frame Sync is not used by this driver to transfer data out.
>  */
>
> -/* After reset, there is a prelude of low level pulse when transmit data first
> - * time. No addtional pulse in following transmit.
> - * According to document:
> - * The SPORTs are ready to start transmitting or receiving data no later than
> - * three serial clock cycles after they are enabled in the SPORTx_TCR1 or
> - * SPORTx_RCR1 register. No serial clock cycles are lost from this point on.
> - * The first internal frame sync will occur one frame sync delay after the
> - * SPORTs are ready. External frame syncs can occur as soon as the SPORT is
> - * ready.
> - */
> +/* #define DEBUG */
>
> -/* Thanks to Axel Alatalo <axel@xxxxxxxxx> for fixing sport rx bug. Sometimes
> - * sport receives data incorrectly. The following is Axel's words.
> - * As EE-191, sport rx samples 3 times of the UART baudrate and takes the
> - * middle smaple of every 3 samples as the data bit. For a 8-N-1 UART setting,
> - * 30 samples will be required for a byte. If transmitter sends a 1/3 bit short
> - * byte due to buadrate drift, then the 30th sample of a byte, this sample is
> - * also the third sample of the stop bit, will happens on the immediately
> - * following start bit which will be thrown away and missed. Thus since parts
> - * of the startbit will be missed and the receiver will begin to drift, the
> - * effect accumulates over time until synchronization is lost.
> - * If only require 2 samples of the stopbit (by sampling in total 29 samples),
> - * then a to short byte as in the case above will be tolerated. Then the 1/3
> - * early startbit will trigger a framesync since the last read is complete
> - * after only 2/3 stopbit and framesync is active during the last 1/3 looking
> - * for a possible early startbit. */
> -
> -//#define DEBUG
> +#define DRV_NAME "bfin-sport-uart"
> +#define DEVICE_NAME    "ttySS"
> +#define pr_fmt(fmt) DRV_NAME ": " fmt
>
>  #include <linux/module.h>
>  #include <linux/ioport.h>
> +#include <linux/io.h>
>  #include <linux/init.h>
>  #include <linux/console.h>
>  #include <linux/sysrq.h>
> @@ -75,23 +38,36 @@
>
>  #include "bfin_sport_uart.h"
>
> +#ifdef CONFIG_SERIAL_BFIN_SPORT0_UART
>  unsigned short bfin_uart_pin_req_sport0[] =
>        {P_SPORT0_TFS, P_SPORT0_DTPRI, P_SPORT0_TSCLK, P_SPORT0_RFS, \
>         P_SPORT0_DRPRI, P_SPORT0_RSCLK, P_SPORT0_DRSEC, P_SPORT0_DTSEC, 0};
> -
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT1_UART
>  unsigned short bfin_uart_pin_req_sport1[] =
>        {P_SPORT1_TFS, P_SPORT1_DTPRI, P_SPORT1_TSCLK, P_SPORT1_RFS, \
>        P_SPORT1_DRPRI, P_SPORT1_RSCLK, P_SPORT1_DRSEC, P_SPORT1_DTSEC, 0};
> -
> -#define DRV_NAME "bfin-sport-uart"
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT2_UART
> +unsigned short bfin_uart_pin_req_sport2[] =
> +       {P_SPORT2_TFS, P_SPORT2_DTPRI, P_SPORT2_TSCLK, P_SPORT2_RFS, \
> +       P_SPORT2_DRPRI, P_SPORT2_RSCLK, P_SPORT2_DRSEC, P_SPORT2_DTSEC, 0};
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT3_UART
> +unsigned short bfin_uart_pin_req_sport3[] =
> +       {P_SPORT3_TFS, P_SPORT3_DTPRI, P_SPORT3_TSCLK, P_SPORT3_RFS, \
> +       P_SPORT3_DRPRI, P_SPORT3_RSCLK, P_SPORT3_DRSEC, P_SPORT3_DTSEC, 0};
> +#endif
>
>  struct sport_uart_port {
>        struct uart_port        port;
> -       char                    *name;
> -
> -       int                     tx_irq;
> -       int                     rx_irq;
>        int                     err_irq;
> +       unsigned short          csize;
> +       unsigned short          rxmask;
> +       unsigned short          txmask1;
> +       unsigned short          txmask2;
> +       unsigned char           stopb;
> +/*     unsigned char           parib; */
>  };
>
>  static void sport_uart_tx_chars(struct sport_uart_port *up);
> @@ -99,36 +75,42 @@ static void sport_stop_tx(struct uart_port *port);
>
>  static inline void tx_one_byte(struct sport_uart_port *up, unsigned int value)
>  {
> -       pr_debug("%s value:%x\n", __func__, value);
> -       /* Place a Start and Stop bit */
> +       pr_debug("%s value:%x, mask1=0x%x, mask2=0x%x\n", __func__, value,
> +               up->txmask1, up->txmask2);
> +
> +       /* Place Start and Stop bits */
>        __asm__ __volatile__ (
> -               "R2 = b#01111111100;"
> -               "R3 = b#10000000001;"
> -               "%0 <<= 2;"
> -               "%0 = %0 & R2;"
> -               "%0 = %0 | R3;"
> -               : "=d"(value)
> -               : "d"(value)
> -               : "ASTAT", "R2", "R3"
> +               "%[val] <<= 1;"
> +               "%[val] = %[val] & %[mask1];"
> +               "%[val] = %[val] | %[mask2];"
> +               : [val]"+d"(value)
> +               : [mask1]"d"(up->txmask1), [mask2]"d"(up->txmask2)
> +               : "ASTAT"
>        );
>        pr_debug("%s value:%x\n", __func__, value);
>
>        SPORT_PUT_TX(up, value);
>  }
>
> -static inline unsigned int rx_one_byte(struct sport_uart_port *up)
> +static inline unsigned char rx_one_byte(struct sport_uart_port *up)
>  {
> -       unsigned int value, extract;
> +       unsigned int value;
> +       unsigned char extract;
>        u32 tmp_mask1, tmp_mask2, tmp_shift, tmp;
>
> -       value = SPORT_GET_RX32(up);
> -       pr_debug("%s value:%x\n", __func__, value);
> +       if ((up->csize + up->stopb) > 7)
> +               value = SPORT_GET_RX32(up);
> +       else
> +               value = SPORT_GET_RX(up);
> +
> +       pr_debug("%s value:%x, cs=%d, mask=0x%x\n", __func__, value,
> +               up->csize, up->rxmask);
>
> -       /* Extract 8 bits data */
> +       /* Extract data */
>        __asm__ __volatile__ (
>                "%[extr] = 0;"
> -               "%[mask1] = 0x1801(Z);"
> -               "%[mask2] = 0x0300(Z);"
> +               "%[mask1] = %[rxmask];"
> +               "%[mask2] = 0x0200(Z);"
>                "%[shift] = 0;"
>                "LSETUP(.Lloop_s, .Lloop_e) LC0 = %[lc];"
>                ".Lloop_s:"
> @@ -138,9 +120,9 @@ static inline unsigned int rx_one_byte(struct sport_uart_port *up)
>                "%[mask1] = %[mask1] - %[mask2];"
>                ".Lloop_e:"
>                "%[shift] += 1;"
> -               : [val]"=d"(value), [extr]"=d"(extract), [shift]"=d"(tmp_shift), [tmp]"=d"(tmp),
> -                 [mask1]"=d"(tmp_mask1), [mask2]"=d"(tmp_mask2)
> -               : "d"(value), [lc]"a"(8)
> +               : [extr]"=&d"(extract), [shift]"=&d"(tmp_shift), [tmp]"=&d"(tmp),
> +                 [mask1]"=&d"(tmp_mask1), [mask2]"=&d"(tmp_mask2)
> +               : [val]"d"(value), [rxmask]"d"(up->rxmask), [lc]"a"(up->csize)
>                : "ASTAT", "LB0", "LC0", "LT0"
>        );
>
> @@ -148,29 +130,28 @@ static inline unsigned int rx_one_byte(struct sport_uart_port *up)
>        return extract;
>  }
>
> -static int sport_uart_setup(struct sport_uart_port *up, int sclk, int baud_rate)
> +static int sport_uart_setup(struct sport_uart_port *up, int size, int baud_rate)
>  {
> -       int tclkdiv, tfsdiv, rclkdiv;
> +       int tclkdiv, rclkdiv;
> +       unsigned int sclk = get_sclk();
>
> -       /* Set TCR1 and TCR2 */
> -       SPORT_PUT_TCR1(up, (LATFS | ITFS | TFSR | TLSBIT | ITCLK));
> -       SPORT_PUT_TCR2(up, 10);
> +       /* Set TCR1 and TCR2, TFSR is not enabled for uart */
> +       SPORT_PUT_TCR1(up, (ITFS | TLSBIT | ITCLK));
> +       SPORT_PUT_TCR2(up, size + 1);
>        pr_debug("%s TCR1:%x, TCR2:%x\n", __func__, SPORT_GET_TCR1(up), SPORT_GET_TCR2(up));
>
>        /* Set RCR1 and RCR2 */
>        SPORT_PUT_RCR1(up, (RCKFE | LARFS | LRFS | RFSR | IRCLK));
> -       SPORT_PUT_RCR2(up, 28);
> +       SPORT_PUT_RCR2(up, (size + 1) * 2 - 1);
>        pr_debug("%s RCR1:%x, RCR2:%x\n", __func__, SPORT_GET_RCR1(up), SPORT_GET_RCR2(up));
>
> -       tclkdiv = sclk/(2 * baud_rate) - 1;
> -       tfsdiv = 12;
> -       rclkdiv = sclk/(2 * baud_rate * 3) - 1;
> +       tclkdiv = sclk / (2 * baud_rate) - 1;
> +       rclkdiv = sclk / (2 * baud_rate * 2) - 1;
>        SPORT_PUT_TCLKDIV(up, tclkdiv);
> -       SPORT_PUT_TFSDIV(up, tfsdiv);
>        SPORT_PUT_RCLKDIV(up, rclkdiv);
>        SSYNC();
> -       pr_debug("%s sclk:%d, baud_rate:%d, tclkdiv:%d, tfsdiv:%d, rclkdiv:%d\n",
> -                       __func__, sclk, baud_rate, tclkdiv, tfsdiv, rclkdiv);
> +       pr_debug("%s sclk:%d, baud_rate:%d, tclkdiv:%d, rclkdiv:%d\n",
> +                       __func__, sclk, baud_rate, tclkdiv, rclkdiv);
>
>        return 0;
>  }
> @@ -181,23 +162,29 @@ static irqreturn_t sport_uart_rx_irq(int irq, void *dev_id)
>        struct tty_struct *tty = up->port.info->port.tty;
>        unsigned int ch;
>
> -       do {
> +       spin_lock(&up->port.lock);
> +
> +       while (SPORT_GET_STAT(up) & RXNE) {
>                ch = rx_one_byte(up);
>                up->port.icount.rx++;
>
> -               if (uart_handle_sysrq_char(&up->port, ch))
> -                       ;
> -               else
> +               if (!uart_handle_sysrq_char(&up->port, ch))
>                        tty_insert_flip_char(tty, ch, TTY_NORMAL);
> -       } while (SPORT_GET_STAT(up) & RXNE);
> +       }
>        tty_flip_buffer_push(tty);
>
> +       spin_unlock(&up->port.lock);
> +
>        return IRQ_HANDLED;
>  }
>
>  static irqreturn_t sport_uart_tx_irq(int irq, void *dev_id)
>  {
> -       sport_uart_tx_chars(dev_id);
> +       struct sport_uart_port *up = dev_id;
> +
> +       spin_lock(&up->port.lock);
> +       sport_uart_tx_chars(up);
> +       spin_unlock(&up->port.lock);
>
>        return IRQ_HANDLED;
>  }
> @@ -208,6 +195,8 @@ static irqreturn_t sport_uart_err_irq(int irq, void *dev_id)
>        struct tty_struct *tty = up->port.info->port.tty;
>        unsigned int stat = SPORT_GET_STAT(up);
>
> +       spin_lock(&up->port.lock);
> +
>        /* Overflow in RX FIFO */
>        if (stat & ROVF) {
>                up->port.icount.overrun++;
> @@ -216,15 +205,16 @@ static irqreturn_t sport_uart_err_irq(int irq, void *dev_id)
>        }
>        /* These should not happen */
>        if (stat & (TOVF | TUVF | RUVF)) {
> -               printk(KERN_ERR "SPORT Error:%s %s %s\n",
> -                               (stat & TOVF)?"TX overflow":"",
> -                               (stat & TUVF)?"TX underflow":"",
> -                               (stat & RUVF)?"RX underflow":"");
> +               pr_err("SPORT Error:%s %s %s\n",
> +                      (stat & TOVF) ? "TX overflow" : "",
> +                      (stat & TUVF) ? "TX underflow" : "",
> +                      (stat & RUVF) ? "RX underflow" : "");
>                SPORT_PUT_TCR1(up, SPORT_GET_TCR1(up) & ~TSPEN);
>                SPORT_PUT_RCR1(up, SPORT_GET_RCR1(up) & ~RSPEN);
>        }
>        SSYNC();
>
> +       spin_unlock(&up->port.lock);
>        return IRQ_HANDLED;
>  }
>
> @@ -232,60 +222,37 @@ static irqreturn_t sport_uart_err_irq(int irq, void *dev_id)
>  static int sport_startup(struct uart_port *port)
>  {
>        struct sport_uart_port *up = (struct sport_uart_port *)port;
> -       char buffer[20];
> -       int retval;
> +       int ret;
>
>        pr_debug("%s enter\n", __func__);
> -       snprintf(buffer, 20, "%s rx", up->name);
> -       retval = request_irq(up->rx_irq, sport_uart_rx_irq, IRQF_SAMPLE_RANDOM, buffer, up);
> -       if (retval) {
> -               printk(KERN_ERR "Unable to request interrupt %s\n", buffer);
> -               return retval;
> +       ret = request_irq(up->port.irq, sport_uart_rx_irq, 0,
> +               "SPORT_UART_RX", up);
> +       if (ret) {
> +               dev_err(port->dev, "unable to request SPORT RX interrupt\n");
> +               return ret;
>        }
>
> -       snprintf(buffer, 20, "%s tx", up->name);
> -       retval = request_irq(up->tx_irq, sport_uart_tx_irq, IRQF_SAMPLE_RANDOM, buffer, up);
> -       if (retval) {
> -               printk(KERN_ERR "Unable to request interrupt %s\n", buffer);
> +       ret = request_irq(up->port.irq+1, sport_uart_tx_irq, 0,
> +               "SPORT_UART_TX", up);
> +       if (ret) {
> +               dev_err(port->dev, "unable to request SPORT TX interrupt\n");
>                goto fail1;
>        }
>
> -       snprintf(buffer, 20, "%s err", up->name);
> -       retval = request_irq(up->err_irq, sport_uart_err_irq, IRQF_SAMPLE_RANDOM, buffer, up);
> -       if (retval) {
> -               printk(KERN_ERR "Unable to request interrupt %s\n", buffer);
> +       ret = request_irq(up->err_irq, sport_uart_err_irq, 0,
> +               "SPORT_UART_STATUS", up);
> +       if (ret) {
> +               dev_err(port->dev, "unable to request SPORT status interrupt\n");
>                goto fail2;
>        }
>
> -       if (port->line) {
> -               if (peripheral_request_list(bfin_uart_pin_req_sport1, DRV_NAME))
> -                       goto fail3;
> -       } else {
> -               if (peripheral_request_list(bfin_uart_pin_req_sport0, DRV_NAME))
> -                       goto fail3;
> -       }
> -
> -       sport_uart_setup(up, get_sclk(), port->uartclk);
> -
> -       /* Enable receive interrupt */
> -       SPORT_PUT_RCR1(up, (SPORT_GET_RCR1(up) | RSPEN));
> -       SSYNC();
> -
>        return 0;
> + fail2:
> +       free_irq(up->port.irq+1, up);
> + fail1:
> +       free_irq(up->port.irq, up);
>
> -
> -fail3:
> -       printk(KERN_ERR DRV_NAME
> -               ": Requesting Peripherals failed\n");
> -
> -       free_irq(up->err_irq, up);
> -fail2:
> -       free_irq(up->tx_irq, up);
> -fail1:
> -       free_irq(up->rx_irq, up);
> -
> -       return retval;
> -
> +       return ret;
>  }
>
>  static void sport_uart_tx_chars(struct sport_uart_port *up)
> @@ -344,20 +311,17 @@ static void sport_set_mctrl(struct uart_port *port, unsigned int mctrl)
>  static void sport_stop_tx(struct uart_port *port)
>  {
>        struct sport_uart_port *up = (struct sport_uart_port *)port;
> -       unsigned int stat;
>
>        pr_debug("%s enter\n", __func__);
>
> -       stat = SPORT_GET_STAT(up);
> -       while(!(stat & TXHRE)) {
> -               udelay(1);
> -               stat = SPORT_GET_STAT(up);
> -       }
>        /* Although the hold register is empty, last byte is still in shift
> -        * register and not sent out yet. If baud rate is lower than default,
> -        * delay should be longer. For example, if the baud rate is 9600,
> -        * the delay must be at least 2ms by experience */
> -       udelay(500);
> +        * register and not sent out yet. So, put a dummy data into TX FIFO.
> +        * Then, sport tx stops when last byte is shift out and the dummy
> +        * data is moved into the shift register.
> +        */
> +       SPORT_PUT_TX(up, 0xffff);
> +       while (!(SPORT_GET_STAT(up) & TXHRE))
> +               cpu_relax();
>
>        SPORT_PUT_TCR1(up, (SPORT_GET_TCR1(up) & ~TSPEN));
>        SSYNC();
> @@ -370,6 +334,7 @@ static void sport_start_tx(struct uart_port *port)
>        struct sport_uart_port *up = (struct sport_uart_port *)port;
>
>        pr_debug("%s enter\n", __func__);
> +
>        /* Write data into SPORT FIFO before enable SPROT to transmit */
>        sport_uart_tx_chars(up);
>
> @@ -403,37 +368,24 @@ static void sport_shutdown(struct uart_port *port)
>  {
>        struct sport_uart_port *up = (struct sport_uart_port *)port;
>
> -       pr_debug("%s enter\n", __func__);
> +       dev_dbg(port->dev, "%s enter\n", __func__);
>
>        /* Disable sport */
>        SPORT_PUT_TCR1(up, (SPORT_GET_TCR1(up) & ~TSPEN));
>        SPORT_PUT_RCR1(up, (SPORT_GET_RCR1(up) & ~RSPEN));
>        SSYNC();
>
> -       if (port->line) {
> -               peripheral_free_list(bfin_uart_pin_req_sport1);
> -       } else {
> -               peripheral_free_list(bfin_uart_pin_req_sport0);
> -       }
> -
> -       free_irq(up->rx_irq, up);
> -       free_irq(up->tx_irq, up);
> +       free_irq(up->port.irq, up);
> +       free_irq(up->port.irq+1, up);
>        free_irq(up->err_irq, up);
>  }
>
> -static void sport_set_termios(struct uart_port *port,
> -               struct ktermios *termios, struct ktermios *old)
> -{
> -       pr_debug("%s enter, c_cflag:%08x\n", __func__, termios->c_cflag);
> -       uart_update_timeout(port, CS8 ,port->uartclk);
> -}
> -
>  static const char *sport_type(struct uart_port *port)
>  {
>        struct sport_uart_port *up = (struct sport_uart_port *)port;
>
>        pr_debug("%s enter\n", __func__);
> -       return up->name;
> +       return up->port.type == PORT_BFIN_SPORT ? "BFIN-SPORT-UART" : NULL;
>  }
>
>  static void sport_release_port(struct uart_port *port)
> @@ -461,6 +413,110 @@ static int sport_verify_port(struct uart_port *port, struct serial_struct *ser)
>        return 0;
>  }
>
> +static void sport_set_termios(struct uart_port *port,
> +               struct ktermios *termios, struct ktermios *old)
> +{
> +       struct sport_uart_port *up = (struct sport_uart_port *)port;
> +       unsigned long flags;
> +       int i;
> +
> +       pr_debug("%s enter, c_cflag:%08x\n", __func__, termios->c_cflag);
> +
> +       switch (termios->c_cflag & CSIZE) {
> +       case CS8:
> +               up->csize = 8;
> +               break;
> +       case CS7:
> +               up->csize = 7;
> +               break;
> +       case CS6:
> +               up->csize = 6;
> +               break;
> +       case CS5:
> +               up->csize = 5;
> +               break;
> +       default:
> +               pr_warning("requested word length not supported\n");
> +       }
> +
> +       if (termios->c_cflag & CSTOPB) {
> +               up->stopb = 1;
> +       }
> +       if (termios->c_cflag & PARENB) {
> +               pr_warning("PAREN bits is not supported yet\n");
> +               /* up->parib = 1; */
> +       }
> +
> +       port->read_status_mask = OE;
> +       if (termios->c_iflag & INPCK)
> +               port->read_status_mask |= (FE | PE);
> +       if (termios->c_iflag & (BRKINT | PARMRK))
> +               port->read_status_mask |= BI;
> +
> +       /*
> +        * Characters to ignore
> +        */
> +       port->ignore_status_mask = 0;
> +       if (termios->c_iflag & IGNPAR)
> +               port->ignore_status_mask |= FE | PE;
> +       if (termios->c_iflag & IGNBRK) {
> +               port->ignore_status_mask |= BI;
> +               /*
> +                * If we're ignoring parity and break indicators,
> +                * ignore overruns too (for real raw support).
> +                */
> +               if (termios->c_iflag & IGNPAR)
> +                       port->ignore_status_mask |= OE;
> +       }
> +
> +       /* RX extract mask */
> +       up->rxmask = 0x01 | (((up->csize + up->stopb) * 2 - 1) << 0x8);
> +       /* TX masks, 8 bit data and 1 bit stop for example:
> +        * mask1 = b#0111111110
> +        * mask2 = b#1000000000
> +        */
> +       for (i = 0, up->txmask1 = 0; i < up->csize; i++)
> +               up->txmask1 |= (1<<i);
> +       up->txmask2 = (1<<i);
> +       if (up->stopb) {
> +               ++i;
> +               up->txmask2 |= (1<<i);
> +       }
> +       up->txmask1 <<= 1;
> +       up->txmask2 <<= 1;
> +       /* uart baud rate */
> +       port->uartclk = uart_get_baud_rate(port, termios, old, 0, get_sclk()/16);
> +
> +       spin_lock_irqsave(&up->port.lock, flags);
> +
> +       /* Disable UART */
> +       SPORT_PUT_TCR1(up, SPORT_GET_TCR1(up) & ~TSPEN);
> +       SPORT_PUT_RCR1(up, SPORT_GET_RCR1(up) & ~RSPEN);
> +
> +       sport_uart_setup(up, up->csize + up->stopb, port->uartclk);
> +
> +       /* driver TX line high after config, one dummy data is
> +        * necessary to stop sport after shift one byte
> +        */
> +       SPORT_PUT_TX(up, 0xffff);
> +       SPORT_PUT_TX(up, 0xffff);
> +       SPORT_PUT_TCR1(up, (SPORT_GET_TCR1(up) | TSPEN));
> +       SSYNC();
> +       while (!(SPORT_GET_STAT(up) & TXHRE))
> +               cpu_relax();
> +       SPORT_PUT_TCR1(up, SPORT_GET_TCR1(up) & ~TSPEN);
> +       SSYNC();
> +
> +       /* Port speed changed, update the per-port timeout. */
> +       uart_update_timeout(port, termios->c_cflag, port->uartclk);
> +
> +       /* Enable sport rx */
> +       SPORT_PUT_RCR1(up, SPORT_GET_RCR1(up) | RSPEN);
> +       SSYNC();
> +
> +       spin_unlock_irqrestore(&up->port.lock, flags);
> +}
> +
>  struct uart_ops sport_uart_ops = {
>        .tx_empty       = sport_tx_empty,
>        .set_mctrl      = sport_set_mctrl,
> @@ -480,104 +536,327 @@ struct uart_ops sport_uart_ops = {
>        .verify_port    = sport_verify_port,
>  };
>
> -static struct sport_uart_port sport_uart_ports[] = {
> -       { /* SPORT 0 */
> -               .name   = "SPORT0",
> -               .tx_irq = IRQ_SPORT0_TX,
> -               .rx_irq = IRQ_SPORT0_RX,
> -               .err_irq= IRQ_SPORT0_ERROR,
> -               .port   = {
> -                       .type           = PORT_BFIN_SPORT,
> -                       .iotype         = UPIO_MEM,
> -                       .membase        = (void __iomem *)SPORT0_TCR1,
> -                       .mapbase        = SPORT0_TCR1,
> -                       .irq            = IRQ_SPORT0_RX,
> -                       .uartclk        = CONFIG_SPORT_BAUD_RATE,
> -                       .fifosize       = 8,
> -                       .ops            = &sport_uart_ops,
> -                       .line           = 0,
> -               },
> -       }, { /* SPORT 1 */
> -               .name   = "SPORT1",
> -               .tx_irq = IRQ_SPORT1_TX,
> -               .rx_irq = IRQ_SPORT1_RX,
> -               .err_irq= IRQ_SPORT1_ERROR,
> -               .port   = {
> -                       .type           = PORT_BFIN_SPORT,
> -                       .iotype         = UPIO_MEM,
> -                       .membase        = (void __iomem *)SPORT1_TCR1,
> -                       .mapbase        = SPORT1_TCR1,
> -                       .irq            = IRQ_SPORT1_RX,
> -                       .uartclk        = CONFIG_SPORT_BAUD_RATE,
> -                       .fifosize       = 8,
> -                       .ops            = &sport_uart_ops,
> -                       .line           = 1,
> -               },
> +#define BFIN_SPORT_UART_MAX_PORTS 4
> +
> +static struct sport_uart_port bfin_sport_uart_ports[BFIN_SPORT_UART_MAX_PORTS];
> +
> +static unsigned long bfin_sport_uart_console_base_addr[] = {
> +#ifdef CONFIG_SERIAL_BFIN_SPORT0_UART
> +               SPORT0_TCR1,
> +#else
> +               0,
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT1_UART
> +               SPORT1_TCR1,
> +#else
> +               0,
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT2_UART
> +               SPORT2_TCR1,
> +#else
> +               0,
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT3_UART
> +               SPORT3_TCR1,
> +#else
> +               0,
> +#endif
> +};
> +
> +static int __init sport_uart_init_ports(void)
> +{
> +       static int first = 1;
> +       int i, ret;
> +
> +       if (!first)
> +               return 0;
> +
> +#ifdef CONFIG_SERIAL_BFIN_SPORT0_UART
> +       ret = peripheral_request_list(bfin_uart_pin_req_sport0, DRV_NAME);
> +       if (ret) {
> +               pr_err("requesting SPORT0 peripherals failed\n");
> +               goto err_out0;
> +       }
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT1_UART
> +       ret = peripheral_request_list(bfin_uart_pin_req_sport1, DRV_NAME);
> +       if (ret) {
> +               pr_err("requesting SPORT1 peripherals failed\n");
> +               goto err_out1;
> +       }
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT2_UART
> +       ret = peripheral_request_list(bfin_uart_pin_req_sport2, DRV_NAME);
> +       if (ret) {
> +               pr_err("requesting SPORT2 peripherals failed\n");
> +               goto err_out2;
> +       }
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT3_UART
> +       ret = peripheral_request_list(bfin_uart_pin_req_sport3, DRV_NAME);
> +       if (ret) {
> +               pr_err("requesting SPORT3 peripherals failed\n");
> +               goto err_out3;
> +       }
> +#endif
> +       for (i = 0; i < BFIN_SPORT_UART_MAX_PORTS; i++) {
> +               spin_lock_init(&bfin_sport_uart_ports[i].port.lock);
> +               bfin_sport_uart_ports[i].port.fifosize  = SPORT_TX_FIFO_SIZE,
> +               bfin_sport_uart_ports[i].port.ops       = &sport_uart_ops;
> +               bfin_sport_uart_ports[i].port.line      = i;
> +               bfin_sport_uart_ports[i].port.iotype    = UPIO_MEM;
> +               bfin_sport_uart_ports[i].port.flags     = UPF_BOOT_AUTOCONF;
> +               bfin_sport_uart_ports[i].port.membase   =
> +                       (void __iomem *)bfin_sport_uart_console_base_addr[i];
> +       }
> +
> +       first = 0;
> +
> +       return 0;
> +#ifdef CONFIG_SERIAL_BFIN_SPORT3_UART
> +err_out3:
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT2_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport2);
> +err_out2:
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT1_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport1);
> +err_out1:
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT0_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport0);
> +err_out0:
> +#endif
> +
> +       return ret;
> +}
> +
> +#ifdef CONFIG_SERIAL_BFIN_SPORT_CONSOLE
> +static int __init
> +sport_uart_console_setup(struct console *co, char *options)
> +{
> +       struct sport_uart_port *up;
> +       int baud = 57600;
> +       int bits = 8;
> +       int parity = 'n';
> +       int flow = 'n';
> +
> +       /* Check whether an invalid uart number has been specified */
> +       if (co->index == -1 || co->index >= BFIN_SPORT_UART_MAX_PORTS)
> +               return -ENODEV;
> +
> +       up = &bfin_sport_uart_ports[co->index];
> +
> +       if (options)
> +               uart_parse_options(options, &baud, &parity, &bits, &flow);
> +
> +       return uart_set_options(&up->port, co, baud, parity, bits, flow);
> +}
> +
> +static void sport_uart_console_putchar(struct uart_port *port, int ch)
> +{
> +       struct sport_uart_port *up = (struct sport_uart_port *)port;
> +
> +       while (SPORT_GET_STAT(up) & TXF)
> +               barrier();
> +
> +       tx_one_byte(up, ch);
> +}
> +
> +/*
> + * Interrupts are disabled on entering
> + */
> +static void
> +sport_uart_console_write(struct console *co, const char *s, unsigned int count)
> +{
> +       struct sport_uart_port *up = &bfin_sport_uart_ports[co->index];
> +       unsigned long flags;
> +
> +       spin_lock_irqsave(&up->port.lock, flags);
> +
> +       if (SPORT_GET_TCR1(up) & TSPEN)
> +               uart_console_write(&up->port, s, count, sport_uart_console_putchar);
> +       else {
> +               /* dummy data to start sport */
> +               while (SPORT_GET_STAT(up) & TXF)
> +                       barrier();
> +               SPORT_PUT_TX(up, 0xffff);
> +               /* Enable transmit, then an interrupt will generated */
> +               SPORT_PUT_TCR1(up, (SPORT_GET_TCR1(up) | TSPEN));
> +               SSYNC();
> +
> +               uart_console_write(&up->port, s, count, sport_uart_console_putchar);
> +
> +               /* Although the hold register is empty, last byte is still in shift
> +                * register and not sent out yet. So, put a dummy data into TX FIFO.
> +                * Then, sport tx stops when last byte is shift out and the dummy
> +                * data is moved into the shift register.
> +                */
> +               while (SPORT_GET_STAT(up) & TXF)
> +                       barrier();
> +               SPORT_PUT_TX(up, 0xffff);
> +               while (!(SPORT_GET_STAT(up) & TXHRE))
> +                       barrier();
> +
> +               /* Stop sport tx transfer */
> +               SPORT_PUT_TCR1(up, (SPORT_GET_TCR1(up) & ~TSPEN));
> +               SSYNC();
>        }
> +
> +       spin_unlock_irqrestore(&up->port.lock, flags);
> +}
> +
> +static struct uart_driver sport_uart_reg;
> +
> +static struct console sport_uart_console = {
> +       .name           = DEVICE_NAME,
> +       .write          = sport_uart_console_write,
> +       .device         = uart_console_device,
> +       .setup          = sport_uart_console_setup,
> +       .flags          = CON_PRINTBUFFER,
> +       .index          = -1,
> +       .data           = &sport_uart_reg,
>  };
>
> +static int __init sport_uart_rs_console_init(void)
> +{
> +       int ret = sport_uart_init_ports();
> +       if (ret)
> +               return ret;
> +
> +       register_console(&sport_uart_console);
> +
> +       return 0;
> +}
> +console_initcall(sport_uart_rs_console_init);
> +
> +#define SPORT_UART_CONSOLE     (&sport_uart_console)
> +#else
> +#define SPORT_UART_CONSOLE     NULL
> +#endif /* CONFIG_SERIAL_BFIN_CONSOLE */
> +
> +
>  static struct uart_driver sport_uart_reg = {
>        .owner          = THIS_MODULE,
> -       .driver_name    = "SPORT-UART",
> -       .dev_name       = "ttySS",
> +       .driver_name    = DRV_NAME,
> +       .dev_name       = DEVICE_NAME,
>        .major          = 204,
>        .minor          = 84,
> -       .nr             = ARRAY_SIZE(sport_uart_ports),
> -       .cons           = NULL,
> +       .nr             = BFIN_SPORT_UART_MAX_PORTS,
> +       .cons           = SPORT_UART_CONSOLE,
>  };
>
> -static int sport_uart_suspend(struct platform_device *dev, pm_message_t state)
> +#ifdef CONFIG_PM
> +static int sport_uart_suspend(struct device *dev)
>  {
> -       struct sport_uart_port *sport = platform_get_drvdata(dev);
> +       struct sport_uart_port *sport = dev_get_drvdata(dev);
>
> -       pr_debug("%s enter\n", __func__);
> +       dev_dbg(dev, "%s enter\n", __func__);
>        if (sport)
>                uart_suspend_port(&sport_uart_reg, &sport->port);
>
>        return 0;
>  }
>
> -static int sport_uart_resume(struct platform_device *dev)
> +static int sport_uart_resume(struct device *dev)
>  {
> -       struct sport_uart_port *sport = platform_get_drvdata(dev);
> +       struct sport_uart_port *sport = dev_get_drvdata(dev);
>
> -       pr_debug("%s enter\n", __func__);
> +       dev_dbg(dev, "%s enter\n", __func__);
>        if (sport)
>                uart_resume_port(&sport_uart_reg, &sport->port);
>
>        return 0;
>  }
>
> -static int sport_uart_probe(struct platform_device *dev)
> +static struct dev_pm_ops bfin_sport_uart_dev_pm_ops = {
> +       .suspend        = sport_uart_suspend,
> +       .resume         = sport_uart_resume,
> +};
> +#endif
> +
> +static int __devinit sport_uart_probe(struct platform_device *pdev)
>  {
> -       pr_debug("%s enter\n", __func__);
> -       sport_uart_ports[dev->id].port.dev = &dev->dev;
> -       uart_add_one_port(&sport_uart_reg, &sport_uart_ports[dev->id].port);
> -       platform_set_drvdata(dev, &sport_uart_ports[dev->id]);
> +       struct resource *res;
> +       struct sport_uart_port *sport;
> +       int ret = 0;
> +       int index;
>
> -       return 0;
> +       dev_dbg(&pdev->dev, "%s enter\n", __func__);
> +
> +       res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> +       if (res == NULL) {
> +               dev_err(&pdev->dev, "Cannot get IORESOURCE_MEM\n");
> +               return -ENOENT;
> +       }
> +
> +       for (index = 0; index < BFIN_SPORT_UART_MAX_PORTS; index++)
> +               if (res->start == bfin_sport_uart_console_base_addr[index])
> +                       break;
> +
> +       if (index == BFIN_SPORT_UART_MAX_PORTS) {
> +               dev_err(&pdev->dev, "Wrong sport uart platform device\n");
> +               return -ENOENT;
> +       }
> +
> +       sport = &bfin_sport_uart_ports[index];
> +
> +       sport->port.membase = ioremap(res->start, res->end - res->start);
> +       if (!sport->port.membase) {
> +               dev_err(&pdev->dev, "Cannot map sport IO\n");
> +               return -ENXIO;
> +       }
> +
> +       sport->port.irq = platform_get_irq(pdev, 0);
> +       if (sport->port.irq < 0) {
> +               dev_err(&pdev->dev, "No sport RX/TX IRQ specified\n");
> +               ret = -ENOENT;
> +               goto out_error;
> +       }
> +
> +       sport->err_irq = platform_get_irq(pdev, 1);
> +       if (sport->err_irq < 0) {
> +               dev_err(&pdev->dev, "No sport status IRQ specified\n");
> +               ret = -ENOENT;
> +               goto out_error;
> +       }
> +
> +       sport->port.dev = &pdev->dev;
> +       dev_set_drvdata(&pdev->dev, sport);
> +       ret = uart_add_one_port(&sport_uart_reg, &sport->port);
> +       if (!ret)
> +               return 0;
> +
> +out_error:
> +       iounmap(sport->port.membase);
> +
> +       return ret;
>  }
>
> -static int sport_uart_remove(struct platform_device *dev)
> +static int __devexit sport_uart_remove(struct platform_device *pdev)
>  {
> -       struct sport_uart_port *sport = platform_get_drvdata(dev);
> +       struct sport_uart_port *sport = platform_get_drvdata(pdev);
>
> -       pr_debug("%s enter\n", __func__);
> -       platform_set_drvdata(dev, NULL);
> +       dev_dbg(&pdev->dev, "%s enter\n", __func__);
> +       dev_set_drvdata(&pdev->dev, NULL);
>
>        if (sport)
>                uart_remove_one_port(&sport_uart_reg, &sport->port);
>
> +       iounmap(sport->port.membase);
> +
>        return 0;
>  }
>
>  static struct platform_driver sport_uart_driver = {
>        .probe          = sport_uart_probe,
> -       .remove         = sport_uart_remove,
> -       .suspend        = sport_uart_suspend,
> -       .resume         = sport_uart_resume,
> +       .remove         = __devexit_p(sport_uart_remove),
>        .driver         = {
>                .name   = DRV_NAME,
> +#ifdef CONFIG_PM
> +               .pm     = &bfin_sport_uart_dev_pm_ops,
> +#endif
>        },
>  };
>
> @@ -585,33 +864,49 @@ static int __init sport_uart_init(void)
>  {
>        int ret;
>
> -       pr_debug("%s enter\n", __func__);
> +       pr_info("Serial: Blackfin uart over sport driver\n");
> +
> +       ret = sport_uart_init_ports();
> +       if (ret)
> +               return ret;
> +
>        ret = uart_register_driver(&sport_uart_reg);
> -       if (ret != 0) {
> -               printk(KERN_ERR "Failed to register %s:%d\n",
> +       if (ret) {
> +               pr_err("failed to register %s:%d\n",
>                                sport_uart_reg.driver_name, ret);
>                return ret;
>        }
>
>        ret = platform_driver_register(&sport_uart_driver);
> -       if (ret != 0) {
> -               printk(KERN_ERR "Failed to register sport uart driver:%d\n", ret);
> +       if (ret) {
> +               pr_err("failed to register sport uart driver:%d\n", ret);
>                uart_unregister_driver(&sport_uart_reg);
>        }
>
> -
> -       pr_debug("%s exit\n", __func__);
>        return ret;
>  }
> +module_init(sport_uart_init);
>
>  static void __exit sport_uart_exit(void)
>  {
> -       pr_debug("%s enter\n", __func__);
>        platform_driver_unregister(&sport_uart_driver);
>        uart_unregister_driver(&sport_uart_reg);
> -}
>
> -module_init(sport_uart_init);
> +#ifdef CONFIG_SERIAL_BFIN_SPORT0_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport0);
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT1_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport1);
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT2_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport2);
> +#endif
> +#ifdef CONFIG_SERIAL_BFIN_SPORT3_UART
> +       peripheral_free_list(bfin_uart_pin_req_sport3);
> +#endif
> +}
>  module_exit(sport_uart_exit);
>
> +MODULE_AUTHOR("Sonic Zhang, Roy Huang");
> +MODULE_DESCRIPTION("Blackfin serial over SPORT driver");
>  MODULE_LICENSE("GPL");
> diff --git a/drivers/serial/bfin_sport_uart.h b/drivers/serial/bfin_sport_uart.h
> index 46e793e..19d814f 100644
> --- a/drivers/serial/bfin_sport_uart.h
> +++ b/drivers/serial/bfin_sport_uart.h
> @@ -1,29 +1,23 @@
>  /*
> - * File:       linux/drivers/serial/bfin_sport_uart.h
> + * Blackfin On-Chip Sport Emulated UART Driver
>  *
> - * Based on:   include/asm-blackfin/mach-533/bfin_serial_5xx.h
> - * Author:     Roy Huang <roy.huang>analog.com>
> + * Copyright 2006-2008 Analog Devices Inc.
>  *
> - * Created:    Nov 22, 2006
> - * Copyright:  (C) Analog Device Inc.
> - * Description: this driver enable SPORTs on Blackfin emulate UART.
> + * Enter bugs at http://blackfin.uclinux.org/
>  *
> - * 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 the file COPYING, or write
> - * to the Free Software Foundation, Inc.,
> - * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
> + * Licensed under the GPL-2 or later.
>  */
>
> +/*
> + * This driver and the hardware supported are in term of EE-191 of ADI.
> + * http://www.analog.com/UploadedFiles/Application_Notes/399447663EE191.pdf
> + * This application note describe how to implement a UART on a Sharc DSP,
> + * but this driver is implemented on Blackfin Processor.
> + * Transmit Frame Sync is not used by this driver to transfer data out.
> + */
> +
> +#ifndef _BFIN_SPORT_UART_H
> +#define _BFIN_SPORT_UART_H
>
>  #define OFFSET_TCR1            0x00    /* Transmit Configuration 1 Register */
>  #define OFFSET_TCR2            0x04    /* Transmit Configuration 2 Register */
> @@ -74,3 +68,7 @@
>  #define SPORT_PUT_RCLKDIV(sport, v)    bfin_write16(((sport)->port.membase + OFFSET_RCLKDIV), v)
>  #define SPORT_PUT_RFSDIV(sport, v)     bfin_write16(((sport)->port.membase + OFFSET_RFSDIV), v)
>  #define SPORT_PUT_STAT(sport, v)       bfin_write16(((sport)->port.membase + OFFSET_STAT), v)
> +
> +#define SPORT_TX_FIFO_SIZE     8
> +
> +#endif /* _BFIN_SPORT_UART_H */
> --
> 1.6.0
>
>
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
> Please read the FAQ at  http://www.tux.org/lkml/
>
--
To unsubscribe from this list: send the line "unsubscribe linux-serial" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Kernel Newbies]     [Security]     [Netfilter]     [Bugtraq]     [Linux PPP]     [Linux FS]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Video 4 Linux]     [Linmodem]     [Device Mapper]     [Linux Kernel for ARM]

  Powered by Linux