| /* |
| * early_printk_intel_mid.c - early consoles for Intel MID platforms |
| * |
| * Copyright (c) 2008-2010, Intel Corporation |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License |
| * as published by the Free Software Foundation; version 2 |
| * of the License. |
| */ |
| |
| /* |
| * This file implements two early consoles named mrst and hsu. |
| * mrst is based on Maxim3110 spi-uart device, it exists in both |
| * Moorestown and Medfield platforms, while hsu is based on a High |
| * Speed UART device which only exists in the Medfield platform |
| */ |
| |
| #include <linux/serial_reg.h> |
| #include <linux/serial_mfd.h> |
| #include <linux/kmsg_dump.h> |
| #include <linux/console.h> |
| #include <linux/kernel.h> |
| #include <linux/delay.h> |
| #include <linux/init.h> |
| #include <linux/io.h> |
| |
| #include <asm/fixmap.h> |
| #include <asm/pgtable.h> |
| #include <asm/intel-mid.h> |
| |
| #define MRST_SPI_TIMEOUT 0x200000 |
| #define MRST_REGBASE_SPI0 0xff128000 |
| #define MRST_REGBASE_SPI1 0xff128400 |
| #define MRST_CLK_SPI0_REG 0xff11d86c |
| |
| /* Bit fields in CTRLR0 */ |
| #define SPI_DFS_OFFSET 0 |
| |
| #define SPI_FRF_OFFSET 4 |
| #define SPI_FRF_SPI 0x0 |
| #define SPI_FRF_SSP 0x1 |
| #define SPI_FRF_MICROWIRE 0x2 |
| #define SPI_FRF_RESV 0x3 |
| |
| #define SPI_MODE_OFFSET 6 |
| #define SPI_SCPH_OFFSET 6 |
| #define SPI_SCOL_OFFSET 7 |
| #define SPI_TMOD_OFFSET 8 |
| #define SPI_TMOD_TR 0x0 /* xmit & recv */ |
| #define SPI_TMOD_TO 0x1 /* xmit only */ |
| #define SPI_TMOD_RO 0x2 /* recv only */ |
| #define SPI_TMOD_EPROMREAD 0x3 /* eeprom read mode */ |
| |
| #define SPI_SLVOE_OFFSET 10 |
| #define SPI_SRL_OFFSET 11 |
| #define SPI_CFS_OFFSET 12 |
| |
| /* Bit fields in SR, 7 bits */ |
| #define SR_MASK 0x7f /* cover 7 bits */ |
| #define SR_BUSY (1 << 0) |
| #define SR_TF_NOT_FULL (1 << 1) |
| #define SR_TF_EMPT (1 << 2) |
| #define SR_RF_NOT_EMPT (1 << 3) |
| #define SR_RF_FULL (1 << 4) |
| #define SR_TX_ERR (1 << 5) |
| #define SR_DCOL (1 << 6) |
| |
| struct dw_spi_reg { |
| u32 ctrl0; |
| u32 ctrl1; |
| u32 ssienr; |
| u32 mwcr; |
| u32 ser; |
| u32 baudr; |
| u32 txfltr; |
| u32 rxfltr; |
| u32 txflr; |
| u32 rxflr; |
| u32 sr; |
| u32 imr; |
| u32 isr; |
| u32 risr; |
| u32 txoicr; |
| u32 rxoicr; |
| u32 rxuicr; |
| u32 msticr; |
| u32 icr; |
| u32 dmacr; |
| u32 dmatdlr; |
| u32 dmardlr; |
| u32 idr; |
| u32 version; |
| |
| /* Currently operates as 32 bits, though only the low 16 bits matter */ |
| u32 dr; |
| } __packed; |
| |
| #define dw_readl(dw, name) __raw_readl(&(dw)->name) |
| #define dw_writel(dw, name, val) __raw_writel((val), &(dw)->name) |
| |
| /* Default use SPI0 register for mrst, we will detect Penwell and use SPI1 */ |
| static unsigned long mrst_spi_paddr = MRST_REGBASE_SPI0; |
| |
| static u32 *pclk_spi0; |
| /* Always contains an accessible address, start with 0 */ |
| static struct dw_spi_reg *pspi; |
| |
| static struct kmsg_dumper dw_dumper; |
| static int dumper_registered; |
| |
| static void dw_kmsg_dump(struct kmsg_dumper *dumper, |
| enum kmsg_dump_reason reason) |
| { |
| static char line[1024]; |
| size_t len; |
| |
| /* When run to this, we'd better re-init the HW */ |
| mrst_early_console_init(); |
| |
| while (kmsg_dump_get_line(dumper, true, line, sizeof(line), &len)) |
| early_mrst_console.write(&early_mrst_console, line, len); |
| } |
| |
| /* Set the ratio rate to 115200, 8n1, IRQ disabled */ |
| static void max3110_write_config(void) |
| { |
| u16 config; |
| |
| config = 0xc001; |
| dw_writel(pspi, dr, config); |
| } |
| |
| /* Translate char to a eligible word and send to max3110 */ |
| static void max3110_write_data(char c) |
| { |
| u16 data; |
| |
| data = 0x8000 | c; |
| dw_writel(pspi, dr, data); |
| } |
| |
| void mrst_early_console_init(void) |
| { |
| u32 ctrlr0 = 0; |
| u32 spi0_cdiv; |
| u32 freq; /* Freqency info only need be searched once */ |
| |
| /* Base clk is 100 MHz, the actual clk = 100M / (clk_divider + 1) */ |
| pclk_spi0 = (void *)set_fixmap_offset_nocache(FIX_EARLYCON_MEM_BASE, |
| MRST_CLK_SPI0_REG); |
| spi0_cdiv = ((*pclk_spi0) & 0xe00) >> 9; |
| freq = 100000000 / (spi0_cdiv + 1); |
| |
| if (intel_mid_identify_cpu() == INTEL_MID_CPU_CHIP_PENWELL) |
| mrst_spi_paddr = MRST_REGBASE_SPI1; |
| |
| pspi = (void *)set_fixmap_offset_nocache(FIX_EARLYCON_MEM_BASE, |
| mrst_spi_paddr); |
| |
| /* Disable SPI controller */ |
| dw_writel(pspi, ssienr, 0); |
| |
| /* Set control param, 8 bits, transmit only mode */ |
| ctrlr0 = dw_readl(pspi, ctrl0); |
| |
| ctrlr0 &= 0xfcc0; |
| ctrlr0 |= 0xf | (SPI_FRF_SPI << SPI_FRF_OFFSET) |
| | (SPI_TMOD_TO << SPI_TMOD_OFFSET); |
| dw_writel(pspi, ctrl0, ctrlr0); |
| |
| /* |
| * Change the spi0 clk to comply with 115200 bps, use 100000 to |
| * calculate the clk dividor to make the clock a little slower |
| * than real baud rate. |
| */ |
| dw_writel(pspi, baudr, freq/100000); |
| |
| /* Disable all INT for early phase */ |
| dw_writel(pspi, imr, 0x0); |
| |
| /* Set the cs to spi-uart */ |
| dw_writel(pspi, ser, 0x2); |
| |
| /* Enable the HW, the last step for HW init */ |
| dw_writel(pspi, ssienr, 0x1); |
| |
| /* Set the default configuration */ |
| max3110_write_config(); |
| |
| /* Register the kmsg dumper */ |
| if (!dumper_registered) { |
| dw_dumper.dump = dw_kmsg_dump; |
| kmsg_dump_register(&dw_dumper); |
| dumper_registered = 1; |
| } |
| } |
| |
| /* Slave select should be called in the read/write function */ |
| static void early_mrst_spi_putc(char c) |
| { |
| unsigned int timeout; |
| u32 sr; |
| |
| timeout = MRST_SPI_TIMEOUT; |
| /* Early putc needs to make sure the TX FIFO is not full */ |
| while (--timeout) { |
| sr = dw_readl(pspi, sr); |
| if (!(sr & SR_TF_NOT_FULL)) |
| cpu_relax(); |
| else |
| break; |
| } |
| |
| if (!timeout) |
| pr_warn("MRST earlycon: timed out\n"); |
| else |
| max3110_write_data(c); |
| } |
| |
| /* Early SPI only uses polling mode */ |
| static void early_mrst_spi_write(struct console *con, const char *str, |
| unsigned n) |
| { |
| int i; |
| |
| for (i = 0; i < n && *str; i++) { |
| if (*str == '\n') |
| early_mrst_spi_putc('\r'); |
| early_mrst_spi_putc(*str); |
| str++; |
| } |
| } |
| |
| struct console early_mrst_console = { |
| .name = "earlymrst", |
| .write = early_mrst_spi_write, |
| .flags = CON_PRINTBUFFER, |
| .index = -1, |
| }; |
| |
| /* |
| * Following is the early console based on Medfield HSU (High |
| * Speed UART) device. |
| */ |
| #define HSU_PORT_BASE 0xffa28080 |
| |
| static void __iomem *phsu; |
| |
| void hsu_early_console_init(const char *s) |
| { |
| unsigned long paddr, port = 0; |
| u8 lcr; |
| |
| /* |
| * Select the early HSU console port if specified by user in the |
| * kernel command line. |
| */ |
| if (*s && !kstrtoul(s, 10, &port)) |
| port = clamp_val(port, 0, 2); |
| |
| paddr = HSU_PORT_BASE + port * 0x80; |
| phsu = (void *)set_fixmap_offset_nocache(FIX_EARLYCON_MEM_BASE, paddr); |
| |
| /* Disable FIFO */ |
| writeb(0x0, phsu + UART_FCR); |
| |
| /* Set to default 115200 bps, 8n1 */ |
| lcr = readb(phsu + UART_LCR); |
| writeb((0x80 | lcr), phsu + UART_LCR); |
| writeb(0x18, phsu + UART_DLL); |
| writeb(lcr, phsu + UART_LCR); |
| writel(0x3600, phsu + UART_MUL*4); |
| |
| writeb(0x8, phsu + UART_MCR); |
| writeb(0x7, phsu + UART_FCR); |
| writeb(0x3, phsu + UART_LCR); |
| |
| /* Clear IRQ status */ |
| readb(phsu + UART_LSR); |
| readb(phsu + UART_RX); |
| readb(phsu + UART_IIR); |
| readb(phsu + UART_MSR); |
| |
| /* Enable FIFO */ |
| writeb(0x7, phsu + UART_FCR); |
| } |
| |
| #define BOTH_EMPTY (UART_LSR_TEMT | UART_LSR_THRE) |
| |
| static void early_hsu_putc(char ch) |
| { |
| unsigned int timeout = 10000; /* 10ms */ |
| u8 status; |
| |
| while (--timeout) { |
| status = readb(phsu + UART_LSR); |
| if (status & BOTH_EMPTY) |
| break; |
| udelay(1); |
| } |
| |
| /* Only write the char when there was no timeout */ |
| if (timeout) |
| writeb(ch, phsu + UART_TX); |
| } |
| |
| static void early_hsu_write(struct console *con, const char *str, unsigned n) |
| { |
| int i; |
| |
| for (i = 0; i < n && *str; i++) { |
| if (*str == '\n') |
| early_hsu_putc('\r'); |
| early_hsu_putc(*str); |
| str++; |
| } |
| } |
| |
| struct console early_hsu_console = { |
| .name = "earlyhsu", |
| .write = early_hsu_write, |
| .flags = CON_PRINTBUFFER, |
| .index = -1, |
| }; |