aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/tty/serial/sh-sci.c
diff options
context:
space:
mode:
authorPaul Mundt <lethal@linux-sh.org>2011-06-28 02:25:36 -0400
committerPaul Mundt <lethal@linux-sh.org>2011-06-28 02:25:36 -0400
commit9174fc8f111982e024a00512c521ad8f1056fccb (patch)
treee8cd3f0bea432af9d970152b451eb96448000157 /drivers/tty/serial/sh-sci.c
parent23241d43eac88f63a7f0bf4d5c12bbc496651585 (diff)
serial: sh-sci: Fix up pretty name printing for port IRQs.
Presently these were all using the same static string with no regard to dev_name() and the like. This implements a bit of rework to name the IRQ dynamically, as it should have been doing all along anyways. Signed-off-by: Paul Mundt <lethal@linux-sh.org>
Diffstat (limited to 'drivers/tty/serial/sh-sci.c')
-rw-r--r--drivers/tty/serial/sh-sci.c118
1 files changed, 85 insertions, 33 deletions
diff --git a/drivers/tty/serial/sh-sci.c b/drivers/tty/serial/sh-sci.c
index 9c8624d9c803..d0a56235c50e 100644
--- a/drivers/tty/serial/sh-sci.c
+++ b/drivers/tty/serial/sh-sci.c
@@ -71,6 +71,8 @@ struct sci_port {
71 /* Function clock */ 71 /* Function clock */
72 struct clk *fclk; 72 struct clk *fclk;
73 73
74 char *irqstr[SCIx_NR_IRQS];
75
74 struct dma_chan *chan_tx; 76 struct dma_chan *chan_tx;
75 struct dma_chan *chan_rx; 77 struct dma_chan *chan_rx;
76 78
@@ -954,53 +956,102 @@ static int sci_notifier(struct notifier_block *self,
954 return NOTIFY_OK; 956 return NOTIFY_OK;
955} 957}
956 958
959static struct sci_irq_desc {
960 const char *desc;
961 irq_handler_t handler;
962} sci_irq_desc[] = {
963 /*
964 * Split out handlers, the default case.
965 */
966 [SCIx_ERI_IRQ] = {
967 .desc = "rx err",
968 .handler = sci_er_interrupt,
969 },
970
971 [SCIx_RXI_IRQ] = {
972 .desc = "rx full",
973 .handler = sci_rx_interrupt,
974 },
975
976 [SCIx_TXI_IRQ] = {
977 .desc = "tx empty",
978 .handler = sci_tx_interrupt,
979 },
980
981 [SCIx_BRI_IRQ] = {
982 .desc = "break",
983 .handler = sci_br_interrupt,
984 },
985
986 /*
987 * Special muxed handler.
988 */
989 [SCIx_MUX_IRQ] = {
990 .desc = "mux",
991 .handler = sci_mpxed_interrupt,
992 },
993};
994
957static int sci_request_irq(struct sci_port *port) 995static int sci_request_irq(struct sci_port *port)
958{ 996{
959 int i; 997 struct uart_port *up = &port->port;
960 irqreturn_t (*handlers[4])(int irq, void *ptr) = { 998 int i, j, ret = 0;
961 sci_er_interrupt, sci_rx_interrupt, sci_tx_interrupt, 999
962 sci_br_interrupt, 1000 for (i = j = 0; i < SCIx_NR_IRQS; i++, j++) {
963 }; 1001 struct sci_irq_desc *desc;
964 const char *desc[] = { "SCI Receive Error", "SCI Receive Data Full", 1002 unsigned int irq;
965 "SCI Transmit Data Empty", "SCI Break" }; 1003
966 1004 if (SCIx_IRQ_IS_MUXED(port)) {
967 if (port->cfg->irqs[0] == port->cfg->irqs[1]) { 1005 i = SCIx_MUX_IRQ;
968 if (unlikely(!port->cfg->irqs[0])) 1006 irq = up->irq;
969 return -ENODEV; 1007 } else
970 1008 irq = port->cfg->irqs[i];
971 if (request_irq(port->cfg->irqs[0], sci_mpxed_interrupt, 1009
972 IRQF_DISABLED, "sci", port)) { 1010 desc = sci_irq_desc + i;
973 dev_err(port->port.dev, "Can't allocate IRQ\n"); 1011 port->irqstr[j] = kasprintf(GFP_KERNEL, "%s:%s",
974 return -ENODEV; 1012 dev_name(up->dev), desc->desc);
1013 if (!port->irqstr[j]) {
1014 dev_err(up->dev, "Failed to allocate %s IRQ string\n",
1015 desc->desc);
1016 goto out_nomem;
975 } 1017 }
976 } else { 1018
977 for (i = 0; i < ARRAY_SIZE(handlers); i++) { 1019 ret = request_irq(irq, desc->handler, up->irqflags,
978 if (unlikely(!port->cfg->irqs[i])) 1020 port->irqstr[j], port);
979 continue; 1021 if (unlikely(ret)) {
980 1022 dev_err(up->dev, "Can't allocate %s IRQ\n", desc->desc);
981 if (request_irq(port->cfg->irqs[i], handlers[i], 1023 goto out_noirq;
982 IRQF_DISABLED, desc[i], port)) {
983 dev_err(port->port.dev, "Can't allocate IRQ\n");
984 return -ENODEV;
985 }
986 } 1024 }
987 } 1025 }
988 1026
989 return 0; 1027 return 0;
1028
1029out_noirq:
1030 while (--i >= 0)
1031 free_irq(port->cfg->irqs[i], port);
1032
1033out_nomem:
1034 while (--j >= 0)
1035 kfree(port->irqstr[j]);
1036
1037 return ret;
990} 1038}
991 1039
992static void sci_free_irq(struct sci_port *port) 1040static void sci_free_irq(struct sci_port *port)
993{ 1041{
994 int i; 1042 int i;
995 1043
996 if (port->cfg->irqs[0] == port->cfg->irqs[1]) 1044 /*
997 free_irq(port->cfg->irqs[0], port); 1045 * Intentionally in reverse order so we iterate over the muxed
998 else { 1046 * IRQ first.
999 for (i = 0; i < ARRAY_SIZE(port->cfg->irqs); i++) { 1047 */
1000 if (!port->cfg->irqs[i]) 1048 for (i = 0; i < SCIx_NR_IRQS; i++) {
1001 continue; 1049 free_irq(port->cfg->irqs[i], port);
1050 kfree(port->irqstr[i]);
1002 1051
1003 free_irq(port->cfg->irqs[i], port); 1052 if (SCIx_IRQ_IS_MUXED(port)) {
1053 /* If there's only one IRQ, we're done. */
1054 return;
1004 } 1055 }
1005 } 1056 }
1006} 1057}
@@ -1910,6 +1961,7 @@ static int __devinit sci_init_single(struct platform_device *dev,
1910 * For the muxed case there's nothing more to do. 1961 * For the muxed case there's nothing more to do.
1911 */ 1962 */
1912 port->irq = p->irqs[SCIx_RXI_IRQ]; 1963 port->irq = p->irqs[SCIx_RXI_IRQ];
1964 port->irqflags = IRQF_DISABLED;
1913 1965
1914 port->serial_in = sci_serial_in; 1966 port->serial_in = sci_serial_in;
1915 port->serial_out = sci_serial_out; 1967 port->serial_out = sci_serial_out;