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

clk: ingenic: Allow divider value to be divided

The JZ4780's MSC clock divider registers multiply the clock divider by 2.
This means that MMC devices run at half their expected speed. Add the
ability to divide the clock divider in order to solve this.

Signed-off-by: Harvey Hunt <harvey.hunt@imgtec.com>
Cc: Paul Burton <paul.burton@imgtec.com>
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: linux-clk@vger.kernel.org
Cc: linux-kernel@vger.kernel.org
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>

authored by

Harvey Hunt and committed by
Stephen Boyd
4afe2d1a 5707291c

+47 -34
+10 -1
drivers/clk/ingenic/cgu.c
··· 325 325 div = (div_reg >> clk_info->div.shift) & 326 326 GENMASK(clk_info->div.bits - 1, 0); 327 327 div += 1; 328 + div *= clk_info->div.div; 328 329 329 330 rate /= div; 330 331 } ··· 345 344 /* and impose hardware constraints */ 346 345 div = min_t(unsigned, div, 1 << clk_info->div.bits); 347 346 div = max_t(unsigned, div, 1); 347 + 348 + /* 349 + * If the divider value itself must be divided before being written to 350 + * the divider register, we must ensure we don't have any bits set that 351 + * would be lost as a result of doing so. 352 + */ 353 + div /= clk_info->div.div; 354 + div *= clk_info->div.div; 348 355 349 356 return div; 350 357 } ··· 404 395 /* update the divide */ 405 396 mask = GENMASK(clk_info->div.bits - 1, 0); 406 397 reg &= ~(mask << clk_info->div.shift); 407 - reg |= (div - 1) << clk_info->div.shift; 398 + reg |= ((div / clk_info->div.div) - 1) << clk_info->div.shift; 408 399 409 400 /* clear the stop bit */ 410 401 if (clk_info->div.stop_bit != -1)
+5 -1
drivers/clk/ingenic/cgu.h
··· 76 76 /** 77 77 * struct ingenic_cgu_div_info - information about a divider 78 78 * @reg: offset of the divider control register within the CGU 79 - * @shift: number of bits to shift the divide value by (ie. the index of 79 + * @shift: number of bits to left shift the divide value by (ie. the index of 80 80 * the lowest bit of the divide value within its control register) 81 + * @div: number of bits to divide the divider value by (i.e. if the 82 + * effective divider value is the value written to the register 83 + * multiplied by some constant) 81 84 * @bits: the size of the divide value in bits 82 85 * @ce_bit: the index of the change enable bit within reg, or -1 if there 83 86 * isn't one ··· 90 87 struct ingenic_cgu_div_info { 91 88 unsigned reg; 92 89 u8 shift; 90 + u8 div; 93 91 u8 bits; 94 92 s8 ce_bit; 95 93 s8 busy_bit;
+12 -12
drivers/clk/ingenic/jz4740-cgu.c
··· 90 90 [JZ4740_CLK_PLL_HALF] = { 91 91 "pll half", CGU_CLK_DIV, 92 92 .parents = { JZ4740_CLK_PLL, -1, -1, -1 }, 93 - .div = { CGU_REG_CPCCR, 21, 1, -1, -1, -1 }, 93 + .div = { CGU_REG_CPCCR, 21, 1, 1, -1, -1, -1 }, 94 94 }, 95 95 96 96 [JZ4740_CLK_CCLK] = { 97 97 "cclk", CGU_CLK_DIV, 98 98 .parents = { JZ4740_CLK_PLL, -1, -1, -1 }, 99 - .div = { CGU_REG_CPCCR, 0, 4, 22, -1, -1 }, 99 + .div = { CGU_REG_CPCCR, 0, 1, 4, 22, -1, -1 }, 100 100 }, 101 101 102 102 [JZ4740_CLK_HCLK] = { 103 103 "hclk", CGU_CLK_DIV, 104 104 .parents = { JZ4740_CLK_PLL, -1, -1, -1 }, 105 - .div = { CGU_REG_CPCCR, 4, 4, 22, -1, -1 }, 105 + .div = { CGU_REG_CPCCR, 4, 1, 4, 22, -1, -1 }, 106 106 }, 107 107 108 108 [JZ4740_CLK_PCLK] = { 109 109 "pclk", CGU_CLK_DIV, 110 110 .parents = { JZ4740_CLK_PLL, -1, -1, -1 }, 111 - .div = { CGU_REG_CPCCR, 8, 4, 22, -1, -1 }, 111 + .div = { CGU_REG_CPCCR, 8, 1, 4, 22, -1, -1 }, 112 112 }, 113 113 114 114 [JZ4740_CLK_MCLK] = { 115 115 "mclk", CGU_CLK_DIV, 116 116 .parents = { JZ4740_CLK_PLL, -1, -1, -1 }, 117 - .div = { CGU_REG_CPCCR, 12, 4, 22, -1, -1 }, 117 + .div = { CGU_REG_CPCCR, 12, 1, 4, 22, -1, -1 }, 118 118 }, 119 119 120 120 [JZ4740_CLK_LCD] = { 121 121 "lcd", CGU_CLK_DIV | CGU_CLK_GATE, 122 122 .parents = { JZ4740_CLK_PLL_HALF, -1, -1, -1 }, 123 - .div = { CGU_REG_CPCCR, 16, 5, 22, -1, -1 }, 123 + .div = { CGU_REG_CPCCR, 16, 1, 5, 22, -1, -1 }, 124 124 .gate = { CGU_REG_CLKGR, 10 }, 125 125 }, 126 126 127 127 [JZ4740_CLK_LCD_PCLK] = { 128 128 "lcd_pclk", CGU_CLK_DIV, 129 129 .parents = { JZ4740_CLK_PLL_HALF, -1, -1, -1 }, 130 - .div = { CGU_REG_LPCDR, 0, 11, -1, -1, -1 }, 130 + .div = { CGU_REG_LPCDR, 0, 1, 11, -1, -1, -1 }, 131 131 }, 132 132 133 133 [JZ4740_CLK_I2S] = { 134 134 "i2s", CGU_CLK_MUX | CGU_CLK_DIV | CGU_CLK_GATE, 135 135 .parents = { JZ4740_CLK_EXT, JZ4740_CLK_PLL_HALF, -1, -1 }, 136 136 .mux = { CGU_REG_CPCCR, 31, 1 }, 137 - .div = { CGU_REG_I2SCDR, 0, 8, -1, -1, -1 }, 137 + .div = { CGU_REG_I2SCDR, 0, 1, 8, -1, -1, -1 }, 138 138 .gate = { CGU_REG_CLKGR, 6 }, 139 139 }, 140 140 ··· 142 142 "spi", CGU_CLK_MUX | CGU_CLK_DIV | CGU_CLK_GATE, 143 143 .parents = { JZ4740_CLK_EXT, JZ4740_CLK_PLL, -1, -1 }, 144 144 .mux = { CGU_REG_SSICDR, 31, 1 }, 145 - .div = { CGU_REG_SSICDR, 0, 4, -1, -1, -1 }, 145 + .div = { CGU_REG_SSICDR, 0, 1, 4, -1, -1, -1 }, 146 146 .gate = { CGU_REG_CLKGR, 4 }, 147 147 }, 148 148 149 149 [JZ4740_CLK_MMC] = { 150 150 "mmc", CGU_CLK_DIV | CGU_CLK_GATE, 151 151 .parents = { JZ4740_CLK_PLL_HALF, -1, -1, -1 }, 152 - .div = { CGU_REG_MSCCDR, 0, 5, -1, -1, -1 }, 152 + .div = { CGU_REG_MSCCDR, 0, 1, 5, -1, -1, -1 }, 153 153 .gate = { CGU_REG_CLKGR, 7 }, 154 154 }, 155 155 156 156 [JZ4740_CLK_UHC] = { 157 157 "uhc", CGU_CLK_DIV | CGU_CLK_GATE, 158 158 .parents = { JZ4740_CLK_PLL_HALF, -1, -1, -1 }, 159 - .div = { CGU_REG_UHCCDR, 0, 4, -1, -1, -1 }, 159 + .div = { CGU_REG_UHCCDR, 0, 1, 4, -1, -1, -1 }, 160 160 .gate = { CGU_REG_CLKGR, 14 }, 161 161 }, 162 162 ··· 164 164 "udc", CGU_CLK_MUX | CGU_CLK_DIV, 165 165 .parents = { JZ4740_CLK_EXT, JZ4740_CLK_PLL_HALF, -1, -1 }, 166 166 .mux = { CGU_REG_CPCCR, 29, 1 }, 167 - .div = { CGU_REG_CPCCR, 23, 6, -1, -1, -1 }, 167 + .div = { CGU_REG_CPCCR, 23, 1, 6, -1, -1, -1 }, 168 168 .gate = { CGU_REG_SCR, 6 }, 169 169 }, 170 170
+20 -20
drivers/clk/ingenic/jz4780-cgu.c
··· 296 296 [JZ4780_CLK_CPU] = { 297 297 "cpu", CGU_CLK_DIV, 298 298 .parents = { JZ4780_CLK_CPUMUX, -1, -1, -1 }, 299 - .div = { CGU_REG_CLOCKCONTROL, 0, 4, 22, -1, -1 }, 299 + .div = { CGU_REG_CLOCKCONTROL, 0, 1, 4, 22, -1, -1 }, 300 300 }, 301 301 302 302 [JZ4780_CLK_L2CACHE] = { 303 303 "l2cache", CGU_CLK_DIV, 304 304 .parents = { JZ4780_CLK_CPUMUX, -1, -1, -1 }, 305 - .div = { CGU_REG_CLOCKCONTROL, 4, 4, -1, -1, -1 }, 305 + .div = { CGU_REG_CLOCKCONTROL, 4, 1, 4, -1, -1, -1 }, 306 306 }, 307 307 308 308 [JZ4780_CLK_AHB0] = { ··· 310 310 .parents = { -1, JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 311 311 JZ4780_CLK_EPLL }, 312 312 .mux = { CGU_REG_CLOCKCONTROL, 26, 2 }, 313 - .div = { CGU_REG_CLOCKCONTROL, 8, 4, 21, -1, -1 }, 313 + .div = { CGU_REG_CLOCKCONTROL, 8, 1, 4, 21, -1, -1 }, 314 314 }, 315 315 316 316 [JZ4780_CLK_AHB2PMUX] = { ··· 323 323 [JZ4780_CLK_AHB2] = { 324 324 "ahb2", CGU_CLK_DIV, 325 325 .parents = { JZ4780_CLK_AHB2PMUX, -1, -1, -1 }, 326 - .div = { CGU_REG_CLOCKCONTROL, 12, 4, 20, -1, -1 }, 326 + .div = { CGU_REG_CLOCKCONTROL, 12, 1, 4, 20, -1, -1 }, 327 327 }, 328 328 329 329 [JZ4780_CLK_PCLK] = { 330 330 "pclk", CGU_CLK_DIV, 331 331 .parents = { JZ4780_CLK_AHB2PMUX, -1, -1, -1 }, 332 - .div = { CGU_REG_CLOCKCONTROL, 16, 4, 20, -1, -1 }, 332 + .div = { CGU_REG_CLOCKCONTROL, 16, 1, 4, 20, -1, -1 }, 333 333 }, 334 334 335 335 [JZ4780_CLK_DDR] = { 336 336 "ddr", CGU_CLK_MUX | CGU_CLK_DIV, 337 337 .parents = { -1, JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, -1 }, 338 338 .mux = { CGU_REG_DDRCDR, 30, 2 }, 339 - .div = { CGU_REG_DDRCDR, 0, 4, 29, 28, 27 }, 339 + .div = { CGU_REG_DDRCDR, 0, 1, 4, 29, 28, 27 }, 340 340 }, 341 341 342 342 [JZ4780_CLK_VPU] = { ··· 344 344 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 345 345 JZ4780_CLK_EPLL, -1 }, 346 346 .mux = { CGU_REG_VPUCDR, 30, 2 }, 347 - .div = { CGU_REG_VPUCDR, 0, 4, 29, 28, 27 }, 347 + .div = { CGU_REG_VPUCDR, 0, 1, 4, 29, 28, 27 }, 348 348 .gate = { CGU_REG_CLKGR1, 2 }, 349 349 }, 350 350 ··· 352 352 "i2s_pll", CGU_CLK_MUX | CGU_CLK_DIV, 353 353 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_EPLL, -1, -1 }, 354 354 .mux = { CGU_REG_I2SCDR, 30, 1 }, 355 - .div = { CGU_REG_I2SCDR, 0, 8, 29, 28, 27 }, 355 + .div = { CGU_REG_I2SCDR, 0, 1, 8, 29, 28, 27 }, 356 356 }, 357 357 358 358 [JZ4780_CLK_I2S] = { ··· 366 366 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 367 367 JZ4780_CLK_VPLL, -1 }, 368 368 .mux = { CGU_REG_LP0CDR, 30, 2 }, 369 - .div = { CGU_REG_LP0CDR, 0, 8, 28, 27, 26 }, 369 + .div = { CGU_REG_LP0CDR, 0, 1, 8, 28, 27, 26 }, 370 370 }, 371 371 372 372 [JZ4780_CLK_LCD1PIXCLK] = { ··· 374 374 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 375 375 JZ4780_CLK_VPLL, -1 }, 376 376 .mux = { CGU_REG_LP1CDR, 30, 2 }, 377 - .div = { CGU_REG_LP1CDR, 0, 8, 28, 27, 26 }, 377 + .div = { CGU_REG_LP1CDR, 0, 1, 8, 28, 27, 26 }, 378 378 }, 379 379 380 380 [JZ4780_CLK_MSCMUX] = { ··· 386 386 [JZ4780_CLK_MSC0] = { 387 387 "msc0", CGU_CLK_DIV | CGU_CLK_GATE, 388 388 .parents = { JZ4780_CLK_MSCMUX, -1, -1, -1 }, 389 - .div = { CGU_REG_MSC0CDR, 0, 8, 29, 28, 27 }, 389 + .div = { CGU_REG_MSC0CDR, 0, 2, 8, 29, 28, 27 }, 390 390 .gate = { CGU_REG_CLKGR0, 3 }, 391 391 }, 392 392 393 393 [JZ4780_CLK_MSC1] = { 394 394 "msc1", CGU_CLK_DIV | CGU_CLK_GATE, 395 395 .parents = { JZ4780_CLK_MSCMUX, -1, -1, -1 }, 396 - .div = { CGU_REG_MSC1CDR, 0, 8, 29, 28, 27 }, 396 + .div = { CGU_REG_MSC1CDR, 0, 2, 8, 29, 28, 27 }, 397 397 .gate = { CGU_REG_CLKGR0, 11 }, 398 398 }, 399 399 400 400 [JZ4780_CLK_MSC2] = { 401 401 "msc2", CGU_CLK_DIV | CGU_CLK_GATE, 402 402 .parents = { JZ4780_CLK_MSCMUX, -1, -1, -1 }, 403 - .div = { CGU_REG_MSC2CDR, 0, 8, 29, 28, 27 }, 403 + .div = { CGU_REG_MSC2CDR, 0, 2, 8, 29, 28, 27 }, 404 404 .gate = { CGU_REG_CLKGR0, 12 }, 405 405 }, 406 406 ··· 409 409 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 410 410 JZ4780_CLK_EPLL, JZ4780_CLK_OTGPHY }, 411 411 .mux = { CGU_REG_UHCCDR, 30, 2 }, 412 - .div = { CGU_REG_UHCCDR, 0, 8, 29, 28, 27 }, 412 + .div = { CGU_REG_UHCCDR, 0, 1, 8, 29, 28, 27 }, 413 413 .gate = { CGU_REG_CLKGR0, 24 }, 414 414 }, 415 415 ··· 417 417 "ssi_pll", CGU_CLK_MUX | CGU_CLK_DIV, 418 418 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, -1, -1 }, 419 419 .mux = { CGU_REG_SSICDR, 30, 1 }, 420 - .div = { CGU_REG_SSICDR, 0, 8, 29, 28, 27 }, 420 + .div = { CGU_REG_SSICDR, 0, 1, 8, 29, 28, 27 }, 421 421 }, 422 422 423 423 [JZ4780_CLK_SSI] = { ··· 430 430 "cim_mclk", CGU_CLK_MUX | CGU_CLK_DIV, 431 431 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, -1, -1 }, 432 432 .mux = { CGU_REG_CIMCDR, 31, 1 }, 433 - .div = { CGU_REG_CIMCDR, 0, 8, 30, 29, 28 }, 433 + .div = { CGU_REG_CIMCDR, 0, 1, 8, 30, 29, 28 }, 434 434 }, 435 435 436 436 [JZ4780_CLK_PCMPLL] = { ··· 438 438 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 439 439 JZ4780_CLK_EPLL, JZ4780_CLK_VPLL }, 440 440 .mux = { CGU_REG_PCMCDR, 29, 2 }, 441 - .div = { CGU_REG_PCMCDR, 0, 8, 28, 27, 26 }, 441 + .div = { CGU_REG_PCMCDR, 0, 1, 8, 28, 27, 26 }, 442 442 }, 443 443 444 444 [JZ4780_CLK_PCM] = { ··· 453 453 .parents = { -1, JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 454 454 JZ4780_CLK_EPLL }, 455 455 .mux = { CGU_REG_GPUCDR, 30, 2 }, 456 - .div = { CGU_REG_GPUCDR, 0, 4, 29, 28, 27 }, 456 + .div = { CGU_REG_GPUCDR, 0, 1, 4, 29, 28, 27 }, 457 457 .gate = { CGU_REG_CLKGR1, 4 }, 458 458 }, 459 459 ··· 462 462 .parents = { JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 463 463 JZ4780_CLK_VPLL, -1 }, 464 464 .mux = { CGU_REG_HDMICDR, 30, 2 }, 465 - .div = { CGU_REG_HDMICDR, 0, 8, 29, 28, 26 }, 465 + .div = { CGU_REG_HDMICDR, 0, 1, 8, 29, 28, 26 }, 466 466 .gate = { CGU_REG_CLKGR1, 9 }, 467 467 }, 468 468 ··· 471 471 .parents = { -1, JZ4780_CLK_SCLKA, JZ4780_CLK_MPLL, 472 472 JZ4780_CLK_EPLL }, 473 473 .mux = { CGU_REG_BCHCDR, 30, 2 }, 474 - .div = { CGU_REG_BCHCDR, 0, 4, 29, 28, 27 }, 474 + .div = { CGU_REG_BCHCDR, 0, 1, 4, 29, 28, 27 }, 475 475 .gate = { CGU_REG_CLKGR0, 1 }, 476 476 }, 477 477