162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Qualcomm SDX55 APCS clock controller driver
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (c) 2020, Linaro Limited
662306a36Sopenharmony_ci * Author: Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/clk.h>
1062306a36Sopenharmony_ci#include <linux/clk-provider.h>
1162306a36Sopenharmony_ci#include <linux/cpu.h>
1262306a36Sopenharmony_ci#include <linux/kernel.h>
1362306a36Sopenharmony_ci#include <linux/module.h>
1462306a36Sopenharmony_ci#include <linux/platform_device.h>
1562306a36Sopenharmony_ci#include <linux/pm_domain.h>
1662306a36Sopenharmony_ci#include <linux/regmap.h>
1762306a36Sopenharmony_ci#include <linux/slab.h>
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci#include "clk-regmap.h"
2062306a36Sopenharmony_ci#include "clk-regmap-mux-div.h"
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic const u32 apcs_mux_clk_parent_map[] = { 0, 1, 5 };
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_cistatic const struct clk_parent_data pdata[] = {
2562306a36Sopenharmony_ci	{ .fw_name = "ref" },
2662306a36Sopenharmony_ci	{ .fw_name = "aux" },
2762306a36Sopenharmony_ci	{ .fw_name = "pll" },
2862306a36Sopenharmony_ci};
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci/*
3162306a36Sopenharmony_ci * We use the notifier function for switching to a temporary safe configuration
3262306a36Sopenharmony_ci * (mux and divider), while the A7 PLL is reconfigured.
3362306a36Sopenharmony_ci */
3462306a36Sopenharmony_cistatic int a7cc_notifier_cb(struct notifier_block *nb, unsigned long event,
3562306a36Sopenharmony_ci			    void *data)
3662306a36Sopenharmony_ci{
3762306a36Sopenharmony_ci	int ret = 0;
3862306a36Sopenharmony_ci	struct clk_regmap_mux_div *md = container_of(nb,
3962306a36Sopenharmony_ci						     struct clk_regmap_mux_div,
4062306a36Sopenharmony_ci						     clk_nb);
4162306a36Sopenharmony_ci	if (event == PRE_RATE_CHANGE)
4262306a36Sopenharmony_ci		/* set the mux and divider to safe frequency (400mhz) */
4362306a36Sopenharmony_ci		ret = mux_div_set_src_div(md, 1, 2);
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ci	return notifier_from_errno(ret);
4662306a36Sopenharmony_ci}
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_cistatic int qcom_apcs_sdx55_clk_probe(struct platform_device *pdev)
4962306a36Sopenharmony_ci{
5062306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
5162306a36Sopenharmony_ci	struct device *parent = dev->parent;
5262306a36Sopenharmony_ci	struct device *cpu_dev;
5362306a36Sopenharmony_ci	struct clk_regmap_mux_div *a7cc;
5462306a36Sopenharmony_ci	struct regmap *regmap;
5562306a36Sopenharmony_ci	struct clk_init_data init = { };
5662306a36Sopenharmony_ci	int ret;
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_ci	regmap = dev_get_regmap(parent, NULL);
5962306a36Sopenharmony_ci	if (!regmap) {
6062306a36Sopenharmony_ci		dev_err(dev, "Failed to get parent regmap\n");
6162306a36Sopenharmony_ci		return -ENODEV;
6262306a36Sopenharmony_ci	}
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_ci	a7cc = devm_kzalloc(dev, sizeof(*a7cc), GFP_KERNEL);
6562306a36Sopenharmony_ci	if (!a7cc)
6662306a36Sopenharmony_ci		return -ENOMEM;
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	init.name = "a7mux";
6962306a36Sopenharmony_ci	init.parent_data = pdata;
7062306a36Sopenharmony_ci	init.num_parents = ARRAY_SIZE(pdata);
7162306a36Sopenharmony_ci	init.ops = &clk_regmap_mux_div_ops;
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	a7cc->clkr.hw.init = &init;
7462306a36Sopenharmony_ci	a7cc->clkr.regmap = regmap;
7562306a36Sopenharmony_ci	a7cc->reg_offset = 0x8;
7662306a36Sopenharmony_ci	a7cc->hid_width = 5;
7762306a36Sopenharmony_ci	a7cc->hid_shift = 0;
7862306a36Sopenharmony_ci	a7cc->src_width = 3;
7962306a36Sopenharmony_ci	a7cc->src_shift = 8;
8062306a36Sopenharmony_ci	a7cc->parent_map = apcs_mux_clk_parent_map;
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ci	a7cc->pclk = devm_clk_get(parent, "pll");
8362306a36Sopenharmony_ci	if (IS_ERR(a7cc->pclk))
8462306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(a7cc->pclk),
8562306a36Sopenharmony_ci				     "Failed to get PLL clk\n");
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	a7cc->clk_nb.notifier_call = a7cc_notifier_cb;
8862306a36Sopenharmony_ci	ret = clk_notifier_register(a7cc->pclk, &a7cc->clk_nb);
8962306a36Sopenharmony_ci	if (ret)
9062306a36Sopenharmony_ci		return dev_err_probe(dev, ret,
9162306a36Sopenharmony_ci				     "Failed to register clock notifier\n");
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci	ret = devm_clk_register_regmap(dev, &a7cc->clkr);
9462306a36Sopenharmony_ci	if (ret) {
9562306a36Sopenharmony_ci		dev_err_probe(dev, ret, "Failed to register regmap clock\n");
9662306a36Sopenharmony_ci		goto err;
9762306a36Sopenharmony_ci	}
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	ret = devm_of_clk_add_hw_provider(dev, of_clk_hw_simple_get,
10062306a36Sopenharmony_ci					  &a7cc->clkr.hw);
10162306a36Sopenharmony_ci	if (ret) {
10262306a36Sopenharmony_ci		dev_err_probe(dev, ret, "Failed to add clock provider\n");
10362306a36Sopenharmony_ci		goto err;
10462306a36Sopenharmony_ci	}
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	platform_set_drvdata(pdev, a7cc);
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci	/*
10962306a36Sopenharmony_ci	 * Attach the power domain to cpudev. Since there is no dedicated driver
11062306a36Sopenharmony_ci	 * for CPUs and the SDX55 platform lacks hardware specific CPUFreq
11162306a36Sopenharmony_ci	 * driver, there seems to be no better place to do this. So do it here!
11262306a36Sopenharmony_ci	 */
11362306a36Sopenharmony_ci	cpu_dev = get_cpu_device(0);
11462306a36Sopenharmony_ci	dev_pm_domain_attach(cpu_dev, true);
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_ci	return 0;
11762306a36Sopenharmony_ci
11862306a36Sopenharmony_cierr:
11962306a36Sopenharmony_ci	clk_notifier_unregister(a7cc->pclk, &a7cc->clk_nb);
12062306a36Sopenharmony_ci	return ret;
12162306a36Sopenharmony_ci}
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_cistatic void qcom_apcs_sdx55_clk_remove(struct platform_device *pdev)
12462306a36Sopenharmony_ci{
12562306a36Sopenharmony_ci	struct device *cpu_dev = get_cpu_device(0);
12662306a36Sopenharmony_ci	struct clk_regmap_mux_div *a7cc = platform_get_drvdata(pdev);
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	clk_notifier_unregister(a7cc->pclk, &a7cc->clk_nb);
12962306a36Sopenharmony_ci	dev_pm_domain_detach(cpu_dev, true);
13062306a36Sopenharmony_ci}
13162306a36Sopenharmony_ci
13262306a36Sopenharmony_cistatic struct platform_driver qcom_apcs_sdx55_clk_driver = {
13362306a36Sopenharmony_ci	.probe = qcom_apcs_sdx55_clk_probe,
13462306a36Sopenharmony_ci	.remove_new = qcom_apcs_sdx55_clk_remove,
13562306a36Sopenharmony_ci	.driver = {
13662306a36Sopenharmony_ci		.name = "qcom-sdx55-acps-clk",
13762306a36Sopenharmony_ci	},
13862306a36Sopenharmony_ci};
13962306a36Sopenharmony_cimodule_platform_driver(qcom_apcs_sdx55_clk_driver);
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_ciMODULE_AUTHOR("Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>");
14262306a36Sopenharmony_ciMODULE_LICENSE("GPL v2");
14362306a36Sopenharmony_ciMODULE_DESCRIPTION("Qualcomm SDX55 APCS clock driver");
144