Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

clk: qcom: Add support for SR2 PLLs

Add support for SR2 type pll operations. SR2 is optimized for Time Interval
Error (TIE) or absolute jitter.

Signed-off-by: Georgi Djakov <georgi.djakov@linaro.org>
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>

authored by

Georgi Djakov and committed by
Stephen Boyd
d4f76de3 a8a8db47

+76
+75
drivers/clk/qcom/clk-pll.c
··· 292 292 clk_pll_set_fsm_mode(pll, regmap, 0); 293 293 } 294 294 EXPORT_SYMBOL_GPL(clk_pll_configure_sr_hpm_lp); 295 + 296 + static int clk_pll_sr2_enable(struct clk_hw *hw) 297 + { 298 + struct clk_pll *pll = to_clk_pll(hw); 299 + int ret; 300 + u32 mode; 301 + 302 + ret = regmap_read(pll->clkr.regmap, pll->mode_reg, &mode); 303 + if (ret) 304 + return ret; 305 + 306 + /* Disable PLL bypass mode. */ 307 + ret = regmap_update_bits(pll->clkr.regmap, pll->mode_reg, PLL_BYPASSNL, 308 + PLL_BYPASSNL); 309 + if (ret) 310 + return ret; 311 + 312 + /* 313 + * H/W requires a 5us delay between disabling the bypass and 314 + * de-asserting the reset. Delay 10us just to be safe. 315 + */ 316 + udelay(10); 317 + 318 + /* De-assert active-low PLL reset. */ 319 + ret = regmap_update_bits(pll->clkr.regmap, pll->mode_reg, PLL_RESET_N, 320 + PLL_RESET_N); 321 + if (ret) 322 + return ret; 323 + 324 + ret = wait_for_pll(pll); 325 + if (ret) 326 + return ret; 327 + 328 + /* Enable PLL output. */ 329 + return regmap_update_bits(pll->clkr.regmap, pll->mode_reg, PLL_OUTCTRL, 330 + PLL_OUTCTRL); 331 + } 332 + 333 + static int 334 + clk_pll_sr2_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long prate) 335 + { 336 + struct clk_pll *pll = to_clk_pll(hw); 337 + const struct pll_freq_tbl *f; 338 + bool enabled; 339 + u32 mode; 340 + u32 enable_mask = PLL_OUTCTRL | PLL_BYPASSNL | PLL_RESET_N; 341 + 342 + f = find_freq(pll->freq_tbl, rate); 343 + if (!f) 344 + return -EINVAL; 345 + 346 + regmap_read(pll->clkr.regmap, pll->mode_reg, &mode); 347 + enabled = (mode & enable_mask) == enable_mask; 348 + 349 + if (enabled) 350 + clk_pll_disable(hw); 351 + 352 + regmap_update_bits(pll->clkr.regmap, pll->l_reg, 0x3ff, f->l); 353 + regmap_update_bits(pll->clkr.regmap, pll->m_reg, 0x7ffff, f->m); 354 + regmap_update_bits(pll->clkr.regmap, pll->n_reg, 0x7ffff, f->n); 355 + 356 + if (enabled) 357 + clk_pll_sr2_enable(hw); 358 + 359 + return 0; 360 + } 361 + 362 + const struct clk_ops clk_pll_sr2_ops = { 363 + .enable = clk_pll_sr2_enable, 364 + .disable = clk_pll_disable, 365 + .set_rate = clk_pll_sr2_set_rate, 366 + .recalc_rate = clk_pll_recalc_rate, 367 + .determine_rate = clk_pll_determine_rate, 368 + }; 369 + EXPORT_SYMBOL_GPL(clk_pll_sr2_ops);
+1
drivers/clk/qcom/clk-pll.h
··· 62 62 63 63 extern const struct clk_ops clk_pll_ops; 64 64 extern const struct clk_ops clk_pll_vote_ops; 65 + extern const struct clk_ops clk_pll_sr2_ops; 65 66 66 67 #define to_clk_pll(_hw) container_of(to_clk_regmap(_hw), struct clk_pll, clkr) 67 68