1 /* 2 * Copyright (c) 2013 Samsung Electronics Co., Ltd. 3 * Copyright (c) 2013 Linaro Ltd. 4 * 5 * This program is free software; you can redistribute it and/or modify 6 * it under the terms of the GNU General Public License version 2 as 7 * published by the Free Software Foundation. 8 * 9 * This file contains the utility functions to register the pll clocks. 10 */ 11 12 #include <linux/errno.h> 13 #include <linux/hrtimer.h> 14 #include <linux/delay.h> 15 #include <linux/slab.h> 16 #include <linux/clk-provider.h> 17 #include <linux/io.h> 18 #include "clk.h" 19 #include "clk-pll.h" 20 21 #define PLL_TIMEOUT_MS 10 22 23 struct samsung_clk_pll { 24 struct clk_hw hw; 25 void __iomem *lock_reg; 26 void __iomem *con_reg; 27 /* PLL enable control bit offset in @con_reg register */ 28 unsigned short enable_offs; 29 /* PLL lock status bit offset in @con_reg register */ 30 unsigned short lock_offs; 31 enum samsung_pll_type type; 32 unsigned int rate_count; 33 const struct samsung_pll_rate_table *rate_table; 34 }; 35 36 #define to_clk_pll(_hw) container_of(_hw, struct samsung_clk_pll, hw) 37 38 static const struct samsung_pll_rate_table *samsung_get_pll_settings( 39 struct samsung_clk_pll *pll, unsigned long rate) 40 { 41 const struct samsung_pll_rate_table *rate_table = pll->rate_table; 42 int i; 43 44 for (i = 0; i < pll->rate_count; i++) { 45 if (rate == rate_table[i].rate) 46 return &rate_table[i]; 47 } 48 49 return NULL; 50 } 51 52 static long samsung_pll_round_rate(struct clk_hw *hw, 53 unsigned long drate, unsigned long *prate) 54 { 55 struct samsung_clk_pll *pll = to_clk_pll(hw); 56 const struct samsung_pll_rate_table *rate_table = pll->rate_table; 57 int i; 58 59 /* Assumming rate_table is in descending order */ 60 for (i = 0; i < pll->rate_count; i++) { 61 if (drate >= rate_table[i].rate) 62 return rate_table[i].rate; 63 } 64 65 /* return minimum supported value */ 66 return rate_table[i - 1].rate; 67 } 68 69 static int samsung_pll3xxx_enable(struct clk_hw *hw) 70 { 71 struct samsung_clk_pll *pll = to_clk_pll(hw); 72 u32 tmp; 73 74 tmp = readl_relaxed(pll->con_reg); 75 tmp |= BIT(pll->enable_offs); 76 writel_relaxed(tmp, pll->con_reg); 77 78 /* wait lock time */ 79 do { 80 cpu_relax(); 81 tmp = readl_relaxed(pll->con_reg); 82 } while (!(tmp & BIT(pll->lock_offs))); 83 84 return 0; 85 } 86 87 static void samsung_pll3xxx_disable(struct clk_hw *hw) 88 { 89 struct samsung_clk_pll *pll = to_clk_pll(hw); 90 u32 tmp; 91 92 tmp = readl_relaxed(pll->con_reg); 93 tmp &= ~BIT(pll->enable_offs); 94 writel_relaxed(tmp, pll->con_reg); 95 } 96 97 /* 98 * PLL2126 Clock Type 99 */ 100 101 #define PLL2126_MDIV_MASK (0xff) 102 #define PLL2126_PDIV_MASK (0x3f) 103 #define PLL2126_SDIV_MASK (0x3) 104 #define PLL2126_MDIV_SHIFT (16) 105 #define PLL2126_PDIV_SHIFT (8) 106 #define PLL2126_SDIV_SHIFT (0) 107 108 static unsigned long samsung_pll2126_recalc_rate(struct clk_hw *hw, 109 unsigned long parent_rate) 110 { 111 struct samsung_clk_pll *pll = to_clk_pll(hw); 112 u32 pll_con, mdiv, pdiv, sdiv; 113 u64 fvco = parent_rate; 114 115 pll_con = readl_relaxed(pll->con_reg); 116 mdiv = (pll_con >> PLL2126_MDIV_SHIFT) & PLL2126_MDIV_MASK; 117 pdiv = (pll_con >> PLL2126_PDIV_SHIFT) & PLL2126_PDIV_MASK; 118 sdiv = (pll_con >> PLL2126_SDIV_SHIFT) & PLL2126_SDIV_MASK; 119 120 fvco *= (mdiv + 8); 121 do_div(fvco, (pdiv + 2) << sdiv); 122 123 return (unsigned long)fvco; 124 } 125 126 static const struct clk_ops samsung_pll2126_clk_ops = { 127 .recalc_rate = samsung_pll2126_recalc_rate, 128 }; 129 130 /* 131 * PLL3000 Clock Type 132 */ 133 134 #define PLL3000_MDIV_MASK (0xff) 135 #define PLL3000_PDIV_MASK (0x3) 136 #define PLL3000_SDIV_MASK (0x3) 137 #define PLL3000_MDIV_SHIFT (16) 138 #define PLL3000_PDIV_SHIFT (8) 139 #define PLL3000_SDIV_SHIFT (0) 140 141 static unsigned long samsung_pll3000_recalc_rate(struct clk_hw *hw, 142 unsigned long parent_rate) 143 { 144 struct samsung_clk_pll *pll = to_clk_pll(hw); 145 u32 pll_con, mdiv, pdiv, sdiv; 146 u64 fvco = parent_rate; 147 148 pll_con = readl_relaxed(pll->con_reg); 149 mdiv = (pll_con >> PLL3000_MDIV_SHIFT) & PLL3000_MDIV_MASK; 150 pdiv = (pll_con >> PLL3000_PDIV_SHIFT) & PLL3000_PDIV_MASK; 151 sdiv = (pll_con >> PLL3000_SDIV_SHIFT) & PLL3000_SDIV_MASK; 152 153 fvco *= (2 * (mdiv + 8)); 154 do_div(fvco, pdiv << sdiv); 155 156 return (unsigned long)fvco; 157 } 158 159 static const struct clk_ops samsung_pll3000_clk_ops = { 160 .recalc_rate = samsung_pll3000_recalc_rate, 161 }; 162 163 /* 164 * PLL35xx Clock Type 165 */ 166 /* Maximum lock time can be 270 * PDIV cycles */ 167 #define PLL35XX_LOCK_FACTOR (270) 168 169 #define PLL35XX_MDIV_MASK (0x3FF) 170 #define PLL35XX_PDIV_MASK (0x3F) 171 #define PLL35XX_SDIV_MASK (0x7) 172 #define PLL35XX_MDIV_SHIFT (16) 173 #define PLL35XX_PDIV_SHIFT (8) 174 #define PLL35XX_SDIV_SHIFT (0) 175 #define PLL35XX_LOCK_STAT_SHIFT (29) 176 #define PLL35XX_ENABLE_SHIFT (31) 177 178 static unsigned long samsung_pll35xx_recalc_rate(struct clk_hw *hw, 179 unsigned long parent_rate) 180 { 181 struct samsung_clk_pll *pll = to_clk_pll(hw); 182 u32 mdiv, pdiv, sdiv, pll_con; 183 u64 fvco = parent_rate; 184 185 pll_con = readl_relaxed(pll->con_reg); 186 mdiv = (pll_con >> PLL35XX_MDIV_SHIFT) & PLL35XX_MDIV_MASK; 187 pdiv = (pll_con >> PLL35XX_PDIV_SHIFT) & PLL35XX_PDIV_MASK; 188 sdiv = (pll_con >> PLL35XX_SDIV_SHIFT) & PLL35XX_SDIV_MASK; 189 190 fvco *= mdiv; 191 do_div(fvco, (pdiv << sdiv)); 192 193 return (unsigned long)fvco; 194 } 195 196 static inline bool samsung_pll35xx_mp_change( 197 const struct samsung_pll_rate_table *rate, u32 pll_con) 198 { 199 u32 old_mdiv, old_pdiv; 200 201 old_mdiv = (pll_con >> PLL35XX_MDIV_SHIFT) & PLL35XX_MDIV_MASK; 202 old_pdiv = (pll_con >> PLL35XX_PDIV_SHIFT) & PLL35XX_PDIV_MASK; 203 204 return (rate->mdiv != old_mdiv || rate->pdiv != old_pdiv); 205 } 206 207 static int samsung_pll35xx_set_rate(struct clk_hw *hw, unsigned long drate, 208 unsigned long prate) 209 { 210 struct samsung_clk_pll *pll = to_clk_pll(hw); 211 const struct samsung_pll_rate_table *rate; 212 u32 tmp; 213 214 /* Get required rate settings from table */ 215 rate = samsung_get_pll_settings(pll, drate); 216 if (!rate) { 217 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 218 drate, clk_hw_get_name(hw)); 219 return -EINVAL; 220 } 221 222 tmp = readl_relaxed(pll->con_reg); 223 224 if (!(samsung_pll35xx_mp_change(rate, tmp))) { 225 /* If only s change, change just s value only*/ 226 tmp &= ~(PLL35XX_SDIV_MASK << PLL35XX_SDIV_SHIFT); 227 tmp |= rate->sdiv << PLL35XX_SDIV_SHIFT; 228 writel_relaxed(tmp, pll->con_reg); 229 230 return 0; 231 } 232 233 /* Set PLL lock time. */ 234 writel_relaxed(rate->pdiv * PLL35XX_LOCK_FACTOR, 235 pll->lock_reg); 236 237 /* Change PLL PMS values */ 238 tmp &= ~((PLL35XX_MDIV_MASK << PLL35XX_MDIV_SHIFT) | 239 (PLL35XX_PDIV_MASK << PLL35XX_PDIV_SHIFT) | 240 (PLL35XX_SDIV_MASK << PLL35XX_SDIV_SHIFT)); 241 tmp |= (rate->mdiv << PLL35XX_MDIV_SHIFT) | 242 (rate->pdiv << PLL35XX_PDIV_SHIFT) | 243 (rate->sdiv << PLL35XX_SDIV_SHIFT); 244 writel_relaxed(tmp, pll->con_reg); 245 246 /* Wait until the PLL is locked if it is enabled. */ 247 if (tmp & BIT(pll->enable_offs)) { 248 do { 249 cpu_relax(); 250 tmp = readl_relaxed(pll->con_reg); 251 } while (!(tmp & BIT(pll->lock_offs))); 252 } 253 return 0; 254 } 255 256 static const struct clk_ops samsung_pll35xx_clk_ops = { 257 .recalc_rate = samsung_pll35xx_recalc_rate, 258 .round_rate = samsung_pll_round_rate, 259 .set_rate = samsung_pll35xx_set_rate, 260 .enable = samsung_pll3xxx_enable, 261 .disable = samsung_pll3xxx_disable, 262 }; 263 264 static const struct clk_ops samsung_pll35xx_clk_min_ops = { 265 .recalc_rate = samsung_pll35xx_recalc_rate, 266 }; 267 268 /* 269 * PLL36xx Clock Type 270 */ 271 /* Maximum lock time can be 3000 * PDIV cycles */ 272 #define PLL36XX_LOCK_FACTOR (3000) 273 274 #define PLL36XX_KDIV_MASK (0xFFFF) 275 #define PLL36XX_MDIV_MASK (0x1FF) 276 #define PLL36XX_PDIV_MASK (0x3F) 277 #define PLL36XX_SDIV_MASK (0x7) 278 #define PLL36XX_MDIV_SHIFT (16) 279 #define PLL36XX_PDIV_SHIFT (8) 280 #define PLL36XX_SDIV_SHIFT (0) 281 #define PLL36XX_KDIV_SHIFT (0) 282 #define PLL36XX_LOCK_STAT_SHIFT (29) 283 #define PLL36XX_ENABLE_SHIFT (31) 284 285 static unsigned long samsung_pll36xx_recalc_rate(struct clk_hw *hw, 286 unsigned long parent_rate) 287 { 288 struct samsung_clk_pll *pll = to_clk_pll(hw); 289 u32 mdiv, pdiv, sdiv, pll_con0, pll_con1; 290 s16 kdiv; 291 u64 fvco = parent_rate; 292 293 pll_con0 = readl_relaxed(pll->con_reg); 294 pll_con1 = readl_relaxed(pll->con_reg + 4); 295 mdiv = (pll_con0 >> PLL36XX_MDIV_SHIFT) & PLL36XX_MDIV_MASK; 296 pdiv = (pll_con0 >> PLL36XX_PDIV_SHIFT) & PLL36XX_PDIV_MASK; 297 sdiv = (pll_con0 >> PLL36XX_SDIV_SHIFT) & PLL36XX_SDIV_MASK; 298 kdiv = (s16)(pll_con1 & PLL36XX_KDIV_MASK); 299 300 fvco *= (mdiv << 16) + kdiv; 301 do_div(fvco, (pdiv << sdiv)); 302 fvco >>= 16; 303 304 return (unsigned long)fvco; 305 } 306 307 static inline bool samsung_pll36xx_mpk_change( 308 const struct samsung_pll_rate_table *rate, u32 pll_con0, u32 pll_con1) 309 { 310 u32 old_mdiv, old_pdiv, old_kdiv; 311 312 old_mdiv = (pll_con0 >> PLL36XX_MDIV_SHIFT) & PLL36XX_MDIV_MASK; 313 old_pdiv = (pll_con0 >> PLL36XX_PDIV_SHIFT) & PLL36XX_PDIV_MASK; 314 old_kdiv = (pll_con1 >> PLL36XX_KDIV_SHIFT) & PLL36XX_KDIV_MASK; 315 316 return (rate->mdiv != old_mdiv || rate->pdiv != old_pdiv || 317 rate->kdiv != old_kdiv); 318 } 319 320 static int samsung_pll36xx_set_rate(struct clk_hw *hw, unsigned long drate, 321 unsigned long parent_rate) 322 { 323 struct samsung_clk_pll *pll = to_clk_pll(hw); 324 u32 tmp, pll_con0, pll_con1; 325 const struct samsung_pll_rate_table *rate; 326 327 rate = samsung_get_pll_settings(pll, drate); 328 if (!rate) { 329 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 330 drate, clk_hw_get_name(hw)); 331 return -EINVAL; 332 } 333 334 pll_con0 = readl_relaxed(pll->con_reg); 335 pll_con1 = readl_relaxed(pll->con_reg + 4); 336 337 if (!(samsung_pll36xx_mpk_change(rate, pll_con0, pll_con1))) { 338 /* If only s change, change just s value only*/ 339 pll_con0 &= ~(PLL36XX_SDIV_MASK << PLL36XX_SDIV_SHIFT); 340 pll_con0 |= (rate->sdiv << PLL36XX_SDIV_SHIFT); 341 writel_relaxed(pll_con0, pll->con_reg); 342 343 return 0; 344 } 345 346 /* Set PLL lock time. */ 347 writel_relaxed(rate->pdiv * PLL36XX_LOCK_FACTOR, pll->lock_reg); 348 349 /* Change PLL PMS values */ 350 pll_con0 &= ~((PLL36XX_MDIV_MASK << PLL36XX_MDIV_SHIFT) | 351 (PLL36XX_PDIV_MASK << PLL36XX_PDIV_SHIFT) | 352 (PLL36XX_SDIV_MASK << PLL36XX_SDIV_SHIFT)); 353 pll_con0 |= (rate->mdiv << PLL36XX_MDIV_SHIFT) | 354 (rate->pdiv << PLL36XX_PDIV_SHIFT) | 355 (rate->sdiv << PLL36XX_SDIV_SHIFT); 356 writel_relaxed(pll_con0, pll->con_reg); 357 358 pll_con1 &= ~(PLL36XX_KDIV_MASK << PLL36XX_KDIV_SHIFT); 359 pll_con1 |= rate->kdiv << PLL36XX_KDIV_SHIFT; 360 writel_relaxed(pll_con1, pll->con_reg + 4); 361 362 /* wait_lock_time */ 363 if (pll_con0 & BIT(pll->enable_offs)) { 364 do { 365 cpu_relax(); 366 tmp = readl_relaxed(pll->con_reg); 367 } while (!(tmp & BIT(pll->lock_offs))); 368 } 369 370 return 0; 371 } 372 373 static const struct clk_ops samsung_pll36xx_clk_ops = { 374 .recalc_rate = samsung_pll36xx_recalc_rate, 375 .set_rate = samsung_pll36xx_set_rate, 376 .round_rate = samsung_pll_round_rate, 377 .enable = samsung_pll3xxx_enable, 378 .disable = samsung_pll3xxx_disable, 379 }; 380 381 static const struct clk_ops samsung_pll36xx_clk_min_ops = { 382 .recalc_rate = samsung_pll36xx_recalc_rate, 383 }; 384 385 /* 386 * PLL45xx Clock Type 387 */ 388 #define PLL4502_LOCK_FACTOR 400 389 #define PLL4508_LOCK_FACTOR 240 390 391 #define PLL45XX_MDIV_MASK (0x3FF) 392 #define PLL45XX_PDIV_MASK (0x3F) 393 #define PLL45XX_SDIV_MASK (0x7) 394 #define PLL45XX_AFC_MASK (0x1F) 395 #define PLL45XX_MDIV_SHIFT (16) 396 #define PLL45XX_PDIV_SHIFT (8) 397 #define PLL45XX_SDIV_SHIFT (0) 398 #define PLL45XX_AFC_SHIFT (0) 399 400 #define PLL45XX_ENABLE BIT(31) 401 #define PLL45XX_LOCKED BIT(29) 402 403 static unsigned long samsung_pll45xx_recalc_rate(struct clk_hw *hw, 404 unsigned long parent_rate) 405 { 406 struct samsung_clk_pll *pll = to_clk_pll(hw); 407 u32 mdiv, pdiv, sdiv, pll_con; 408 u64 fvco = parent_rate; 409 410 pll_con = readl_relaxed(pll->con_reg); 411 mdiv = (pll_con >> PLL45XX_MDIV_SHIFT) & PLL45XX_MDIV_MASK; 412 pdiv = (pll_con >> PLL45XX_PDIV_SHIFT) & PLL45XX_PDIV_MASK; 413 sdiv = (pll_con >> PLL45XX_SDIV_SHIFT) & PLL45XX_SDIV_MASK; 414 415 if (pll->type == pll_4508) 416 sdiv = sdiv - 1; 417 418 fvco *= mdiv; 419 do_div(fvco, (pdiv << sdiv)); 420 421 return (unsigned long)fvco; 422 } 423 424 static bool samsung_pll45xx_mp_change(u32 pll_con0, u32 pll_con1, 425 const struct samsung_pll_rate_table *rate) 426 { 427 u32 old_mdiv, old_pdiv, old_afc; 428 429 old_mdiv = (pll_con0 >> PLL45XX_MDIV_SHIFT) & PLL45XX_MDIV_MASK; 430 old_pdiv = (pll_con0 >> PLL45XX_PDIV_SHIFT) & PLL45XX_PDIV_MASK; 431 old_afc = (pll_con1 >> PLL45XX_AFC_SHIFT) & PLL45XX_AFC_MASK; 432 433 return (old_mdiv != rate->mdiv || old_pdiv != rate->pdiv 434 || old_afc != rate->afc); 435 } 436 437 static int samsung_pll45xx_set_rate(struct clk_hw *hw, unsigned long drate, 438 unsigned long prate) 439 { 440 struct samsung_clk_pll *pll = to_clk_pll(hw); 441 const struct samsung_pll_rate_table *rate; 442 u32 con0, con1; 443 ktime_t start; 444 445 /* Get required rate settings from table */ 446 rate = samsung_get_pll_settings(pll, drate); 447 if (!rate) { 448 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 449 drate, clk_hw_get_name(hw)); 450 return -EINVAL; 451 } 452 453 con0 = readl_relaxed(pll->con_reg); 454 con1 = readl_relaxed(pll->con_reg + 0x4); 455 456 if (!(samsung_pll45xx_mp_change(con0, con1, rate))) { 457 /* If only s change, change just s value only*/ 458 con0 &= ~(PLL45XX_SDIV_MASK << PLL45XX_SDIV_SHIFT); 459 con0 |= rate->sdiv << PLL45XX_SDIV_SHIFT; 460 writel_relaxed(con0, pll->con_reg); 461 462 return 0; 463 } 464 465 /* Set PLL PMS values. */ 466 con0 &= ~((PLL45XX_MDIV_MASK << PLL45XX_MDIV_SHIFT) | 467 (PLL45XX_PDIV_MASK << PLL45XX_PDIV_SHIFT) | 468 (PLL45XX_SDIV_MASK << PLL45XX_SDIV_SHIFT)); 469 con0 |= (rate->mdiv << PLL45XX_MDIV_SHIFT) | 470 (rate->pdiv << PLL45XX_PDIV_SHIFT) | 471 (rate->sdiv << PLL45XX_SDIV_SHIFT); 472 473 /* Set PLL AFC value. */ 474 con1 = readl_relaxed(pll->con_reg + 0x4); 475 con1 &= ~(PLL45XX_AFC_MASK << PLL45XX_AFC_SHIFT); 476 con1 |= (rate->afc << PLL45XX_AFC_SHIFT); 477 478 /* Set PLL lock time. */ 479 switch (pll->type) { 480 case pll_4502: 481 writel_relaxed(rate->pdiv * PLL4502_LOCK_FACTOR, pll->lock_reg); 482 break; 483 case pll_4508: 484 writel_relaxed(rate->pdiv * PLL4508_LOCK_FACTOR, pll->lock_reg); 485 break; 486 default: 487 break; 488 } 489 490 /* Set new configuration. */ 491 writel_relaxed(con1, pll->con_reg + 0x4); 492 writel_relaxed(con0, pll->con_reg); 493 494 /* Wait for locking. */ 495 start = ktime_get(); 496 while (!(readl_relaxed(pll->con_reg) & PLL45XX_LOCKED)) { 497 ktime_t delta = ktime_sub(ktime_get(), start); 498 499 if (ktime_to_ms(delta) > PLL_TIMEOUT_MS) { 500 pr_err("%s: could not lock PLL %s\n", 501 __func__, clk_hw_get_name(hw)); 502 return -EFAULT; 503 } 504 505 cpu_relax(); 506 } 507 508 return 0; 509 } 510 511 static const struct clk_ops samsung_pll45xx_clk_ops = { 512 .recalc_rate = samsung_pll45xx_recalc_rate, 513 .round_rate = samsung_pll_round_rate, 514 .set_rate = samsung_pll45xx_set_rate, 515 }; 516 517 static const struct clk_ops samsung_pll45xx_clk_min_ops = { 518 .recalc_rate = samsung_pll45xx_recalc_rate, 519 }; 520 521 /* 522 * PLL46xx Clock Type 523 */ 524 #define PLL46XX_LOCK_FACTOR 3000 525 526 #define PLL46XX_VSEL_MASK (1) 527 #define PLL46XX_MDIV_MASK (0x1FF) 528 #define PLL1460X_MDIV_MASK (0x3FF) 529 530 #define PLL46XX_PDIV_MASK (0x3F) 531 #define PLL46XX_SDIV_MASK (0x7) 532 #define PLL46XX_VSEL_SHIFT (27) 533 #define PLL46XX_MDIV_SHIFT (16) 534 #define PLL46XX_PDIV_SHIFT (8) 535 #define PLL46XX_SDIV_SHIFT (0) 536 537 #define PLL46XX_KDIV_MASK (0xFFFF) 538 #define PLL4650C_KDIV_MASK (0xFFF) 539 #define PLL46XX_KDIV_SHIFT (0) 540 #define PLL46XX_MFR_MASK (0x3F) 541 #define PLL46XX_MRR_MASK (0x1F) 542 #define PLL46XX_KDIV_SHIFT (0) 543 #define PLL46XX_MFR_SHIFT (16) 544 #define PLL46XX_MRR_SHIFT (24) 545 546 #define PLL46XX_ENABLE BIT(31) 547 #define PLL46XX_LOCKED BIT(29) 548 #define PLL46XX_VSEL BIT(27) 549 550 static unsigned long samsung_pll46xx_recalc_rate(struct clk_hw *hw, 551 unsigned long parent_rate) 552 { 553 struct samsung_clk_pll *pll = to_clk_pll(hw); 554 u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1, shift; 555 u64 fvco = parent_rate; 556 557 pll_con0 = readl_relaxed(pll->con_reg); 558 pll_con1 = readl_relaxed(pll->con_reg + 4); 559 mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & ((pll->type == pll_1460x) ? 560 PLL1460X_MDIV_MASK : PLL46XX_MDIV_MASK); 561 pdiv = (pll_con0 >> PLL46XX_PDIV_SHIFT) & PLL46XX_PDIV_MASK; 562 sdiv = (pll_con0 >> PLL46XX_SDIV_SHIFT) & PLL46XX_SDIV_MASK; 563 kdiv = pll->type == pll_4650c ? pll_con1 & PLL4650C_KDIV_MASK : 564 pll_con1 & PLL46XX_KDIV_MASK; 565 566 shift = ((pll->type == pll_4600) || (pll->type == pll_1460x)) ? 16 : 10; 567 568 fvco *= (mdiv << shift) + kdiv; 569 do_div(fvco, (pdiv << sdiv)); 570 fvco >>= shift; 571 572 return (unsigned long)fvco; 573 } 574 575 static bool samsung_pll46xx_mpk_change(u32 pll_con0, u32 pll_con1, 576 const struct samsung_pll_rate_table *rate) 577 { 578 u32 old_mdiv, old_pdiv, old_kdiv; 579 580 old_mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & PLL46XX_MDIV_MASK; 581 old_pdiv = (pll_con0 >> PLL46XX_PDIV_SHIFT) & PLL46XX_PDIV_MASK; 582 old_kdiv = (pll_con1 >> PLL46XX_KDIV_SHIFT) & PLL46XX_KDIV_MASK; 583 584 return (old_mdiv != rate->mdiv || old_pdiv != rate->pdiv 585 || old_kdiv != rate->kdiv); 586 } 587 588 static int samsung_pll46xx_set_rate(struct clk_hw *hw, unsigned long drate, 589 unsigned long prate) 590 { 591 struct samsung_clk_pll *pll = to_clk_pll(hw); 592 const struct samsung_pll_rate_table *rate; 593 u32 con0, con1, lock; 594 ktime_t start; 595 596 /* Get required rate settings from table */ 597 rate = samsung_get_pll_settings(pll, drate); 598 if (!rate) { 599 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 600 drate, clk_hw_get_name(hw)); 601 return -EINVAL; 602 } 603 604 con0 = readl_relaxed(pll->con_reg); 605 con1 = readl_relaxed(pll->con_reg + 0x4); 606 607 if (!(samsung_pll46xx_mpk_change(con0, con1, rate))) { 608 /* If only s change, change just s value only*/ 609 con0 &= ~(PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT); 610 con0 |= rate->sdiv << PLL46XX_SDIV_SHIFT; 611 writel_relaxed(con0, pll->con_reg); 612 613 return 0; 614 } 615 616 /* Set PLL lock time. */ 617 lock = rate->pdiv * PLL46XX_LOCK_FACTOR; 618 if (lock > 0xffff) 619 /* Maximum lock time bitfield is 16-bit. */ 620 lock = 0xffff; 621 622 /* Set PLL PMS and VSEL values. */ 623 if (pll->type == pll_1460x) { 624 con0 &= ~((PLL1460X_MDIV_MASK << PLL46XX_MDIV_SHIFT) | 625 (PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) | 626 (PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT)); 627 } else { 628 con0 &= ~((PLL46XX_MDIV_MASK << PLL46XX_MDIV_SHIFT) | 629 (PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) | 630 (PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT) | 631 (PLL46XX_VSEL_MASK << PLL46XX_VSEL_SHIFT)); 632 con0 |= rate->vsel << PLL46XX_VSEL_SHIFT; 633 } 634 635 con0 |= (rate->mdiv << PLL46XX_MDIV_SHIFT) | 636 (rate->pdiv << PLL46XX_PDIV_SHIFT) | 637 (rate->sdiv << PLL46XX_SDIV_SHIFT); 638 639 /* Set PLL K, MFR and MRR values. */ 640 con1 = readl_relaxed(pll->con_reg + 0x4); 641 con1 &= ~((PLL46XX_KDIV_MASK << PLL46XX_KDIV_SHIFT) | 642 (PLL46XX_MFR_MASK << PLL46XX_MFR_SHIFT) | 643 (PLL46XX_MRR_MASK << PLL46XX_MRR_SHIFT)); 644 con1 |= (rate->kdiv << PLL46XX_KDIV_SHIFT) | 645 (rate->mfr << PLL46XX_MFR_SHIFT) | 646 (rate->mrr << PLL46XX_MRR_SHIFT); 647 648 /* Write configuration to PLL */ 649 writel_relaxed(lock, pll->lock_reg); 650 writel_relaxed(con0, pll->con_reg); 651 writel_relaxed(con1, pll->con_reg + 0x4); 652 653 /* Wait for locking. */ 654 start = ktime_get(); 655 while (!(readl_relaxed(pll->con_reg) & PLL46XX_LOCKED)) { 656 ktime_t delta = ktime_sub(ktime_get(), start); 657 658 if (ktime_to_ms(delta) > PLL_TIMEOUT_MS) { 659 pr_err("%s: could not lock PLL %s\n", 660 __func__, clk_hw_get_name(hw)); 661 return -EFAULT; 662 } 663 664 cpu_relax(); 665 } 666 667 return 0; 668 } 669 670 static const struct clk_ops samsung_pll46xx_clk_ops = { 671 .recalc_rate = samsung_pll46xx_recalc_rate, 672 .round_rate = samsung_pll_round_rate, 673 .set_rate = samsung_pll46xx_set_rate, 674 }; 675 676 static const struct clk_ops samsung_pll46xx_clk_min_ops = { 677 .recalc_rate = samsung_pll46xx_recalc_rate, 678 }; 679 680 /* 681 * PLL6552 Clock Type 682 */ 683 684 #define PLL6552_MDIV_MASK 0x3ff 685 #define PLL6552_PDIV_MASK 0x3f 686 #define PLL6552_SDIV_MASK 0x7 687 #define PLL6552_MDIV_SHIFT 16 688 #define PLL6552_MDIV_SHIFT_2416 14 689 #define PLL6552_PDIV_SHIFT 8 690 #define PLL6552_PDIV_SHIFT_2416 5 691 #define PLL6552_SDIV_SHIFT 0 692 693 static unsigned long samsung_pll6552_recalc_rate(struct clk_hw *hw, 694 unsigned long parent_rate) 695 { 696 struct samsung_clk_pll *pll = to_clk_pll(hw); 697 u32 mdiv, pdiv, sdiv, pll_con; 698 u64 fvco = parent_rate; 699 700 pll_con = readl_relaxed(pll->con_reg); 701 if (pll->type == pll_6552_s3c2416) { 702 mdiv = (pll_con >> PLL6552_MDIV_SHIFT_2416) & PLL6552_MDIV_MASK; 703 pdiv = (pll_con >> PLL6552_PDIV_SHIFT_2416) & PLL6552_PDIV_MASK; 704 } else { 705 mdiv = (pll_con >> PLL6552_MDIV_SHIFT) & PLL6552_MDIV_MASK; 706 pdiv = (pll_con >> PLL6552_PDIV_SHIFT) & PLL6552_PDIV_MASK; 707 } 708 sdiv = (pll_con >> PLL6552_SDIV_SHIFT) & PLL6552_SDIV_MASK; 709 710 fvco *= mdiv; 711 do_div(fvco, (pdiv << sdiv)); 712 713 return (unsigned long)fvco; 714 } 715 716 static const struct clk_ops samsung_pll6552_clk_ops = { 717 .recalc_rate = samsung_pll6552_recalc_rate, 718 }; 719 720 /* 721 * PLL6553 Clock Type 722 */ 723 724 #define PLL6553_MDIV_MASK 0xff 725 #define PLL6553_PDIV_MASK 0x3f 726 #define PLL6553_SDIV_MASK 0x7 727 #define PLL6553_KDIV_MASK 0xffff 728 #define PLL6553_MDIV_SHIFT 16 729 #define PLL6553_PDIV_SHIFT 8 730 #define PLL6553_SDIV_SHIFT 0 731 #define PLL6553_KDIV_SHIFT 0 732 733 static unsigned long samsung_pll6553_recalc_rate(struct clk_hw *hw, 734 unsigned long parent_rate) 735 { 736 struct samsung_clk_pll *pll = to_clk_pll(hw); 737 u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1; 738 u64 fvco = parent_rate; 739 740 pll_con0 = readl_relaxed(pll->con_reg); 741 pll_con1 = readl_relaxed(pll->con_reg + 0x4); 742 mdiv = (pll_con0 >> PLL6553_MDIV_SHIFT) & PLL6553_MDIV_MASK; 743 pdiv = (pll_con0 >> PLL6553_PDIV_SHIFT) & PLL6553_PDIV_MASK; 744 sdiv = (pll_con0 >> PLL6553_SDIV_SHIFT) & PLL6553_SDIV_MASK; 745 kdiv = (pll_con1 >> PLL6553_KDIV_SHIFT) & PLL6553_KDIV_MASK; 746 747 fvco *= (mdiv << 16) + kdiv; 748 do_div(fvco, (pdiv << sdiv)); 749 fvco >>= 16; 750 751 return (unsigned long)fvco; 752 } 753 754 static const struct clk_ops samsung_pll6553_clk_ops = { 755 .recalc_rate = samsung_pll6553_recalc_rate, 756 }; 757 758 /* 759 * PLL Clock Type of S3C24XX before S3C2443 760 */ 761 762 #define PLLS3C2410_MDIV_MASK (0xff) 763 #define PLLS3C2410_PDIV_MASK (0x1f) 764 #define PLLS3C2410_SDIV_MASK (0x3) 765 #define PLLS3C2410_MDIV_SHIFT (12) 766 #define PLLS3C2410_PDIV_SHIFT (4) 767 #define PLLS3C2410_SDIV_SHIFT (0) 768 769 #define PLLS3C2410_ENABLE_REG_OFFSET 0x10 770 771 static unsigned long samsung_s3c2410_pll_recalc_rate(struct clk_hw *hw, 772 unsigned long parent_rate) 773 { 774 struct samsung_clk_pll *pll = to_clk_pll(hw); 775 u32 pll_con, mdiv, pdiv, sdiv; 776 u64 fvco = parent_rate; 777 778 pll_con = readl_relaxed(pll->con_reg); 779 mdiv = (pll_con >> PLLS3C2410_MDIV_SHIFT) & PLLS3C2410_MDIV_MASK; 780 pdiv = (pll_con >> PLLS3C2410_PDIV_SHIFT) & PLLS3C2410_PDIV_MASK; 781 sdiv = (pll_con >> PLLS3C2410_SDIV_SHIFT) & PLLS3C2410_SDIV_MASK; 782 783 fvco *= (mdiv + 8); 784 do_div(fvco, (pdiv + 2) << sdiv); 785 786 return (unsigned int)fvco; 787 } 788 789 static unsigned long samsung_s3c2440_mpll_recalc_rate(struct clk_hw *hw, 790 unsigned long parent_rate) 791 { 792 struct samsung_clk_pll *pll = to_clk_pll(hw); 793 u32 pll_con, mdiv, pdiv, sdiv; 794 u64 fvco = parent_rate; 795 796 pll_con = readl_relaxed(pll->con_reg); 797 mdiv = (pll_con >> PLLS3C2410_MDIV_SHIFT) & PLLS3C2410_MDIV_MASK; 798 pdiv = (pll_con >> PLLS3C2410_PDIV_SHIFT) & PLLS3C2410_PDIV_MASK; 799 sdiv = (pll_con >> PLLS3C2410_SDIV_SHIFT) & PLLS3C2410_SDIV_MASK; 800 801 fvco *= (2 * (mdiv + 8)); 802 do_div(fvco, (pdiv + 2) << sdiv); 803 804 return (unsigned int)fvco; 805 } 806 807 static int samsung_s3c2410_pll_set_rate(struct clk_hw *hw, unsigned long drate, 808 unsigned long prate) 809 { 810 struct samsung_clk_pll *pll = to_clk_pll(hw); 811 const struct samsung_pll_rate_table *rate; 812 u32 tmp; 813 814 /* Get required rate settings from table */ 815 rate = samsung_get_pll_settings(pll, drate); 816 if (!rate) { 817 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 818 drate, clk_hw_get_name(hw)); 819 return -EINVAL; 820 } 821 822 tmp = readl_relaxed(pll->con_reg); 823 824 /* Change PLL PMS values */ 825 tmp &= ~((PLLS3C2410_MDIV_MASK << PLLS3C2410_MDIV_SHIFT) | 826 (PLLS3C2410_PDIV_MASK << PLLS3C2410_PDIV_SHIFT) | 827 (PLLS3C2410_SDIV_MASK << PLLS3C2410_SDIV_SHIFT)); 828 tmp |= (rate->mdiv << PLLS3C2410_MDIV_SHIFT) | 829 (rate->pdiv << PLLS3C2410_PDIV_SHIFT) | 830 (rate->sdiv << PLLS3C2410_SDIV_SHIFT); 831 writel_relaxed(tmp, pll->con_reg); 832 833 /* Time to settle according to the manual */ 834 udelay(300); 835 836 return 0; 837 } 838 839 static int samsung_s3c2410_pll_enable(struct clk_hw *hw, int bit, bool enable) 840 { 841 struct samsung_clk_pll *pll = to_clk_pll(hw); 842 u32 pll_en = readl_relaxed(pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET); 843 u32 pll_en_orig = pll_en; 844 845 if (enable) 846 pll_en &= ~BIT(bit); 847 else 848 pll_en |= BIT(bit); 849 850 writel_relaxed(pll_en, pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET); 851 852 /* if we started the UPLL, then allow to settle */ 853 if (enable && (pll_en_orig & BIT(bit))) 854 udelay(300); 855 856 return 0; 857 } 858 859 static int samsung_s3c2410_mpll_enable(struct clk_hw *hw) 860 { 861 return samsung_s3c2410_pll_enable(hw, 5, true); 862 } 863 864 static void samsung_s3c2410_mpll_disable(struct clk_hw *hw) 865 { 866 samsung_s3c2410_pll_enable(hw, 5, false); 867 } 868 869 static int samsung_s3c2410_upll_enable(struct clk_hw *hw) 870 { 871 return samsung_s3c2410_pll_enable(hw, 7, true); 872 } 873 874 static void samsung_s3c2410_upll_disable(struct clk_hw *hw) 875 { 876 samsung_s3c2410_pll_enable(hw, 7, false); 877 } 878 879 static const struct clk_ops samsung_s3c2410_mpll_clk_min_ops = { 880 .recalc_rate = samsung_s3c2410_pll_recalc_rate, 881 .enable = samsung_s3c2410_mpll_enable, 882 .disable = samsung_s3c2410_mpll_disable, 883 }; 884 885 static const struct clk_ops samsung_s3c2410_upll_clk_min_ops = { 886 .recalc_rate = samsung_s3c2410_pll_recalc_rate, 887 .enable = samsung_s3c2410_upll_enable, 888 .disable = samsung_s3c2410_upll_disable, 889 }; 890 891 static const struct clk_ops samsung_s3c2440_mpll_clk_min_ops = { 892 .recalc_rate = samsung_s3c2440_mpll_recalc_rate, 893 .enable = samsung_s3c2410_mpll_enable, 894 .disable = samsung_s3c2410_mpll_disable, 895 }; 896 897 static const struct clk_ops samsung_s3c2410_mpll_clk_ops = { 898 .recalc_rate = samsung_s3c2410_pll_recalc_rate, 899 .enable = samsung_s3c2410_mpll_enable, 900 .disable = samsung_s3c2410_mpll_disable, 901 .round_rate = samsung_pll_round_rate, 902 .set_rate = samsung_s3c2410_pll_set_rate, 903 }; 904 905 static const struct clk_ops samsung_s3c2410_upll_clk_ops = { 906 .recalc_rate = samsung_s3c2410_pll_recalc_rate, 907 .enable = samsung_s3c2410_upll_enable, 908 .disable = samsung_s3c2410_upll_disable, 909 .round_rate = samsung_pll_round_rate, 910 .set_rate = samsung_s3c2410_pll_set_rate, 911 }; 912 913 static const struct clk_ops samsung_s3c2440_mpll_clk_ops = { 914 .recalc_rate = samsung_s3c2440_mpll_recalc_rate, 915 .enable = samsung_s3c2410_mpll_enable, 916 .disable = samsung_s3c2410_mpll_disable, 917 .round_rate = samsung_pll_round_rate, 918 .set_rate = samsung_s3c2410_pll_set_rate, 919 }; 920 921 /* 922 * PLL2550x Clock Type 923 */ 924 925 #define PLL2550X_R_MASK (0x1) 926 #define PLL2550X_P_MASK (0x3F) 927 #define PLL2550X_M_MASK (0x3FF) 928 #define PLL2550X_S_MASK (0x7) 929 #define PLL2550X_R_SHIFT (20) 930 #define PLL2550X_P_SHIFT (14) 931 #define PLL2550X_M_SHIFT (4) 932 #define PLL2550X_S_SHIFT (0) 933 934 static unsigned long samsung_pll2550x_recalc_rate(struct clk_hw *hw, 935 unsigned long parent_rate) 936 { 937 struct samsung_clk_pll *pll = to_clk_pll(hw); 938 u32 r, p, m, s, pll_stat; 939 u64 fvco = parent_rate; 940 941 pll_stat = readl_relaxed(pll->con_reg); 942 r = (pll_stat >> PLL2550X_R_SHIFT) & PLL2550X_R_MASK; 943 if (!r) 944 return 0; 945 p = (pll_stat >> PLL2550X_P_SHIFT) & PLL2550X_P_MASK; 946 m = (pll_stat >> PLL2550X_M_SHIFT) & PLL2550X_M_MASK; 947 s = (pll_stat >> PLL2550X_S_SHIFT) & PLL2550X_S_MASK; 948 949 fvco *= m; 950 do_div(fvco, (p << s)); 951 952 return (unsigned long)fvco; 953 } 954 955 static const struct clk_ops samsung_pll2550x_clk_ops = { 956 .recalc_rate = samsung_pll2550x_recalc_rate, 957 }; 958 959 /* 960 * PLL2550xx Clock Type 961 */ 962 963 /* Maximum lock time can be 270 * PDIV cycles */ 964 #define PLL2550XX_LOCK_FACTOR 270 965 966 #define PLL2550XX_M_MASK 0x3FF 967 #define PLL2550XX_P_MASK 0x3F 968 #define PLL2550XX_S_MASK 0x7 969 #define PLL2550XX_LOCK_STAT_MASK 0x1 970 #define PLL2550XX_M_SHIFT 9 971 #define PLL2550XX_P_SHIFT 3 972 #define PLL2550XX_S_SHIFT 0 973 #define PLL2550XX_LOCK_STAT_SHIFT 21 974 975 static unsigned long samsung_pll2550xx_recalc_rate(struct clk_hw *hw, 976 unsigned long parent_rate) 977 { 978 struct samsung_clk_pll *pll = to_clk_pll(hw); 979 u32 mdiv, pdiv, sdiv, pll_con; 980 u64 fvco = parent_rate; 981 982 pll_con = readl_relaxed(pll->con_reg); 983 mdiv = (pll_con >> PLL2550XX_M_SHIFT) & PLL2550XX_M_MASK; 984 pdiv = (pll_con >> PLL2550XX_P_SHIFT) & PLL2550XX_P_MASK; 985 sdiv = (pll_con >> PLL2550XX_S_SHIFT) & PLL2550XX_S_MASK; 986 987 fvco *= mdiv; 988 do_div(fvco, (pdiv << sdiv)); 989 990 return (unsigned long)fvco; 991 } 992 993 static inline bool samsung_pll2550xx_mp_change(u32 mdiv, u32 pdiv, u32 pll_con) 994 { 995 u32 old_mdiv, old_pdiv; 996 997 old_mdiv = (pll_con >> PLL2550XX_M_SHIFT) & PLL2550XX_M_MASK; 998 old_pdiv = (pll_con >> PLL2550XX_P_SHIFT) & PLL2550XX_P_MASK; 999 1000 return mdiv != old_mdiv || pdiv != old_pdiv; 1001 } 1002 1003 static int samsung_pll2550xx_set_rate(struct clk_hw *hw, unsigned long drate, 1004 unsigned long prate) 1005 { 1006 struct samsung_clk_pll *pll = to_clk_pll(hw); 1007 const struct samsung_pll_rate_table *rate; 1008 u32 tmp; 1009 1010 /* Get required rate settings from table */ 1011 rate = samsung_get_pll_settings(pll, drate); 1012 if (!rate) { 1013 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 1014 drate, clk_hw_get_name(hw)); 1015 return -EINVAL; 1016 } 1017 1018 tmp = readl_relaxed(pll->con_reg); 1019 1020 if (!(samsung_pll2550xx_mp_change(rate->mdiv, rate->pdiv, tmp))) { 1021 /* If only s change, change just s value only*/ 1022 tmp &= ~(PLL2550XX_S_MASK << PLL2550XX_S_SHIFT); 1023 tmp |= rate->sdiv << PLL2550XX_S_SHIFT; 1024 writel_relaxed(tmp, pll->con_reg); 1025 1026 return 0; 1027 } 1028 1029 /* Set PLL lock time. */ 1030 writel_relaxed(rate->pdiv * PLL2550XX_LOCK_FACTOR, pll->lock_reg); 1031 1032 /* Change PLL PMS values */ 1033 tmp &= ~((PLL2550XX_M_MASK << PLL2550XX_M_SHIFT) | 1034 (PLL2550XX_P_MASK << PLL2550XX_P_SHIFT) | 1035 (PLL2550XX_S_MASK << PLL2550XX_S_SHIFT)); 1036 tmp |= (rate->mdiv << PLL2550XX_M_SHIFT) | 1037 (rate->pdiv << PLL2550XX_P_SHIFT) | 1038 (rate->sdiv << PLL2550XX_S_SHIFT); 1039 writel_relaxed(tmp, pll->con_reg); 1040 1041 /* wait_lock_time */ 1042 do { 1043 cpu_relax(); 1044 tmp = readl_relaxed(pll->con_reg); 1045 } while (!(tmp & (PLL2550XX_LOCK_STAT_MASK 1046 << PLL2550XX_LOCK_STAT_SHIFT))); 1047 1048 return 0; 1049 } 1050 1051 static const struct clk_ops samsung_pll2550xx_clk_ops = { 1052 .recalc_rate = samsung_pll2550xx_recalc_rate, 1053 .round_rate = samsung_pll_round_rate, 1054 .set_rate = samsung_pll2550xx_set_rate, 1055 }; 1056 1057 static const struct clk_ops samsung_pll2550xx_clk_min_ops = { 1058 .recalc_rate = samsung_pll2550xx_recalc_rate, 1059 }; 1060 1061 /* 1062 * PLL2650x Clock Type 1063 */ 1064 1065 /* Maximum lock time can be 3000 * PDIV cycles */ 1066 #define PLL2650X_LOCK_FACTOR 3000 1067 1068 #define PLL2650X_M_MASK 0x1ff 1069 #define PLL2650X_P_MASK 0x3f 1070 #define PLL2650X_S_MASK 0x7 1071 #define PLL2650X_K_MASK 0xffff 1072 #define PLL2650X_LOCK_STAT_MASK 0x1 1073 #define PLL2650X_M_SHIFT 16 1074 #define PLL2650X_P_SHIFT 8 1075 #define PLL2650X_S_SHIFT 0 1076 #define PLL2650X_K_SHIFT 0 1077 #define PLL2650X_LOCK_STAT_SHIFT 29 1078 #define PLL2650X_PLL_ENABLE_SHIFT 31 1079 1080 static unsigned long samsung_pll2650x_recalc_rate(struct clk_hw *hw, 1081 unsigned long parent_rate) 1082 { 1083 struct samsung_clk_pll *pll = to_clk_pll(hw); 1084 u64 fout = parent_rate; 1085 u32 mdiv, pdiv, sdiv, pll_con0, pll_con1; 1086 s16 kdiv; 1087 1088 pll_con0 = readl_relaxed(pll->con_reg); 1089 mdiv = (pll_con0 >> PLL2650X_M_SHIFT) & PLL2650X_M_MASK; 1090 pdiv = (pll_con0 >> PLL2650X_P_SHIFT) & PLL2650X_P_MASK; 1091 sdiv = (pll_con0 >> PLL2650X_S_SHIFT) & PLL2650X_S_MASK; 1092 1093 pll_con1 = readl_relaxed(pll->con_reg + 4); 1094 kdiv = (s16)((pll_con1 >> PLL2650X_K_SHIFT) & PLL2650X_K_MASK); 1095 1096 fout *= (mdiv << 16) + kdiv; 1097 do_div(fout, (pdiv << sdiv)); 1098 fout >>= 16; 1099 1100 return (unsigned long)fout; 1101 } 1102 1103 static int samsung_pll2650x_set_rate(struct clk_hw *hw, unsigned long drate, 1104 unsigned long prate) 1105 { 1106 struct samsung_clk_pll *pll = to_clk_pll(hw); 1107 const struct samsung_pll_rate_table *rate; 1108 u32 con0, con1; 1109 1110 /* Get required rate settings from table */ 1111 rate = samsung_get_pll_settings(pll, drate); 1112 if (!rate) { 1113 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 1114 drate, clk_hw_get_name(hw)); 1115 return -EINVAL; 1116 } 1117 1118 con0 = readl_relaxed(pll->con_reg); 1119 con1 = readl_relaxed(pll->con_reg + 4); 1120 1121 /* Set PLL lock time. */ 1122 writel_relaxed(rate->pdiv * PLL2650X_LOCK_FACTOR, pll->lock_reg); 1123 1124 /* Change PLL PMS values */ 1125 con0 &= ~((PLL2650X_M_MASK << PLL2650X_M_SHIFT) | 1126 (PLL2650X_P_MASK << PLL2650X_P_SHIFT) | 1127 (PLL2650X_S_MASK << PLL2650X_S_SHIFT)); 1128 con0 |= (rate->mdiv << PLL2650X_M_SHIFT) | 1129 (rate->pdiv << PLL2650X_P_SHIFT) | 1130 (rate->sdiv << PLL2650X_S_SHIFT); 1131 con0 |= (1 << PLL2650X_PLL_ENABLE_SHIFT); 1132 writel_relaxed(con0, pll->con_reg); 1133 1134 con1 &= ~(PLL2650X_K_MASK << PLL2650X_K_SHIFT); 1135 con1 |= ((rate->kdiv & PLL2650X_K_MASK) << PLL2650X_K_SHIFT); 1136 writel_relaxed(con1, pll->con_reg + 4); 1137 1138 do { 1139 cpu_relax(); 1140 con0 = readl_relaxed(pll->con_reg); 1141 } while (!(con0 & (PLL2650X_LOCK_STAT_MASK 1142 << PLL2650X_LOCK_STAT_SHIFT))); 1143 1144 return 0; 1145 } 1146 1147 static const struct clk_ops samsung_pll2650x_clk_ops = { 1148 .recalc_rate = samsung_pll2650x_recalc_rate, 1149 .round_rate = samsung_pll_round_rate, 1150 .set_rate = samsung_pll2650x_set_rate, 1151 }; 1152 1153 static const struct clk_ops samsung_pll2650x_clk_min_ops = { 1154 .recalc_rate = samsung_pll2650x_recalc_rate, 1155 }; 1156 1157 /* 1158 * PLL2650XX Clock Type 1159 */ 1160 1161 /* Maximum lock time can be 3000 * PDIV cycles */ 1162 #define PLL2650XX_LOCK_FACTOR 3000 1163 1164 #define PLL2650XX_MDIV_SHIFT 9 1165 #define PLL2650XX_PDIV_SHIFT 3 1166 #define PLL2650XX_SDIV_SHIFT 0 1167 #define PLL2650XX_KDIV_SHIFT 0 1168 #define PLL2650XX_MDIV_MASK 0x1ff 1169 #define PLL2650XX_PDIV_MASK 0x3f 1170 #define PLL2650XX_SDIV_MASK 0x7 1171 #define PLL2650XX_KDIV_MASK 0xffff 1172 #define PLL2650XX_PLL_ENABLE_SHIFT 23 1173 #define PLL2650XX_PLL_LOCKTIME_SHIFT 21 1174 #define PLL2650XX_PLL_FOUTMASK_SHIFT 31 1175 1176 static unsigned long samsung_pll2650xx_recalc_rate(struct clk_hw *hw, 1177 unsigned long parent_rate) 1178 { 1179 struct samsung_clk_pll *pll = to_clk_pll(hw); 1180 u32 mdiv, pdiv, sdiv, pll_con0, pll_con2; 1181 s16 kdiv; 1182 u64 fvco = parent_rate; 1183 1184 pll_con0 = readl_relaxed(pll->con_reg); 1185 pll_con2 = readl_relaxed(pll->con_reg + 8); 1186 mdiv = (pll_con0 >> PLL2650XX_MDIV_SHIFT) & PLL2650XX_MDIV_MASK; 1187 pdiv = (pll_con0 >> PLL2650XX_PDIV_SHIFT) & PLL2650XX_PDIV_MASK; 1188 sdiv = (pll_con0 >> PLL2650XX_SDIV_SHIFT) & PLL2650XX_SDIV_MASK; 1189 kdiv = (s16)(pll_con2 & PLL2650XX_KDIV_MASK); 1190 1191 fvco *= (mdiv << 16) + kdiv; 1192 do_div(fvco, (pdiv << sdiv)); 1193 fvco >>= 16; 1194 1195 return (unsigned long)fvco; 1196 } 1197 1198 static int samsung_pll2650xx_set_rate(struct clk_hw *hw, unsigned long drate, 1199 unsigned long parent_rate) 1200 { 1201 struct samsung_clk_pll *pll = to_clk_pll(hw); 1202 u32 tmp, pll_con0, pll_con2; 1203 const struct samsung_pll_rate_table *rate; 1204 1205 rate = samsung_get_pll_settings(pll, drate); 1206 if (!rate) { 1207 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__, 1208 drate, clk_hw_get_name(hw)); 1209 return -EINVAL; 1210 } 1211 1212 pll_con0 = readl_relaxed(pll->con_reg); 1213 pll_con2 = readl_relaxed(pll->con_reg + 8); 1214 1215 /* Change PLL PMS values */ 1216 pll_con0 &= ~(PLL2650XX_MDIV_MASK << PLL2650XX_MDIV_SHIFT | 1217 PLL2650XX_PDIV_MASK << PLL2650XX_PDIV_SHIFT | 1218 PLL2650XX_SDIV_MASK << PLL2650XX_SDIV_SHIFT); 1219 pll_con0 |= rate->mdiv << PLL2650XX_MDIV_SHIFT; 1220 pll_con0 |= rate->pdiv << PLL2650XX_PDIV_SHIFT; 1221 pll_con0 |= rate->sdiv << PLL2650XX_SDIV_SHIFT; 1222 pll_con0 |= 1 << PLL2650XX_PLL_ENABLE_SHIFT; 1223 pll_con0 |= 1 << PLL2650XX_PLL_FOUTMASK_SHIFT; 1224 1225 pll_con2 &= ~(PLL2650XX_KDIV_MASK << PLL2650XX_KDIV_SHIFT); 1226 pll_con2 |= ((~(rate->kdiv) + 1) & PLL2650XX_KDIV_MASK) 1227 << PLL2650XX_KDIV_SHIFT; 1228 1229 /* Set PLL lock time. */ 1230 writel_relaxed(PLL2650XX_LOCK_FACTOR * rate->pdiv, pll->lock_reg); 1231 1232 writel_relaxed(pll_con0, pll->con_reg); 1233 writel_relaxed(pll_con2, pll->con_reg + 8); 1234 1235 do { 1236 tmp = readl_relaxed(pll->con_reg); 1237 } while (!(tmp & (0x1 << PLL2650XX_PLL_LOCKTIME_SHIFT))); 1238 1239 return 0; 1240 } 1241 1242 static const struct clk_ops samsung_pll2650xx_clk_ops = { 1243 .recalc_rate = samsung_pll2650xx_recalc_rate, 1244 .set_rate = samsung_pll2650xx_set_rate, 1245 .round_rate = samsung_pll_round_rate, 1246 }; 1247 1248 static const struct clk_ops samsung_pll2650xx_clk_min_ops = { 1249 .recalc_rate = samsung_pll2650xx_recalc_rate, 1250 }; 1251 1252 static void __init _samsung_clk_register_pll(struct samsung_clk_provider *ctx, 1253 const struct samsung_pll_clock *pll_clk, 1254 void __iomem *base) 1255 { 1256 struct samsung_clk_pll *pll; 1257 struct clk_init_data init; 1258 int ret, len; 1259 1260 pll = kzalloc(sizeof(*pll), GFP_KERNEL); 1261 if (!pll) { 1262 pr_err("%s: could not allocate pll clk %s\n", 1263 __func__, pll_clk->name); 1264 return; 1265 } 1266 1267 init.name = pll_clk->name; 1268 init.flags = pll_clk->flags; 1269 init.parent_names = &pll_clk->parent_name; 1270 init.num_parents = 1; 1271 1272 if (pll_clk->rate_table) { 1273 /* find count of rates in rate_table */ 1274 for (len = 0; pll_clk->rate_table[len].rate != 0; ) 1275 len++; 1276 1277 pll->rate_count = len; 1278 pll->rate_table = kmemdup(pll_clk->rate_table, 1279 pll->rate_count * 1280 sizeof(struct samsung_pll_rate_table), 1281 GFP_KERNEL); 1282 WARN(!pll->rate_table, 1283 "%s: could not allocate rate table for %s\n", 1284 __func__, pll_clk->name); 1285 } 1286 1287 switch (pll_clk->type) { 1288 case pll_2126: 1289 init.ops = &samsung_pll2126_clk_ops; 1290 break; 1291 case pll_3000: 1292 init.ops = &samsung_pll3000_clk_ops; 1293 break; 1294 /* clk_ops for 35xx and 2550 are similar */ 1295 case pll_35xx: 1296 case pll_2550: 1297 case pll_1450x: 1298 case pll_1451x: 1299 case pll_1452x: 1300 pll->enable_offs = PLL35XX_ENABLE_SHIFT; 1301 pll->lock_offs = PLL35XX_LOCK_STAT_SHIFT; 1302 if (!pll->rate_table) 1303 init.ops = &samsung_pll35xx_clk_min_ops; 1304 else 1305 init.ops = &samsung_pll35xx_clk_ops; 1306 break; 1307 case pll_4500: 1308 init.ops = &samsung_pll45xx_clk_min_ops; 1309 break; 1310 case pll_4502: 1311 case pll_4508: 1312 if (!pll->rate_table) 1313 init.ops = &samsung_pll45xx_clk_min_ops; 1314 else 1315 init.ops = &samsung_pll45xx_clk_ops; 1316 break; 1317 /* clk_ops for 36xx and 2650 are similar */ 1318 case pll_36xx: 1319 case pll_2650: 1320 pll->enable_offs = PLL36XX_ENABLE_SHIFT; 1321 pll->lock_offs = PLL36XX_LOCK_STAT_SHIFT; 1322 if (!pll->rate_table) 1323 init.ops = &samsung_pll36xx_clk_min_ops; 1324 else 1325 init.ops = &samsung_pll36xx_clk_ops; 1326 break; 1327 case pll_6552: 1328 case pll_6552_s3c2416: 1329 init.ops = &samsung_pll6552_clk_ops; 1330 break; 1331 case pll_6553: 1332 init.ops = &samsung_pll6553_clk_ops; 1333 break; 1334 case pll_4600: 1335 case pll_4650: 1336 case pll_4650c: 1337 case pll_1460x: 1338 if (!pll->rate_table) 1339 init.ops = &samsung_pll46xx_clk_min_ops; 1340 else 1341 init.ops = &samsung_pll46xx_clk_ops; 1342 break; 1343 case pll_s3c2410_mpll: 1344 if (!pll->rate_table) 1345 init.ops = &samsung_s3c2410_mpll_clk_min_ops; 1346 else 1347 init.ops = &samsung_s3c2410_mpll_clk_ops; 1348 break; 1349 case pll_s3c2410_upll: 1350 if (!pll->rate_table) 1351 init.ops = &samsung_s3c2410_upll_clk_min_ops; 1352 else 1353 init.ops = &samsung_s3c2410_upll_clk_ops; 1354 break; 1355 case pll_s3c2440_mpll: 1356 if (!pll->rate_table) 1357 init.ops = &samsung_s3c2440_mpll_clk_min_ops; 1358 else 1359 init.ops = &samsung_s3c2440_mpll_clk_ops; 1360 break; 1361 case pll_2550x: 1362 init.ops = &samsung_pll2550x_clk_ops; 1363 break; 1364 case pll_2550xx: 1365 if (!pll->rate_table) 1366 init.ops = &samsung_pll2550xx_clk_min_ops; 1367 else 1368 init.ops = &samsung_pll2550xx_clk_ops; 1369 break; 1370 case pll_2650x: 1371 if (!pll->rate_table) 1372 init.ops = &samsung_pll2650x_clk_min_ops; 1373 else 1374 init.ops = &samsung_pll2650x_clk_ops; 1375 break; 1376 case pll_2650xx: 1377 if (!pll->rate_table) 1378 init.ops = &samsung_pll2650xx_clk_min_ops; 1379 else 1380 init.ops = &samsung_pll2650xx_clk_ops; 1381 break; 1382 default: 1383 pr_warn("%s: Unknown pll type for pll clk %s\n", 1384 __func__, pll_clk->name); 1385 } 1386 1387 pll->hw.init = &init; 1388 pll->type = pll_clk->type; 1389 pll->lock_reg = base + pll_clk->lock_offset; 1390 pll->con_reg = base + pll_clk->con_offset; 1391 1392 ret = clk_hw_register(ctx->dev, &pll->hw); 1393 if (ret) { 1394 pr_err("%s: failed to register pll clock %s : %d\n", 1395 __func__, pll_clk->name, ret); 1396 kfree(pll); 1397 return; 1398 } 1399 1400 samsung_clk_add_lookup(ctx, &pll->hw, pll_clk->id); 1401 } 1402 1403 void __init samsung_clk_register_pll(struct samsung_clk_provider *ctx, 1404 const struct samsung_pll_clock *pll_list, 1405 unsigned int nr_pll, void __iomem *base) 1406 { 1407 int cnt; 1408 1409 for (cnt = 0; cnt < nr_pll; cnt++) 1410 _samsung_clk_register_pll(ctx, &pll_list[cnt], base); 1411 } 1412