2 * Copyright (C) 2007 Atmel Corporation
4 * Driver for the AT32AP700X PS/2 controller (PSIF).
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License version 2 as published
8 * by the Free Software Foundation.
10 #include <linux/kernel.h>
11 #include <linux/module.h>
12 #include <linux/device.h>
13 #include <linux/init.h>
14 #include <linux/serio.h>
15 #include <linux/interrupt.h>
16 #include <linux/delay.h>
17 #include <linux/err.h>
19 #include <linux/clk.h>
20 #include <linux/platform_device.h>
21 #include <linux/slab.h>
23 /* PSIF register offsets */
33 /* Bitfields in control register. */
34 #define PSIF_CR_RXDIS_OFFSET 1
35 #define PSIF_CR_RXDIS_SIZE 1
36 #define PSIF_CR_RXEN_OFFSET 0
37 #define PSIF_CR_RXEN_SIZE 1
38 #define PSIF_CR_SWRST_OFFSET 15
39 #define PSIF_CR_SWRST_SIZE 1
40 #define PSIF_CR_TXDIS_OFFSET 9
41 #define PSIF_CR_TXDIS_SIZE 1
42 #define PSIF_CR_TXEN_OFFSET 8
43 #define PSIF_CR_TXEN_SIZE 1
45 /* Bitfields in interrupt disable, enable, mask and status register. */
46 #define PSIF_NACK_OFFSET 8
47 #define PSIF_NACK_SIZE 1
48 #define PSIF_OVRUN_OFFSET 5
49 #define PSIF_OVRUN_SIZE 1
50 #define PSIF_PARITY_OFFSET 9
51 #define PSIF_PARITY_SIZE 1
52 #define PSIF_RXRDY_OFFSET 4
53 #define PSIF_RXRDY_SIZE 1
54 #define PSIF_TXEMPTY_OFFSET 1
55 #define PSIF_TXEMPTY_SIZE 1
56 #define PSIF_TXRDY_OFFSET 0
57 #define PSIF_TXRDY_SIZE 1
59 /* Bitfields in prescale register. */
60 #define PSIF_PSR_PRSCV_OFFSET 0
61 #define PSIF_PSR_PRSCV_SIZE 12
63 /* Bitfields in receive hold register. */
64 #define PSIF_RHR_RXDATA_OFFSET 0
65 #define PSIF_RHR_RXDATA_SIZE 8
67 /* Bitfields in transmit hold register. */
68 #define PSIF_THR_TXDATA_OFFSET 0
69 #define PSIF_THR_TXDATA_SIZE 8
71 /* Bit manipulation macros */
72 #define PSIF_BIT(name) \
73 (1 << PSIF_##name##_OFFSET)
75 #define PSIF_BF(name, value) \
76 (((value) & ((1 << PSIF_##name##_SIZE) - 1)) \
77 << PSIF_##name##_OFFSET)
79 #define PSIF_BFEXT(name, value) \
80 (((value) >> PSIF_##name##_OFFSET) \
81 & ((1 << PSIF_##name##_SIZE) - 1))
83 #define PSIF_BFINS(name, value, old) \
84 (((old) & ~(((1 << PSIF_##name##_SIZE) - 1) \
85 << PSIF_##name##_OFFSET)) \
86 | PSIF_BF(name, value))
88 /* Register access macros */
89 #define psif_readl(port, reg) \
90 __raw_readl((port)->regs + PSIF_##reg)
92 #define psif_writel(port, reg, value) \
93 __raw_writel((value), (port)->regs + PSIF_##reg)
96 struct platform_device
*pdev
;
102 /* Prevent concurrent writes to PSIF THR. */
106 static irqreturn_t
psif_interrupt(int irq
, void *_ptr
)
108 struct psif
*psif
= _ptr
;
109 int retval
= IRQ_NONE
;
110 unsigned int io_flags
= 0;
111 unsigned long status
;
113 status
= psif_readl(psif
, SR
);
115 if (status
& PSIF_BIT(RXRDY
)) {
116 unsigned char val
= (unsigned char) psif_readl(psif
, RHR
);
118 if (status
& PSIF_BIT(PARITY
))
119 io_flags
|= SERIO_PARITY
;
120 if (status
& PSIF_BIT(OVRUN
))
121 dev_err(&psif
->pdev
->dev
, "overrun read error\n");
123 serio_interrupt(psif
->io
, val
, io_flags
);
125 retval
= IRQ_HANDLED
;
131 static int psif_write(struct serio
*io
, unsigned char val
)
133 struct psif
*psif
= io
->port_data
;
138 spin_lock_irqsave(&psif
->lock
, flags
);
140 while (!(psif_readl(psif
, SR
) & PSIF_BIT(TXEMPTY
)) && timeout
--)
144 psif_writel(psif
, THR
, val
);
146 dev_dbg(&psif
->pdev
->dev
, "timeout writing to THR\n");
150 spin_unlock_irqrestore(&psif
->lock
, flags
);
155 static int psif_open(struct serio
*io
)
157 struct psif
*psif
= io
->port_data
;
160 retval
= clk_enable(psif
->pclk
);
164 psif_writel(psif
, CR
, PSIF_BIT(CR_TXEN
) | PSIF_BIT(CR_RXEN
));
165 psif_writel(psif
, IER
, PSIF_BIT(RXRDY
));
172 static void psif_close(struct serio
*io
)
174 struct psif
*psif
= io
->port_data
;
178 psif_writel(psif
, IDR
, ~0UL);
179 psif_writel(psif
, CR
, PSIF_BIT(CR_TXDIS
) | PSIF_BIT(CR_RXDIS
));
181 clk_disable(psif
->pclk
);
184 static void psif_set_prescaler(struct psif
*psif
)
187 unsigned long rate
= clk_get_rate(psif
->pclk
);
189 /* PRSCV = Pulse length (100 us) * PSIF module frequency. */
190 prscv
= 100 * (rate
/ 1000000UL);
192 if (prscv
> ((1<<PSIF_PSR_PRSCV_SIZE
) - 1)) {
193 prscv
= (1<<PSIF_PSR_PRSCV_SIZE
) - 1;
194 dev_dbg(&psif
->pdev
->dev
, "pclk too fast, "
195 "prescaler set to max\n");
198 clk_enable(psif
->pclk
);
199 psif_writel(psif
, PSR
, prscv
);
200 clk_disable(psif
->pclk
);
203 static int __init
psif_probe(struct platform_device
*pdev
)
205 struct resource
*regs
;
212 psif
= kzalloc(sizeof(struct psif
), GFP_KERNEL
);
214 dev_dbg(&pdev
->dev
, "out of memory\n");
220 io
= kzalloc(sizeof(struct serio
), GFP_KERNEL
);
222 dev_dbg(&pdev
->dev
, "out of memory\n");
228 regs
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
230 dev_dbg(&pdev
->dev
, "no mmio resources defined\n");
235 psif
->regs
= ioremap(regs
->start
, resource_size(regs
));
238 dev_dbg(&pdev
->dev
, "could not map I/O memory\n");
242 pclk
= clk_get(&pdev
->dev
, "pclk");
244 dev_dbg(&pdev
->dev
, "could not get peripheral clock\n");
250 /* Reset the PSIF to enter at a known state. */
251 ret
= clk_enable(pclk
);
253 dev_dbg(&pdev
->dev
, "could not enable pclk\n");
256 psif_writel(psif
, CR
, PSIF_BIT(CR_SWRST
));
259 irq
= platform_get_irq(pdev
, 0);
261 dev_dbg(&pdev
->dev
, "could not get irq\n");
265 ret
= request_irq(irq
, psif_interrupt
, IRQF_SHARED
, "at32psif", psif
);
267 dev_dbg(&pdev
->dev
, "could not request irq %d\n", irq
);
272 io
->id
.type
= SERIO_8042
;
273 io
->write
= psif_write
;
274 io
->open
= psif_open
;
275 io
->close
= psif_close
;
276 snprintf(io
->name
, sizeof(io
->name
), "AVR32 PS/2 port%d", pdev
->id
);
277 snprintf(io
->phys
, sizeof(io
->phys
), "at32psif/serio%d", pdev
->id
);
278 io
->port_data
= psif
;
279 io
->dev
.parent
= &pdev
->dev
;
281 psif_set_prescaler(psif
);
283 spin_lock_init(&psif
->lock
);
284 serio_register_port(psif
->io
);
285 platform_set_drvdata(pdev
, psif
);
287 dev_info(&pdev
->dev
, "Atmel AVR32 PSIF PS/2 driver on 0x%08x irq %d\n",
288 (int)psif
->regs
, psif
->irq
);
304 static int __exit
psif_remove(struct platform_device
*pdev
)
306 struct psif
*psif
= platform_get_drvdata(pdev
);
308 psif_writel(psif
, IDR
, ~0UL);
309 psif_writel(psif
, CR
, PSIF_BIT(CR_TXDIS
) | PSIF_BIT(CR_RXDIS
));
311 serio_unregister_port(psif
->io
);
313 free_irq(psif
->irq
, psif
);
317 platform_set_drvdata(pdev
, NULL
);
323 static int psif_suspend(struct platform_device
*pdev
, pm_message_t state
)
325 struct psif
*psif
= platform_get_drvdata(pdev
);
328 psif_writel(psif
, CR
, PSIF_BIT(CR_RXDIS
) | PSIF_BIT(CR_TXDIS
));
329 clk_disable(psif
->pclk
);
335 static int psif_resume(struct platform_device
*pdev
)
337 struct psif
*psif
= platform_get_drvdata(pdev
);
340 clk_enable(psif
->pclk
);
341 psif_set_prescaler(psif
);
342 psif_writel(psif
, CR
, PSIF_BIT(CR_RXEN
) | PSIF_BIT(CR_TXEN
));
348 #define psif_suspend NULL
349 #define psif_resume NULL
352 static struct platform_driver psif_driver
= {
353 .remove
= __exit_p(psif_remove
),
355 .name
= "atmel_psif",
356 .owner
= THIS_MODULE
,
358 .suspend
= psif_suspend
,
359 .resume
= psif_resume
,
362 static int __init
psif_init(void)
364 return platform_driver_probe(&psif_driver
, psif_probe
);
367 static void __exit
psif_exit(void)
369 platform_driver_unregister(&psif_driver
);
372 module_init(psif_init
);
373 module_exit(psif_exit
);
375 MODULE_AUTHOR("Hans-Christian Egtvedt <hans-christian.egtvedt@atmel.com>");
376 MODULE_DESCRIPTION("Atmel AVR32 PSIF PS/2 driver");
377 MODULE_LICENSE("GPL");