/* * arch/arm/mach-tegra/tegra30_clocks.c * * Copyright (c) 2010-2012 NVIDIA CORPORATION. All rights reserved. * * 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 program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "clock.h" #include "fuse.h" #include "iomap.h" #define USE_PLL_LOCK_BITS 0 #define RST_DEVICES_L 0x004 #define RST_DEVICES_H 0x008 #define RST_DEVICES_U 0x00C #define RST_DEVICES_V 0x358 #define RST_DEVICES_W 0x35C #define RST_DEVICES_SET_L 0x300 #define RST_DEVICES_CLR_L 0x304 #define RST_DEVICES_SET_V 0x430 #define RST_DEVICES_CLR_V 0x434 #define RST_DEVICES_NUM 5 #define CLK_OUT_ENB_L 0x010 #define CLK_OUT_ENB_H 0x014 #define CLK_OUT_ENB_U 0x018 #define CLK_OUT_ENB_V 0x360 #define CLK_OUT_ENB_W 0x364 #define CLK_OUT_ENB_SET_L 0x320 #define CLK_OUT_ENB_CLR_L 0x324 #define CLK_OUT_ENB_SET_V 0x440 #define CLK_OUT_ENB_CLR_V 0x444 #define CLK_OUT_ENB_NUM 5 #define RST_DEVICES_V_SWR_CPULP_RST_DIS (0x1 << 1) #define CLK_OUT_ENB_V_CLK_ENB_CPULP_EN (0x1 << 1) #define PERIPH_CLK_TO_BIT(c) (1 << (c->u.periph.clk_num % 32)) #define PERIPH_CLK_TO_RST_REG(c) \ periph_clk_to_reg((c), RST_DEVICES_L, RST_DEVICES_V, 4) #define PERIPH_CLK_TO_RST_SET_REG(c) \ periph_clk_to_reg((c), RST_DEVICES_SET_L, RST_DEVICES_SET_V, 8) #define PERIPH_CLK_TO_RST_CLR_REG(c) \ periph_clk_to_reg((c), RST_DEVICES_CLR_L, RST_DEVICES_CLR_V, 8) #define PERIPH_CLK_TO_ENB_REG(c) \ periph_clk_to_reg((c), CLK_OUT_ENB_L, CLK_OUT_ENB_V, 4) #define PERIPH_CLK_TO_ENB_SET_REG(c) \ periph_clk_to_reg((c), CLK_OUT_ENB_SET_L, CLK_OUT_ENB_SET_V, 8) #define PERIPH_CLK_TO_ENB_CLR_REG(c) \ periph_clk_to_reg((c), CLK_OUT_ENB_CLR_L, CLK_OUT_ENB_CLR_V, 8) #define CLK_MASK_ARM 0x44 #define MISC_CLK_ENB 0x48 #define OSC_CTRL 0x50 #define OSC_CTRL_OSC_FREQ_MASK (0xF<<28) #define OSC_CTRL_OSC_FREQ_13MHZ (0x0<<28) #define OSC_CTRL_OSC_FREQ_19_2MHZ (0x4<<28) #define OSC_CTRL_OSC_FREQ_12MHZ (0x8<<28) #define OSC_CTRL_OSC_FREQ_26MHZ (0xC<<28) #define OSC_CTRL_OSC_FREQ_16_8MHZ (0x1<<28) #define OSC_CTRL_OSC_FREQ_38_4MHZ (0x5<<28) #define OSC_CTRL_OSC_FREQ_48MHZ (0x9<<28) #define OSC_CTRL_MASK (0x3f2 | OSC_CTRL_OSC_FREQ_MASK) #define OSC_CTRL_PLL_REF_DIV_MASK (3<<26) #define OSC_CTRL_PLL_REF_DIV_1 (0<<26) #define OSC_CTRL_PLL_REF_DIV_2 (1<<26) #define OSC_CTRL_PLL_REF_DIV_4 (2<<26) #define OSC_FREQ_DET 0x58 #define OSC_FREQ_DET_TRIG (1<<31) #define OSC_FREQ_DET_STATUS 0x5C #define OSC_FREQ_DET_BUSY (1<<31) #define OSC_FREQ_DET_CNT_MASK 0xFFFF #define PERIPH_CLK_SOURCE_I2S1 0x100 #define PERIPH_CLK_SOURCE_EMC 0x19c #define PERIPH_CLK_SOURCE_OSC 0x1fc #define PERIPH_CLK_SOURCE_NUM1 \ ((PERIPH_CLK_SOURCE_OSC - PERIPH_CLK_SOURCE_I2S1) / 4) #define PERIPH_CLK_SOURCE_G3D2 0x3b0 #define PERIPH_CLK_SOURCE_SE 0x42c #define PERIPH_CLK_SOURCE_NUM2 \ ((PERIPH_CLK_SOURCE_SE - PERIPH_CLK_SOURCE_G3D2) / 4 + 1) #define AUDIO_DLY_CLK 0x49c #define AUDIO_SYNC_CLK_SPDIF 0x4b4 #define PERIPH_CLK_SOURCE_NUM3 \ ((AUDIO_SYNC_CLK_SPDIF - AUDIO_DLY_CLK) / 4 + 1) #define PERIPH_CLK_SOURCE_NUM (PERIPH_CLK_SOURCE_NUM1 + \ PERIPH_CLK_SOURCE_NUM2 + \ PERIPH_CLK_SOURCE_NUM3) #define CPU_SOFTRST_CTRL 0x380 #define PERIPH_CLK_SOURCE_DIVU71_MASK 0xFF #define PERIPH_CLK_SOURCE_DIVU16_MASK 0xFFFF #define PERIPH_CLK_SOURCE_DIV_SHIFT 0 #define PERIPH_CLK_SOURCE_DIVIDLE_SHIFT 8 #define PERIPH_CLK_SOURCE_DIVIDLE_VAL 50 #define PERIPH_CLK_UART_DIV_ENB (1<<24) #define PERIPH_CLK_VI_SEL_EX_SHIFT 24 #define PERIPH_CLK_VI_SEL_EX_MASK (0x3<reg_shift - 24) * 4) #define PLL_BASE 0x0 #define PLL_BASE_BYPASS (1<<31) #define PLL_BASE_ENABLE (1<<30) #define PLL_BASE_REF_ENABLE (1<<29) #define PLL_BASE_OVERRIDE (1<<28) #define PLL_BASE_LOCK (1<<27) #define PLL_BASE_DIVP_MASK (0x7<<20) #define PLL_BASE_DIVP_SHIFT 20 #define PLL_BASE_DIVN_MASK (0x3FF<<8) #define PLL_BASE_DIVN_SHIFT 8 #define PLL_BASE_DIVM_MASK (0x1F) #define PLL_BASE_DIVM_SHIFT 0 #define PLL_OUT_RATIO_MASK (0xFF<<8) #define PLL_OUT_RATIO_SHIFT 8 #define PLL_OUT_OVERRIDE (1<<2) #define PLL_OUT_CLKEN (1<<1) #define PLL_OUT_RESET_DISABLE (1<<0) #define PLL_MISC(c) \ (((c)->flags & PLL_ALT_MISC_REG) ? 0x4 : 0xc) #define PLL_MISC_LOCK_ENABLE(c) \ (((c)->flags & (PLLU | PLLD)) ? (1<<22) : (1<<18)) #define PLL_MISC_DCCON_SHIFT 20 #define PLL_MISC_CPCON_SHIFT 8 #define PLL_MISC_CPCON_MASK (0xF<hw.clk; return clk_set_rate(clk, (__clk_get_rate(__clk_get_parent(clk)) + n - 1) / n); } static inline u32 periph_clk_to_reg( struct clk_tegra *c, u32 reg_L, u32 reg_V, int offs) { u32 reg = c->u.periph.clk_num / 32; BUG_ON(reg >= RST_DEVICES_NUM); if (reg < 3) reg = reg_L + (reg * offs); else reg = reg_V + ((reg - 3) * offs); return reg; } static unsigned long clk_measure_input_freq(void) { u32 clock_autodetect; clk_writel(OSC_FREQ_DET_TRIG | 1, OSC_FREQ_DET); do {} while (clk_readl(OSC_FREQ_DET_STATUS) & OSC_FREQ_DET_BUSY); clock_autodetect = clk_readl(OSC_FREQ_DET_STATUS); if (clock_autodetect >= 732 - 3 && clock_autodetect <= 732 + 3) { return 12000000; } else if (clock_autodetect >= 794 - 3 && clock_autodetect <= 794 + 3) { return 13000000; } else if (clock_autodetect >= 1172 - 3 && clock_autodetect <= 1172 + 3) { return 19200000; } else if (clock_autodetect >= 1587 - 3 && clock_autodetect <= 1587 + 3) { return 26000000; } else if (clock_autodetect >= 1025 - 3 && clock_autodetect <= 1025 + 3) { return 16800000; } else if (clock_autodetect >= 2344 - 3 && clock_autodetect <= 2344 + 3) { return 38400000; } else if (clock_autodetect >= 2928 - 3 && clock_autodetect <= 2928 + 3) { return 48000000; } else { pr_err("%s: Unexpected clock autodetect value %d", __func__, clock_autodetect); BUG(); return 0; } } static int clk_div71_get_divider(unsigned long parent_rate, unsigned long rate, u32 flags, u32 round_mode) { s64 divider_u71 = parent_rate; if (!rate) return -EINVAL; if (!(flags & DIV_U71_INT)) divider_u71 *= 2; if (round_mode == ROUND_DIVIDER_UP) divider_u71 += rate - 1; do_div(divider_u71, rate); if (flags & DIV_U71_INT) divider_u71 *= 2; if (divider_u71 - 2 < 0) return 0; if (divider_u71 - 2 > 255) return -EINVAL; return divider_u71 - 2; } static int clk_div16_get_divider(unsigned long parent_rate, unsigned long rate) { s64 divider_u16; divider_u16 = parent_rate; if (!rate) return -EINVAL; divider_u16 += rate - 1; do_div(divider_u16, rate); if (divider_u16 - 1 < 0) return 0; if (divider_u16 - 1 > 0xFFFF) return -EINVAL; return divider_u16 - 1; } static unsigned long tegra30_clk_fixed_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { return to_clk_tegra(hw)->fixed_rate; } struct clk_ops tegra30_clk_32k_ops = { .recalc_rate = tegra30_clk_fixed_recalc_rate, }; /* clk_m functions */ static unsigned long tegra30_clk_m_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { if (!to_clk_tegra(hw)->fixed_rate) to_clk_tegra(hw)->fixed_rate = clk_measure_input_freq(); return to_clk_tegra(hw)->fixed_rate; } static void tegra30_clk_m_init(struct clk_hw *hw) { u32 osc_ctrl = clk_readl(OSC_CTRL); u32 auto_clock_control = osc_ctrl & ~OSC_CTRL_OSC_FREQ_MASK; u32 pll_ref_div = osc_ctrl & OSC_CTRL_PLL_REF_DIV_MASK; switch (to_clk_tegra(hw)->fixed_rate) { case 12000000: auto_clock_control |= OSC_CTRL_OSC_FREQ_12MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1); break; case 13000000: auto_clock_control |= OSC_CTRL_OSC_FREQ_13MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1); break; case 19200000: auto_clock_control |= OSC_CTRL_OSC_FREQ_19_2MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1); break; case 26000000: auto_clock_control |= OSC_CTRL_OSC_FREQ_26MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1); break; case 16800000: auto_clock_control |= OSC_CTRL_OSC_FREQ_16_8MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1); break; case 38400000: auto_clock_control |= OSC_CTRL_OSC_FREQ_38_4MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_2); break; case 48000000: auto_clock_control |= OSC_CTRL_OSC_FREQ_48MHZ; BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_4); break; default: pr_err("%s: Unexpected clock rate %ld", __func__, to_clk_tegra(hw)->fixed_rate); BUG(); } clk_writel(auto_clock_control, OSC_CTRL); } struct clk_ops tegra30_clk_m_ops = { .init = tegra30_clk_m_init, .recalc_rate = tegra30_clk_m_recalc_rate, }; static unsigned long tegra30_clk_m_div_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } struct clk_ops tegra_clk_m_div_ops = { .recalc_rate = tegra30_clk_m_div_recalc_rate, }; /* PLL reference divider functions */ static unsigned long tegra30_pll_ref_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long rate = parent_rate; u32 pll_ref_div = clk_readl(OSC_CTRL) & OSC_CTRL_PLL_REF_DIV_MASK; switch (pll_ref_div) { case OSC_CTRL_PLL_REF_DIV_1: c->div = 1; break; case OSC_CTRL_PLL_REF_DIV_2: c->div = 2; break; case OSC_CTRL_PLL_REF_DIV_4: c->div = 4; break; default: pr_err("%s: Invalid pll ref divider %d", __func__, pll_ref_div); BUG(); } c->mul = 1; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } struct clk_ops tegra_pll_ref_ops = { .recalc_rate = tegra30_pll_ref_recalc_rate, }; /* super clock functions */ /* "super clocks" on tegra30 have two-stage muxes, fractional 7.1 divider and * clock skipping super divider. We will ignore the clock skipping divider, * since we can't lower the voltage when using the clock skip, but we can if * we lower the PLL frequency. We will use 7.1 divider for CPU super-clock * only when its parent is a fixed rate PLL, since we can't change PLL rate * in this case. */ static void tegra30_super_clk_init(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); struct clk_tegra *p = to_clk_tegra(__clk_get_hw(__clk_get_parent(hw->clk))); c->state = ON; if (c->flags & DIV_U71) { /* Init safe 7.1 divider value (does not affect PLLX path) */ clk_writel(SUPER_CLOCK_DIV_U71_MIN << SUPER_CLOCK_DIV_U71_SHIFT, c->reg + SUPER_CLK_DIVIDER); c->mul = 2; c->div = 2; if (!(p->flags & PLLX)) c->div += SUPER_CLOCK_DIV_U71_MIN; } else clk_writel(0, c->reg + SUPER_CLK_DIVIDER); } static u8 tegra30_super_clk_get_parent(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; int source; int shift; val = clk_readl(c->reg + SUPER_CLK_MUX); BUG_ON(((val & SUPER_STATE_MASK) != SUPER_STATE_RUN) && ((val & SUPER_STATE_MASK) != SUPER_STATE_IDLE)); shift = ((val & SUPER_STATE_MASK) == SUPER_STATE_IDLE) ? SUPER_IDLE_SOURCE_SHIFT : SUPER_RUN_SOURCE_SHIFT; source = (val >> shift) & SUPER_SOURCE_MASK; if (c->flags & DIV_2) source |= val & SUPER_LP_DIV2_BYPASS; return source; } static int tegra30_super_clk_set_parent(struct clk_hw *hw, u8 index) { struct clk_tegra *c = to_clk_tegra(hw); struct clk_tegra *p = to_clk_tegra(__clk_get_hw(clk_get_parent(hw->clk))); u32 val; int shift; val = clk_readl(c->reg + SUPER_CLK_MUX); BUG_ON(((val & SUPER_STATE_MASK) != SUPER_STATE_RUN) && ((val & SUPER_STATE_MASK) != SUPER_STATE_IDLE)); shift = ((val & SUPER_STATE_MASK) == SUPER_STATE_IDLE) ? SUPER_IDLE_SOURCE_SHIFT : SUPER_RUN_SOURCE_SHIFT; /* For LP mode super-clock switch between PLLX direct and divided-by-2 outputs is allowed only when other than PLLX clock source is current parent */ if ((c->flags & DIV_2) && (p->flags & PLLX) && ((index ^ val) & SUPER_LP_DIV2_BYPASS)) { if (p->flags & PLLX) return -EINVAL; val ^= SUPER_LP_DIV2_BYPASS; clk_writel_delay(val, c->reg); } val &= ~(SUPER_SOURCE_MASK << shift); val |= (index & SUPER_SOURCE_MASK) << shift; /* 7.1 divider for CPU super-clock does not affect PLLX path */ if (c->flags & DIV_U71) { u32 div = 0; if (!(p->flags & PLLX)) { div = clk_readl(c->reg + SUPER_CLK_DIVIDER); div &= SUPER_CLOCK_DIV_U71_MASK; div >>= SUPER_CLOCK_DIV_U71_SHIFT; } c->div = div + 2; c->mul = 2; } clk_writel_delay(val, c->reg); return 0; } /* * Do not use super clocks "skippers", since dividing using a clock skipper * does not allow the voltage to be scaled down. Instead adjust the rate of * the parent clock. This requires that the parent of a super clock have no * other children, otherwise the rate will change underneath the other * children. Special case: if fixed rate PLL is CPU super clock parent the * rate of this PLL can't be changed, and it has many other children. In * this case use 7.1 fractional divider to adjust the super clock rate. */ static int tegra30_super_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); struct clk *parent = __clk_get_parent(hw->clk); struct clk_tegra *cparent = to_clk_tegra(__clk_get_hw(parent)); if ((c->flags & DIV_U71) && (cparent->flags & PLL_FIXED)) { int div = clk_div71_get_divider(parent_rate, rate, c->flags, ROUND_DIVIDER_DOWN); div = max(div, SUPER_CLOCK_DIV_U71_MIN); clk_writel(div << SUPER_CLOCK_DIV_U71_SHIFT, c->reg + SUPER_CLK_DIVIDER); c->div = div + 2; c->mul = 2; return 0; } return 0; } static unsigned long tegra30_super_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } static long tegra30_super_clk_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { struct clk_tegra *c = to_clk_tegra(hw); struct clk *parent = __clk_get_parent(hw->clk); struct clk_tegra *cparent = to_clk_tegra(__clk_get_hw(parent)); int mul = 2; int div; if ((c->flags & DIV_U71) && (cparent->flags & PLL_FIXED)) { div = clk_div71_get_divider(*prate, rate, c->flags, ROUND_DIVIDER_DOWN); div = max(div, SUPER_CLOCK_DIV_U71_MIN) + 2; rate = *prate * mul; rate += div - 1; /* round up */ do_div(rate, c->div); return rate; } return *prate; } struct clk_ops tegra30_super_ops = { .init = tegra30_super_clk_init, .set_parent = tegra30_super_clk_set_parent, .get_parent = tegra30_super_clk_get_parent, .recalc_rate = tegra30_super_clk_recalc_rate, .round_rate = tegra30_super_clk_round_rate, .set_rate = tegra30_super_clk_set_rate, }; static unsigned long tegra30_twd_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } struct clk_ops tegra30_twd_ops = { .recalc_rate = tegra30_twd_clk_recalc_rate, }; /* bus clock functions */ static int tegra30_bus_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); c->state = ((val >> c->reg_shift) & BUS_CLK_DISABLE) ? OFF : ON; return c->state; } static int tegra30_bus_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; val = clk_readl(c->reg); val &= ~(BUS_CLK_DISABLE << c->reg_shift); clk_writel(val, c->reg); return 0; } static void tegra30_bus_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; val = clk_readl(c->reg); val |= BUS_CLK_DISABLE << c->reg_shift; clk_writel(val, c->reg); } static unsigned long tegra30_bus_clk_recalc_rate(struct clk_hw *hw, unsigned long prate) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); u64 rate = prate; c->div = ((val >> c->reg_shift) & BUS_CLK_DIV_MASK) + 1; c->mul = 1; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } static int tegra30_bus_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); int ret = -EINVAL; u32 val; int i; val = clk_readl(c->reg); for (i = 1; i <= 4; i++) { if (rate == parent_rate / i) { val &= ~(BUS_CLK_DIV_MASK << c->reg_shift); val |= (i - 1) << c->reg_shift; clk_writel(val, c->reg); c->div = i; c->mul = 1; ret = 0; break; } } return ret; } static long tegra30_bus_clk_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { unsigned long parent_rate = *prate; s64 divider; if (rate >= parent_rate) return parent_rate; divider = parent_rate; divider += rate - 1; do_div(divider, rate); if (divider < 0) return divider; if (divider > 4) divider = 4; do_div(parent_rate, divider); return parent_rate; } struct clk_ops tegra30_bus_ops = { .is_enabled = tegra30_bus_clk_is_enabled, .enable = tegra30_bus_clk_enable, .disable = tegra30_bus_clk_disable, .set_rate = tegra30_bus_clk_set_rate, .round_rate = tegra30_bus_clk_round_rate, .recalc_rate = tegra30_bus_clk_recalc_rate, }; /* Blink output functions */ static int tegra30_blink_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; val = pmc_readl(PMC_CTRL); c->state = (val & PMC_CTRL_BLINK_ENB) ? ON : OFF; return c->state; } static int tegra30_blink_clk_enable(struct clk_hw *hw) { u32 val; val = pmc_readl(PMC_DPD_PADS_ORIDE); pmc_writel(val | PMC_DPD_PADS_ORIDE_BLINK_ENB, PMC_DPD_PADS_ORIDE); val = pmc_readl(PMC_CTRL); pmc_writel(val | PMC_CTRL_BLINK_ENB, PMC_CTRL); return 0; } static void tegra30_blink_clk_disable(struct clk_hw *hw) { u32 val; val = pmc_readl(PMC_CTRL); pmc_writel(val & ~PMC_CTRL_BLINK_ENB, PMC_CTRL); val = pmc_readl(PMC_DPD_PADS_ORIDE); pmc_writel(val & ~PMC_DPD_PADS_ORIDE_BLINK_ENB, PMC_DPD_PADS_ORIDE); } static int tegra30_blink_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); if (rate >= parent_rate) { c->div = 1; pmc_writel(0, c->reg); } else { unsigned int on_off; u32 val; on_off = DIV_ROUND_UP(parent_rate / 8, rate); c->div = on_off * 8; val = (on_off & PMC_BLINK_TIMER_DATA_ON_MASK) << PMC_BLINK_TIMER_DATA_ON_SHIFT; on_off &= PMC_BLINK_TIMER_DATA_OFF_MASK; on_off <<= PMC_BLINK_TIMER_DATA_OFF_SHIFT; val |= on_off; val |= PMC_BLINK_TIMER_ENB; pmc_writel(val, c->reg); } return 0; } static unsigned long tegra30_blink_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; u32 val; u32 mul; u32 div; u32 on_off; mul = 1; val = pmc_readl(c->reg); if (val & PMC_BLINK_TIMER_ENB) { on_off = (val >> PMC_BLINK_TIMER_DATA_ON_SHIFT) & PMC_BLINK_TIMER_DATA_ON_MASK; val >>= PMC_BLINK_TIMER_DATA_OFF_SHIFT; val &= PMC_BLINK_TIMER_DATA_OFF_MASK; on_off += val; /* each tick in the blink timer is 4 32KHz clocks */ div = on_off * 4; } else { div = 1; } if (mul != 0 && div != 0) { rate *= mul; rate += div - 1; /* round up */ do_div(rate, div); } return rate; } static long tegra30_blink_clk_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { int div; int mul; long round_rate = *prate; mul = 1; if (rate >= *prate) { div = 1; } else { div = DIV_ROUND_UP(*prate / 8, rate); div *= 8; } round_rate *= mul; round_rate += div - 1; do_div(round_rate, div); return round_rate; } struct clk_ops tegra30_blink_clk_ops = { .is_enabled = tegra30_blink_clk_is_enabled, .enable = tegra30_blink_clk_enable, .disable = tegra30_blink_clk_disable, .recalc_rate = tegra30_blink_clk_recalc_rate, .round_rate = tegra30_blink_clk_round_rate, .set_rate = tegra30_blink_clk_set_rate, }; static void tegra30_utmi_param_configure(struct clk_hw *hw) { unsigned long main_rate = __clk_get_rate(__clk_get_parent(__clk_get_parent(hw->clk))); u32 reg; int i; for (i = 0; i < ARRAY_SIZE(utmi_parameters); i++) { if (main_rate == utmi_parameters[i].osc_frequency) break; } if (i >= ARRAY_SIZE(utmi_parameters)) { pr_err("%s: Unexpected main rate %lu\n", __func__, main_rate); return; } reg = clk_readl(UTMIP_PLL_CFG2); /* Program UTMIP PLL stable and active counts */ /* [FIXME] arclk_rst.h says WRONG! This should be 1ms -> 0x50 Check! */ reg &= ~UTMIP_PLL_CFG2_STABLE_COUNT(~0); reg |= UTMIP_PLL_CFG2_STABLE_COUNT( utmi_parameters[i].stable_count); reg &= ~UTMIP_PLL_CFG2_ACTIVE_DLY_COUNT(~0); reg |= UTMIP_PLL_CFG2_ACTIVE_DLY_COUNT( utmi_parameters[i].active_delay_count); /* Remove power downs from UTMIP PLL control bits */ reg &= ~UTMIP_PLL_CFG2_FORCE_PD_SAMP_A_POWERDOWN; reg &= ~UTMIP_PLL_CFG2_FORCE_PD_SAMP_B_POWERDOWN; reg &= ~UTMIP_PLL_CFG2_FORCE_PD_SAMP_C_POWERDOWN; clk_writel(reg, UTMIP_PLL_CFG2); /* Program UTMIP PLL delay and oscillator frequency counts */ reg = clk_readl(UTMIP_PLL_CFG1); reg &= ~UTMIP_PLL_CFG1_ENABLE_DLY_COUNT(~0); reg |= UTMIP_PLL_CFG1_ENABLE_DLY_COUNT( utmi_parameters[i].enable_delay_count); reg &= ~UTMIP_PLL_CFG1_XTAL_FREQ_COUNT(~0); reg |= UTMIP_PLL_CFG1_XTAL_FREQ_COUNT( utmi_parameters[i].xtal_freq_count); /* Remove power downs from UTMIP PLL control bits */ reg &= ~UTMIP_PLL_CFG1_FORCE_PLL_ENABLE_POWERDOWN; reg &= ~UTMIP_PLL_CFG1_FORCE_PLL_ACTIVE_POWERDOWN; reg &= ~UTMIP_PLL_CFG1_FORCE_PLLU_POWERDOWN; clk_writel(reg, UTMIP_PLL_CFG1); } /* PLL Functions */ static int tegra30_pll_clk_wait_for_lock(struct clk_tegra *c, u32 lock_reg, u32 lock_bit) { int ret = 0; #if USE_PLL_LOCK_BITS int i; for (i = 0; i < c->u.pll.lock_delay; i++) { if (clk_readl(lock_reg) & lock_bit) { udelay(PLL_POST_LOCK_DELAY); return 0; } udelay(2); /* timeout = 2 * lock time */ } pr_err("Timed out waiting for lock bit on pll %s", __clk_get_name(hw->clk)); ret = -1; #else udelay(c->u.pll.lock_delay); #endif return ret; } static int tegra30_pll_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg + PLL_BASE); c->state = (val & PLL_BASE_ENABLE) ? ON : OFF; return c->state; } static void tegra30_pll_clk_init(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); if (c->flags & PLLU) tegra30_utmi_param_configure(hw); } static int tegra30_pll_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; pr_debug("%s on clock %s\n", __func__, __clk_get_name(hw->clk)); #if USE_PLL_LOCK_BITS val = clk_readl(c->reg + PLL_MISC(c)); val |= PLL_MISC_LOCK_ENABLE(c); clk_writel(val, c->reg + PLL_MISC(c)); #endif val = clk_readl(c->reg + PLL_BASE); val &= ~PLL_BASE_BYPASS; val |= PLL_BASE_ENABLE; clk_writel(val, c->reg + PLL_BASE); if (c->flags & PLLM) { val = pmc_readl(PMC_PLLP_WB0_OVERRIDE); val |= PMC_PLLP_WB0_OVERRIDE_PLLM_ENABLE; pmc_writel(val, PMC_PLLP_WB0_OVERRIDE); } tegra30_pll_clk_wait_for_lock(c, c->reg + PLL_BASE, PLL_BASE_LOCK); return 0; } static void tegra30_pll_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; pr_debug("%s on clock %s\n", __func__, __clk_get_name(hw->clk)); val = clk_readl(c->reg); val &= ~(PLL_BASE_BYPASS | PLL_BASE_ENABLE); clk_writel(val, c->reg); if (c->flags & PLLM) { val = pmc_readl(PMC_PLLP_WB0_OVERRIDE); val &= ~PMC_PLLP_WB0_OVERRIDE_PLLM_ENABLE; pmc_writel(val, PMC_PLLP_WB0_OVERRIDE); } } static int tegra30_pll_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u32 val, p_div, old_base; unsigned long input_rate; const struct clk_pll_freq_table *sel; struct clk_pll_freq_table cfg; if (c->flags & PLL_FIXED) { int ret = 0; if (rate != c->u.pll.fixed_rate) { pr_err("%s: Can not change %s fixed rate %lu to %lu\n", __func__, __clk_get_name(hw->clk), c->u.pll.fixed_rate, rate); ret = -EINVAL; } return ret; } if (c->flags & PLLM) { if (rate != __clk_get_rate(hw->clk)) { pr_err("%s: Can not change memory %s rate in flight\n", __func__, __clk_get_name(hw->clk)); return -EINVAL; } } p_div = 0; input_rate = parent_rate; /* Check if the target rate is tabulated */ for (sel = c->u.pll.freq_table; sel->input_rate != 0; sel++) { if (sel->input_rate == input_rate && sel->output_rate == rate) { if (c->flags & PLLU) { BUG_ON(sel->p < 1 || sel->p > 2); if (sel->p == 1) p_div = PLLU_BASE_POST_DIV; } else { BUG_ON(sel->p < 1); for (val = sel->p; val > 1; val >>= 1) p_div++; p_div <<= PLL_BASE_DIVP_SHIFT; } break; } } /* Configure out-of-table rate */ if (sel->input_rate == 0) { unsigned long cfreq; BUG_ON(c->flags & PLLU); sel = &cfg; switch (input_rate) { case 12000000: case 26000000: cfreq = (rate <= 1000000 * 1000) ? 1000000 : 2000000; break; case 13000000: cfreq = (rate <= 1000000 * 1000) ? 1000000 : 2600000; break; case 16800000: case 19200000: cfreq = (rate <= 1200000 * 1000) ? 1200000 : 2400000; break; default: pr_err("%s: Unexpected reference rate %lu\n", __func__, input_rate); BUG(); } /* Raise VCO to guarantee 0.5% accuracy */ for (cfg.output_rate = rate; cfg.output_rate < 200 * cfreq; cfg.output_rate <<= 1) p_div++; cfg.p = 0x1 << p_div; cfg.m = input_rate / cfreq; cfg.n = cfg.output_rate / cfreq; cfg.cpcon = OUT_OF_TABLE_CPCON; if ((cfg.m > (PLL_BASE_DIVM_MASK >> PLL_BASE_DIVM_SHIFT)) || (cfg.n > (PLL_BASE_DIVN_MASK >> PLL_BASE_DIVN_SHIFT)) || (p_div > (PLL_BASE_DIVP_MASK >> PLL_BASE_DIVP_SHIFT)) || (cfg.output_rate > c->u.pll.vco_max)) { pr_err("%s: Failed to set %s out-of-table rate %lu\n", __func__, __clk_get_name(hw->clk), rate); return -EINVAL; } p_div <<= PLL_BASE_DIVP_SHIFT; } c->mul = sel->n; c->div = sel->m * sel->p; old_base = val = clk_readl(c->reg + PLL_BASE); val &= ~(PLL_BASE_DIVM_MASK | PLL_BASE_DIVN_MASK | ((c->flags & PLLU) ? PLLU_BASE_POST_DIV : PLL_BASE_DIVP_MASK)); val |= (sel->m << PLL_BASE_DIVM_SHIFT) | (sel->n << PLL_BASE_DIVN_SHIFT) | p_div; if (val == old_base) return 0; if (c->state == ON) { tegra30_pll_clk_disable(hw); val &= ~(PLL_BASE_BYPASS | PLL_BASE_ENABLE); } clk_writel(val, c->reg + PLL_BASE); if (c->flags & PLL_HAS_CPCON) { val = clk_readl(c->reg + PLL_MISC(c)); val &= ~PLL_MISC_CPCON_MASK; val |= sel->cpcon << PLL_MISC_CPCON_SHIFT; if (c->flags & (PLLU | PLLD)) { val &= ~PLL_MISC_LFCON_MASK; if (sel->n >= PLLDU_LFCON_SET_DIVN) val |= 0x1 << PLL_MISC_LFCON_SHIFT; } else if (c->flags & (PLLX | PLLM)) { val &= ~(0x1 << PLL_MISC_DCCON_SHIFT); if (rate >= (c->u.pll.vco_max >> 1)) val |= 0x1 << PLL_MISC_DCCON_SHIFT; } clk_writel(val, c->reg + PLL_MISC(c)); } if (c->state == ON) tegra30_pll_clk_enable(hw); c->u.pll.fixed_rate = rate; return 0; } static long tegra30_pll_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long input_rate = *prate; u64 output_rate = *prate; const struct clk_pll_freq_table *sel; struct clk_pll_freq_table cfg; int mul; int div; u32 p_div; u32 val; if (c->flags & PLL_FIXED) return c->u.pll.fixed_rate; if (c->flags & PLLM) return __clk_get_rate(hw->clk); p_div = 0; /* Check if the target rate is tabulated */ for (sel = c->u.pll.freq_table; sel->input_rate != 0; sel++) { if (sel->input_rate == input_rate && sel->output_rate == rate) { if (c->flags & PLLU) { BUG_ON(sel->p < 1 || sel->p > 2); if (sel->p == 1) p_div = PLLU_BASE_POST_DIV; } else { BUG_ON(sel->p < 1); for (val = sel->p; val > 1; val >>= 1) p_div++; p_div <<= PLL_BASE_DIVP_SHIFT; } break; } } if (sel->input_rate == 0) { unsigned long cfreq; BUG_ON(c->flags & PLLU); sel = &cfg; switch (input_rate) { case 12000000: case 26000000: cfreq = (rate <= 1000000 * 1000) ? 1000000 : 2000000; break; case 13000000: cfreq = (rate <= 1000000 * 1000) ? 1000000 : 2600000; break; case 16800000: case 19200000: cfreq = (rate <= 1200000 * 1000) ? 1200000 : 2400000; break; default: pr_err("%s: Unexpected reference rate %lu\n", __func__, input_rate); BUG(); } /* Raise VCO to guarantee 0.5% accuracy */ for (cfg.output_rate = rate; cfg.output_rate < 200 * cfreq; cfg.output_rate <<= 1) p_div++; cfg.p = 0x1 << p_div; cfg.m = input_rate / cfreq; cfg.n = cfg.output_rate / cfreq; } mul = sel->n; div = sel->m * sel->p; output_rate *= mul; output_rate += div - 1; /* round up */ do_div(output_rate, div); return output_rate; } static unsigned long tegra30_pll_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; u32 val = clk_readl(c->reg + PLL_BASE); if (c->flags & PLL_FIXED && !(val & PLL_BASE_OVERRIDE)) { const struct clk_pll_freq_table *sel; for (sel = c->u.pll.freq_table; sel->input_rate != 0; sel++) { if (sel->input_rate == parent_rate && sel->output_rate == c->u.pll.fixed_rate) { c->mul = sel->n; c->div = sel->m * sel->p; break; } } pr_err("Clock %s has unknown fixed frequency\n", __clk_get_name(hw->clk)); BUG(); } else if (val & PLL_BASE_BYPASS) { c->mul = 1; c->div = 1; } else { c->mul = (val & PLL_BASE_DIVN_MASK) >> PLL_BASE_DIVN_SHIFT; c->div = (val & PLL_BASE_DIVM_MASK) >> PLL_BASE_DIVM_SHIFT; if (c->flags & PLLU) c->div *= (val & PLLU_BASE_POST_DIV) ? 1 : 2; else c->div *= (0x1 << ((val & PLL_BASE_DIVP_MASK) >> PLL_BASE_DIVP_SHIFT)); } if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } struct clk_ops tegra30_pll_ops = { .is_enabled = tegra30_pll_clk_is_enabled, .init = tegra30_pll_clk_init, .enable = tegra30_pll_clk_enable, .disable = tegra30_pll_clk_disable, .recalc_rate = tegra30_pll_recalc_rate, .round_rate = tegra30_pll_round_rate, .set_rate = tegra30_pll_clk_set_rate, }; int tegra30_plld_clk_cfg_ex(struct clk_hw *hw, enum tegra_clk_ex_param p, u32 setting) { struct clk_tegra *c = to_clk_tegra(hw); u32 val, mask, reg; switch (p) { case TEGRA_CLK_PLLD_CSI_OUT_ENB: mask = PLLD_BASE_CSI_CLKENABLE; reg = c->reg + PLL_BASE; break; case TEGRA_CLK_PLLD_DSI_OUT_ENB: mask = PLLD_MISC_DSI_CLKENABLE; reg = c->reg + PLL_MISC(c); break; case TEGRA_CLK_PLLD_MIPI_MUX_SEL: if (!(c->flags & PLL_ALT_MISC_REG)) { mask = PLLD_BASE_DSIB_MUX_MASK; reg = c->reg + PLL_BASE; break; } /* fall through - error since PLLD2 does not have MUX_SEL control */ default: return -EINVAL; } val = clk_readl(reg); if (setting) val |= mask; else val &= ~mask; clk_writel(val, reg); return 0; } static int tegra30_plle_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; val = clk_readl(c->reg + PLL_BASE); c->state = (val & PLLE_BASE_ENABLE) ? ON : OFF; return c->state; } static void tegra30_plle_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; val = clk_readl(c->reg + PLL_BASE); val &= ~(PLLE_BASE_CML_ENABLE | PLLE_BASE_ENABLE); clk_writel(val, c->reg + PLL_BASE); } static void tegra30_plle_training(struct clk_tegra *c) { u32 val; /* PLLE is already disabled, and setup cleared; * create falling edge on PLLE IDDQ input */ val = pmc_readl(PMC_SATA_PWRGT); val |= PMC_SATA_PWRGT_PLLE_IDDQ_VALUE; pmc_writel(val, PMC_SATA_PWRGT); val = pmc_readl(PMC_SATA_PWRGT); val |= PMC_SATA_PWRGT_PLLE_IDDQ_SWCTL; pmc_writel(val, PMC_SATA_PWRGT); val = pmc_readl(PMC_SATA_PWRGT); val &= ~PMC_SATA_PWRGT_PLLE_IDDQ_VALUE; pmc_writel(val, PMC_SATA_PWRGT); do { val = clk_readl(c->reg + PLL_MISC(c)); } while (!(val & PLLE_MISC_READY)); } static int tegra30_plle_configure(struct clk_hw *hw, bool force_training) { struct clk_tegra *c = to_clk_tegra(hw); struct clk *parent = __clk_get_parent(hw->clk); const struct clk_pll_freq_table *sel; u32 val; unsigned long rate = c->u.pll.fixed_rate; unsigned long input_rate = __clk_get_rate(parent); for (sel = c->u.pll.freq_table; sel->input_rate != 0; sel++) { if (sel->input_rate == input_rate && sel->output_rate == rate) break; } if (sel->input_rate == 0) return -ENOSYS; /* disable PLLE, clear setup fiels */ tegra30_plle_clk_disable(hw); val = clk_readl(c->reg + PLL_MISC(c)); val &= ~(PLLE_MISC_LOCK_ENABLE | PLLE_MISC_SETUP_MASK); clk_writel(val, c->reg + PLL_MISC(c)); /* training */ val = clk_readl(c->reg + PLL_MISC(c)); if (force_training || (!(val & PLLE_MISC_READY))) tegra30_plle_training(c); /* configure dividers, setup, disable SS */ val = clk_readl(c->reg + PLL_BASE); val &= ~PLLE_BASE_DIV_MASK; val |= PLLE_BASE_DIV(sel->m, sel->n, sel->p, sel->cpcon); clk_writel(val, c->reg + PLL_BASE); c->mul = sel->n; c->div = sel->m * sel->p; val = clk_readl(c->reg + PLL_MISC(c)); val |= PLLE_MISC_SETUP_VALUE; val |= PLLE_MISC_LOCK_ENABLE; clk_writel(val, c->reg + PLL_MISC(c)); val = clk_readl(PLLE_SS_CTRL); val |= PLLE_SS_DISABLE; clk_writel(val, PLLE_SS_CTRL); /* enable and lock PLLE*/ val = clk_readl(c->reg + PLL_BASE); val |= (PLLE_BASE_CML_ENABLE | PLLE_BASE_ENABLE); clk_writel(val, c->reg + PLL_BASE); tegra30_pll_clk_wait_for_lock(c, c->reg + PLL_MISC(c), PLLE_MISC_LOCK); return 0; } static int tegra30_plle_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); return tegra30_plle_configure(hw, !c->set); } static unsigned long tegra30_plle_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long rate = parent_rate; u32 val; val = clk_readl(c->reg + PLL_BASE); c->mul = (val & PLLE_BASE_DIVN_MASK) >> PLLE_BASE_DIVN_SHIFT; c->div = (val & PLLE_BASE_DIVM_MASK) >> PLLE_BASE_DIVM_SHIFT; c->div *= (val & PLLE_BASE_DIVP_MASK) >> PLLE_BASE_DIVP_SHIFT; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } struct clk_ops tegra30_plle_ops = { .is_enabled = tegra30_plle_clk_is_enabled, .enable = tegra30_plle_clk_enable, .disable = tegra30_plle_clk_disable, .recalc_rate = tegra30_plle_clk_recalc_rate, }; /* Clock divider ops */ static int tegra30_pll_div_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); if (c->flags & DIV_U71) { u32 val = clk_readl(c->reg); val >>= c->reg_shift; c->state = (val & PLL_OUT_CLKEN) ? ON : OFF; if (!(val & PLL_OUT_RESET_DISABLE)) c->state = OFF; } else { c->state = ON; } return c->state; } static int tegra30_pll_div_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; u32 new_val; pr_debug("%s: %s\n", __func__, __clk_get_name(hw->clk)); if (c->flags & DIV_U71) { val = clk_readl(c->reg); new_val = val >> c->reg_shift; new_val &= 0xFFFF; new_val |= PLL_OUT_CLKEN | PLL_OUT_RESET_DISABLE; val &= ~(0xFFFF << c->reg_shift); val |= new_val << c->reg_shift; clk_writel_delay(val, c->reg); return 0; } else if (c->flags & DIV_2) { return 0; } return -EINVAL; } static void tegra30_pll_div_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; u32 new_val; pr_debug("%s: %s\n", __func__, __clk_get_name(hw->clk)); if (c->flags & DIV_U71) { val = clk_readl(c->reg); new_val = val >> c->reg_shift; new_val &= 0xFFFF; new_val &= ~(PLL_OUT_CLKEN | PLL_OUT_RESET_DISABLE); val &= ~(0xFFFF << c->reg_shift); val |= new_val << c->reg_shift; clk_writel_delay(val, c->reg); } } static int tegra30_pll_div_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; u32 new_val; int divider_u71; if (c->flags & DIV_U71) { divider_u71 = clk_div71_get_divider( parent_rate, rate, c->flags, ROUND_DIVIDER_UP); if (divider_u71 >= 0) { val = clk_readl(c->reg); new_val = val >> c->reg_shift; new_val &= 0xFFFF; if (c->flags & DIV_U71_FIXED) new_val |= PLL_OUT_OVERRIDE; new_val &= ~PLL_OUT_RATIO_MASK; new_val |= divider_u71 << PLL_OUT_RATIO_SHIFT; val &= ~(0xFFFF << c->reg_shift); val |= new_val << c->reg_shift; clk_writel_delay(val, c->reg); c->div = divider_u71 + 2; c->mul = 2; c->fixed_rate = rate; return 0; } } else if (c->flags & DIV_2) { c->fixed_rate = rate; return 0; } return -EINVAL; } static unsigned long tegra30_pll_div_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; if (c->flags & DIV_U71) { u32 divu71; u32 val = clk_readl(c->reg); val >>= c->reg_shift; divu71 = (val & PLL_OUT_RATIO_MASK) >> PLL_OUT_RATIO_SHIFT; c->div = (divu71 + 2); c->mul = 2; } else if (c->flags & DIV_2) { if (c->flags & (PLLD | PLLX)) { c->div = 2; c->mul = 1; } else BUG(); } else { c->div = 1; c->mul = 1; } if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } static long tegra30_pll_div_clk_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long parent_rate = __clk_get_rate(__clk_get_parent(hw->clk)); int divider; if (prate) parent_rate = *prate; if (c->flags & DIV_U71) { divider = clk_div71_get_divider( parent_rate, rate, c->flags, ROUND_DIVIDER_UP); if (divider < 0) return divider; return DIV_ROUND_UP(parent_rate * 2, divider + 2); } else if (c->flags & DIV_2) { *prate = rate * 2; return rate; } return -EINVAL; } struct clk_ops tegra30_pll_div_ops = { .is_enabled = tegra30_pll_div_clk_is_enabled, .enable = tegra30_pll_div_clk_enable, .disable = tegra30_pll_div_clk_disable, .set_rate = tegra30_pll_div_clk_set_rate, .recalc_rate = tegra30_pll_div_clk_recalc_rate, .round_rate = tegra30_pll_div_clk_round_rate, }; /* Periph clk ops */ static inline u32 periph_clk_source_mask(struct clk_tegra *c) { if (c->flags & MUX8) return 7 << 29; else if (c->flags & MUX_PWM) return 3 << 28; else if (c->flags & MUX_CLK_OUT) return 3 << (c->u.periph.clk_num + 4); else if (c->flags & PLLD) return PLLD_BASE_DSIB_MUX_MASK; else return 3 << 30; } static inline u32 periph_clk_source_shift(struct clk_tegra *c) { if (c->flags & MUX8) return 29; else if (c->flags & MUX_PWM) return 28; else if (c->flags & MUX_CLK_OUT) return c->u.periph.clk_num + 4; else if (c->flags & PLLD) return PLLD_BASE_DSIB_MUX_SHIFT; else return 30; } static int tegra30_periph_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); c->state = ON; if (!(clk_readl(PERIPH_CLK_TO_ENB_REG(c)) & PERIPH_CLK_TO_BIT(c))) c->state = OFF; if (!(c->flags & PERIPH_NO_RESET)) if (clk_readl(PERIPH_CLK_TO_RST_REG(c)) & PERIPH_CLK_TO_BIT(c)) c->state = OFF; return c->state; } static int tegra30_periph_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); tegra_periph_clk_enable_refcount[c->u.periph.clk_num]++; if (tegra_periph_clk_enable_refcount[c->u.periph.clk_num] > 1) return 0; clk_writel_delay(PERIPH_CLK_TO_BIT(c), PERIPH_CLK_TO_ENB_SET_REG(c)); if (!(c->flags & PERIPH_NO_RESET) && !(c->flags & PERIPH_MANUAL_RESET)) { if (clk_readl(PERIPH_CLK_TO_RST_REG(c)) & PERIPH_CLK_TO_BIT(c)) { udelay(5); /* reset propagation delay */ clk_writel(PERIPH_CLK_TO_BIT(c), PERIPH_CLK_TO_RST_CLR_REG(c)); } } return 0; } static void tegra30_periph_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long val; tegra_periph_clk_enable_refcount[c->u.periph.clk_num]--; if (tegra_periph_clk_enable_refcount[c->u.periph.clk_num] > 0) return; /* If peripheral is in the APB bus then read the APB bus to * flush the write operation in apb bus. This will avoid the * peripheral access after disabling clock*/ if (c->flags & PERIPH_ON_APB) val = chipid_readl(); clk_writel_delay(PERIPH_CLK_TO_BIT(c), PERIPH_CLK_TO_ENB_CLR_REG(c)); } void tegra30_periph_clk_reset(struct clk_hw *hw, bool assert) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long val; if (!(c->flags & PERIPH_NO_RESET)) { if (assert) { /* If peripheral is in the APB bus then read the APB * bus to flush the write operation in apb bus. This * will avoid the peripheral access after disabling * clock */ if (c->flags & PERIPH_ON_APB) val = chipid_readl(); clk_writel(PERIPH_CLK_TO_BIT(c), PERIPH_CLK_TO_RST_SET_REG(c)); } else clk_writel(PERIPH_CLK_TO_BIT(c), PERIPH_CLK_TO_RST_CLR_REG(c)); } } static int tegra30_periph_clk_set_parent(struct clk_hw *hw, u8 index) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; if (!(c->flags & MUX)) return (index == 0) ? 0 : (-EINVAL); val = clk_readl(c->reg); val &= ~periph_clk_source_mask(c); val |= (index << periph_clk_source_shift(c)); clk_writel_delay(val, c->reg); return 0; } static u8 tegra30_periph_clk_get_parent(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); int source = (val & periph_clk_source_mask(c)) >> periph_clk_source_shift(c); if (!(c->flags & MUX)) return 0; return source; } static int tegra30_periph_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; int divider; if (c->flags & DIV_U71) { divider = clk_div71_get_divider( parent_rate, rate, c->flags, ROUND_DIVIDER_UP); if (divider >= 0) { val = clk_readl(c->reg); val &= ~PERIPH_CLK_SOURCE_DIVU71_MASK; val |= divider; if (c->flags & DIV_U71_UART) { if (divider) val |= PERIPH_CLK_UART_DIV_ENB; else val &= ~PERIPH_CLK_UART_DIV_ENB; } clk_writel_delay(val, c->reg); c->div = divider + 2; c->mul = 2; return 0; } } else if (c->flags & DIV_U16) { divider = clk_div16_get_divider(parent_rate, rate); if (divider >= 0) { val = clk_readl(c->reg); val &= ~PERIPH_CLK_SOURCE_DIVU16_MASK; val |= divider; clk_writel_delay(val, c->reg); c->div = divider + 1; c->mul = 1; return 0; } } else if (parent_rate <= rate) { c->div = 1; c->mul = 1; return 0; } return -EINVAL; } static long tegra30_periph_clk_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { struct clk_tegra *c = to_clk_tegra(hw); unsigned long parent_rate = __clk_get_rate(__clk_get_parent(hw->clk)); int divider; if (prate) parent_rate = *prate; if (c->flags & DIV_U71) { divider = clk_div71_get_divider( parent_rate, rate, c->flags, ROUND_DIVIDER_UP); if (divider < 0) return divider; return DIV_ROUND_UP(parent_rate * 2, divider + 2); } else if (c->flags & DIV_U16) { divider = clk_div16_get_divider(parent_rate, rate); if (divider < 0) return divider; return DIV_ROUND_UP(parent_rate, divider + 1); } return -EINVAL; } static unsigned long tegra30_periph_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; u32 val = clk_readl(c->reg); if (c->flags & DIV_U71) { u32 divu71 = val & PERIPH_CLK_SOURCE_DIVU71_MASK; if ((c->flags & DIV_U71_UART) && (!(val & PERIPH_CLK_UART_DIV_ENB))) { divu71 = 0; } if (c->flags & DIV_U71_IDLE) { val &= ~(PERIPH_CLK_SOURCE_DIVU71_MASK << PERIPH_CLK_SOURCE_DIVIDLE_SHIFT); val |= (PERIPH_CLK_SOURCE_DIVIDLE_VAL << PERIPH_CLK_SOURCE_DIVIDLE_SHIFT); clk_writel(val, c->reg); } c->div = divu71 + 2; c->mul = 2; } else if (c->flags & DIV_U16) { u32 divu16 = val & PERIPH_CLK_SOURCE_DIVU16_MASK; c->div = divu16 + 1; c->mul = 1; } else { c->div = 1; c->mul = 1; } if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } struct clk_ops tegra30_periph_clk_ops = { .is_enabled = tegra30_periph_clk_is_enabled, .enable = tegra30_periph_clk_enable, .disable = tegra30_periph_clk_disable, .set_parent = tegra30_periph_clk_set_parent, .get_parent = tegra30_periph_clk_get_parent, .set_rate = tegra30_periph_clk_set_rate, .round_rate = tegra30_periph_clk_round_rate, .recalc_rate = tegra30_periph_clk_recalc_rate, }; static int tegra30_dsib_clk_set_parent(struct clk_hw *hw, u8 index) { struct clk *d = clk_get_sys(NULL, "pll_d"); /* The DSIB parent selection bit is in PLLD base register */ tegra_clk_cfg_ex( d, TEGRA_CLK_PLLD_MIPI_MUX_SEL, index); return 0; } struct clk_ops tegra30_dsib_clk_ops = { .is_enabled = tegra30_periph_clk_is_enabled, .enable = &tegra30_periph_clk_enable, .disable = &tegra30_periph_clk_disable, .set_parent = &tegra30_dsib_clk_set_parent, .get_parent = &tegra30_periph_clk_get_parent, .set_rate = &tegra30_periph_clk_set_rate, .round_rate = &tegra30_periph_clk_round_rate, .recalc_rate = &tegra30_periph_clk_recalc_rate, }; /* Periph extended clock configuration ops */ int tegra30_vi_clk_cfg_ex(struct clk_hw *hw, enum tegra_clk_ex_param p, u32 setting) { struct clk_tegra *c = to_clk_tegra(hw); if (p == TEGRA_CLK_VI_INP_SEL) { u32 val = clk_readl(c->reg); val &= ~PERIPH_CLK_VI_SEL_EX_MASK; val |= (setting << PERIPH_CLK_VI_SEL_EX_SHIFT) & PERIPH_CLK_VI_SEL_EX_MASK; clk_writel(val, c->reg); return 0; } return -EINVAL; } int tegra30_nand_clk_cfg_ex(struct clk_hw *hw, enum tegra_clk_ex_param p, u32 setting) { struct clk_tegra *c = to_clk_tegra(hw); if (p == TEGRA_CLK_NAND_PAD_DIV2_ENB) { u32 val = clk_readl(c->reg); if (setting) val |= PERIPH_CLK_NAND_DIV_EX_ENB; else val &= ~PERIPH_CLK_NAND_DIV_EX_ENB; clk_writel(val, c->reg); return 0; } return -EINVAL; } int tegra30_dtv_clk_cfg_ex(struct clk_hw *hw, enum tegra_clk_ex_param p, u32 setting) { struct clk_tegra *c = to_clk_tegra(hw); if (p == TEGRA_CLK_DTV_INVERT) { u32 val = clk_readl(c->reg); if (setting) val |= PERIPH_CLK_DTV_POLARITY_INV; else val &= ~PERIPH_CLK_DTV_POLARITY_INV; clk_writel(val, c->reg); return 0; } return -EINVAL; } /* Output clock ops */ static DEFINE_SPINLOCK(clk_out_lock); static int tegra30_clk_out_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = pmc_readl(c->reg); c->state = (val & (0x1 << c->u.periph.clk_num)) ? ON : OFF; c->mul = 1; c->div = 1; return c->state; } static int tegra30_clk_out_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; unsigned long flags; spin_lock_irqsave(&clk_out_lock, flags); val = pmc_readl(c->reg); val |= (0x1 << c->u.periph.clk_num); pmc_writel(val, c->reg); spin_unlock_irqrestore(&clk_out_lock, flags); return 0; } static void tegra30_clk_out_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; unsigned long flags; spin_lock_irqsave(&clk_out_lock, flags); val = pmc_readl(c->reg); val &= ~(0x1 << c->u.periph.clk_num); pmc_writel(val, c->reg); spin_unlock_irqrestore(&clk_out_lock, flags); } static int tegra30_clk_out_set_parent(struct clk_hw *hw, u8 index) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; unsigned long flags; spin_lock_irqsave(&clk_out_lock, flags); val = pmc_readl(c->reg); val &= ~periph_clk_source_mask(c); val |= (index << periph_clk_source_shift(c)); pmc_writel(val, c->reg); spin_unlock_irqrestore(&clk_out_lock, flags); return 0; } static u8 tegra30_clk_out_get_parent(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = pmc_readl(c->reg); int source; source = (val & periph_clk_source_mask(c)) >> periph_clk_source_shift(c); return source; } struct clk_ops tegra_clk_out_ops = { .is_enabled = tegra30_clk_out_is_enabled, .enable = tegra30_clk_out_enable, .disable = tegra30_clk_out_disable, .set_parent = tegra30_clk_out_set_parent, .get_parent = tegra30_clk_out_get_parent, .recalc_rate = tegra30_clk_fixed_recalc_rate, }; /* Clock doubler ops */ static int tegra30_clk_double_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); c->state = ON; if (!(clk_readl(PERIPH_CLK_TO_ENB_REG(c)) & PERIPH_CLK_TO_BIT(c))) c->state = OFF; return c->state; }; static int tegra30_clk_double_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; if (rate == parent_rate) { val = clk_readl(c->reg) | (0x1 << c->reg_shift); clk_writel(val, c->reg); c->mul = 1; c->div = 1; return 0; } else if (rate == 2 * parent_rate) { val = clk_readl(c->reg) & (~(0x1 << c->reg_shift)); clk_writel(val, c->reg); c->mul = 2; c->div = 1; return 0; } return -EINVAL; } static unsigned long tegra30_clk_double_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct clk_tegra *c = to_clk_tegra(hw); u64 rate = parent_rate; u32 val = clk_readl(c->reg); c->mul = val & (0x1 << c->reg_shift) ? 1 : 2; c->div = 1; if (c->mul != 0 && c->div != 0) { rate *= c->mul; rate += c->div - 1; /* round up */ do_div(rate, c->div); } return rate; } static long tegra30_clk_double_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *prate) { unsigned long output_rate = *prate; do_div(output_rate, 2); return output_rate; } struct clk_ops tegra30_clk_double_ops = { .is_enabled = tegra30_clk_double_is_enabled, .enable = tegra30_periph_clk_enable, .disable = tegra30_periph_clk_disable, .recalc_rate = tegra30_clk_double_recalc_rate, .round_rate = tegra30_clk_double_round_rate, .set_rate = tegra30_clk_double_set_rate, }; /* Audio sync clock ops */ struct clk_ops tegra_sync_source_ops = { .recalc_rate = tegra30_clk_fixed_recalc_rate, }; static int tegra30_audio_sync_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); c->state = (val & AUDIO_SYNC_DISABLE_BIT) ? OFF : ON; return c->state; } static int tegra30_audio_sync_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); clk_writel((val & (~AUDIO_SYNC_DISABLE_BIT)), c->reg); return 0; } static void tegra30_audio_sync_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); clk_writel((val | AUDIO_SYNC_DISABLE_BIT), c->reg); } static int tegra30_audio_sync_clk_set_parent(struct clk_hw *hw, u8 index) { struct clk_tegra *c = to_clk_tegra(hw); u32 val; val = clk_readl(c->reg); val &= ~AUDIO_SYNC_SOURCE_MASK; val |= index; clk_writel(val, c->reg); return 0; } static u8 tegra30_audio_sync_clk_get_parent(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); int source; source = val & AUDIO_SYNC_SOURCE_MASK; return source; } struct clk_ops tegra30_audio_sync_clk_ops = { .is_enabled = tegra30_audio_sync_clk_is_enabled, .enable = tegra30_audio_sync_clk_enable, .disable = tegra30_audio_sync_clk_disable, .set_parent = tegra30_audio_sync_clk_set_parent, .get_parent = tegra30_audio_sync_clk_get_parent, .recalc_rate = tegra30_clk_fixed_recalc_rate, }; /* cml0 (pcie), and cml1 (sata) clock ops */ static int tegra30_cml_clk_is_enabled(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); c->state = val & (0x1 << c->u.periph.clk_num) ? ON : OFF; return c->state; } static int tegra30_cml_clk_enable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); val |= (0x1 << c->u.periph.clk_num); clk_writel(val, c->reg); return 0; } static void tegra30_cml_clk_disable(struct clk_hw *hw) { struct clk_tegra *c = to_clk_tegra(hw); u32 val = clk_readl(c->reg); val &= ~(0x1 << c->u.periph.clk_num); clk_writel(val, c->reg); } struct clk_ops tegra_cml_clk_ops = { .is_enabled = tegra30_cml_clk_is_enabled, .enable = tegra30_cml_clk_enable, .disable = tegra30_cml_clk_disable, .recalc_rate = tegra30_clk_fixed_recalc_rate, }; struct clk_ops tegra_pciex_clk_ops = { .recalc_rate = tegra30_clk_fixed_recalc_rate, }; /* Tegra30 CPU clock and reset control functions */ static void tegra30_wait_cpu_in_reset(u32 cpu) { unsigned int reg; do { reg = readl(reg_clk_base + TEGRA30_CLK_RST_CONTROLLER_CPU_CMPLX_STATUS); cpu_relax(); } while (!(reg & (1 << cpu))); /* check CPU been reset or not */ return; } static void tegra30_put_cpu_in_reset(u32 cpu) { writel(CPU_RESET(cpu), reg_clk_base + TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_SET); dmb(); } static void tegra30_cpu_out_of_reset(u32 cpu) { writel(CPU_RESET(cpu), reg_clk_base + TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_CLR); wmb(); } static void tegra30_enable_cpu_clock(u32 cpu) { unsigned int reg; writel(CPU_CLOCK(cpu), reg_clk_base + TEGRA30_CLK_RST_CONTROLLER_CLK_CPU_CMPLX_CLR); reg = readl(reg_clk_base + TEGRA30_CLK_RST_CONTROLLER_CLK_CPU_CMPLX_CLR); } static void tegra30_disable_cpu_clock(u32 cpu) { unsigned int reg; reg = readl(reg_clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX); writel(reg | CPU_CLOCK(cpu), reg_clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX); } #ifdef CONFIG_PM_SLEEP static bool tegra30_cpu_rail_off_ready(void) { unsigned int cpu_rst_status; int cpu_pwr_status; cpu_rst_status = readl(reg_clk_base + TEGRA30_CLK_RST_CONTROLLER_CPU_CMPLX_STATUS); cpu_pwr_status = tegra_powergate_is_powered(TEGRA_POWERGATE_CPU1) || tegra_powergate_is_powered(TEGRA_POWERGATE_CPU2) || tegra_powergate_is_powered(TEGRA_POWERGATE_CPU3); if (((cpu_rst_status & 0xE) != 0xE) || cpu_pwr_status) return false; return true; } static void tegra30_cpu_clock_suspend(void) { /* switch coresite to clk_m, save off original source */ tegra30_cpu_clk_sctx.clk_csite_src = readl(reg_clk_base + CLK_RESET_SOURCE_CSITE); writel(3<<30, reg_clk_base + CLK_RESET_SOURCE_CSITE); tegra30_cpu_clk_sctx.cpu_burst = readl(reg_clk_base + CLK_RESET_CCLK_BURST); tegra30_cpu_clk_sctx.pllx_base = readl(reg_clk_base + CLK_RESET_PLLX_BASE); tegra30_cpu_clk_sctx.pllx_misc = readl(reg_clk_base + CLK_RESET_PLLX_MISC); tegra30_cpu_clk_sctx.cclk_divider = readl(reg_clk_base + CLK_RESET_CCLK_DIVIDER); } static void tegra30_cpu_clock_resume(void) { unsigned int reg, policy; /* Is CPU complex already running on PLLX? */ reg = readl(reg_clk_base + CLK_RESET_CCLK_BURST); policy = (reg >> CLK_RESET_CCLK_BURST_POLICY_SHIFT) & 0xF; if (policy == CLK_RESET_CCLK_IDLE_POLICY) reg = (reg >> CLK_RESET_CCLK_IDLE_POLICY_SHIFT) & 0xF; else if (policy == CLK_RESET_CCLK_RUN_POLICY) reg = (reg >> CLK_RESET_CCLK_RUN_POLICY_SHIFT) & 0xF; else BUG(); if (reg != CLK_RESET_CCLK_BURST_POLICY_PLLX) { /* restore PLLX settings if CPU is on different PLL */ writel(tegra30_cpu_clk_sctx.pllx_misc, reg_clk_base + CLK_RESET_PLLX_MISC); writel(tegra30_cpu_clk_sctx.pllx_base, reg_clk_base + CLK_RESET_PLLX_BASE); /* wait for PLL stabilization if PLLX was enabled */ if (tegra30_cpu_clk_sctx.pllx_base & (1 << 30)) udelay(300); } /* * Restore original burst policy setting for calls resulting from CPU * LP2 in idle or system suspend. */ writel(tegra30_cpu_clk_sctx.cclk_divider, reg_clk_base + CLK_RESET_CCLK_DIVIDER); writel(tegra30_cpu_clk_sctx.cpu_burst, reg_clk_base + CLK_RESET_CCLK_BURST); writel(tegra30_cpu_clk_sctx.clk_csite_src, reg_clk_base + CLK_RESET_SOURCE_CSITE); } #endif static struct tegra_cpu_car_ops tegra30_cpu_car_ops = { .wait_for_reset = tegra30_wait_cpu_in_reset, .put_in_reset = tegra30_put_cpu_in_reset, .out_of_reset = tegra30_cpu_out_of_reset, .enable_clock = tegra30_enable_cpu_clock, .disable_clock = tegra30_disable_cpu_clock, #ifdef CONFIG_PM_SLEEP .rail_off_ready = tegra30_cpu_rail_off_ready, .suspend = tegra30_cpu_clock_suspend, .resume = tegra30_cpu_clock_resume, #endif }; void __init tegra30_cpu_car_ops_init(void) { tegra_cpu_car_ops = &tegra30_cpu_car_ops; }