Index: sys/arm64/rockchip/clk/rk3399_cru.c =================================================================== --- sys/arm64/rockchip/clk/rk3399_cru.c +++ sys/arm64/rockchip/clk/rk3399_cru.c @@ -775,7 +775,7 @@ static struct rk_clk rk3399_clks[] = { /* External clocks */ LINK("xin24m"), - FRATE(0, "xin32k", 32768), + LINK("xin32k"), FFACT(0, "xin12m", "xin24m", 1, 2), FRATE(0, "clkin_i2s", 0), FRATE(0, "pclkin_cif", 0), Index: sys/arm64/rockchip/rk805.c =================================================================== --- sys/arm64/rockchip/rk805.c +++ sys/arm64/rockchip/rk805.c @@ -42,10 +42,12 @@ #include #include +#include #include #include +#include "clkdev_if.h" #include "regdev_if.h" MALLOC_DEFINE(M_RK805_REG, "RK805 regulator", "RK805 power regulator"); @@ -582,6 +584,124 @@ return (reg_sc); } +/* -------------------------------------------------------------------------- */ + +/* Clock class and method */ +struct rk808_clk_sc { + device_t base_dev; +}; + +#define CLK32OUT_REG 0x20 +#define CLK32OUT_CLKOUT2_EN 1 + +static int +rk808_clk_set_gate_1(struct clknode *clk, bool enable) +{ + struct rk808_clk_sc *sc; + uint8_t val; + + sc = clknode_get_softc(clk); + + (void) rk805_read(sc->base_dev, CLK32OUT_REG, &val, sizeof(val)); + if (enable) + val |= CLK32OUT_CLKOUT2_EN; + else + val &= ~CLK32OUT_CLKOUT2_EN; + (void) rk805_write(sc->base_dev, CLK32OUT_REG, val); + + return (0); +} + +static int +rk808_clk_recalc(struct clknode *clk, uint64_t *freq) +{ + + *freq = 32768; + return (0); +} + +static clknode_method_t rk808_clk_clknode_methods_0[] = { + /* Device interface */ + CLKNODEMETHOD(clknode_recalc_freq, rk808_clk_recalc), + + CLKNODEMETHOD_END +}; + +DEFINE_CLASS_1(rk808_clk_clknode_0, rk808_clk_clknode_class_0, + rk808_clk_clknode_methods_0, sizeof(struct rk808_clk_sc), + clknode_class); + +static clknode_method_t rk808_clk_clknode_methods_1[] = { + /* Device interface */ + CLKNODEMETHOD(clknode_recalc_freq, rk808_clk_recalc), + CLKNODEMETHOD(clknode_set_gate, rk808_clk_set_gate_1), + + CLKNODEMETHOD_END +}; + +DEFINE_CLASS_1(rk808_clk_clknode_1, rk808_clk_clknode_class_1, + rk808_clk_clknode_methods_1, sizeof(struct rk808_clk_sc), + clknode_class); + +static int +rk808_export_clocks(device_t dev) +{ + struct clkdom *clkdom; + struct clknode_init_def clkidef; + struct clknode *clk; + struct rk808_clk_sc *clksc; + const char **clknames; + phandle_t node; + int nclks, rv; + + node = ofw_bus_get_node(dev); + + /* clock-output-names are optional. Could use them for clkidef.name. */ + nclks = ofw_bus_string_list_to_array(node, "clock-output-names", + &clknames); + + clkdom = clkdom_create(dev); + + /* Fixed clock removed from rk3399_cru.c. */ + memset(&clkidef, 0, sizeof(clkidef)); + clkidef.id = 0; + clkidef.name = (nclks = 2) ? clknames[0] : "xin32k"; + clk = clknode_create(clkdom, &rk808_clk_clknode_class_0, &clkidef); + if (clk == NULL) { + device_printf(dev, "Cannot create clknode 0.\n"); + return (ENXIO); + } + clksc = clknode_get_softc(clk); + clksc->base_dev = dev; + clknode_register(clkdom, clk); + + memset(&clkidef, 0, sizeof(clkidef)); + clkidef.id = 1; + clkidef.name = (nclks = 2) ? clknames[1] : "rk808-clk1"; + clk = clknode_create(clkdom, &rk808_clk_clknode_class_1, &clkidef); + if (clk == NULL) { + device_printf(dev, "Cannot create clknode 1.\n"); + return (ENXIO); + } + clksc = clknode_get_softc(clk); + clksc->base_dev = dev; + clknode_register(clkdom, clk); + + rv = clkdom_finit(clkdom); + if (rv != 0) { + device_printf(dev, "Cannot finalize clkdom initialization: " + "%d\n", rv); + return (ENXIO); + } + + if (bootverbose) + clkdom_dump(clkdom); + + return (0); +} + +/* -------------------------------------------------------------------------- */ + static int rk805_probe(device_t dev) { @@ -639,13 +759,18 @@ sc = device_get_softc(dev); + sc->type = ofw_bus_search_compatible(dev, compat_data)->ocd_data; + if (sc->type == RK808) { + i = rk808_export_clocks(dev); + if (i != 0) + device_printf(dev, "Could not export clocks: %d\n", i); + } + sc->intr_hook.ich_func = rk805_start; sc->intr_hook.ich_arg = dev; - if (config_intrhook_establish(&sc->intr_hook) != 0) return (ENOMEM); - sc->type = ofw_bus_search_compatible(dev, compat_data)->ocd_data; switch (sc->type) { case RK805: regdefs = rk805_regdefs;