2 * early_printk_intel_mid.c - early consoles for Intel MID platforms
4 * Copyright (c) 2008-2010, Intel Corporation
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; version 2
13 * This file implements two early consoles named mrst and hsu.
14 * mrst is based on Maxim3110 spi-uart device, it exists in both
15 * Moorestown and Medfield platforms, while hsu is based on a High
16 * Speed UART device which only exists in the Medfield platform
19 #include <linux/serial_reg.h>
20 #include <linux/serial_mfd.h>
21 #include <linux/kmsg_dump.h>
22 #include <linux/console.h>
23 #include <linux/kernel.h>
24 #include <linux/delay.h>
27 #include <asm/fixmap.h>
28 #include <asm/pgtable.h>
29 #include <asm/intel-mid.h>
31 #define MRST_SPI_TIMEOUT 0x200000
32 #define MRST_REGBASE_SPI0 0xff128000
33 #define MRST_REGBASE_SPI1 0xff128400
34 #define MRST_CLK_SPI0_REG 0xff11d86c
36 /* Bit fields in CTRLR0 */
37 #define SPI_DFS_OFFSET 0
39 #define SPI_FRF_OFFSET 4
40 #define SPI_FRF_SPI 0x0
41 #define SPI_FRF_SSP 0x1
42 #define SPI_FRF_MICROWIRE 0x2
43 #define SPI_FRF_RESV 0x3
45 #define SPI_MODE_OFFSET 6
46 #define SPI_SCPH_OFFSET 6
47 #define SPI_SCOL_OFFSET 7
48 #define SPI_TMOD_OFFSET 8
49 #define SPI_TMOD_TR 0x0 /* xmit & recv */
50 #define SPI_TMOD_TO 0x1 /* xmit only */
51 #define SPI_TMOD_RO 0x2 /* recv only */
52 #define SPI_TMOD_EPROMREAD 0x3 /* eeprom read mode */
54 #define SPI_SLVOE_OFFSET 10
55 #define SPI_SRL_OFFSET 11
56 #define SPI_CFS_OFFSET 12
58 /* Bit fields in SR, 7 bits */
59 #define SR_MASK 0x7f /* cover 7 bits */
60 #define SR_BUSY (1 << 0)
61 #define SR_TF_NOT_FULL (1 << 1)
62 #define SR_TF_EMPT (1 << 2)
63 #define SR_RF_NOT_EMPT (1 << 3)
64 #define SR_RF_FULL (1 << 4)
65 #define SR_TX_ERR (1 << 5)
66 #define SR_DCOL (1 << 6)
94 /* Currently operates as 32 bits, though only the low 16 bits matter */
98 #define dw_readl(dw, name) __raw_readl(&(dw)->name)
99 #define dw_writel(dw, name, val) __raw_writel((val), &(dw)->name)
101 /* Default use SPI0 register for mrst, we will detect Penwell and use SPI1 */
102 static unsigned long mrst_spi_paddr
= MRST_REGBASE_SPI0
;
104 static u32
*pclk_spi0
;
105 /* Always contains an accessible address, start with 0 */
106 static struct dw_spi_reg
*pspi
;
108 static struct kmsg_dumper dw_dumper
;
109 static int dumper_registered
;
111 static void dw_kmsg_dump(struct kmsg_dumper
*dumper
,
112 enum kmsg_dump_reason reason
)
114 static char line
[1024];
117 /* When run to this, we'd better re-init the HW */
118 mrst_early_console_init();
120 while (kmsg_dump_get_line(dumper
, true, line
, sizeof(line
), &len
))
121 early_mrst_console
.write(&early_mrst_console
, line
, len
);
124 /* Set the ratio rate to 115200, 8n1, IRQ disabled */
125 static void max3110_write_config(void)
130 dw_writel(pspi
, dr
, config
);
133 /* Translate char to a eligible word and send to max3110 */
134 static void max3110_write_data(char c
)
139 dw_writel(pspi
, dr
, data
);
142 void mrst_early_console_init(void)
146 u32 freq
; /* Freqency info only need be searched once */
148 /* Base clk is 100 MHz, the actual clk = 100M / (clk_divider + 1) */
149 pclk_spi0
= (void *)set_fixmap_offset_nocache(FIX_EARLYCON_MEM_BASE
,
151 spi0_cdiv
= ((*pclk_spi0
) & 0xe00) >> 9;
152 freq
= 100000000 / (spi0_cdiv
+ 1);
154 if (intel_mid_identify_cpu() == INTEL_MID_CPU_CHIP_PENWELL
)
155 mrst_spi_paddr
= MRST_REGBASE_SPI1
;
157 pspi
= (void *)set_fixmap_offset_nocache(FIX_EARLYCON_MEM_BASE
,
160 /* Disable SPI controller */
161 dw_writel(pspi
, ssienr
, 0);
163 /* Set control param, 8 bits, transmit only mode */
164 ctrlr0
= dw_readl(pspi
, ctrl0
);
167 ctrlr0
|= 0xf | (SPI_FRF_SPI
<< SPI_FRF_OFFSET
)
168 | (SPI_TMOD_TO
<< SPI_TMOD_OFFSET
);
169 dw_writel(pspi
, ctrl0
, ctrlr0
);
172 * Change the spi0 clk to comply with 115200 bps, use 100000 to
173 * calculate the clk dividor to make the clock a little slower
174 * than real baud rate.
176 dw_writel(pspi
, baudr
, freq
/100000);
178 /* Disable all INT for early phase */
179 dw_writel(pspi
, imr
, 0x0);
181 /* Set the cs to spi-uart */
182 dw_writel(pspi
, ser
, 0x2);
184 /* Enable the HW, the last step for HW init */
185 dw_writel(pspi
, ssienr
, 0x1);
187 /* Set the default configuration */
188 max3110_write_config();
190 /* Register the kmsg dumper */
191 if (!dumper_registered
) {
192 dw_dumper
.dump
= dw_kmsg_dump
;
193 kmsg_dump_register(&dw_dumper
);
194 dumper_registered
= 1;
198 /* Slave select should be called in the read/write function */
199 static void early_mrst_spi_putc(char c
)
201 unsigned int timeout
;
204 timeout
= MRST_SPI_TIMEOUT
;
205 /* Early putc needs to make sure the TX FIFO is not full */
207 sr
= dw_readl(pspi
, sr
);
208 if (!(sr
& SR_TF_NOT_FULL
))
215 pr_warn("MRST earlycon: timed out\n");
217 max3110_write_data(c
);
220 /* Early SPI only uses polling mode */
221 static void early_mrst_spi_write(struct console
*con
, const char *str
,
226 for (i
= 0; i
< n
&& *str
; i
++) {
228 early_mrst_spi_putc('\r');
229 early_mrst_spi_putc(*str
);
234 struct console early_mrst_console
= {
236 .write
= early_mrst_spi_write
,
237 .flags
= CON_PRINTBUFFER
,
242 * Following is the early console based on Medfield HSU (High
243 * Speed UART) device.
245 #define HSU_PORT_BASE 0xffa28080
247 static void __iomem
*phsu
;
249 void hsu_early_console_init(const char *s
)
251 unsigned long paddr
, port
= 0;
255 * Select the early HSU console port if specified by user in the
256 * kernel command line.
258 if (*s
&& !kstrtoul(s
, 10, &port
))
259 port
= clamp_val(port
, 0, 2);
261 paddr
= HSU_PORT_BASE
+ port
* 0x80;
262 phsu
= (void *)set_fixmap_offset_nocache(FIX_EARLYCON_MEM_BASE
, paddr
);
265 writeb(0x0, phsu
+ UART_FCR
);
267 /* Set to default 115200 bps, 8n1 */
268 lcr
= readb(phsu
+ UART_LCR
);
269 writeb((0x80 | lcr
), phsu
+ UART_LCR
);
270 writeb(0x18, phsu
+ UART_DLL
);
271 writeb(lcr
, phsu
+ UART_LCR
);
272 writel(0x3600, phsu
+ UART_MUL
*4);
274 writeb(0x8, phsu
+ UART_MCR
);
275 writeb(0x7, phsu
+ UART_FCR
);
276 writeb(0x3, phsu
+ UART_LCR
);
278 /* Clear IRQ status */
279 readb(phsu
+ UART_LSR
);
280 readb(phsu
+ UART_RX
);
281 readb(phsu
+ UART_IIR
);
282 readb(phsu
+ UART_MSR
);
285 writeb(0x7, phsu
+ UART_FCR
);
288 #define BOTH_EMPTY (UART_LSR_TEMT | UART_LSR_THRE)
290 static void early_hsu_putc(char ch
)
292 unsigned int timeout
= 10000; /* 10ms */
296 status
= readb(phsu
+ UART_LSR
);
297 if (status
& BOTH_EMPTY
)
302 /* Only write the char when there was no timeout */
304 writeb(ch
, phsu
+ UART_TX
);
307 static void early_hsu_write(struct console
*con
, const char *str
, unsigned n
)
311 for (i
= 0; i
< n
&& *str
; i
++) {
313 early_hsu_putc('\r');
314 early_hsu_putc(*str
);
319 struct console early_hsu_console
= {
321 .write
= early_hsu_write
,
322 .flags
= CON_PRINTBUFFER
,