// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2019 DENX Software Engineering * Lukasz Majewski, DENX Software Engineering, lukma@denx.de */ #define LOG_CATEGORY UCLASS_CLK #include #include #include #include #include #include #include #include int clk_register(struct clk *clk, const char *drv_name, const char *name, const char *parent_name) { struct udevice *parent; struct driver *drv; int ret; ret = uclass_get_device_by_name(UCLASS_CLK, parent_name, &parent); if (ret) { log_err("%s: failed to get %s device (parent of %s)\n", __func__, parent_name, name); } else { log_debug("%s: name: %s parent: %s [0x%p]\n", __func__, name, parent->name, parent); } drv = lists_driver_lookup_name(drv_name); if (!drv) { log_err("%s: %s is not a valid driver name\n", __func__, drv_name); return -ENOENT; } ret = device_bind(parent, drv, name, NULL, ofnode_null(), &clk->dev); if (ret) { log_err("%s: CLK: %s driver bind error [%d]!\n", __func__, name, ret); return ret; } clk->enable_count = 0; /* Store back pointer to clk from udevice */ /* FIXME: This is not allowed...should be allocated by driver model */ dev_set_uclass_priv(clk->dev, clk); return 0; } ulong clk_generic_get_rate(struct clk *clk) { return clk_get_parent_rate(clk); } const char *clk_hw_get_name(const struct clk *hw) { assert(hw); assert(hw->dev); return hw->dev->name; } bool clk_dev_binded(struct clk *clk) { if (clk->dev && (dev_get_flags(clk->dev) & DM_FLAG_BOUND)) return true; return false; } /* Helper functions for clock ops */ ulong ccf_clk_get_rate(struct clk *clk) { struct clk *c; int err = clk_get_by_id(clk->id, &c); if (err) return err; return clk_get_rate(c); } ulong ccf_clk_set_rate(struct clk *clk, unsigned long rate) { struct clk *c; int err = clk_get_by_id(clk->id, &c); if (err) return err; return clk_set_rate(c, rate); } int ccf_clk_set_parent(struct clk *clk, struct clk *parent) { struct clk *c, *p; int err = clk_get_by_id(clk->id, &c); if (err) return err; err = clk_get_by_id(parent->id, &p); if (err) return err; return clk_set_parent(c, p); } static int ccf_clk_endisable(struct clk *clk, bool enable) { struct clk *c; int err = clk_get_by_id(clk->id, &c); if (err) return err; return enable ? clk_enable(c) : clk_disable(c); } int ccf_clk_enable(struct clk *clk) { return ccf_clk_endisable(clk, true); } int ccf_clk_disable(struct clk *clk) { return ccf_clk_endisable(clk, false); } const struct clk_ops ccf_clk_ops = { .set_rate = ccf_clk_set_rate, .get_rate = ccf_clk_get_rate, .set_parent = ccf_clk_set_parent, .enable = ccf_clk_enable, .disable = ccf_clk_disable, };