10aa0c95fSHaojian Zhuang /* 20aa0c95fSHaojian Zhuang * Hisilicon clock driver 30aa0c95fSHaojian Zhuang * 40aa0c95fSHaojian Zhuang * Copyright (c) 2012-2013 Hisilicon Limited. 50aa0c95fSHaojian Zhuang * Copyright (c) 2012-2013 Linaro Limited. 60aa0c95fSHaojian Zhuang * 70aa0c95fSHaojian Zhuang * Author: Haojian Zhuang <haojian.zhuang@linaro.org> 80aa0c95fSHaojian Zhuang * Xin Li <li.xin@linaro.org> 90aa0c95fSHaojian Zhuang * 100aa0c95fSHaojian Zhuang * This program is free software; you can redistribute it and/or modify 110aa0c95fSHaojian Zhuang * it under the terms of the GNU General Public License as published by 120aa0c95fSHaojian Zhuang * the Free Software Foundation; either version 2 of the License, or 130aa0c95fSHaojian Zhuang * (at your option) any later version. 140aa0c95fSHaojian Zhuang * 150aa0c95fSHaojian Zhuang * This program is distributed in the hope that it will be useful, 160aa0c95fSHaojian Zhuang * but WITHOUT ANY WARRANTY; without even the implied warranty of 170aa0c95fSHaojian Zhuang * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 180aa0c95fSHaojian Zhuang * GNU General Public License for more details. 190aa0c95fSHaojian Zhuang * 200aa0c95fSHaojian Zhuang * You should have received a copy of the GNU General Public License along 210aa0c95fSHaojian Zhuang * with this program; if not, write to the Free Software Foundation, Inc., 220aa0c95fSHaojian Zhuang * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. 230aa0c95fSHaojian Zhuang * 240aa0c95fSHaojian Zhuang */ 250aa0c95fSHaojian Zhuang 260aa0c95fSHaojian Zhuang #include <linux/kernel.h> 270aa0c95fSHaojian Zhuang #include <linux/clkdev.h> 28593438e4SStephen Boyd #include <linux/clk-provider.h> 290aa0c95fSHaojian Zhuang #include <linux/delay.h> 300aa0c95fSHaojian Zhuang #include <linux/io.h> 310aa0c95fSHaojian Zhuang #include <linux/of.h> 320aa0c95fSHaojian Zhuang #include <linux/of_address.h> 330aa0c95fSHaojian Zhuang #include <linux/of_device.h> 340aa0c95fSHaojian Zhuang #include <linux/slab.h> 350aa0c95fSHaojian Zhuang 360aa0c95fSHaojian Zhuang #include "clk.h" 370aa0c95fSHaojian Zhuang 380aa0c95fSHaojian Zhuang static DEFINE_SPINLOCK(hisi_clk_lock); 390aa0c95fSHaojian Zhuang 4032226916SJiancheng Xue struct hisi_clock_data *hisi_clk_alloc(struct platform_device *pdev, 4132226916SJiancheng Xue int nr_clks) 4232226916SJiancheng Xue { 4332226916SJiancheng Xue struct hisi_clock_data *clk_data; 4432226916SJiancheng Xue struct resource *res; 4532226916SJiancheng Xue struct clk **clk_table; 4632226916SJiancheng Xue 4732226916SJiancheng Xue clk_data = devm_kmalloc(&pdev->dev, sizeof(*clk_data), GFP_KERNEL); 4832226916SJiancheng Xue if (!clk_data) 4932226916SJiancheng Xue return NULL; 5032226916SJiancheng Xue 5132226916SJiancheng Xue res = platform_get_resource(pdev, IORESOURCE_MEM, 0); 52*c744b63bSWei Yongjun if (!res) 53*c744b63bSWei Yongjun return NULL; 5432226916SJiancheng Xue clk_data->base = devm_ioremap(&pdev->dev, 5532226916SJiancheng Xue res->start, resource_size(res)); 5632226916SJiancheng Xue if (!clk_data->base) 5732226916SJiancheng Xue return NULL; 5832226916SJiancheng Xue 598d9bdc46SMarkus Elfring clk_table = devm_kmalloc_array(&pdev->dev, nr_clks, 608d9bdc46SMarkus Elfring sizeof(*clk_table), 6132226916SJiancheng Xue GFP_KERNEL); 6232226916SJiancheng Xue if (!clk_table) 6332226916SJiancheng Xue return NULL; 6432226916SJiancheng Xue 6532226916SJiancheng Xue clk_data->clk_data.clks = clk_table; 6632226916SJiancheng Xue clk_data->clk_data.clk_num = nr_clks; 6732226916SJiancheng Xue 6832226916SJiancheng Xue return clk_data; 6932226916SJiancheng Xue } 7032226916SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_alloc); 7132226916SJiancheng Xue 72f6ff57c8SJiancheng Xue struct hisi_clock_data *hisi_clk_init(struct device_node *np, 7375af25f5SHaojian Zhuang int nr_clks) 740aa0c95fSHaojian Zhuang { 7575af25f5SHaojian Zhuang struct hisi_clock_data *clk_data; 7675af25f5SHaojian Zhuang struct clk **clk_table; 7775af25f5SHaojian Zhuang void __iomem *base; 7875af25f5SHaojian Zhuang 7975af25f5SHaojian Zhuang base = of_iomap(np, 0); 8075af25f5SHaojian Zhuang if (!base) { 811fb6dd9dSLeo Yan pr_err("%s: failed to map clock registers\n", __func__); 8275af25f5SHaojian Zhuang goto err; 8375af25f5SHaojian Zhuang } 8475af25f5SHaojian Zhuang 8575af25f5SHaojian Zhuang clk_data = kzalloc(sizeof(*clk_data), GFP_KERNEL); 86840e5632SMarkus Elfring if (!clk_data) 8775af25f5SHaojian Zhuang goto err; 88840e5632SMarkus Elfring 8975af25f5SHaojian Zhuang clk_data->base = base; 907b9bae17SMarkus Elfring clk_table = kcalloc(nr_clks, sizeof(*clk_table), GFP_KERNEL); 91840e5632SMarkus Elfring if (!clk_table) 9275af25f5SHaojian Zhuang goto err_data; 93840e5632SMarkus Elfring 9475af25f5SHaojian Zhuang clk_data->clk_data.clks = clk_table; 9575af25f5SHaojian Zhuang clk_data->clk_data.clk_num = nr_clks; 9675af25f5SHaojian Zhuang of_clk_add_provider(np, of_clk_src_onecell_get, &clk_data->clk_data); 9775af25f5SHaojian Zhuang return clk_data; 9875af25f5SHaojian Zhuang err_data: 9975af25f5SHaojian Zhuang kfree(clk_data); 10075af25f5SHaojian Zhuang err: 10175af25f5SHaojian Zhuang return NULL; 1020aa0c95fSHaojian Zhuang } 103f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_init); 1040aa0c95fSHaojian Zhuang 1055497f668SJiancheng Xue int hisi_clk_register_fixed_rate(const struct hisi_fixed_rate_clock *clks, 10675af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 1070aa0c95fSHaojian Zhuang { 1080aa0c95fSHaojian Zhuang struct clk *clk; 1090aa0c95fSHaojian Zhuang int i; 1100aa0c95fSHaojian Zhuang 1110aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 1120aa0c95fSHaojian Zhuang clk = clk_register_fixed_rate(NULL, clks[i].name, 1130aa0c95fSHaojian Zhuang clks[i].parent_name, 1140aa0c95fSHaojian Zhuang clks[i].flags, 1150aa0c95fSHaojian Zhuang clks[i].fixed_rate); 1160aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 1170aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 1180aa0c95fSHaojian Zhuang __func__, clks[i].name); 1195497f668SJiancheng Xue goto err; 1200aa0c95fSHaojian Zhuang } 12175af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 1220aa0c95fSHaojian Zhuang } 1235497f668SJiancheng Xue 1245497f668SJiancheng Xue return 0; 1255497f668SJiancheng Xue 1265497f668SJiancheng Xue err: 1275497f668SJiancheng Xue while (i--) 1285497f668SJiancheng Xue clk_unregister_fixed_rate(data->clk_data.clks[clks[i].id]); 1295497f668SJiancheng Xue 1305497f668SJiancheng Xue return PTR_ERR(clk); 1310aa0c95fSHaojian Zhuang } 132f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_fixed_rate); 1330aa0c95fSHaojian Zhuang 1345497f668SJiancheng Xue int hisi_clk_register_fixed_factor(const struct hisi_fixed_factor_clock *clks, 13575af25f5SHaojian Zhuang int nums, 13675af25f5SHaojian Zhuang struct hisi_clock_data *data) 1370aa0c95fSHaojian Zhuang { 1380aa0c95fSHaojian Zhuang struct clk *clk; 1390aa0c95fSHaojian Zhuang int i; 1400aa0c95fSHaojian Zhuang 1410aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 1420aa0c95fSHaojian Zhuang clk = clk_register_fixed_factor(NULL, clks[i].name, 1430aa0c95fSHaojian Zhuang clks[i].parent_name, 1440aa0c95fSHaojian Zhuang clks[i].flags, clks[i].mult, 1450aa0c95fSHaojian Zhuang clks[i].div); 1460aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 1470aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 1480aa0c95fSHaojian Zhuang __func__, clks[i].name); 1495497f668SJiancheng Xue goto err; 1500aa0c95fSHaojian Zhuang } 15175af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 1520aa0c95fSHaojian Zhuang } 1535497f668SJiancheng Xue 1545497f668SJiancheng Xue return 0; 1555497f668SJiancheng Xue 1565497f668SJiancheng Xue err: 1575497f668SJiancheng Xue while (i--) 1585497f668SJiancheng Xue clk_unregister_fixed_factor(data->clk_data.clks[clks[i].id]); 1595497f668SJiancheng Xue 1605497f668SJiancheng Xue return PTR_ERR(clk); 1610aa0c95fSHaojian Zhuang } 162f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_fixed_factor); 1630aa0c95fSHaojian Zhuang 1645497f668SJiancheng Xue int hisi_clk_register_mux(const struct hisi_mux_clock *clks, 16575af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 1660aa0c95fSHaojian Zhuang { 1670aa0c95fSHaojian Zhuang struct clk *clk; 16875af25f5SHaojian Zhuang void __iomem *base = data->base; 1690aa0c95fSHaojian Zhuang int i; 1700aa0c95fSHaojian Zhuang 1710aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 172156342a1SZhangfei Gao u32 mask = BIT(clks[i].width) - 1; 173156342a1SZhangfei Gao 174156342a1SZhangfei Gao clk = clk_register_mux_table(NULL, clks[i].name, 175156342a1SZhangfei Gao clks[i].parent_names, 1760aa0c95fSHaojian Zhuang clks[i].num_parents, clks[i].flags, 1770aa0c95fSHaojian Zhuang base + clks[i].offset, clks[i].shift, 178156342a1SZhangfei Gao mask, clks[i].mux_flags, 179156342a1SZhangfei Gao clks[i].table, &hisi_clk_lock); 1800aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 1810aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 1820aa0c95fSHaojian Zhuang __func__, clks[i].name); 1835497f668SJiancheng Xue goto err; 1840aa0c95fSHaojian Zhuang } 1850aa0c95fSHaojian Zhuang 1860aa0c95fSHaojian Zhuang if (clks[i].alias) 1870aa0c95fSHaojian Zhuang clk_register_clkdev(clk, clks[i].alias, NULL); 1880aa0c95fSHaojian Zhuang 18975af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 1900aa0c95fSHaojian Zhuang } 1915497f668SJiancheng Xue 1925497f668SJiancheng Xue return 0; 1935497f668SJiancheng Xue 1945497f668SJiancheng Xue err: 1955497f668SJiancheng Xue while (i--) 1965497f668SJiancheng Xue clk_unregister_mux(data->clk_data.clks[clks[i].id]); 1975497f668SJiancheng Xue 1985497f668SJiancheng Xue return PTR_ERR(clk); 1990aa0c95fSHaojian Zhuang } 200f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_mux); 2010aa0c95fSHaojian Zhuang 202811f67ccStianshuliang int hisi_clk_register_phase(struct device *dev, 203811f67ccStianshuliang const struct hisi_phase_clock *clks, 204811f67ccStianshuliang int nums, struct hisi_clock_data *data) 205811f67ccStianshuliang { 206811f67ccStianshuliang void __iomem *base = data->base; 207811f67ccStianshuliang struct clk *clk; 208811f67ccStianshuliang int i; 209811f67ccStianshuliang 210811f67ccStianshuliang for (i = 0; i < nums; i++) { 211811f67ccStianshuliang clk = clk_register_hisi_phase(dev, &clks[i], base, 212811f67ccStianshuliang &hisi_clk_lock); 213811f67ccStianshuliang if (IS_ERR(clk)) { 214811f67ccStianshuliang pr_err("%s: failed to register clock %s\n", __func__, 215811f67ccStianshuliang clks[i].name); 216811f67ccStianshuliang return PTR_ERR(clk); 217811f67ccStianshuliang } 218811f67ccStianshuliang 219811f67ccStianshuliang data->clk_data.clks[clks[i].id] = clk; 220811f67ccStianshuliang } 221811f67ccStianshuliang 222811f67ccStianshuliang return 0; 223811f67ccStianshuliang } 224811f67ccStianshuliang EXPORT_SYMBOL_GPL(hisi_clk_register_phase); 225811f67ccStianshuliang 2265497f668SJiancheng Xue int hisi_clk_register_divider(const struct hisi_divider_clock *clks, 22775af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 2280aa0c95fSHaojian Zhuang { 2290aa0c95fSHaojian Zhuang struct clk *clk; 23075af25f5SHaojian Zhuang void __iomem *base = data->base; 2310aa0c95fSHaojian Zhuang int i; 2320aa0c95fSHaojian Zhuang 2330aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 2340aa0c95fSHaojian Zhuang clk = clk_register_divider_table(NULL, clks[i].name, 2350aa0c95fSHaojian Zhuang clks[i].parent_name, 2360aa0c95fSHaojian Zhuang clks[i].flags, 2370aa0c95fSHaojian Zhuang base + clks[i].offset, 2380aa0c95fSHaojian Zhuang clks[i].shift, clks[i].width, 2390aa0c95fSHaojian Zhuang clks[i].div_flags, 2400aa0c95fSHaojian Zhuang clks[i].table, 2410aa0c95fSHaojian Zhuang &hisi_clk_lock); 2420aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 2430aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 2440aa0c95fSHaojian Zhuang __func__, clks[i].name); 2455497f668SJiancheng Xue goto err; 2460aa0c95fSHaojian Zhuang } 2470aa0c95fSHaojian Zhuang 2480aa0c95fSHaojian Zhuang if (clks[i].alias) 2490aa0c95fSHaojian Zhuang clk_register_clkdev(clk, clks[i].alias, NULL); 2500aa0c95fSHaojian Zhuang 25175af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 2520aa0c95fSHaojian Zhuang } 2535497f668SJiancheng Xue 2545497f668SJiancheng Xue return 0; 2555497f668SJiancheng Xue 2565497f668SJiancheng Xue err: 2575497f668SJiancheng Xue while (i--) 2585497f668SJiancheng Xue clk_unregister_divider(data->clk_data.clks[clks[i].id]); 2595497f668SJiancheng Xue 2605497f668SJiancheng Xue return PTR_ERR(clk); 2610aa0c95fSHaojian Zhuang } 262f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_divider); 2630aa0c95fSHaojian Zhuang 2645497f668SJiancheng Xue int hisi_clk_register_gate(const struct hisi_gate_clock *clks, 2658b9dcb6cSZhangfei Gao int nums, struct hisi_clock_data *data) 2668b9dcb6cSZhangfei Gao { 2678b9dcb6cSZhangfei Gao struct clk *clk; 2688b9dcb6cSZhangfei Gao void __iomem *base = data->base; 2698b9dcb6cSZhangfei Gao int i; 2708b9dcb6cSZhangfei Gao 2718b9dcb6cSZhangfei Gao for (i = 0; i < nums; i++) { 2728b9dcb6cSZhangfei Gao clk = clk_register_gate(NULL, clks[i].name, 2738b9dcb6cSZhangfei Gao clks[i].parent_name, 2748b9dcb6cSZhangfei Gao clks[i].flags, 2758b9dcb6cSZhangfei Gao base + clks[i].offset, 2768b9dcb6cSZhangfei Gao clks[i].bit_idx, 2778b9dcb6cSZhangfei Gao clks[i].gate_flags, 2788b9dcb6cSZhangfei Gao &hisi_clk_lock); 2798b9dcb6cSZhangfei Gao if (IS_ERR(clk)) { 2808b9dcb6cSZhangfei Gao pr_err("%s: failed to register clock %s\n", 2818b9dcb6cSZhangfei Gao __func__, clks[i].name); 2825497f668SJiancheng Xue goto err; 2838b9dcb6cSZhangfei Gao } 2848b9dcb6cSZhangfei Gao 2858b9dcb6cSZhangfei Gao if (clks[i].alias) 2868b9dcb6cSZhangfei Gao clk_register_clkdev(clk, clks[i].alias, NULL); 2878b9dcb6cSZhangfei Gao 2888b9dcb6cSZhangfei Gao data->clk_data.clks[clks[i].id] = clk; 2898b9dcb6cSZhangfei Gao } 2905497f668SJiancheng Xue 2915497f668SJiancheng Xue return 0; 2925497f668SJiancheng Xue 2935497f668SJiancheng Xue err: 2945497f668SJiancheng Xue while (i--) 2955497f668SJiancheng Xue clk_unregister_gate(data->clk_data.clks[clks[i].id]); 2965497f668SJiancheng Xue 2975497f668SJiancheng Xue return PTR_ERR(clk); 2988b9dcb6cSZhangfei Gao } 299f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_gate); 3008b9dcb6cSZhangfei Gao 301f6ff57c8SJiancheng Xue void hisi_clk_register_gate_sep(const struct hisi_gate_clock *clks, 30275af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 3030aa0c95fSHaojian Zhuang { 3040aa0c95fSHaojian Zhuang struct clk *clk; 30575af25f5SHaojian Zhuang void __iomem *base = data->base; 3060aa0c95fSHaojian Zhuang int i; 3070aa0c95fSHaojian Zhuang 3080aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 3090aa0c95fSHaojian Zhuang clk = hisi_register_clkgate_sep(NULL, clks[i].name, 3100aa0c95fSHaojian Zhuang clks[i].parent_name, 3110aa0c95fSHaojian Zhuang clks[i].flags, 3120aa0c95fSHaojian Zhuang base + clks[i].offset, 3130aa0c95fSHaojian Zhuang clks[i].bit_idx, 3140aa0c95fSHaojian Zhuang clks[i].gate_flags, 3150aa0c95fSHaojian Zhuang &hisi_clk_lock); 3160aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 3170aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 3180aa0c95fSHaojian Zhuang __func__, clks[i].name); 3190aa0c95fSHaojian Zhuang continue; 3200aa0c95fSHaojian Zhuang } 3210aa0c95fSHaojian Zhuang 3220aa0c95fSHaojian Zhuang if (clks[i].alias) 3230aa0c95fSHaojian Zhuang clk_register_clkdev(clk, clks[i].alias, NULL); 3240aa0c95fSHaojian Zhuang 32575af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 3260aa0c95fSHaojian Zhuang } 3270aa0c95fSHaojian Zhuang } 328f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_gate_sep); 32972ea4861SBintian Wang 330f6ff57c8SJiancheng Xue void __init hi6220_clk_register_divider(const struct hi6220_divider_clock *clks, 33172ea4861SBintian Wang int nums, struct hisi_clock_data *data) 33272ea4861SBintian Wang { 33372ea4861SBintian Wang struct clk *clk; 33472ea4861SBintian Wang void __iomem *base = data->base; 33572ea4861SBintian Wang int i; 33672ea4861SBintian Wang 33772ea4861SBintian Wang for (i = 0; i < nums; i++) { 33872ea4861SBintian Wang clk = hi6220_register_clkdiv(NULL, clks[i].name, 33972ea4861SBintian Wang clks[i].parent_name, 34072ea4861SBintian Wang clks[i].flags, 34172ea4861SBintian Wang base + clks[i].offset, 34272ea4861SBintian Wang clks[i].shift, 34372ea4861SBintian Wang clks[i].width, 34472ea4861SBintian Wang clks[i].mask_bit, 34572ea4861SBintian Wang &hisi_clk_lock); 34672ea4861SBintian Wang if (IS_ERR(clk)) { 34772ea4861SBintian Wang pr_err("%s: failed to register clock %s\n", 34872ea4861SBintian Wang __func__, clks[i].name); 34972ea4861SBintian Wang continue; 35072ea4861SBintian Wang } 35172ea4861SBintian Wang 35272ea4861SBintian Wang if (clks[i].alias) 35372ea4861SBintian Wang clk_register_clkdev(clk, clks[i].alias, NULL); 35472ea4861SBintian Wang 35572ea4861SBintian Wang data->clk_data.clks[clks[i].id] = clk; 35672ea4861SBintian Wang } 35772ea4861SBintian Wang } 358