serial.c 9.9 KB
Newer Older
1
/*
2
 * arch/arm/mach-omap2/serial.c
3 4 5
 *
 * OMAP2 serial support.
 *
6
 * Copyright (C) 2005-2008 Nokia Corporation
7 8
 * Author: Paul Mundt <paul.mundt@nokia.com>
 *
9 10
 * Major rework for PM support by Kevin Hilman
 *
11 12
 * Based off of arch/arm/mach-omap/omap1/serial.c
 *
13 14 15
 * Copyright (C) 2009 Texas Instruments
 * Added OMAP4 support - Santosh Shilimkar <santosh.shilimkar@ti.com
 *
16 17 18 19 20 21
 * This file is subject to the terms and conditions of the GNU General Public
 * License. See the file "COPYING" in the main directory of this archive
 * for more details.
 */
#include <linux/kernel.h>
#include <linux/init.h>
22
#include <linux/clk.h>
23
#include <linux/io.h>
24
#include <linux/delay.h>
25 26
#include <linux/platform_device.h>
#include <linux/slab.h>
27
#include <linux/pm_runtime.h>
28
#include <linux/console.h>
29 30

#include <plat/omap-serial.h>
31
#include "common.h"
32
#include <plat/board.h>
33 34 35
#include <plat/dma.h>
#include <plat/omap_hwmod.h>
#include <plat/omap_device.h>
36
#include <plat/omap-pm.h>
37

38
#include "prm2xxx_3xxx.h"
39
#include "pm.h"
40
#include "cm2xxx_3xxx.h"
41
#include "prm-regbits-34xx.h"
42
#include "control.h"
43
#include "mux.h"
44

45 46 47 48 49 50 51
/*
 * NOTE: By default the serial timeout is disabled as it causes lost characters
 * over the serial ports. This means that the UART clocks will stay on until
 * disabled via sysfs. This also causes that any deeper omap sleep states are
 * blocked. 
 */
#define DEFAULT_TIMEOUT 0
52

53 54
#define MAX_UART_HWMOD_NAME_LEN		16

55 56 57 58 59
struct omap_uart_state {
	int num;
	int can_sleep;

	struct list_head node;
60 61
	struct omap_hwmod *oh;
	struct platform_device *pdev;
62 63 64
};

static LIST_HEAD(uart_list);
65
static u8 num_uarts;
66

67 68 69 70 71 72 73 74 75 76 77 78 79 80
#ifdef CONFIG_PM

int omap_uart_can_sleep(void)
{
	struct omap_uart_state *uart;
	int can_sleep = 1;

	list_for_each_entry(uart, &uart_list, node) {
		if (!uart->clocked)
			continue;

		if (!uart->can_sleep) {
			can_sleep = 0;
			continue;
81
		}
82 83 84

		/* This UART can now safely sleep. */
		omap_uart_allow_sleep(uart);
85
	}
86 87

	return can_sleep;
88 89
}

90
static void omap_uart_enable_wakeup(struct platform_device *pdev, bool enable)
91
{
92
	struct omap_device *od = to_omap_device(pdev);
93

94 95
	if (!od)
		return;
96

97 98 99 100
	if (enable)
		omap_hwmod_enable_wakeup(od->hwmods[0]);
	else
		omap_hwmod_disable_wakeup(od->hwmods[0]);
101 102
}

103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
/*
 * Errata i291: [UART]:Cannot Acknowledge Idle Requests
 * in Smartidle Mode When Configured for DMA Operations.
 * WA: configure uart in force idle mode.
 */
static void omap_uart_set_noidle(struct platform_device *pdev)
{
	struct omap_device *od = to_omap_device(pdev);

	omap_hwmod_set_slave_idlemode(od->hwmods[0], HWMOD_IDLEMODE_NO);
}

static void omap_uart_set_forceidle(struct platform_device *pdev)
{
	struct omap_device *od = to_omap_device(pdev);

	omap_hwmod_set_slave_idlemode(od->hwmods[0], HWMOD_IDLEMODE_FORCE);
}

122
#else
123 124
static void omap_uart_enable_wakeup(struct platform_device *pdev, bool enable)
{}
125 126
static void omap_uart_set_noidle(struct platform_device *pdev) {}
static void omap_uart_set_forceidle(struct platform_device *pdev) {}
127 128
#endif /* CONFIG_PM */

129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252
#ifdef CONFIG_OMAP_MUX
static struct omap_device_pad default_uart1_pads[] __initdata = {
	{
		.name	= "uart1_cts.uart1_cts",
		.enable	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart1_rts.uart1_rts",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart1_tx.uart1_tx",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart1_rx.uart1_rx",
		.flags	= OMAP_DEVICE_PAD_REMUX | OMAP_DEVICE_PAD_WAKEUP,
		.enable	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
		.idle	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
	},
};

static struct omap_device_pad default_uart2_pads[] __initdata = {
	{
		.name	= "uart2_cts.uart2_cts",
		.enable	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart2_rts.uart2_rts",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart2_tx.uart2_tx",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart2_rx.uart2_rx",
		.flags	= OMAP_DEVICE_PAD_REMUX | OMAP_DEVICE_PAD_WAKEUP,
		.enable	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
		.idle	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
	},
};

static struct omap_device_pad default_uart3_pads[] __initdata = {
	{
		.name	= "uart3_cts_rctx.uart3_cts_rctx",
		.enable	= OMAP_PIN_INPUT_PULLUP | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart3_rts_sd.uart3_rts_sd",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart3_tx_irtx.uart3_tx_irtx",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart3_rx_irrx.uart3_rx_irrx",
		.flags	= OMAP_DEVICE_PAD_REMUX | OMAP_DEVICE_PAD_WAKEUP,
		.enable	= OMAP_PIN_INPUT | OMAP_MUX_MODE0,
		.idle	= OMAP_PIN_INPUT | OMAP_MUX_MODE0,
	},
};

static struct omap_device_pad default_omap36xx_uart4_pads[] __initdata = {
	{
		.name   = "gpmc_wait2.uart4_tx",
		.enable = OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "gpmc_wait3.uart4_rx",
		.flags	= OMAP_DEVICE_PAD_REMUX | OMAP_DEVICE_PAD_WAKEUP,
		.enable	= OMAP_PIN_INPUT | OMAP_MUX_MODE2,
		.idle	= OMAP_PIN_INPUT | OMAP_MUX_MODE2,
	},
};

static struct omap_device_pad default_omap4_uart4_pads[] __initdata = {
	{
		.name	= "uart4_tx.uart4_tx",
		.enable	= OMAP_PIN_OUTPUT | OMAP_MUX_MODE0,
	},
	{
		.name	= "uart4_rx.uart4_rx",
		.flags	= OMAP_DEVICE_PAD_REMUX | OMAP_DEVICE_PAD_WAKEUP,
		.enable	= OMAP_PIN_INPUT | OMAP_MUX_MODE0,
		.idle	= OMAP_PIN_INPUT | OMAP_MUX_MODE0,
	},
};

static void omap_serial_fill_default_pads(struct omap_board_data *bdata)
{
	switch (bdata->id) {
	case 0:
		bdata->pads = default_uart1_pads;
		bdata->pads_cnt = ARRAY_SIZE(default_uart1_pads);
		break;
	case 1:
		bdata->pads = default_uart2_pads;
		bdata->pads_cnt = ARRAY_SIZE(default_uart2_pads);
		break;
	case 2:
		bdata->pads = default_uart3_pads;
		bdata->pads_cnt = ARRAY_SIZE(default_uart3_pads);
		break;
	case 3:
		if (cpu_is_omap44xx()) {
			bdata->pads = default_omap4_uart4_pads;
			bdata->pads_cnt =
				ARRAY_SIZE(default_omap4_uart4_pads);
		} else if (cpu_is_omap3630()) {
			bdata->pads = default_omap36xx_uart4_pads;
			bdata->pads_cnt =
				ARRAY_SIZE(default_omap36xx_uart4_pads);
		}
		break;
	default:
		break;
	}
}
#else
static void omap_serial_fill_default_pads(struct omap_board_data *bdata) {}
#endif

253
static int __init omap_serial_early_init(void)
254
{
255
	int i = 0;
256

257 258 259 260
	do {
		char oh_name[MAX_UART_HWMOD_NAME_LEN];
		struct omap_hwmod *oh;
		struct omap_uart_state *uart;
261

262 263 264 265 266 267 268 269
		snprintf(oh_name, MAX_UART_HWMOD_NAME_LEN,
			 "uart%d", i + 1);
		oh = omap_hwmod_lookup(oh_name);
		if (!oh)
			break;

		uart = kzalloc(sizeof(struct omap_uart_state), GFP_KERNEL);
		if (WARN_ON(!uart))
270
			return -ENODEV;
271

272 273 274 275
		uart->oh = oh;
		uart->num = i++;
		list_add_tail(&uart->node, &uart_list);
		num_uarts++;
276

277
		/*
278
		 * NOTE: omap_hwmod_setup*() has not yet been called,
279
		 *       so no hwmod functions will work yet.
280
		 */
281

282 283 284 285 286 287 288
		/*
		 * During UART early init, device need to be probed
		 * to determine SoC specific init before omap_device
		 * is ready.  Therefore, don't allow idle here
		 */
		uart->oh->flags |= HWMOD_INIT_NO_IDLE | HWMOD_INIT_NO_RESET;
	} while (1);
289 290

	return 0;
291
}
292
core_initcall(omap_serial_early_init);
293

294 295
/**
 * omap_serial_init_port() - initialize single serial port
296
 * @bdata: port specific board data pointer
297
 *
298
 * This function initialies serial driver for given port only.
299 300 301 302 303 304
 * Platforms can call this function instead of omap_serial_init()
 * if they don't plan to use all available UARTs as serial ports.
 *
 * Don't mix calls to omap_serial_init_port() and omap_serial_init(),
 * use only one of the two.
 */
305
void __init omap_serial_init_port(struct omap_board_data *bdata)
306
{
307
	struct omap_uart_state *uart;
308
	struct omap_hwmod *oh;
309
	struct platform_device *pdev;
310 311 312 313
	void *pdata = NULL;
	u32 pdata_size = 0;
	char *name;
	struct omap_uart_port_info omap_up;
314

315
	if (WARN_ON(!bdata))
316
		return;
317 318 319
	if (WARN_ON(bdata->id < 0))
		return;
	if (WARN_ON(bdata->id >= num_uarts))
320
		return;
321

322
	list_for_each_entry(uart, &uart_list, node)
323
		if (bdata->id == uart->num)
324
			break;
325

326 327 328 329 330
	oh = uart->oh;
	name = DRIVER_NAME;

	omap_up.dma_enabled = uart->dma_enabled;
	omap_up.uartclk = OMAP24XX_BASE_BAUD * 16;
331
	omap_up.flags = UPF_BOOT_AUTOCONF;
332
	omap_up.get_context_loss_count = omap_pm_get_dev_context_loss_count;
333 334
	omap_up.set_forceidle = omap_uart_set_forceidle;
	omap_up.set_noidle = omap_uart_set_noidle;
335
	omap_up.enable_wakeup = omap_uart_enable_wakeup;
336 337 338 339 340 341 342 343

	/* Enable the MDR1 Errata i202 for OMAP2430/3xxx/44xx */
	if (!cpu_is_omap2420() && !cpu_is_ti816x())
		omap_up.errata |= UART_ERRATA_i202_MDR1_ACCESS;

	/* Enable DMA Mode Force Idle Errata i291 for omap34xx/3630 */
	if (cpu_is_omap34xx() || cpu_is_omap3630())
		omap_up.errata |= UART_ERRATA_i291_DMA_FORCEIDLE;
344 345 346 347 348 349 350

	pdata = &omap_up;
	pdata_size = sizeof(struct omap_uart_port_info);

	if (WARN_ON(!oh))
		return;

351
	pdev = omap_device_build(name, uart->num, oh, pdata, pdata_size,
352
				 NULL, 0, false);
353
	WARN(IS_ERR(pdev), "Could not build omap_device for %s: %s.\n",
354 355
	     name, oh->name);

356
	omap_device_disable_idle_on_suspend(pdev);
357 358
	oh->mux = omap_hwmod_mux_init(bdata->pads, bdata->pads_cnt);

359
	uart->pdev = pdev;
360 361 362

	oh->dev_attr = uart;

363
	console_lock(); /* in case the earlycon is on the UART */
364

365 366 367 368 369 370 371 372 373 374
	/*
	 * Because of early UART probing, UART did not get idled
	 * on init.  Now that omap_device is ready, ensure full idle
	 * before doing omap_device_enable().
	 */
	omap_hwmod_idle(uart->oh);

	omap_device_enable(uart->pdev);
	omap_device_idle(uart->pdev);

375
	console_unlock();
376

377
	if ((cpu_is_omap34xx() || cpu_is_omap44xx()) && bdata->pads)
378
		device_init_wakeup(&pdev->dev, true);
379 380 381
}

/**
382
 * omap_serial_init() - initialize all supported serial ports
383 384 385 386 387 388 389
 *
 * Initializes all available UARTs as serial ports. Platforms
 * can call this function when they want to have default behaviour
 * for serial ports (e.g initialize them all as serial ports).
 */
void __init omap_serial_init(void)
{
390
	struct omap_uart_state *uart;
391
	struct omap_board_data bdata;
392

393 394 395 396 397
	list_for_each_entry(uart, &uart_list, node) {
		bdata.id = uart->num;
		bdata.flags = 0;
		bdata.pads = NULL;
		bdata.pads_cnt = 0;
398 399 400 401

		if (cpu_is_omap44xx() || cpu_is_omap34xx())
			omap_serial_fill_default_pads(&bdata);

402 403 404
		omap_serial_init_port(&bdata);

	}
405
}