| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * Copyright (C) 2016 Stefan Roese <sr@denx.de> |
| */ |
| |
| #include <common.h> |
| #include <dm.h> |
| #include <serial.h> |
| #include <asm/io.h> |
| #include <asm/arch/cpu.h> |
| |
| struct mvebu_plat { |
| void __iomem *base; |
| }; |
| |
| /* |
| * Register offset |
| */ |
| #define UART_RX_REG 0x00 |
| #define UART_TX_REG 0x04 |
| #define UART_CTRL_REG 0x08 |
| #define UART_STATUS_REG 0x0c |
| #define UART_BAUD_REG 0x10 |
| #define UART_POSSR_REG 0x14 |
| |
| #define UART_STATUS_RX_RDY 0x10 |
| #define UART_STATUS_TX_EMPTY 0x40 |
| #define UART_STATUS_TXFIFO_FULL 0x800 |
| |
| #define UART_CTRL_RXFIFO_RESET 0x4000 |
| #define UART_CTRL_TXFIFO_RESET 0x8000 |
| |
| static int mvebu_serial_putc(struct udevice *dev, const char ch) |
| { |
| struct mvebu_plat *plat = dev_get_plat(dev); |
| void __iomem *base = plat->base; |
| |
| while (readl(base + UART_STATUS_REG) & UART_STATUS_TXFIFO_FULL) |
| ; |
| |
| writel(ch, base + UART_TX_REG); |
| |
| return 0; |
| } |
| |
| static int mvebu_serial_getc(struct udevice *dev) |
| { |
| struct mvebu_plat *plat = dev_get_plat(dev); |
| void __iomem *base = plat->base; |
| |
| while (!(readl(base + UART_STATUS_REG) & UART_STATUS_RX_RDY)) |
| ; |
| |
| return readl(base + UART_RX_REG) & 0xff; |
| } |
| |
| static int mvebu_serial_pending(struct udevice *dev, bool input) |
| { |
| struct mvebu_plat *plat = dev_get_plat(dev); |
| void __iomem *base = plat->base; |
| |
| if (input) { |
| if (readl(base + UART_STATUS_REG) & UART_STATUS_RX_RDY) |
| return 1; |
| } else { |
| if (!(readl(base + UART_STATUS_REG) & UART_STATUS_TX_EMPTY)) |
| return 1; |
| } |
| |
| return 0; |
| } |
| |
| static int mvebu_serial_setbrg(struct udevice *dev, int baudrate) |
| { |
| struct mvebu_plat *plat = dev_get_plat(dev); |
| void __iomem *base = plat->base; |
| u32 parent_rate, divider; |
| |
| /* |
| * Calculate divider |
| * baudrate = clock / 16 / divider |
| */ |
| parent_rate = get_ref_clk() * 1000000; |
| divider = DIV_ROUND_CLOSEST(parent_rate, baudrate * 16); |
| writel(divider, base + UART_BAUD_REG); |
| |
| /* |
| * Set Programmable Oversampling Stack to 0, |
| * UART defaults to 16x scheme |
| */ |
| writel(0, base + UART_POSSR_REG); |
| |
| return 0; |
| } |
| |
| static int mvebu_serial_probe(struct udevice *dev) |
| { |
| struct mvebu_plat *plat = dev_get_plat(dev); |
| void __iomem *base = plat->base; |
| |
| /* reset FIFOs */ |
| writel(UART_CTRL_RXFIFO_RESET | UART_CTRL_TXFIFO_RESET, |
| base + UART_CTRL_REG); |
| |
| /* No Parity, 1 Stop */ |
| writel(0, base + UART_CTRL_REG); |
| |
| return 0; |
| } |
| |
| static int mvebu_serial_of_to_plat(struct udevice *dev) |
| { |
| struct mvebu_plat *plat = dev_get_plat(dev); |
| |
| plat->base = dev_read_addr_ptr(dev); |
| |
| return 0; |
| } |
| |
| static const struct dm_serial_ops mvebu_serial_ops = { |
| .putc = mvebu_serial_putc, |
| .pending = mvebu_serial_pending, |
| .getc = mvebu_serial_getc, |
| .setbrg = mvebu_serial_setbrg, |
| }; |
| |
| static const struct udevice_id mvebu_serial_ids[] = { |
| { .compatible = "marvell,armada-3700-uart" }, |
| { } |
| }; |
| |
| U_BOOT_DRIVER(serial_mvebu) = { |
| .name = "serial_mvebu", |
| .id = UCLASS_SERIAL, |
| .of_match = mvebu_serial_ids, |
| .of_to_plat = mvebu_serial_of_to_plat, |
| .plat_auto = sizeof(struct mvebu_plat), |
| .probe = mvebu_serial_probe, |
| .ops = &mvebu_serial_ops, |
| }; |
| |
| #ifdef CONFIG_DEBUG_MVEBU_A3700_UART |
| |
| #include <debug_uart.h> |
| |
| static inline void _debug_uart_init(void) |
| { |
| void __iomem *base = (void __iomem *)CONFIG_DEBUG_UART_BASE; |
| u32 baudrate, parent_rate, divider; |
| |
| /* reset FIFOs */ |
| writel(UART_CTRL_RXFIFO_RESET | UART_CTRL_TXFIFO_RESET, |
| base + UART_CTRL_REG); |
| |
| /* No Parity, 1 Stop */ |
| writel(0, base + UART_CTRL_REG); |
| |
| /* |
| * Calculate divider |
| * baudrate = clock / 16 / divider |
| */ |
| baudrate = 115200; |
| parent_rate = get_ref_clk() * 1000000; |
| divider = DIV_ROUND_CLOSEST(parent_rate, baudrate * 16); |
| writel(divider, base + UART_BAUD_REG); |
| |
| /* |
| * Set Programmable Oversampling Stack to 0, |
| * UART defaults to 16x scheme |
| */ |
| writel(0, base + UART_POSSR_REG); |
| } |
| |
| static inline void _debug_uart_putc(int ch) |
| { |
| void __iomem *base = (void __iomem *)CONFIG_DEBUG_UART_BASE; |
| |
| while (readl(base + UART_STATUS_REG) & UART_STATUS_TXFIFO_FULL) |
| ; |
| |
| writel(ch, base + UART_TX_REG); |
| } |
| |
| DEBUG_UART_FUNCS |
| #endif |