diff options
Diffstat (limited to 'arch/arm/cpu/tegra-common')
| -rw-r--r-- | arch/arm/cpu/tegra-common/Makefile | 3 | ||||
| -rw-r--r-- | arch/arm/cpu/tegra-common/ap.c | 18 | ||||
| -rw-r--r-- | arch/arm/cpu/tegra-common/board.c | 10 | ||||
| -rw-r--r-- | arch/arm/cpu/tegra-common/cache.c | 10 | ||||
| -rw-r--r-- | arch/arm/cpu/tegra-common/clock.c | 128 | ||||
| -rw-r--r-- | arch/arm/cpu/tegra-common/sys_info.c | 2 | 
6 files changed, 145 insertions, 26 deletions
| diff --git a/arch/arm/cpu/tegra-common/Makefile b/arch/arm/cpu/tegra-common/Makefile index edfc1a83a95..34d57349afb 100644 --- a/arch/arm/cpu/tegra-common/Makefile +++ b/arch/arm/cpu/tegra-common/Makefile @@ -8,4 +8,5 @@  #  obj-y += lowlevel_init.o -obj-y	+= ap.o board.o sys_info.o clock.o cache.o +obj-y	+= ap.o board.o clock.o cache.o +obj-$(CONFIG_DISPLAY_CPUINFO) += sys_info.o diff --git a/arch/arm/cpu/tegra-common/ap.c b/arch/arm/cpu/tegra-common/ap.c index 60d71a6c30a..91d70da6566 100644 --- a/arch/arm/cpu/tegra-common/ap.c +++ b/arch/arm/cpu/tegra-common/ap.c @@ -1,5 +1,5 @@  /* -* (C) Copyright 2010-2011 +* (C) Copyright 2010-2014  * NVIDIA Corporation <www.nvidia.com>  *   * SPDX-License-Identifier:	GPL-2.0+ @@ -27,7 +27,7 @@ int tegra_get_chip(void)  	/*  	 * This is undocumented, Chip ID is bits 15:8 of the register  	 * APB_MISC + 0x804, and has value 0x20 for Tegra20, 0x30 for -	 * Tegra30, and 0x35 for T114. +	 * Tegra30, 0x35 for T114, and 0x40 for Tegra124.  	 */  	rev = (readl(&gp->hidrev) & HIDREV_CHIPID_MASK) >> HIDREV_CHIPID_SHIFT;  	debug("%s: CHIPID is 0x%02X\n", __func__, rev); @@ -72,6 +72,7 @@ int tegra_get_chip_sku(void)  		case SKU_ID_T33:  		case SKU_ID_T30:  		case SKU_ID_TM30MQS_P_A3: +		default:  			return TEGRA_SOC_T30;  		}  		break; @@ -79,10 +80,19 @@ int tegra_get_chip_sku(void)  		switch (sku_id) {  		case SKU_ID_T114_ENG:  		case SKU_ID_T114_1: +		default:  			return TEGRA_SOC_T114;  		}  		break; +	case CHIPID_TEGRA124: +		switch (sku_id) { +		case SKU_ID_T124_ENG: +		default: +			return TEGRA_SOC_T124; +		} +		break;  	} +  	/* unknown chip/sku id */  	printf("%s: ERROR: UNKNOWN CHIP/SKU ID COMBO (0x%02X/0x%02X)\n",  		__func__, chip_id, sku_id); @@ -117,8 +127,8 @@ static u32 get_odmdata(void)  	 * ODMDATA is stored in the BCT in IRAM by the BootROM.  	 * The BCT start and size are stored in the BIT in IRAM.  	 * Read the data @ bct_start + (bct_size - 12). This works -	 * on T20 and T30 BCTs, which are locked down. If this changes -	 * in new chips (T114, etc.), we can revisit this algorithm. +	 * on BCTs for currently supported SoCs, which are locked down. +	 * If this changes in new chips, we can revisit this algorithm.  	 */  	u32 bct_start, odmdata; diff --git a/arch/arm/cpu/tegra-common/board.c b/arch/arm/cpu/tegra-common/board.c index d9cbda8a749..6a6faf4b276 100644 --- a/arch/arm/cpu/tegra-common/board.c +++ b/arch/arm/cpu/tegra-common/board.c @@ -1,5 +1,5 @@  /* - *  (C) Copyright 2010,2011 + *  (C) Copyright 2010-2014   *  NVIDIA Corporation <www.nvidia.com>   *   * SPDX-License-Identifier:	GPL-2.0+ @@ -109,12 +109,18 @@ static int uart_configs[] = {  	-1,  	-1,  	-1, -#else	/* Tegra114 */ +#elif defined(CONFIG_TEGRA114)  	-1,  	-1,  	-1,  	FUNCMUX_UART4_GMI,	/* UARTD */  	-1, +#else	/* Tegra124 */ +	FUNCMUX_UART1_KBC,	/* UARTA */ +	-1, +	-1, +	FUNCMUX_UART4_GPIO,	/* UARTD */ +	-1,  #endif  }; diff --git a/arch/arm/cpu/tegra-common/cache.c b/arch/arm/cpu/tegra-common/cache.c index 48e9319c750..94f5bce90ec 100644 --- a/arch/arm/cpu/tegra-common/cache.c +++ b/arch/arm/cpu/tegra-common/cache.c @@ -1,5 +1,5 @@  /* - * Copyright (c) 2013, NVIDIA CORPORATION.  All rights reserved. + * Copyright (c) 2013-2014, NVIDIA CORPORATION.  All rights reserved.   *   * This program is free software; you can redistribute it and/or modify it   * under the terms and conditions of the GNU General Public License, @@ -23,8 +23,6 @@  void config_cache(void)  { -	struct apb_misc_gp_ctlr *gp = -		(struct apb_misc_gp_ctlr *)NV_PA_APB_MISC_GP_BASE;  	u32 reg = 0;  	/* enable SMP mode and FW for CPU0, by writing to Auxiliary Ctl reg */ @@ -33,10 +31,10 @@ void config_cache(void)  		"orr r0, r0, #0x41\n"  		"mcr p15, 0, r0, c1, c0, 1\n"); -	/* Currently, only T114 needs this L2 cache change to boot Linux */ -	reg = (readl(&gp->hidrev) & HIDREV_CHIPID_MASK); -	if (reg != (CHIPID_TEGRA114 << HIDREV_CHIPID_SHIFT)) +	/* Currently, only Tegra114+ needs this L2 cache change to boot Linux */ +	if (tegra_get_chip() < CHIPID_TEGRA114)  		return; +  	/*  	 * Systems with an architectural L2 cache must not use the PL310.  	 * Config L2CTLR here for a data RAM latency of 3 cycles. diff --git a/arch/arm/cpu/tegra-common/clock.c b/arch/arm/cpu/tegra-common/clock.c index 268fb912b50..11c7435505c 100644 --- a/arch/arm/cpu/tegra-common/clock.c +++ b/arch/arm/cpu/tegra-common/clock.c @@ -1,5 +1,5 @@  /* - * Copyright (c) 2010-2013, NVIDIA CORPORATION.  All rights reserved. + * Copyright (c) 2010-2014, NVIDIA CORPORATION.  All rights reserved.   *   * This program is free software; you can redistribute it and/or modify it   * under the terms and conditions of the GNU General Public License, @@ -142,8 +142,8 @@ void clock_ll_set_source_divisor(enum periph_id periph_id, unsigned source,  	value = readl(reg); -	value &= ~OUT_CLK_SOURCE_MASK; -	value |= source << OUT_CLK_SOURCE_SHIFT; +	value &= ~OUT_CLK_SOURCE_31_30_MASK; +	value |= source << OUT_CLK_SOURCE_31_30_SHIFT;  	value &= ~OUT_CLK_DIVISOR_MASK;  	value |= divisor << OUT_CLK_DIVISOR_SHIFT; @@ -155,8 +155,8 @@ void clock_ll_set_source(enum periph_id periph_id, unsigned source)  {  	u32 *reg = get_periph_source_reg(periph_id); -	clrsetbits_le32(reg, OUT_CLK_SOURCE_MASK, -			source << OUT_CLK_SOURCE_SHIFT); +	clrsetbits_le32(reg, OUT_CLK_SOURCE_31_30_MASK, +			source << OUT_CLK_SOURCE_31_30_SHIFT);  }  /** @@ -304,13 +304,27 @@ static int adjust_periph_pll(enum periph_id periph_id, int source,  	/* work out the source clock and set it */  	if (source < 0)  		return -1; -	if (mux_bits == 4) { -		clrsetbits_le32(reg, OUT_CLK_SOURCE4_MASK, -			source << OUT_CLK_SOURCE4_SHIFT); -	} else { -		clrsetbits_le32(reg, OUT_CLK_SOURCE_MASK, -			source << OUT_CLK_SOURCE_SHIFT); + +	switch (mux_bits) { +	case MASK_BITS_31_30: +		clrsetbits_le32(reg, OUT_CLK_SOURCE_31_30_MASK, +				source << OUT_CLK_SOURCE_31_30_SHIFT); +		break; + +	case MASK_BITS_31_29: +		clrsetbits_le32(reg, OUT_CLK_SOURCE_31_29_MASK, +				source << OUT_CLK_SOURCE_31_29_SHIFT); +		break; + +	case MASK_BITS_31_28: +		clrsetbits_le32(reg, OUT_CLK_SOURCE_31_28_MASK, +				source << OUT_CLK_SOURCE_31_28_SHIFT); +		break; + +	default: +		return -1;  	} +  	udelay(2);  	return 0;  } @@ -561,3 +575,95 @@ void clock_init(void)  	/* Do any special system timer/TSC setup */  	arch_timer_init();  } + +static void set_avp_clock_source(u32 src) +{ +	struct clk_rst_ctlr *clkrst = +			(struct clk_rst_ctlr *)NV_PA_CLK_RST_BASE; +	u32 val; + +	val = (src << SCLK_SWAKEUP_FIQ_SOURCE_SHIFT) | +		(src << SCLK_SWAKEUP_IRQ_SOURCE_SHIFT) | +		(src << SCLK_SWAKEUP_RUN_SOURCE_SHIFT) | +		(src << SCLK_SWAKEUP_IDLE_SOURCE_SHIFT) | +		(SCLK_SYS_STATE_RUN << SCLK_SYS_STATE_SHIFT); +	writel(val, &clkrst->crc_sclk_brst_pol); +	udelay(3); +} + +/* + * This function is useful on Tegra30, and any later SoCs that have compatible + * PLLP configuration registers. + */ +void tegra30_set_up_pllp(void) +{ +	struct clk_rst_ctlr *clkrst = (struct clk_rst_ctlr *)NV_PA_CLK_RST_BASE; +	u32 reg; + +	/* +	 * Based on the Tegra TRM, the system clock (which is the AVP clock) can +	 * run up to 275MHz. On power on, the default sytem clock source is set +	 * to PLLP_OUT0. This function sets PLLP's (hence PLLP_OUT0's) rate to +	 * 408MHz which is beyond system clock's upper limit. +	 * +	 * The fix is to set the system clock to CLK_M before initializing PLLP, +	 * and then switch back to PLLP_OUT4, which has an appropriate divider +	 * configured, after PLLP has been configured +	 */ +	set_avp_clock_source(SCLK_SOURCE_CLKM); + +	/* +	 * PLLP output frequency set to 408Mhz +	 * PLLC output frequency set to 228Mhz +	 */ +	switch (clock_get_osc_freq()) { +	case CLOCK_OSC_FREQ_12_0: /* OSC is 12Mhz */ +		clock_set_rate(CLOCK_ID_PERIPH, 408, 12, 0, 8); +		clock_set_rate(CLOCK_ID_CGENERAL, 456, 12, 1, 8); +		break; + +	case CLOCK_OSC_FREQ_26_0: /* OSC is 26Mhz */ +		clock_set_rate(CLOCK_ID_PERIPH, 408, 26, 0, 8); +		clock_set_rate(CLOCK_ID_CGENERAL, 600, 26, 0, 8); +		break; + +	case CLOCK_OSC_FREQ_13_0: /* OSC is 13Mhz */ +		clock_set_rate(CLOCK_ID_PERIPH, 408, 13, 0, 8); +		clock_set_rate(CLOCK_ID_CGENERAL, 600, 13, 0, 8); +		break; +	case CLOCK_OSC_FREQ_19_2: +	default: +		/* +		 * These are not supported. It is too early to print a +		 * message and the UART likely won't work anyway due to the +		 * oscillator being wrong. +		 */ +		break; +	} + +	/* Set PLLP_OUT1, 2, 3 & 4 freqs to 9.6, 48, 102 & 204MHz */ + +	/* OUT1, 2 */ +	/* Assert RSTN before enable */ +	reg = PLLP_OUT2_RSTN_EN | PLLP_OUT1_RSTN_EN; +	writel(reg, &clkrst->crc_pll[CLOCK_ID_PERIPH].pll_out[0]); +	/* Set divisor and reenable */ +	reg = (IN_408_OUT_48_DIVISOR << PLLP_OUT2_RATIO) +		| PLLP_OUT2_OVR | PLLP_OUT2_CLKEN | PLLP_OUT2_RSTN_DIS +		| (IN_408_OUT_9_6_DIVISOR << PLLP_OUT1_RATIO) +		| PLLP_OUT1_OVR | PLLP_OUT1_CLKEN | PLLP_OUT1_RSTN_DIS; +	writel(reg, &clkrst->crc_pll[CLOCK_ID_PERIPH].pll_out[0]); + +	/* OUT3, 4 */ +	/* Assert RSTN before enable */ +	reg = PLLP_OUT4_RSTN_EN | PLLP_OUT3_RSTN_EN; +	writel(reg, &clkrst->crc_pll[CLOCK_ID_PERIPH].pll_out[1]); +	/* Set divisor and reenable */ +	reg = (IN_408_OUT_204_DIVISOR << PLLP_OUT4_RATIO) +		| PLLP_OUT4_OVR | PLLP_OUT4_CLKEN | PLLP_OUT4_RSTN_DIS +		| (IN_408_OUT_102_DIVISOR << PLLP_OUT3_RATIO) +		| PLLP_OUT3_OVR | PLLP_OUT3_CLKEN | PLLP_OUT3_RSTN_DIS; +	writel(reg, &clkrst->crc_pll[CLOCK_ID_PERIPH].pll_out[1]); + +	set_avp_clock_source(SCLK_SOURCE_PLLP_OUT4); +} diff --git a/arch/arm/cpu/tegra-common/sys_info.c b/arch/arm/cpu/tegra-common/sys_info.c index dc8a2e4d316..de20325ecf3 100644 --- a/arch/arm/cpu/tegra-common/sys_info.c +++ b/arch/arm/cpu/tegra-common/sys_info.c @@ -8,7 +8,6 @@  #include <common.h>  #include <linux/ctype.h> -#ifdef CONFIG_DISPLAY_CPUINFO  void upstring(char *s)  {  	while (*s) { @@ -30,4 +29,3 @@ int print_cpuinfo(void)  	/* TBD: Add printf of major/minor rev info, stepping, etc. */  	return 0;  } -#endif	/* CONFIG_DISPLAY_CPUINFO */ | 
