/* * Bootloader version of the embedded MPSC/UART driver for the Marvell 64x60. * Note: Due to a GT64260A erratum, DMA will be used for UART input (via SDMA). * * Author: Mark A. Greer * * 2001 (c) MontaVista Software, Inc. 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. */ /* This code assumes that the data cache has been disabled (L1, L2, L3). */ #include #include #include #include #include #include #include #ifdef CONFIG_EV64360 #include u32 mv64x60_console_baud = EV64360_DEFAULT_BAUD; u32 mv64x60_mpsc_clk_src = EV64360_MPSC_CLK_SRC; /* TCLK */ u32 mv64x60_mpsc_clk_freq = EV64360_MPSC_CLK_FREQ; #else u32 mv64x60_console_baud = 9600; u32 mv64x60_mpsc_clk_src = 8; /* TCLK */ u32 mv64x60_mpsc_clk_freq = 100000000; #endif extern void udelay(long); static void stop_dma(int chan); static void __iomem *mv64x60_base = (void __iomem *)CONFIG_MV64X60_NEW_BASE; struct sdma_regs { u32 sdc; u32 sdcm; u32 rx_desc; u32 rx_buf_ptr; u32 scrdp; u32 tx_desc; u32 sctdp; u32 sftdp; }; static struct sdma_regs sdma_regs[2]; #define SDMA_REGS_INIT(s, reg_base) { \ (s)->sdc = (reg_base) + SDMA_SDC; \ (s)->sdcm = (reg_base) + SDMA_SDCM; \ (s)->rx_desc = (reg_base) + SDMA_RX_DESC; \ (s)->rx_buf_ptr = (reg_base) + SDMA_RX_BUF_PTR; \ (s)->scrdp = (reg_base) + SDMA_SCRDP; \ (s)->tx_desc = (reg_base) + SDMA_TX_DESC; \ (s)->sctdp = (reg_base) + SDMA_SCTDP; \ (s)->sftdp = (reg_base) + SDMA_SFTDP; \ } static u32 mpsc_base[2] = { MV64x60_MPSC_0_OFFSET, MV64x60_MPSC_1_OFFSET }; struct mv64x60_rx_desc { u16 bufsize; u16 bytecnt; u32 cmd_stat; u32 next_desc_ptr; u32 buffer; }; struct mv64x60_tx_desc { u16 bytecnt; u16 shadow; u32 cmd_stat; u32 next_desc_ptr; u32 buffer; }; #define MAX_RESET_WAIT 10000 #define MAX_TX_WAIT 10000 #define RX_NUM_DESC 2 #define TX_NUM_DESC 2 #define RX_BUF_SIZE 32 #define TX_BUF_SIZE 32 static struct mv64x60_rx_desc rd[2][RX_NUM_DESC] __attribute__ ((aligned(32))); static struct mv64x60_tx_desc td[2][TX_NUM_DESC] __attribute__ ((aligned(32))); static char rx_buf[2][RX_NUM_DESC * RX_BUF_SIZE] __attribute__ ((aligned(32))); static char tx_buf[2][TX_NUM_DESC * TX_BUF_SIZE] __attribute__ ((aligned(32))); static int cur_rd[2] = { 0, 0 }; static int cur_td[2] = { 0, 0 }; static char chan_initialized[2] = { 0, 0 }; #define RX_INIT_RDP(rdp) { \ (rdp)->bufsize = 2; \ (rdp)->bytecnt = 0; \ (rdp)->cmd_stat = SDMA_DESC_CMDSTAT_L | SDMA_DESC_CMDSTAT_F | \ SDMA_DESC_CMDSTAT_O; \ } #ifdef CONFIG_MV64360 static u32 cpu2mem_tab[MV64x60_CPU2MEM_WINDOWS][2] = { { MV64x60_CPU2MEM_0_BASE, MV64x60_CPU2MEM_0_SIZE }, { MV64x60_CPU2MEM_1_BASE, MV64x60_CPU2MEM_1_SIZE }, { MV64x60_CPU2MEM_2_BASE, MV64x60_CPU2MEM_2_SIZE }, { MV64x60_CPU2MEM_3_BASE, MV64x60_CPU2MEM_3_SIZE } }; static u32 com2mem_tab[MV64x60_CPU2MEM_WINDOWS][2] = { { MV64360_MPSC2MEM_0_BASE, MV64360_MPSC2MEM_0_SIZE }, { MV64360_MPSC2MEM_1_BASE, MV64360_MPSC2MEM_1_SIZE }, { MV64360_MPSC2MEM_2_BASE, MV64360_MPSC2MEM_2_SIZE }, { MV64360_MPSC2MEM_3_BASE, MV64360_MPSC2MEM_3_SIZE } }; static u32 dram_selects[MV64x60_CPU2MEM_WINDOWS] = { 0xe, 0xd, 0xb, 0x7 }; #endif unsigned long serial_init(int chan, void *ignored) { u32 mpsc_routing_base, sdma_base, brg_bcr, cdv; int i; chan = (chan == 1); /* default to chan 0 if anything but 1 */ if (chan_initialized[chan]) return chan; chan_initialized[chan] = 1; if (chan == 0) { sdma_base = MV64x60_SDMA_0_OFFSET; brg_bcr = MV64x60_BRG_0_OFFSET + BRG_BCR; SDMA_REGS_INIT(&sdma_regs[0], MV64x60_SDMA_0_OFFSET); } else { sdma_base = MV64x60_SDMA_1_OFFSET; brg_bcr = MV64x60_BRG_1_OFFSET + BRG_BCR; SDMA_REGS_INIT(&sdma_regs[0], MV64x60_SDMA_1_OFFSET); } mpsc_routing_base = MV64x60_MPSC_ROUTING_OFFSET; stop_dma(chan); /* Set up ring buffers */ for (i=0; i= TX_NUM_DESC) cur_td[com_port] = 0; *(unchar *)(tdp->buffer ^ 7) = c; tdp->bytecnt = 1; tdp->shadow = 1; tdp->cmd_stat = SDMA_DESC_CMDSTAT_L | SDMA_DESC_CMDSTAT_F | SDMA_DESC_CMDSTAT_O; out_le32(mv64x60_base + sdma_regs[com_port].sctdp, (int)tdp); out_le32(mv64x60_base + sdma_regs[com_port].sftdp, (int)tdp); out_le32(mv64x60_base + sdma_regs[com_port].sdcm, in_le32(mv64x60_base + sdma_regs[com_port].sdcm) | SDMA_SDCM_TXD); } unsigned char serial_getc(unsigned long com_port) { struct mv64x60_rx_desc *rdp; unchar c = '\0'; rdp = &rd[com_port][cur_rd[com_port]]; if ((rdp->cmd_stat & (SDMA_DESC_CMDSTAT_O|SDMA_DESC_CMDSTAT_ES)) == 0) { c = *(unchar *)(rdp->buffer ^ 7); RX_INIT_RDP(rdp); if (++cur_rd[com_port] >= RX_NUM_DESC) cur_rd[com_port] = 0; } return c; } int serial_tstc(unsigned long com_port) { struct mv64x60_rx_desc *rdp; int loop_count = 0; int rc = 0; rdp = &rd[com_port][cur_rd[com_port]]; /* Go thru rcv desc's until empty looking for one with data (no error)*/ while (((rdp->cmd_stat & SDMA_DESC_CMDSTAT_O) == 0) && (loop_count++ < RX_NUM_DESC)) { /* If there was an error, reinit the desc & continue */ if ((rdp->cmd_stat & SDMA_DESC_CMDSTAT_ES) != 0) { RX_INIT_RDP(rdp); if (++cur_rd[com_port] >= RX_NUM_DESC) cur_rd[com_port] = 0; rdp = (struct mv64x60_rx_desc *)rdp->next_desc_ptr; } else { rc = 1; break; } } return rc; } void serial_close(unsigned long com_port) { stop_dma(com_port); }