Lines Matching +full:clock +full:- +full:tree

1 // SPDX-License-Identifier: GPL-2.0-or-later
7 * common clock driver support for the MPC512x platform
12 #include <linux/clk-provider.h>
21 #include <dt-bindings/clock/mpc512x-clock.h>
69 /* data required for the OF clock provider registration */
89 * interpretation, no CFM, different fourth PSC/CAN mux0 input -- yet
90 * those differences can get folded into this clock provider support
292 val &= (1 << len) - 1; in get_bit_field()
305 spmf = get_bit_field(&clkregs->spmr, 24, 4); in get_spmf_mult()
326 divcode = get_bit_field(&clkregs->scfr2, 26, 6); in get_sys_div_x2()
350 cpmf = get_bit_field(&clkregs->spmr, 16, 4); in get_cpmf_mult_x2()
359 * some of the clock dividers do scale in a linear way, yet not all of
389 np = of_find_compatible_node(NULL, NULL, "fsl,mpc5121-immr"); in get_freq_from_dt()
404 clks[i] = ERR_PTR(-ENODEV); in mpc512x_clk_preset_data()
408 * - receives the "bus frequency" from the caller (that's the IPS clock
409 * rate, the historical source of clock information)
410 * - fetches the system PLL multiplier and divider values as well as the
412 * - determines the REF clock rate either from the XTAL/OSC spec (if
413 * there is a device tree node describing the oscillator) or from the
414 * IPS bus clock (supported for backwards compatibility, such that
416 * - creates the "ref" clock item in the clock tree, such that
417 * subsequent code can create the remainder of the hierarchy (REF ->
418 * SYS -> CSB -> IPS) from the REF clock rate and the returned mul/div
432 *ips_div = get_bit_field(&clkregs->scfr1, 23, 3); in mpc512x_clk_setup_ref_clock()
434 /* lookup the oscillator clock for its rate */ in mpc512x_clk_setup_ref_clock()
442 * in either case the REF clock gets created here and the in mpc512x_clk_setup_ref_clock()
443 * remainder of the clock tree can get spanned from there in mpc512x_clk_setup_ref_clock()
457 calc_freq *= *ips_div; /* IPS -> CSB */ in mpc512x_clk_setup_ref_clock()
458 calc_freq *= 2; /* CSB -> SYS */ in mpc512x_clk_setup_ref_clock()
459 calc_freq *= *sys_div; /* SYS -> PLL out */ in mpc512x_clk_setup_ref_clock()
460 calc_freq /= *sys_mul; /* PLL out -> REF == OSC */ in mpc512x_clk_setup_ref_clock()
471 * suggests that all instances of the "PSC clock generation" are equal,
472 * and that one might re-use the PSC setup for MSCAN clock generation
476 * the details (starting at page 5-20) show differences in the specific
478 * factual non-availability of the second mux stage (it's present yet
481 * the MSCAN clock related registers (starting at page 5-35) all
485 * TODO re-check the document, and clarify whether the RM is correct in
489 * it turns out that the RM rev4 as of 2012-06 talks about "can" for the
490 * PSCs while RM rev3 as of 2008-10 talks about "spdif", so I guess that
497 * it's the very data type dictated by <linux/clk-provider.h>,
501 "sys", "ref", "psc-mclk-in", "spdif-tx",
505 "sys", "ref", "psc-mclk-in", "can-clk-in",
527 "psc" #id "-mux0", \
528 "psc" #id "-en0", \
536 "mscan" #id "-mux0", \
537 "mscan" #id "-en0", \
545 "spdif-mux0", \
546 "spdif-en0", \
548 { "spdif_mclk_div", "spdif-rx", }, \
554 "out" #id "-mux0", \
555 "out" #id "-en0", \
594 /* setup the MCLK clock subtree of an individual PSC/MSCAN/SPDIF */
602 switch (entry->type) { in mpc512x_clk_setup_mclk()
607 mccr_reg = &clkregs->psc_ccr[idx]; in mpc512x_clk_setup_mclk()
613 mccr_reg = &clkregs->mscan_ccr[idx]; in mpc512x_clk_setup_mclk()
619 mccr_reg = &clkregs->spccr; in mpc512x_clk_setup_mclk()
626 mccr_reg = &clkregs->out_ccr[idx]; in mpc512x_clk_setup_mclk()
634 * enforced a specific MCLK divider while the clock was gated in mpc512x_clk_setup_mclk()
641 * adjusted the divider which the clock setup might have left in in mpc512x_clk_setup_mclk()
645 * - MCLK 0 from SYS in mpc512x_clk_setup_mclk()
646 * - MCLK DIV such to not exceed the IPS clock in mpc512x_clk_setup_mclk()
647 * - MCLK 0 enabled in mpc512x_clk_setup_mclk()
648 * - MCLK 1 from MCLK DIV in mpc512x_clk_setup_mclk()
653 out_be32(mccr_reg, (0 << 16) | ((div - 1) << 17)); in mpc512x_clk_setup_mclk()
654 out_be32(mccr_reg, (1 << 16) | ((div - 1) << 17)); in mpc512x_clk_setup_mclk()
657 * create the 'struct clk' items of the MCLK's clock subtree in mpc512x_clk_setup_mclk()
661 * - the "internal" MCLK_DIV and MCLK_OUT signal in turn are in mpc512x_clk_setup_mclk()
665 * - in the absence of "aliases" for clocks we need to create in mpc512x_clk_setup_mclk()
669 * - for easier future maintenance and for better reflection of in mpc512x_clk_setup_mclk()
671 * clock items even for those muxers which actually are NOPs in mpc512x_clk_setup_mclk()
675 entry->name_mux0, in mpc512x_clk_setup_mclk()
682 entry->name_en0, entry->name_mux0, in mpc512x_clk_setup_mclk()
685 entry->name_div0, in mpc512x_clk_setup_mclk()
686 entry->name_en0, CLK_SET_RATE_GATE, in mpc512x_clk_setup_mclk()
688 if (entry->has_mclk1) { in mpc512x_clk_setup_mclk()
690 entry->name_mclk, in mpc512x_clk_setup_mclk()
691 &entry->parent_names_mux1[0], in mpc512x_clk_setup_mclk()
692 ARRAY_SIZE(entry->parent_names_mux1), in mpc512x_clk_setup_mclk()
696 entry->name_mclk, in mpc512x_clk_setup_mclk()
697 entry->parent_names_mux1[0], in mpc512x_clk_setup_mclk()
713 * - consider whether to handle clocks which have both gates and in mpc512x_clk_setup_clock_tree()
715 * - fractional dividers appear to not map well to composites in mpc512x_clk_setup_clock_tree()
720 * - PSC/MSCAN/SPDIF clock generation OTOH already is very in mpc512x_clk_setup_clock_tree()
723 * intermediate clock signals get referenced elsewhere (e.g. in mpc512x_clk_setup_clock_tree()
724 * in the clock frequency measurement, CFM) and thus need in mpc512x_clk_setup_clock_tree()
726 * - the current source layout appropriately reflects the in mpc512x_clk_setup_clock_tree()
734 /* now setup the REF -> SYS -> CSB -> IPS hierarchy */ in mpc512x_clk_setup_clock_tree()
739 &clkregs->scfr1, 23, 3, in mpc512x_clk_setup_clock_tree()
743 clks[MPC512x_CLK_DDR_UG] = mpc512x_clk_factor("ddr-ug", "sys", 1, 2); in mpc512x_clk_setup_clock_tree()
747 * ratios are supported because clock domain synchronization in mpc512x_clk_setup_clock_tree()
756 clks[MPC512x_CLK_SDHC_x4] = mpc512x_clk_factor("sdhc-x4", "csb", 2, 1); in mpc512x_clk_setup_clock_tree()
757 clks[MPC512x_CLK_SDHC_UG] = mpc512x_clk_divider("sdhc-ug", "sdhc-x4", 0, in mpc512x_clk_setup_clock_tree()
758 &clkregs->scfr2, 1, 7, in mpc512x_clk_setup_clock_tree()
762 "sdhc2-ug", "sdhc-x4", 0, &clkregs->scfr2, in mpc512x_clk_setup_clock_tree()
766 clks[MPC512x_CLK_DIU_x4] = mpc512x_clk_factor("diu-x4", "csb", 4, 1); in mpc512x_clk_setup_clock_tree()
767 clks[MPC512x_CLK_DIU_UG] = mpc512x_clk_divider("diu-ug", "diu-x4", 0, in mpc512x_clk_setup_clock_tree()
768 &clkregs->scfr1, 0, 8, in mpc512x_clk_setup_clock_tree()
784 "mbx-bus-ug", "csb", 1, 2); in mpc512x_clk_setup_clock_tree()
786 "mbx-ug", "mbx-bus-ug", &clkregs->scfr1, in mpc512x_clk_setup_clock_tree()
789 "mbx-3d-ug", "mbx-ug", 1, 1); in mpc512x_clk_setup_clock_tree()
793 "pci-ug", "csb", &clkregs->scfr1, in mpc512x_clk_setup_clock_tree()
798 * XXX TODO implement 5125 NFC clock setup logic, in mpc512x_clk_setup_clock_tree()
799 * with high/low period counters in clkregs->scfr3, in mpc512x_clk_setup_clock_tree()
802 clks[MPC512x_CLK_NFC_UG] = ERR_PTR(-ENOTSUPP); in mpc512x_clk_setup_clock_tree()
805 "nfc-ug", "ips", &clkregs->scfr1, in mpc512x_clk_setup_clock_tree()
808 clks[MPC512x_CLK_LPC_UG] = mpc512x_clk_divtable("lpc-ug", "ips", in mpc512x_clk_setup_clock_tree()
809 &clkregs->scfr1, 11, 3, in mpc512x_clk_setup_clock_tree()
812 clks[MPC512x_CLK_LPC] = mpc512x_clk_gated("lpc", "lpc-ug", in mpc512x_clk_setup_clock_tree()
813 &clkregs->sccr1, 30); in mpc512x_clk_setup_clock_tree()
814 clks[MPC512x_CLK_NFC] = mpc512x_clk_gated("nfc", "nfc-ug", in mpc512x_clk_setup_clock_tree()
815 &clkregs->sccr1, 29); in mpc512x_clk_setup_clock_tree()
818 "pata", "ips", &clkregs->sccr1, 28); in mpc512x_clk_setup_clock_tree()
825 name, "ips", &clkregs->sccr1, 27 - mclk_idx); in mpc512x_clk_setup_clock_tree()
828 clks[MPC512x_CLK_PSC_FIFO] = mpc512x_clk_gated("psc-fifo", "ips", in mpc512x_clk_setup_clock_tree()
829 &clkregs->sccr1, 15); in mpc512x_clk_setup_clock_tree()
832 "sata", "ips", &clkregs->sccr1, 14); in mpc512x_clk_setup_clock_tree()
835 &clkregs->sccr1, 13); in mpc512x_clk_setup_clock_tree()
838 "pci", "pci-ug", &clkregs->sccr1, 11); in mpc512x_clk_setup_clock_tree()
840 clks[MPC512x_CLK_DDR] = mpc512x_clk_gated("ddr", "ddr-ug", in mpc512x_clk_setup_clock_tree()
841 &clkregs->sccr1, 10); in mpc512x_clk_setup_clock_tree()
844 "fec2", "ips", &clkregs->sccr1, 9); in mpc512x_clk_setup_clock_tree()
847 clks[MPC512x_CLK_DIU] = mpc512x_clk_gated("diu", "diu-ug", in mpc512x_clk_setup_clock_tree()
848 &clkregs->sccr2, 31); in mpc512x_clk_setup_clock_tree()
851 "axe", "csb", &clkregs->sccr2, 30); in mpc512x_clk_setup_clock_tree()
854 &clkregs->sccr2, 29); in mpc512x_clk_setup_clock_tree()
856 &clkregs->sccr2, 28); in mpc512x_clk_setup_clock_tree()
858 &clkregs->sccr2, 27); in mpc512x_clk_setup_clock_tree()
860 &clkregs->sccr2, 26); in mpc512x_clk_setup_clock_tree()
863 &clkregs->sccr2, 25); in mpc512x_clk_setup_clock_tree()
866 clks[MPC512x_CLK_SDHC] = mpc512x_clk_gated("sdhc", "sdhc-ug", in mpc512x_clk_setup_clock_tree()
867 &clkregs->sccr2, 24); in mpc512x_clk_setup_clock_tree()
871 "spdif", "ips", &clkregs->sccr2, 23); in mpc512x_clk_setup_clock_tree()
876 "mbx-bus", "mbx-bus-ug", &clkregs->sccr2, 22); in mpc512x_clk_setup_clock_tree()
878 "mbx", "mbx-ug", &clkregs->sccr2, 21); in mpc512x_clk_setup_clock_tree()
880 "mbx-3d", "mbx-3d-ug", &clkregs->sccr2, 20); in mpc512x_clk_setup_clock_tree()
883 &clkregs->sccr2, 19); in mpc512x_clk_setup_clock_tree()
886 "viu", "csb", &clkregs->sccr2, 18); in mpc512x_clk_setup_clock_tree()
890 "sdhc-2", "sdhc2-ug", &clkregs->sccr2, 17); in mpc512x_clk_setup_clock_tree()
901 * device tree may specify values which otherwise were unknown) in mpc512x_clk_setup_clock_tree()
924 * pre-enable those "internal" clock items which never get in mpc512x_clk_setup_clock_tree()
925 * claimed by any peripheral driver, to not have the clock in mpc512x_clk_setup_clock_tree()
937 * registers the set of public clocks (those listed in the dt-bindings/
955 * pre-enable those clock items which are not yet appropriately in mpc5121_clk_provide_migration_support()
958 * the PCI clock cannot get acquired by its peripheral driver, in mpc5121_clk_provide_migration_support()
961 * at a point in time where the clock provider has not been in mpc5121_clk_provide_migration_support()
964 * so we "pre-enable" the clock here, to not have the clock in mpc5121_clk_provide_migration_support()
967 * this PCI clock pre-enable workaround only applies when there in mpc5121_clk_provide_migration_support()
968 * are device tree nodes for PCI and thus the peripheral driver in mpc5121_clk_provide_migration_support()
969 * has attached to bridges, otherwise the PCI clock remains in mpc5121_clk_provide_migration_support()
973 np = of_find_compatible_node(NULL, "pci", "fsl,mpc5121-pci"); in mpc5121_clk_provide_migration_support()
989 snprintf(devname, sizeof(devname), "%pa.%s", &res.start, np->name); \
999 clk_register_clkdev(clk, clkname, np->name); \
1001 pr_debug("clock alias name '%s' for dev '%s' pointer %p\n", \
1009 * register source code provided fallback results for clock lookups,
1010 * these get consulted when OF based clock lookup fails (that is in the
1011 * case of not yet adjusted device tree data, where clock related specs
1044 FOR_NODES("fsl,mpc5121-psc-fifo") { in mpc5121_clk_provide_backwards_compat()
1049 FOR_NODES("fsl,mpc5121-nfc") { in mpc5121_clk_provide_backwards_compat()
1054 FOR_NODES("fsl,mpc5121-mscan") { in mpc5121_clk_provide_backwards_compat()
1074 FOR_NODES("fsl,mpc5121-i2c") { in mpc5121_clk_provide_backwards_compat()
1081 * lookup (NULL name spec, which yields the first clock spec) for in mpc5121_clk_provide_backwards_compat()
1082 * which we cannot register an alias -- a _global_ 'ipg' alias that in mpc5121_clk_provide_backwards_compat()
1083 * is not bound to any device name and returns the I2C clock item in mpc5121_clk_provide_backwards_compat()
1087 * silent and non-fatal, and pre-enable the clock item here such in mpc5121_clk_provide_backwards_compat()
1090 * see commit b3bfce2b "i2c: mpc: cleanup clock API use" for in mpc5121_clk_provide_backwards_compat()
1091 * details, adjusting s/NULL/"ipg"/ in i2c-mpc.c would make this in mpc5121_clk_provide_backwards_compat()
1097 FOR_NODES("fsl,mpc5121-diu") { in mpc5121_clk_provide_backwards_compat()
1102 FOR_NODES("fsl,mpc5121-viu") { in mpc5121_clk_provide_backwards_compat()
1108 * note that 2771399a "fs_enet: cleanup clock API use" did use the in mpc5121_clk_provide_backwards_compat()
1109 * "per" string for the clock lookup in contrast to the "ipg" name in mpc5121_clk_provide_backwards_compat()
1110 * which most other nodes are using -- this is not a fatal thing in mpc5121_clk_provide_backwards_compat()
1112 * registration, it's a non-issue with up-to-date device tree data in mpc5121_clk_provide_backwards_compat()
1114 FOR_NODES("fsl,mpc5121-fec") { in mpc5121_clk_provide_backwards_compat()
1118 FOR_NODES("fsl,mpc5121-fec-mdio") { in mpc5121_clk_provide_backwards_compat()
1124 * the clock items don't "form an array" since FEC2 was in mpc5121_clk_provide_backwards_compat()
1126 * clock item indices, so the numbers aren't adjacent in mpc5121_clk_provide_backwards_compat()
1128 FOR_NODES("fsl,mpc5125-fec") { in mpc5121_clk_provide_backwards_compat()
1137 FOR_NODES("fsl,mpc5121-usb2-dr") { in mpc5121_clk_provide_backwards_compat()
1143 FOR_NODES("fsl,mpc5121-pata") { in mpc5121_clk_provide_backwards_compat()
1151 * absence of up-to-date device tree data -- backwards in mpc5121_clk_provide_backwards_compat()
1156 pr_notice("device tree lacks clock specs, adding fallbacks (0x%x,%s%s%s%s%s%s%s%s%s%s)\n", in mpc5121_clk_provide_backwards_compat()
1169 pr_debug("device tree has clock specs, no fallbacks added\n"); in mpc5121_clk_provide_backwards_compat()
1174 * The "fixed-clock" nodes (which includes the oscillator node if the board's
1183 /* map the clock control registers */ in mpc5121_clk_init()
1184 clk_np = of_find_compatible_node(NULL, NULL, "fsl,mpc5121-clock"); in mpc5121_clk_init()
1186 return -ENODEV; in mpc5121_clk_init()
1193 /* invalidate all not yet registered clock slots */ in mpc5121_clk_init()
1197 * add a dummy clock for those situations where a clock spec is in mpc5121_clk_init()
1198 * required yet no real clock is involved in mpc5121_clk_init()
1203 * have all the real nodes in the clock tree populated from REF in mpc5121_clk_init()
1205 * a REF root that was created from the IPS bus clock input in mpc5121_clk_init()
1207 busfreq = get_freq_from_dt("bus-frequency"); in mpc5121_clk_init()
1210 /* register as an OF clock provider */ in mpc5121_clk_init()
1217 * towards fully operational common clock support, and allow in mpc5121_clk_init()
1218 * operation in the absence of clock related device tree specs in mpc5121_clk_init()