mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00
86 lines
2 KiB
C
86 lines
2 KiB
C
![]() |
// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
/*
|
||
|
* Copyright (c) 2024 Collabora Ltd.
|
||
|
* Author: Sebastian Reichel <sebastian.reichel@collabora.com>
|
||
|
*/
|
||
|
|
||
|
#include <linux/clk.h>
|
||
|
#include <linux/platform_device.h>
|
||
|
#include <linux/pm_clock.h>
|
||
|
#include <linux/pm_runtime.h>
|
||
|
#include <linux/property.h>
|
||
|
#include "clk.h"
|
||
|
|
||
|
static int rk_clk_gate_link_register(struct device *dev,
|
||
|
struct rockchip_clk_provider *ctx,
|
||
|
struct rockchip_clk_branch *clkbr)
|
||
|
{
|
||
|
unsigned long flags = clkbr->flags | CLK_SET_RATE_PARENT;
|
||
|
struct clk *clk;
|
||
|
|
||
|
clk = clk_register_gate(dev, clkbr->name, clkbr->parent_names[0],
|
||
|
flags, ctx->reg_base + clkbr->gate_offset,
|
||
|
clkbr->gate_shift, clkbr->gate_flags,
|
||
|
&ctx->lock);
|
||
|
|
||
|
if (IS_ERR(clk))
|
||
|
return PTR_ERR(clk);
|
||
|
|
||
|
rockchip_clk_set_lookup(ctx, clk, clkbr->id);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int rk_clk_gate_link_probe(struct platform_device *pdev)
|
||
|
{
|
||
|
struct rockchip_gate_link_platdata *pdata;
|
||
|
struct device *dev = &pdev->dev;
|
||
|
struct clk *linked_clk;
|
||
|
int ret;
|
||
|
|
||
|
pdata = dev_get_platdata(dev);
|
||
|
if (!pdata)
|
||
|
return dev_err_probe(dev, -ENODEV, "missing platform data");
|
||
|
|
||
|
ret = devm_pm_runtime_enable(dev);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
ret = devm_pm_clk_create(dev);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
linked_clk = rockchip_clk_get_lookup(pdata->ctx, pdata->clkbr->linked_clk_id);
|
||
|
ret = pm_clk_add_clk(dev, linked_clk);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
ret = rk_clk_gate_link_register(dev, pdata->ctx, pdata->clkbr);
|
||
|
if (ret)
|
||
|
goto err;
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
err:
|
||
|
pm_clk_remove_clk(dev, linked_clk);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static const struct dev_pm_ops rk_clk_gate_link_pm_ops = {
|
||
|
SET_RUNTIME_PM_OPS(pm_clk_suspend, pm_clk_resume, NULL)
|
||
|
};
|
||
|
|
||
|
static struct platform_driver rk_clk_gate_link_driver = {
|
||
|
.probe = rk_clk_gate_link_probe,
|
||
|
.driver = {
|
||
|
.name = "rockchip-gate-link-clk",
|
||
|
.pm = &rk_clk_gate_link_pm_ops,
|
||
|
.suppress_bind_attrs = true,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
static int __init rk_clk_gate_link_drv_register(void)
|
||
|
{
|
||
|
return platform_driver_register(&rk_clk_gate_link_driver);
|
||
|
}
|
||
|
core_initcall(rk_clk_gate_link_drv_register);
|