162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2013 - 2014 Texas Instruments Incorporated - https://www.ti.com 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Authors: 662306a36Sopenharmony_ci * Jyri Sarha <jsarha@ti.com> 762306a36Sopenharmony_ci * Sergej Sawazki <ce3a@gmx.de> 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci * Gpio controlled clock implementation 1062306a36Sopenharmony_ci */ 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#include <linux/clk-provider.h> 1362306a36Sopenharmony_ci#include <linux/export.h> 1462306a36Sopenharmony_ci#include <linux/slab.h> 1562306a36Sopenharmony_ci#include <linux/gpio/consumer.h> 1662306a36Sopenharmony_ci#include <linux/err.h> 1762306a36Sopenharmony_ci#include <linux/device.h> 1862306a36Sopenharmony_ci#include <linux/of.h> 1962306a36Sopenharmony_ci#include <linux/platform_device.h> 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci/** 2262306a36Sopenharmony_ci * DOC: basic gpio gated clock which can be enabled and disabled 2362306a36Sopenharmony_ci * with gpio output 2462306a36Sopenharmony_ci * Traits of this clock: 2562306a36Sopenharmony_ci * prepare - clk_(un)prepare only ensures parent is (un)prepared 2662306a36Sopenharmony_ci * enable - clk_enable and clk_disable are functional & control gpio 2762306a36Sopenharmony_ci * rate - inherits rate from parent. No clk_set_rate support 2862306a36Sopenharmony_ci * parent - fixed parent. No clk_set_parent support 2962306a36Sopenharmony_ci */ 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_ci/** 3262306a36Sopenharmony_ci * struct clk_gpio - gpio gated clock 3362306a36Sopenharmony_ci * 3462306a36Sopenharmony_ci * @hw: handle between common and hardware-specific interfaces 3562306a36Sopenharmony_ci * @gpiod: gpio descriptor 3662306a36Sopenharmony_ci * 3762306a36Sopenharmony_ci * Clock with a gpio control for enabling and disabling the parent clock 3862306a36Sopenharmony_ci * or switching between two parents by asserting or deasserting the gpio. 3962306a36Sopenharmony_ci * 4062306a36Sopenharmony_ci * Implements .enable, .disable and .is_enabled or 4162306a36Sopenharmony_ci * .get_parent, .set_parent and .determine_rate depending on which clk_ops 4262306a36Sopenharmony_ci * is used. 4362306a36Sopenharmony_ci */ 4462306a36Sopenharmony_cistruct clk_gpio { 4562306a36Sopenharmony_ci struct clk_hw hw; 4662306a36Sopenharmony_ci struct gpio_desc *gpiod; 4762306a36Sopenharmony_ci}; 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci#define to_clk_gpio(_hw) container_of(_hw, struct clk_gpio, hw) 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_cistatic int clk_gpio_gate_enable(struct clk_hw *hw) 5262306a36Sopenharmony_ci{ 5362306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci gpiod_set_value(clk->gpiod, 1); 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci return 0; 5862306a36Sopenharmony_ci} 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_cistatic void clk_gpio_gate_disable(struct clk_hw *hw) 6162306a36Sopenharmony_ci{ 6262306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci gpiod_set_value(clk->gpiod, 0); 6562306a36Sopenharmony_ci} 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_cistatic int clk_gpio_gate_is_enabled(struct clk_hw *hw) 6862306a36Sopenharmony_ci{ 6962306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci return gpiod_get_value(clk->gpiod); 7262306a36Sopenharmony_ci} 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_cistatic const struct clk_ops clk_gpio_gate_ops = { 7562306a36Sopenharmony_ci .enable = clk_gpio_gate_enable, 7662306a36Sopenharmony_ci .disable = clk_gpio_gate_disable, 7762306a36Sopenharmony_ci .is_enabled = clk_gpio_gate_is_enabled, 7862306a36Sopenharmony_ci}; 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_cistatic int clk_sleeping_gpio_gate_prepare(struct clk_hw *hw) 8162306a36Sopenharmony_ci{ 8262306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_ci gpiod_set_value_cansleep(clk->gpiod, 1); 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_ci return 0; 8762306a36Sopenharmony_ci} 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_cistatic void clk_sleeping_gpio_gate_unprepare(struct clk_hw *hw) 9062306a36Sopenharmony_ci{ 9162306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci gpiod_set_value_cansleep(clk->gpiod, 0); 9462306a36Sopenharmony_ci} 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_cistatic int clk_sleeping_gpio_gate_is_prepared(struct clk_hw *hw) 9762306a36Sopenharmony_ci{ 9862306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci return gpiod_get_value_cansleep(clk->gpiod); 10162306a36Sopenharmony_ci} 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_cistatic const struct clk_ops clk_sleeping_gpio_gate_ops = { 10462306a36Sopenharmony_ci .prepare = clk_sleeping_gpio_gate_prepare, 10562306a36Sopenharmony_ci .unprepare = clk_sleeping_gpio_gate_unprepare, 10662306a36Sopenharmony_ci .is_prepared = clk_sleeping_gpio_gate_is_prepared, 10762306a36Sopenharmony_ci}; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ci/** 11062306a36Sopenharmony_ci * DOC: basic clock multiplexer which can be controlled with a gpio output 11162306a36Sopenharmony_ci * Traits of this clock: 11262306a36Sopenharmony_ci * prepare - clk_prepare only ensures that parents are prepared 11362306a36Sopenharmony_ci * rate - rate is only affected by parent switching. No clk_set_rate support 11462306a36Sopenharmony_ci * parent - parent is adjustable through clk_set_parent 11562306a36Sopenharmony_ci */ 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_cistatic u8 clk_gpio_mux_get_parent(struct clk_hw *hw) 11862306a36Sopenharmony_ci{ 11962306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci return gpiod_get_value_cansleep(clk->gpiod); 12262306a36Sopenharmony_ci} 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_cistatic int clk_gpio_mux_set_parent(struct clk_hw *hw, u8 index) 12562306a36Sopenharmony_ci{ 12662306a36Sopenharmony_ci struct clk_gpio *clk = to_clk_gpio(hw); 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ci gpiod_set_value_cansleep(clk->gpiod, index); 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci return 0; 13162306a36Sopenharmony_ci} 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_cistatic const struct clk_ops clk_gpio_mux_ops = { 13462306a36Sopenharmony_ci .get_parent = clk_gpio_mux_get_parent, 13562306a36Sopenharmony_ci .set_parent = clk_gpio_mux_set_parent, 13662306a36Sopenharmony_ci .determine_rate = __clk_mux_determine_rate, 13762306a36Sopenharmony_ci}; 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_cistatic struct clk_hw *clk_register_gpio(struct device *dev, u8 num_parents, 14062306a36Sopenharmony_ci struct gpio_desc *gpiod, 14162306a36Sopenharmony_ci const struct clk_ops *clk_gpio_ops) 14262306a36Sopenharmony_ci{ 14362306a36Sopenharmony_ci struct clk_gpio *clk_gpio; 14462306a36Sopenharmony_ci struct clk_hw *hw; 14562306a36Sopenharmony_ci struct clk_init_data init = {}; 14662306a36Sopenharmony_ci int err; 14762306a36Sopenharmony_ci const struct clk_parent_data gpio_parent_data[] = { 14862306a36Sopenharmony_ci { .index = 0 }, 14962306a36Sopenharmony_ci { .index = 1 }, 15062306a36Sopenharmony_ci }; 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci clk_gpio = devm_kzalloc(dev, sizeof(*clk_gpio), GFP_KERNEL); 15362306a36Sopenharmony_ci if (!clk_gpio) 15462306a36Sopenharmony_ci return ERR_PTR(-ENOMEM); 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_ci init.name = dev->of_node->name; 15762306a36Sopenharmony_ci init.ops = clk_gpio_ops; 15862306a36Sopenharmony_ci init.parent_data = gpio_parent_data; 15962306a36Sopenharmony_ci init.num_parents = num_parents; 16062306a36Sopenharmony_ci init.flags = CLK_SET_RATE_PARENT; 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci clk_gpio->gpiod = gpiod; 16362306a36Sopenharmony_ci clk_gpio->hw.init = &init; 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_ci hw = &clk_gpio->hw; 16662306a36Sopenharmony_ci err = devm_clk_hw_register(dev, hw); 16762306a36Sopenharmony_ci if (err) 16862306a36Sopenharmony_ci return ERR_PTR(err); 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci return hw; 17162306a36Sopenharmony_ci} 17262306a36Sopenharmony_ci 17362306a36Sopenharmony_cistatic struct clk_hw *clk_hw_register_gpio_gate(struct device *dev, 17462306a36Sopenharmony_ci int num_parents, 17562306a36Sopenharmony_ci struct gpio_desc *gpiod) 17662306a36Sopenharmony_ci{ 17762306a36Sopenharmony_ci const struct clk_ops *ops; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci if (gpiod_cansleep(gpiod)) 18062306a36Sopenharmony_ci ops = &clk_sleeping_gpio_gate_ops; 18162306a36Sopenharmony_ci else 18262306a36Sopenharmony_ci ops = &clk_gpio_gate_ops; 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci return clk_register_gpio(dev, num_parents, gpiod, ops); 18562306a36Sopenharmony_ci} 18662306a36Sopenharmony_ci 18762306a36Sopenharmony_cistatic struct clk_hw *clk_hw_register_gpio_mux(struct device *dev, 18862306a36Sopenharmony_ci struct gpio_desc *gpiod) 18962306a36Sopenharmony_ci{ 19062306a36Sopenharmony_ci return clk_register_gpio(dev, 2, gpiod, &clk_gpio_mux_ops); 19162306a36Sopenharmony_ci} 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_cistatic int gpio_clk_driver_probe(struct platform_device *pdev) 19462306a36Sopenharmony_ci{ 19562306a36Sopenharmony_ci struct device *dev = &pdev->dev; 19662306a36Sopenharmony_ci struct device_node *node = dev->of_node; 19762306a36Sopenharmony_ci const char *gpio_name; 19862306a36Sopenharmony_ci unsigned int num_parents; 19962306a36Sopenharmony_ci struct gpio_desc *gpiod; 20062306a36Sopenharmony_ci struct clk_hw *hw; 20162306a36Sopenharmony_ci bool is_mux; 20262306a36Sopenharmony_ci int ret; 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_ci is_mux = of_device_is_compatible(node, "gpio-mux-clock"); 20562306a36Sopenharmony_ci 20662306a36Sopenharmony_ci num_parents = of_clk_get_parent_count(node); 20762306a36Sopenharmony_ci if (is_mux && num_parents != 2) { 20862306a36Sopenharmony_ci dev_err(dev, "mux-clock must have 2 parents\n"); 20962306a36Sopenharmony_ci return -EINVAL; 21062306a36Sopenharmony_ci } 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_ci gpio_name = is_mux ? "select" : "enable"; 21362306a36Sopenharmony_ci gpiod = devm_gpiod_get(dev, gpio_name, GPIOD_OUT_LOW); 21462306a36Sopenharmony_ci if (IS_ERR(gpiod)) { 21562306a36Sopenharmony_ci ret = PTR_ERR(gpiod); 21662306a36Sopenharmony_ci if (ret == -EPROBE_DEFER) 21762306a36Sopenharmony_ci pr_debug("%pOFn: %s: GPIOs not yet available, retry later\n", 21862306a36Sopenharmony_ci node, __func__); 21962306a36Sopenharmony_ci else 22062306a36Sopenharmony_ci pr_err("%pOFn: %s: Can't get '%s' named GPIO property\n", 22162306a36Sopenharmony_ci node, __func__, 22262306a36Sopenharmony_ci gpio_name); 22362306a36Sopenharmony_ci return ret; 22462306a36Sopenharmony_ci } 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci if (is_mux) 22762306a36Sopenharmony_ci hw = clk_hw_register_gpio_mux(dev, gpiod); 22862306a36Sopenharmony_ci else 22962306a36Sopenharmony_ci hw = clk_hw_register_gpio_gate(dev, num_parents, gpiod); 23062306a36Sopenharmony_ci if (IS_ERR(hw)) 23162306a36Sopenharmony_ci return PTR_ERR(hw); 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_ci return devm_of_clk_add_hw_provider(dev, of_clk_hw_simple_get, hw); 23462306a36Sopenharmony_ci} 23562306a36Sopenharmony_ci 23662306a36Sopenharmony_cistatic const struct of_device_id gpio_clk_match_table[] = { 23762306a36Sopenharmony_ci { .compatible = "gpio-mux-clock" }, 23862306a36Sopenharmony_ci { .compatible = "gpio-gate-clock" }, 23962306a36Sopenharmony_ci { } 24062306a36Sopenharmony_ci}; 24162306a36Sopenharmony_ci 24262306a36Sopenharmony_cistatic struct platform_driver gpio_clk_driver = { 24362306a36Sopenharmony_ci .probe = gpio_clk_driver_probe, 24462306a36Sopenharmony_ci .driver = { 24562306a36Sopenharmony_ci .name = "gpio-clk", 24662306a36Sopenharmony_ci .of_match_table = gpio_clk_match_table, 24762306a36Sopenharmony_ci }, 24862306a36Sopenharmony_ci}; 24962306a36Sopenharmony_cibuiltin_platform_driver(gpio_clk_driver); 250