162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2016 Socionext Inc. 462306a36Sopenharmony_ci * Author: Masahiro Yamada <yamada.masahiro@socionext.com> 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/clk-provider.h> 862306a36Sopenharmony_ci#include <linux/init.h> 962306a36Sopenharmony_ci#include <linux/mfd/syscon.h> 1062306a36Sopenharmony_ci#include <linux/of.h> 1162306a36Sopenharmony_ci#include <linux/platform_device.h> 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#include "clk-uniphier.h" 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_cistatic struct clk_hw *uniphier_clk_register(struct device *dev, 1662306a36Sopenharmony_ci struct regmap *regmap, 1762306a36Sopenharmony_ci const struct uniphier_clk_data *data) 1862306a36Sopenharmony_ci{ 1962306a36Sopenharmony_ci switch (data->type) { 2062306a36Sopenharmony_ci case UNIPHIER_CLK_TYPE_CPUGEAR: 2162306a36Sopenharmony_ci return uniphier_clk_register_cpugear(dev, regmap, data->name, 2262306a36Sopenharmony_ci &data->data.cpugear); 2362306a36Sopenharmony_ci case UNIPHIER_CLK_TYPE_FIXED_FACTOR: 2462306a36Sopenharmony_ci return uniphier_clk_register_fixed_factor(dev, data->name, 2562306a36Sopenharmony_ci &data->data.factor); 2662306a36Sopenharmony_ci case UNIPHIER_CLK_TYPE_FIXED_RATE: 2762306a36Sopenharmony_ci return uniphier_clk_register_fixed_rate(dev, data->name, 2862306a36Sopenharmony_ci &data->data.rate); 2962306a36Sopenharmony_ci case UNIPHIER_CLK_TYPE_GATE: 3062306a36Sopenharmony_ci return uniphier_clk_register_gate(dev, regmap, data->name, 3162306a36Sopenharmony_ci &data->data.gate); 3262306a36Sopenharmony_ci case UNIPHIER_CLK_TYPE_MUX: 3362306a36Sopenharmony_ci return uniphier_clk_register_mux(dev, regmap, data->name, 3462306a36Sopenharmony_ci &data->data.mux); 3562306a36Sopenharmony_ci default: 3662306a36Sopenharmony_ci dev_err(dev, "unsupported clock type\n"); 3762306a36Sopenharmony_ci return ERR_PTR(-EINVAL); 3862306a36Sopenharmony_ci } 3962306a36Sopenharmony_ci} 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_cistatic int uniphier_clk_probe(struct platform_device *pdev) 4262306a36Sopenharmony_ci{ 4362306a36Sopenharmony_ci struct device *dev = &pdev->dev; 4462306a36Sopenharmony_ci struct clk_hw_onecell_data *hw_data; 4562306a36Sopenharmony_ci const struct uniphier_clk_data *p, *data; 4662306a36Sopenharmony_ci struct regmap *regmap; 4762306a36Sopenharmony_ci struct device_node *parent; 4862306a36Sopenharmony_ci int clk_num = 0; 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci data = of_device_get_match_data(dev); 5162306a36Sopenharmony_ci if (WARN_ON(!data)) 5262306a36Sopenharmony_ci return -EINVAL; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci parent = of_get_parent(dev->of_node); /* parent should be syscon node */ 5562306a36Sopenharmony_ci regmap = syscon_node_to_regmap(parent); 5662306a36Sopenharmony_ci of_node_put(parent); 5762306a36Sopenharmony_ci if (IS_ERR(regmap)) { 5862306a36Sopenharmony_ci dev_err(dev, "failed to get regmap (error %ld)\n", 5962306a36Sopenharmony_ci PTR_ERR(regmap)); 6062306a36Sopenharmony_ci return PTR_ERR(regmap); 6162306a36Sopenharmony_ci } 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci for (p = data; p->name; p++) 6462306a36Sopenharmony_ci clk_num = max(clk_num, p->idx + 1); 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci hw_data = devm_kzalloc(dev, struct_size(hw_data, hws, clk_num), 6762306a36Sopenharmony_ci GFP_KERNEL); 6862306a36Sopenharmony_ci if (!hw_data) 6962306a36Sopenharmony_ci return -ENOMEM; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci hw_data->num = clk_num; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci /* avoid returning NULL for unused idx */ 7462306a36Sopenharmony_ci while (--clk_num >= 0) 7562306a36Sopenharmony_ci hw_data->hws[clk_num] = ERR_PTR(-EINVAL); 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci for (p = data; p->name; p++) { 7862306a36Sopenharmony_ci struct clk_hw *hw; 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ci dev_dbg(dev, "register %s (index=%d)\n", p->name, p->idx); 8162306a36Sopenharmony_ci hw = uniphier_clk_register(dev, regmap, p); 8262306a36Sopenharmony_ci if (WARN(IS_ERR(hw), "failed to register %s", p->name)) 8362306a36Sopenharmony_ci continue; 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci if (p->idx >= 0) 8662306a36Sopenharmony_ci hw_data->hws[p->idx] = hw; 8762306a36Sopenharmony_ci } 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci return devm_of_clk_add_hw_provider(dev, of_clk_hw_onecell_get, 9062306a36Sopenharmony_ci hw_data); 9162306a36Sopenharmony_ci} 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_cistatic const struct of_device_id uniphier_clk_match[] = { 9462306a36Sopenharmony_ci /* System clock */ 9562306a36Sopenharmony_ci { 9662306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld4-clock", 9762306a36Sopenharmony_ci .data = uniphier_ld4_sys_clk_data, 9862306a36Sopenharmony_ci }, 9962306a36Sopenharmony_ci { 10062306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro4-clock", 10162306a36Sopenharmony_ci .data = uniphier_pro4_sys_clk_data, 10262306a36Sopenharmony_ci }, 10362306a36Sopenharmony_ci { 10462306a36Sopenharmony_ci .compatible = "socionext,uniphier-sld8-clock", 10562306a36Sopenharmony_ci .data = uniphier_sld8_sys_clk_data, 10662306a36Sopenharmony_ci }, 10762306a36Sopenharmony_ci { 10862306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro5-clock", 10962306a36Sopenharmony_ci .data = uniphier_pro5_sys_clk_data, 11062306a36Sopenharmony_ci }, 11162306a36Sopenharmony_ci { 11262306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs2-clock", 11362306a36Sopenharmony_ci .data = uniphier_pxs2_sys_clk_data, 11462306a36Sopenharmony_ci }, 11562306a36Sopenharmony_ci { 11662306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld11-clock", 11762306a36Sopenharmony_ci .data = uniphier_ld11_sys_clk_data, 11862306a36Sopenharmony_ci }, 11962306a36Sopenharmony_ci { 12062306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld20-clock", 12162306a36Sopenharmony_ci .data = uniphier_ld20_sys_clk_data, 12262306a36Sopenharmony_ci }, 12362306a36Sopenharmony_ci { 12462306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs3-clock", 12562306a36Sopenharmony_ci .data = uniphier_pxs3_sys_clk_data, 12662306a36Sopenharmony_ci }, 12762306a36Sopenharmony_ci { 12862306a36Sopenharmony_ci .compatible = "socionext,uniphier-nx1-clock", 12962306a36Sopenharmony_ci .data = uniphier_nx1_sys_clk_data, 13062306a36Sopenharmony_ci }, 13162306a36Sopenharmony_ci /* Media I/O clock, SD clock */ 13262306a36Sopenharmony_ci { 13362306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld4-mio-clock", 13462306a36Sopenharmony_ci .data = uniphier_ld4_mio_clk_data, 13562306a36Sopenharmony_ci }, 13662306a36Sopenharmony_ci { 13762306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro4-mio-clock", 13862306a36Sopenharmony_ci .data = uniphier_ld4_mio_clk_data, 13962306a36Sopenharmony_ci }, 14062306a36Sopenharmony_ci { 14162306a36Sopenharmony_ci .compatible = "socionext,uniphier-sld8-mio-clock", 14262306a36Sopenharmony_ci .data = uniphier_ld4_mio_clk_data, 14362306a36Sopenharmony_ci }, 14462306a36Sopenharmony_ci { 14562306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro5-sd-clock", 14662306a36Sopenharmony_ci .data = uniphier_pro5_sd_clk_data, 14762306a36Sopenharmony_ci }, 14862306a36Sopenharmony_ci { 14962306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs2-sd-clock", 15062306a36Sopenharmony_ci .data = uniphier_pro5_sd_clk_data, 15162306a36Sopenharmony_ci }, 15262306a36Sopenharmony_ci { 15362306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld11-mio-clock", 15462306a36Sopenharmony_ci .data = uniphier_ld4_mio_clk_data, 15562306a36Sopenharmony_ci }, 15662306a36Sopenharmony_ci { 15762306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld20-sd-clock", 15862306a36Sopenharmony_ci .data = uniphier_pro5_sd_clk_data, 15962306a36Sopenharmony_ci }, 16062306a36Sopenharmony_ci { 16162306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs3-sd-clock", 16262306a36Sopenharmony_ci .data = uniphier_pro5_sd_clk_data, 16362306a36Sopenharmony_ci }, 16462306a36Sopenharmony_ci { 16562306a36Sopenharmony_ci .compatible = "socionext,uniphier-nx1-sd-clock", 16662306a36Sopenharmony_ci .data = uniphier_pro5_sd_clk_data, 16762306a36Sopenharmony_ci }, 16862306a36Sopenharmony_ci /* Peripheral clock */ 16962306a36Sopenharmony_ci { 17062306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld4-peri-clock", 17162306a36Sopenharmony_ci .data = uniphier_ld4_peri_clk_data, 17262306a36Sopenharmony_ci }, 17362306a36Sopenharmony_ci { 17462306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro4-peri-clock", 17562306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 17662306a36Sopenharmony_ci }, 17762306a36Sopenharmony_ci { 17862306a36Sopenharmony_ci .compatible = "socionext,uniphier-sld8-peri-clock", 17962306a36Sopenharmony_ci .data = uniphier_ld4_peri_clk_data, 18062306a36Sopenharmony_ci }, 18162306a36Sopenharmony_ci { 18262306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro5-peri-clock", 18362306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 18462306a36Sopenharmony_ci }, 18562306a36Sopenharmony_ci { 18662306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs2-peri-clock", 18762306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 18862306a36Sopenharmony_ci }, 18962306a36Sopenharmony_ci { 19062306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld11-peri-clock", 19162306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 19262306a36Sopenharmony_ci }, 19362306a36Sopenharmony_ci { 19462306a36Sopenharmony_ci .compatible = "socionext,uniphier-ld20-peri-clock", 19562306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 19662306a36Sopenharmony_ci }, 19762306a36Sopenharmony_ci { 19862306a36Sopenharmony_ci .compatible = "socionext,uniphier-pxs3-peri-clock", 19962306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 20062306a36Sopenharmony_ci }, 20162306a36Sopenharmony_ci { 20262306a36Sopenharmony_ci .compatible = "socionext,uniphier-nx1-peri-clock", 20362306a36Sopenharmony_ci .data = uniphier_pro4_peri_clk_data, 20462306a36Sopenharmony_ci }, 20562306a36Sopenharmony_ci /* SoC-glue clock */ 20662306a36Sopenharmony_ci { 20762306a36Sopenharmony_ci .compatible = "socionext,uniphier-pro4-sg-clock", 20862306a36Sopenharmony_ci .data = uniphier_pro4_sg_clk_data, 20962306a36Sopenharmony_ci }, 21062306a36Sopenharmony_ci { /* sentinel */ } 21162306a36Sopenharmony_ci}; 21262306a36Sopenharmony_ci 21362306a36Sopenharmony_cistatic struct platform_driver uniphier_clk_driver = { 21462306a36Sopenharmony_ci .probe = uniphier_clk_probe, 21562306a36Sopenharmony_ci .driver = { 21662306a36Sopenharmony_ci .name = "uniphier-clk", 21762306a36Sopenharmony_ci .of_match_table = uniphier_clk_match, 21862306a36Sopenharmony_ci }, 21962306a36Sopenharmony_ci}; 22062306a36Sopenharmony_cibuiltin_platform_driver(uniphier_clk_driver); 221