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 40*32226916SJiancheng Xue struct hisi_clock_data *hisi_clk_alloc(struct platform_device *pdev, 41*32226916SJiancheng Xue int nr_clks) 42*32226916SJiancheng Xue { 43*32226916SJiancheng Xue struct hisi_clock_data *clk_data; 44*32226916SJiancheng Xue struct resource *res; 45*32226916SJiancheng Xue struct clk **clk_table; 46*32226916SJiancheng Xue 47*32226916SJiancheng Xue clk_data = devm_kmalloc(&pdev->dev, sizeof(*clk_data), GFP_KERNEL); 48*32226916SJiancheng Xue if (!clk_data) 49*32226916SJiancheng Xue return NULL; 50*32226916SJiancheng Xue 51*32226916SJiancheng Xue res = platform_get_resource(pdev, IORESOURCE_MEM, 0); 52*32226916SJiancheng Xue clk_data->base = devm_ioremap(&pdev->dev, 53*32226916SJiancheng Xue res->start, resource_size(res)); 54*32226916SJiancheng Xue if (!clk_data->base) 55*32226916SJiancheng Xue return NULL; 56*32226916SJiancheng Xue 57*32226916SJiancheng Xue clk_table = devm_kmalloc(&pdev->dev, sizeof(struct clk *) * nr_clks, 58*32226916SJiancheng Xue GFP_KERNEL); 59*32226916SJiancheng Xue if (!clk_table) 60*32226916SJiancheng Xue return NULL; 61*32226916SJiancheng Xue 62*32226916SJiancheng Xue clk_data->clk_data.clks = clk_table; 63*32226916SJiancheng Xue clk_data->clk_data.clk_num = nr_clks; 64*32226916SJiancheng Xue 65*32226916SJiancheng Xue return clk_data; 66*32226916SJiancheng Xue } 67*32226916SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_alloc); 68*32226916SJiancheng Xue 69f6ff57c8SJiancheng Xue struct hisi_clock_data *hisi_clk_init(struct device_node *np, 7075af25f5SHaojian Zhuang int nr_clks) 710aa0c95fSHaojian Zhuang { 7275af25f5SHaojian Zhuang struct hisi_clock_data *clk_data; 7375af25f5SHaojian Zhuang struct clk **clk_table; 7475af25f5SHaojian Zhuang void __iomem *base; 7575af25f5SHaojian Zhuang 7675af25f5SHaojian Zhuang base = of_iomap(np, 0); 7775af25f5SHaojian Zhuang if (!base) { 781fb6dd9dSLeo Yan pr_err("%s: failed to map clock registers\n", __func__); 7975af25f5SHaojian Zhuang goto err; 8075af25f5SHaojian Zhuang } 8175af25f5SHaojian Zhuang 8275af25f5SHaojian Zhuang clk_data = kzalloc(sizeof(*clk_data), GFP_KERNEL); 8375af25f5SHaojian Zhuang if (!clk_data) { 8475af25f5SHaojian Zhuang pr_err("%s: could not allocate clock data\n", __func__); 8575af25f5SHaojian Zhuang goto err; 8675af25f5SHaojian Zhuang } 8775af25f5SHaojian Zhuang clk_data->base = base; 8875af25f5SHaojian Zhuang 890aa0c95fSHaojian Zhuang clk_table = kzalloc(sizeof(struct clk *) * nr_clks, GFP_KERNEL); 900aa0c95fSHaojian Zhuang if (!clk_table) { 910aa0c95fSHaojian Zhuang pr_err("%s: could not allocate clock lookup table\n", __func__); 9275af25f5SHaojian Zhuang goto err_data; 930aa0c95fSHaojian Zhuang } 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 105f6ff57c8SJiancheng Xue void 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); 1190aa0c95fSHaojian Zhuang continue; 1200aa0c95fSHaojian Zhuang } 12175af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 1220aa0c95fSHaojian Zhuang } 1230aa0c95fSHaojian Zhuang } 124f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_fixed_rate); 1250aa0c95fSHaojian Zhuang 126f6ff57c8SJiancheng Xue void hisi_clk_register_fixed_factor(const struct hisi_fixed_factor_clock *clks, 12775af25f5SHaojian Zhuang int nums, 12875af25f5SHaojian Zhuang struct hisi_clock_data *data) 1290aa0c95fSHaojian Zhuang { 1300aa0c95fSHaojian Zhuang struct clk *clk; 1310aa0c95fSHaojian Zhuang int i; 1320aa0c95fSHaojian Zhuang 1330aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 1340aa0c95fSHaojian Zhuang clk = clk_register_fixed_factor(NULL, clks[i].name, 1350aa0c95fSHaojian Zhuang clks[i].parent_name, 1360aa0c95fSHaojian Zhuang clks[i].flags, clks[i].mult, 1370aa0c95fSHaojian Zhuang clks[i].div); 1380aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 1390aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 1400aa0c95fSHaojian Zhuang __func__, clks[i].name); 1410aa0c95fSHaojian Zhuang continue; 1420aa0c95fSHaojian Zhuang } 14375af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 1440aa0c95fSHaojian Zhuang } 1450aa0c95fSHaojian Zhuang } 146f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_fixed_factor); 1470aa0c95fSHaojian Zhuang 148f6ff57c8SJiancheng Xue void hisi_clk_register_mux(const struct hisi_mux_clock *clks, 14975af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 1500aa0c95fSHaojian Zhuang { 1510aa0c95fSHaojian Zhuang struct clk *clk; 15275af25f5SHaojian Zhuang void __iomem *base = data->base; 1530aa0c95fSHaojian Zhuang int i; 1540aa0c95fSHaojian Zhuang 1550aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 156156342a1SZhangfei Gao u32 mask = BIT(clks[i].width) - 1; 157156342a1SZhangfei Gao 158156342a1SZhangfei Gao clk = clk_register_mux_table(NULL, clks[i].name, 159156342a1SZhangfei Gao clks[i].parent_names, 1600aa0c95fSHaojian Zhuang clks[i].num_parents, clks[i].flags, 1610aa0c95fSHaojian Zhuang base + clks[i].offset, clks[i].shift, 162156342a1SZhangfei Gao mask, clks[i].mux_flags, 163156342a1SZhangfei Gao clks[i].table, &hisi_clk_lock); 1640aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 1650aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 1660aa0c95fSHaojian Zhuang __func__, clks[i].name); 1670aa0c95fSHaojian Zhuang continue; 1680aa0c95fSHaojian Zhuang } 1690aa0c95fSHaojian Zhuang 1700aa0c95fSHaojian Zhuang if (clks[i].alias) 1710aa0c95fSHaojian Zhuang clk_register_clkdev(clk, clks[i].alias, NULL); 1720aa0c95fSHaojian Zhuang 17375af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 1740aa0c95fSHaojian Zhuang } 1750aa0c95fSHaojian Zhuang } 176f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_mux); 1770aa0c95fSHaojian Zhuang 178f6ff57c8SJiancheng Xue void hisi_clk_register_divider(const struct hisi_divider_clock *clks, 17975af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 1800aa0c95fSHaojian Zhuang { 1810aa0c95fSHaojian Zhuang struct clk *clk; 18275af25f5SHaojian Zhuang void __iomem *base = data->base; 1830aa0c95fSHaojian Zhuang int i; 1840aa0c95fSHaojian Zhuang 1850aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 1860aa0c95fSHaojian Zhuang clk = clk_register_divider_table(NULL, clks[i].name, 1870aa0c95fSHaojian Zhuang clks[i].parent_name, 1880aa0c95fSHaojian Zhuang clks[i].flags, 1890aa0c95fSHaojian Zhuang base + clks[i].offset, 1900aa0c95fSHaojian Zhuang clks[i].shift, clks[i].width, 1910aa0c95fSHaojian Zhuang clks[i].div_flags, 1920aa0c95fSHaojian Zhuang clks[i].table, 1930aa0c95fSHaojian Zhuang &hisi_clk_lock); 1940aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 1950aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 1960aa0c95fSHaojian Zhuang __func__, clks[i].name); 1970aa0c95fSHaojian Zhuang continue; 1980aa0c95fSHaojian Zhuang } 1990aa0c95fSHaojian Zhuang 2000aa0c95fSHaojian Zhuang if (clks[i].alias) 2010aa0c95fSHaojian Zhuang clk_register_clkdev(clk, clks[i].alias, NULL); 2020aa0c95fSHaojian Zhuang 20375af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 2040aa0c95fSHaojian Zhuang } 2050aa0c95fSHaojian Zhuang } 206f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_divider); 2070aa0c95fSHaojian Zhuang 208f6ff57c8SJiancheng Xue void hisi_clk_register_gate(const struct hisi_gate_clock *clks, 2098b9dcb6cSZhangfei Gao int nums, struct hisi_clock_data *data) 2108b9dcb6cSZhangfei Gao { 2118b9dcb6cSZhangfei Gao struct clk *clk; 2128b9dcb6cSZhangfei Gao void __iomem *base = data->base; 2138b9dcb6cSZhangfei Gao int i; 2148b9dcb6cSZhangfei Gao 2158b9dcb6cSZhangfei Gao for (i = 0; i < nums; i++) { 2168b9dcb6cSZhangfei Gao clk = clk_register_gate(NULL, clks[i].name, 2178b9dcb6cSZhangfei Gao clks[i].parent_name, 2188b9dcb6cSZhangfei Gao clks[i].flags, 2198b9dcb6cSZhangfei Gao base + clks[i].offset, 2208b9dcb6cSZhangfei Gao clks[i].bit_idx, 2218b9dcb6cSZhangfei Gao clks[i].gate_flags, 2228b9dcb6cSZhangfei Gao &hisi_clk_lock); 2238b9dcb6cSZhangfei Gao if (IS_ERR(clk)) { 2248b9dcb6cSZhangfei Gao pr_err("%s: failed to register clock %s\n", 2258b9dcb6cSZhangfei Gao __func__, clks[i].name); 2268b9dcb6cSZhangfei Gao continue; 2278b9dcb6cSZhangfei Gao } 2288b9dcb6cSZhangfei Gao 2298b9dcb6cSZhangfei Gao if (clks[i].alias) 2308b9dcb6cSZhangfei Gao clk_register_clkdev(clk, clks[i].alias, NULL); 2318b9dcb6cSZhangfei Gao 2328b9dcb6cSZhangfei Gao data->clk_data.clks[clks[i].id] = clk; 2338b9dcb6cSZhangfei Gao } 2348b9dcb6cSZhangfei Gao } 235f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_gate); 2368b9dcb6cSZhangfei Gao 237f6ff57c8SJiancheng Xue void hisi_clk_register_gate_sep(const struct hisi_gate_clock *clks, 23875af25f5SHaojian Zhuang int nums, struct hisi_clock_data *data) 2390aa0c95fSHaojian Zhuang { 2400aa0c95fSHaojian Zhuang struct clk *clk; 24175af25f5SHaojian Zhuang void __iomem *base = data->base; 2420aa0c95fSHaojian Zhuang int i; 2430aa0c95fSHaojian Zhuang 2440aa0c95fSHaojian Zhuang for (i = 0; i < nums; i++) { 2450aa0c95fSHaojian Zhuang clk = hisi_register_clkgate_sep(NULL, clks[i].name, 2460aa0c95fSHaojian Zhuang clks[i].parent_name, 2470aa0c95fSHaojian Zhuang clks[i].flags, 2480aa0c95fSHaojian Zhuang base + clks[i].offset, 2490aa0c95fSHaojian Zhuang clks[i].bit_idx, 2500aa0c95fSHaojian Zhuang clks[i].gate_flags, 2510aa0c95fSHaojian Zhuang &hisi_clk_lock); 2520aa0c95fSHaojian Zhuang if (IS_ERR(clk)) { 2530aa0c95fSHaojian Zhuang pr_err("%s: failed to register clock %s\n", 2540aa0c95fSHaojian Zhuang __func__, clks[i].name); 2550aa0c95fSHaojian Zhuang continue; 2560aa0c95fSHaojian Zhuang } 2570aa0c95fSHaojian Zhuang 2580aa0c95fSHaojian Zhuang if (clks[i].alias) 2590aa0c95fSHaojian Zhuang clk_register_clkdev(clk, clks[i].alias, NULL); 2600aa0c95fSHaojian Zhuang 26175af25f5SHaojian Zhuang data->clk_data.clks[clks[i].id] = clk; 2620aa0c95fSHaojian Zhuang } 2630aa0c95fSHaojian Zhuang } 264f6ff57c8SJiancheng Xue EXPORT_SYMBOL_GPL(hisi_clk_register_gate_sep); 26572ea4861SBintian Wang 266f6ff57c8SJiancheng Xue void __init hi6220_clk_register_divider(const struct hi6220_divider_clock *clks, 26772ea4861SBintian Wang int nums, struct hisi_clock_data *data) 26872ea4861SBintian Wang { 26972ea4861SBintian Wang struct clk *clk; 27072ea4861SBintian Wang void __iomem *base = data->base; 27172ea4861SBintian Wang int i; 27272ea4861SBintian Wang 27372ea4861SBintian Wang for (i = 0; i < nums; i++) { 27472ea4861SBintian Wang clk = hi6220_register_clkdiv(NULL, clks[i].name, 27572ea4861SBintian Wang clks[i].parent_name, 27672ea4861SBintian Wang clks[i].flags, 27772ea4861SBintian Wang base + clks[i].offset, 27872ea4861SBintian Wang clks[i].shift, 27972ea4861SBintian Wang clks[i].width, 28072ea4861SBintian Wang clks[i].mask_bit, 28172ea4861SBintian Wang &hisi_clk_lock); 28272ea4861SBintian Wang if (IS_ERR(clk)) { 28372ea4861SBintian Wang pr_err("%s: failed to register clock %s\n", 28472ea4861SBintian Wang __func__, clks[i].name); 28572ea4861SBintian Wang continue; 28672ea4861SBintian Wang } 28772ea4861SBintian Wang 28872ea4861SBintian Wang if (clks[i].alias) 28972ea4861SBintian Wang clk_register_clkdev(clk, clks[i].alias, NULL); 29072ea4861SBintian Wang 29172ea4861SBintian Wang data->clk_data.clks[clks[i].id] = clk; 29272ea4861SBintian Wang } 29372ea4861SBintian Wang } 294