/* * (C) Copyright 2013-2016 * Allwinner Technology Co., Ltd. * * SPDX-License-Identifier: GPL-2.0+ */ /* * COM1 NS16550 support * originally from linux source (arch/powerpc/boot/ns16550.c) * modified to use CONFIG_SYS_ISA_MEM and new defines */ #include #include #include #include #include #define thr rbr #define dll rbr #define dlh ier #define iir fcr static serial_hw_t *serial_ctrl_base = NULL; int serial_init(void) { sunxi_serial_init(0, NULL, 0); return 0; } /* ************************************************************************************************************ * * function * * name : * * parmeters : * * return : * * note : * * ************************************************************************************************************ */ void sunxi_serial_init(int uart_port, void *gpio_cfg, int gpio_max) { u32 reg, i; u32 uart_clk; #ifdef CCMU_UART_BGR_REG /* reset */ reg = readl(CCMU_UART_BGR_REG); reg &= ~(1<<(CCM_UART_RST_OFFSET + uart_port)); writel(reg, CCMU_UART_BGR_REG); for (i = 0; i < 100; i++) ; reg |= (1<<(CCM_UART_RST_OFFSET + uart_port)); writel(reg, CCMU_UART_BGR_REG); /* gate */ reg = readl(CCMU_UART_BGR_REG); reg &= ~(1<<(CCM_UART_GATING_OFFSET + uart_port)); writel(reg, CCMU_UART_BGR_REG); for (i = 0; i < 100; i++) ; reg |= (1<<(CCM_UART_GATING_OFFSET + uart_port)); writel(reg, CCMU_UART_BGR_REG); #else //reset reg = readl(CCMU_BUS_SOFT_RST_REG4); reg &= ~(1<<(CCM_UART_PORT_OFFSET + uart_port)); writel(reg, CCMU_BUS_SOFT_RST_REG4); for( i = 0; i < 100; i++ ); reg |= (1<<(CCM_UART_PORT_OFFSET + uart_port)); writel(reg, CCMU_BUS_SOFT_RST_REG4); //gate reg = readl(CCMU_BUS_CLK_GATING_REG3); reg &= ~(1<<(CCM_UART_PORT_OFFSET + uart_port)); writel(reg, CCMU_BUS_CLK_GATING_REG3); for( i = 0; i < 100; i++ ); reg |= (1<<(CCM_UART_PORT_OFFSET + uart_port)); writel(reg, CCMU_BUS_CLK_GATING_REG3); #endif //gpio //boot_set_gpio(gpio_cfg, gpio_max, 1); //uart init serial_ctrl_base = (serial_hw_t *)(SUNXI_UART0_BASE + uart_port * CCM_UART_ADDR_OFFSET); serial_ctrl_base->mcr = 0x3; uart_clk = (24000000 + 8 * UART_BAUD)/(16 * UART_BAUD); serial_ctrl_base->lcr |= 0x80; serial_ctrl_base->dlh = uart_clk>>8; serial_ctrl_base->dll = uart_clk&0xff; serial_ctrl_base->lcr &= ~0x80; serial_ctrl_base->lcr = ((PARITY&0x03)<<3) | ((STOP&0x01)<<2) | (DLEN&0x03); serial_ctrl_base->fcr = 0x7; return; } /* ************************************************************************************************************ * * function * * name : * * parmeters : * * return : * * note : * * ************************************************************************************************************ */ void sunxi_serial_putc (char c) { while((serial_ctrl_base->lsr & ( 1 << 6 )) == 0); serial_ctrl_base->thr = c; } /* ************************************************************************************************************ * * function * * name : * * parmeters : * * return : * * note : * * ************************************************************************************************************ */ char sunxi_serial_getc (void) { while((serial_ctrl_base->lsr & 1) == 0); return serial_ctrl_base->rbr; } /* ************************************************************************************************************ * * function * * name : * * parmeters : * * return : * * note : * * ************************************************************************************************************ */ int sunxi_serial_tstc (void) { return serial_ctrl_base->lsr & 1; }