101 lines
		
	
	
	
		
			2.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			101 lines
		
	
	
	
		
			2.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /*
 | ||
|  |  * The platform specific code for virtex devices since a boot loader is not | ||
|  |  * always used. | ||
|  |  * | ||
|  |  * (C) Copyright 2008 Xilinx, Inc. | ||
|  |  * | ||
|  |  * This program is free software; you can redistribute it and/or modify it | ||
|  |  * under the terms of the GNU General Public License version 2 as published | ||
|  |  * by the Free Software Foundation. | ||
|  |  */ | ||
|  | 
 | ||
|  | #include "ops.h"
 | ||
|  | #include "io.h"
 | ||
|  | #include "stdio.h"
 | ||
|  | 
 | ||
|  | #define UART_DLL		0	/* Out: Divisor Latch Low */
 | ||
|  | #define UART_DLM		1	/* Out: Divisor Latch High */
 | ||
|  | #define UART_FCR		2	/* Out: FIFO Control Register */
 | ||
|  | #define UART_FCR_CLEAR_RCVR 	0x02 	/* Clear the RCVR FIFO */
 | ||
|  | #define UART_FCR_CLEAR_XMIT	0x04 	/* Clear the XMIT FIFO */
 | ||
|  | #define UART_LCR		3	/* Out: Line Control Register */
 | ||
|  | #define UART_MCR		4	/* Out: Modem Control Register */
 | ||
|  | #define UART_MCR_RTS		0x02 	/* RTS complement */
 | ||
|  | #define UART_MCR_DTR		0x01 	/* DTR complement */
 | ||
|  | #define UART_LCR_DLAB		0x80 	/* Divisor latch access bit */
 | ||
|  | #define UART_LCR_WLEN8		0x03 	/* Wordlength: 8 bits */
 | ||
|  | 
 | ||
|  | static int virtex_ns16550_console_init(void *devp) | ||
|  | { | ||
|  | 	unsigned char *reg_base; | ||
|  | 	u32 reg_shift, reg_offset, clk, spd; | ||
|  | 	u16 divisor; | ||
|  | 	int n; | ||
|  | 
 | ||
|  | 	if (dt_get_virtual_reg(devp, (void **)®_base, 1) < 1) | ||
|  | 		return -1; | ||
|  | 
 | ||
|  | 	n = getprop(devp, "reg-offset", ®_offset, sizeof(reg_offset)); | ||
|  | 	if (n == sizeof(reg_offset)) | ||
|  | 		reg_base += reg_offset; | ||
|  | 
 | ||
|  | 	n = getprop(devp, "reg-shift", ®_shift, sizeof(reg_shift)); | ||
|  | 	if (n != sizeof(reg_shift)) | ||
|  | 		reg_shift = 0; | ||
|  | 
 | ||
|  | 	n = getprop(devp, "current-speed", (void *)&spd, sizeof(spd)); | ||
|  | 	if (n != sizeof(spd)) | ||
|  | 		spd = 9600; | ||
|  | 
 | ||
|  | 	/* should there be a default clock rate?*/ | ||
|  | 	n = getprop(devp, "clock-frequency", (void *)&clk, sizeof(clk)); | ||
|  | 	if (n != sizeof(clk)) | ||
|  | 		return -1; | ||
|  | 
 | ||
|  | 	divisor = clk / (16 * spd); | ||
|  | 
 | ||
|  | 	/* Access baud rate */ | ||
|  | 	out_8(reg_base + (UART_LCR << reg_shift), UART_LCR_DLAB); | ||
|  | 
 | ||
|  | 	/* Baud rate based on input clock */ | ||
|  | 	out_8(reg_base + (UART_DLL << reg_shift), divisor & 0xFF); | ||
|  | 	out_8(reg_base + (UART_DLM << reg_shift), divisor >> 8); | ||
|  | 
 | ||
|  | 	/* 8 data, 1 stop, no parity */ | ||
|  | 	out_8(reg_base + (UART_LCR << reg_shift), UART_LCR_WLEN8); | ||
|  | 
 | ||
|  | 	/* RTS/DTR */ | ||
|  | 	out_8(reg_base + (UART_MCR << reg_shift), UART_MCR_RTS | UART_MCR_DTR); | ||
|  | 
 | ||
|  | 	/* Clear transmitter and receiver */ | ||
|  | 	out_8(reg_base + (UART_FCR << reg_shift), | ||
|  | 				UART_FCR_CLEAR_XMIT | UART_FCR_CLEAR_RCVR); | ||
|  | 	return 0; | ||
|  | } | ||
|  | 
 | ||
|  | /* For virtex, the kernel may be loaded without using a bootloader and if so
 | ||
|  |    some UARTs need more setup than is provided in the normal console init | ||
|  | */ | ||
|  | int platform_specific_init(void) | ||
|  | { | ||
|  | 	void *devp; | ||
|  | 	char devtype[MAX_PROP_LEN]; | ||
|  | 	char path[MAX_PATH_LEN]; | ||
|  | 
 | ||
|  | 	devp = finddevice("/chosen"); | ||
|  | 	if (devp == NULL) | ||
|  | 		return -1; | ||
|  | 
 | ||
|  | 	if (getprop(devp, "linux,stdout-path", path, MAX_PATH_LEN) > 0) { | ||
|  | 		devp = finddevice(path); | ||
|  | 		if (devp == NULL) | ||
|  | 			return -1; | ||
|  | 
 | ||
|  | 		if ((getprop(devp, "device_type", devtype, sizeof(devtype)) > 0) | ||
|  | 				&& !strcmp(devtype, "serial") | ||
|  | 				&& (dt_is_compatible(devp, "ns16550"))) | ||
|  | 				virtex_ns16550_console_init(devp); | ||
|  | 	} | ||
|  | 	return 0; | ||
|  | } |