| /* |
| * AVR power-management chip interface for the Buffalo Linkstation / |
| * Kurobox Platform. |
| * |
| * Author: 2006 (c) G. Liakhovetski |
| * g.liakhovetski@gmx.de |
| * |
| * This file is licensed under the terms of the GNU General Public License |
| * version 2. This program is licensed "as is" without any warranty of |
| * any kind, whether express or implied. |
| */ |
| #include <linux/workqueue.h> |
| #include <linux/string.h> |
| #include <linux/delay.h> |
| #include <linux/serial_reg.h> |
| #include <linux/serial_8250.h> |
| #include <linux/of.h> |
| #include <linux/of_address.h> |
| #include <asm/io.h> |
| #include <asm/termbits.h> |
| |
| #include "mpc10x.h" |
| |
| static void __iomem *avr_addr; |
| static unsigned long avr_clock; |
| |
| static struct work_struct wd_work; |
| |
| static void wd_stop(struct work_struct *unused) |
| { |
| const char string[] = "AAAAFFFFJJJJ>>>>VVVV>>>>ZZZZVVVVKKKK"; |
| int i = 0, rescue = 8; |
| int len = strlen(string); |
| |
| while (rescue--) { |
| int j; |
| char lsr = in_8(avr_addr + UART_LSR); |
| |
| if (lsr & (UART_LSR_THRE | UART_LSR_TEMT)) { |
| for (j = 0; j < 16 && i < len; j++, i++) |
| out_8(avr_addr + UART_TX, string[i]); |
| if (i == len) { |
| /* Read "OK" back: 4ms for the last "KKKK" |
| plus a couple bytes back */ |
| msleep(7); |
| printk("linkstation: disarming the AVR watchdog: "); |
| while (in_8(avr_addr + UART_LSR) & UART_LSR_DR) |
| printk("%c", in_8(avr_addr + UART_RX)); |
| break; |
| } |
| } |
| msleep(17); |
| } |
| printk("\n"); |
| } |
| |
| #define AVR_QUOT(clock) ((clock) + 8 * 9600) / (16 * 9600) |
| |
| void avr_uart_configure(void) |
| { |
| unsigned char cval = UART_LCR_WLEN8; |
| unsigned int quot = AVR_QUOT(avr_clock); |
| |
| if (!avr_addr || !avr_clock) |
| return; |
| |
| out_8(avr_addr + UART_LCR, cval); /* initialise UART */ |
| out_8(avr_addr + UART_MCR, 0); |
| out_8(avr_addr + UART_IER, 0); |
| |
| cval |= UART_LCR_STOP | UART_LCR_PARITY | UART_LCR_EPAR; |
| |
| out_8(avr_addr + UART_LCR, cval); /* Set character format */ |
| |
| out_8(avr_addr + UART_LCR, cval | UART_LCR_DLAB); /* set DLAB */ |
| out_8(avr_addr + UART_DLL, quot & 0xff); /* LS of divisor */ |
| out_8(avr_addr + UART_DLM, quot >> 8); /* MS of divisor */ |
| out_8(avr_addr + UART_LCR, cval); /* reset DLAB */ |
| out_8(avr_addr + UART_FCR, UART_FCR_ENABLE_FIFO); /* enable FIFO */ |
| } |
| |
| void avr_uart_send(const char c) |
| { |
| if (!avr_addr || !avr_clock) |
| return; |
| |
| out_8(avr_addr + UART_TX, c); |
| out_8(avr_addr + UART_TX, c); |
| out_8(avr_addr + UART_TX, c); |
| out_8(avr_addr + UART_TX, c); |
| } |
| |
| static void __init ls_uart_init(void) |
| { |
| local_irq_disable(); |
| |
| #ifndef CONFIG_SERIAL_8250 |
| out_8(avr_addr + UART_FCR, UART_FCR_ENABLE_FIFO); /* enable FIFO */ |
| out_8(avr_addr + UART_FCR, UART_FCR_ENABLE_FIFO | |
| UART_FCR_CLEAR_RCVR | UART_FCR_CLEAR_XMIT); /* clear FIFOs */ |
| out_8(avr_addr + UART_FCR, 0); |
| out_8(avr_addr + UART_IER, 0); |
| |
| /* Clear up interrupts */ |
| (void) in_8(avr_addr + UART_LSR); |
| (void) in_8(avr_addr + UART_RX); |
| (void) in_8(avr_addr + UART_IIR); |
| (void) in_8(avr_addr + UART_MSR); |
| #endif |
| avr_uart_configure(); |
| |
| local_irq_enable(); |
| } |
| |
| static int __init ls_uarts_init(void) |
| { |
| struct device_node *avr; |
| struct resource res; |
| int len, ret; |
| |
| avr = of_find_node_by_path("/soc10x/serial@80004500"); |
| if (!avr) |
| return -EINVAL; |
| |
| avr_clock = *(u32*)of_get_property(avr, "clock-frequency", &len); |
| if (!avr_clock) |
| return -EINVAL; |
| |
| ret = of_address_to_resource(avr, 0, &res); |
| if (ret) |
| return ret; |
| |
| of_node_put(avr); |
| |
| avr_addr = ioremap(res.start, 32); |
| if (!avr_addr) |
| return -EFAULT; |
| |
| ls_uart_init(); |
| |
| INIT_WORK(&wd_work, wd_stop); |
| schedule_work(&wd_work); |
| |
| return 0; |
| } |
| |
| machine_late_initcall(linkstation, ls_uarts_init); |