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