Commit 407c04d6 authored by Stephen Boyd's avatar Stephen Boyd

Merge branches 'clk-microchip', 'clk-si', 'clk-mtk', 'clk-at91' and 'clk-st' into clk-next

 - Clock configuration on Microchip PolarFire SoCs
 - Free allocations on probe error in Mediatek clk driver
 - Modernize Mediatek clk driver by consolidating code

* clk-microchip:
  clk: microchip: Add driver for Microchip PolarFire SoC
  dt-bindings: clk: microchip: Add Microchip PolarFire host binding

* clk-si:
  clk-si5341: replace snprintf in show functions with sysfs_emit
  clk: si5341: fix reported clk_rate when output divider is 2

* clk-mtk: (32 commits)
  clk: mediatek: Warn if clk IDs are duplicated
  clk: mediatek: mt8195: Implement remove functions
  clk: mediatek: mt8195: Implement error handling in probe functions
  clk: mediatek: mt8195: Hook up mtk_clk_simple_remove()
  clk: mediatek: Unregister clks in mtk_clk_simple_probe() error path
  clk: mediatek: mtk: Implement error handling in register APIs
  clk: mediatek: pll: Implement error handling in register API
  clk: mediatek: mux: Implement error handling in register API
  clk: mediatek: mux: Reverse check for existing clk to reduce nesting level
  clk: mediatek: gate: Implement error handling in register API
  clk: mediatek: cpumux: Implement error handling in register API
  clk: mediatek: mtk: Clean up included headers
  clk: mediatek: Add mtk_clk_simple_remove()
  clk: mediatek: Implement mtk_clk_unregister_composites() API
  clk: mediatek: Implement mtk_clk_unregister_divider_clks() API
  clk: mediatek: Implement mtk_clk_unregister_factors() API
  clk: mediatek: Implement mtk_clk_unregister_fixed_clks() API
  clk: mediatek: pll: Clean up included headers
  clk: mediatek: pll: Implement unregister API
  clk: mediatek: pll: Split definitions into separate header file
  ...

* clk-at91:
  clk: at91: clk-master: remove dead code
  clk: at91: sama7g5: fix parents of PDMCs' GCLK
  clk: at91: sama7g5: Allow MCK1 to be exported and referenced in DT
  clk: at91: allow setting PMC_AUDIOPINCK clock parents via DT

* clk-st:
  clk: stm32mp1: Add parent_data to ETHRX clock
  clk: stm32mp1: Split ETHCK_K into separate MUX and GATE clock
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
%YAML 1.2
---
$id: http://devicetree.org/schemas/clock/microchip,mpfs.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: Microchip PolarFire Clock Control Module Binding
maintainers:
- Daire McNamara <daire.mcnamara@microchip.com>
description: |
Microchip PolarFire clock control (CLKCFG) is an integrated clock controller,
which gates and enables all peripheral clocks.
This device tree binding describes 33 gate clocks. Clocks are referenced by
user nodes by the CLKCFG node phandle and the clock index in the group, from
0 to 32.
properties:
compatible:
const: microchip,mpfs-clkcfg
reg:
maxItems: 1
clocks:
maxItems: 1
'#clock-cells':
const: 1
description: |
The clock consumer should specify the desired clock by having the clock
ID in its "clocks" phandle cell. See include/dt-bindings/clock/microchip,mpfs-clock.h
for the full list of PolarFire clock IDs.
required:
- compatible
- reg
- clocks
- '#clock-cells'
additionalProperties: false
examples:
# Clock Config node:
- |
#include <dt-bindings/clock/microchip,mpfs-clock.h>
soc {
#address-cells = <2>;
#size-cells = <2>;
clkcfg: clock-controller@20002000 {
compatible = "microchip,mpfs-clkcfg";
reg = <0x0 0x20002000 0x0 0x1000>;
clocks = <&ref>;
#clock-cells = <1>;
};
};
...@@ -330,9 +330,6 @@ config COMMON_CLK_PXA ...@@ -330,9 +330,6 @@ config COMMON_CLK_PXA
help help
Support for the Marvell PXA SoC. Support for the Marvell PXA SoC.
config COMMON_CLK_PIC32
def_bool COMMON_CLK && MACH_PIC32
config COMMON_CLK_OXNAS config COMMON_CLK_OXNAS
bool "Clock driver for the OXNAS SoC Family" bool "Clock driver for the OXNAS SoC Family"
depends on ARCH_OXNAS || COMPILE_TEST depends on ARCH_OXNAS || COMPILE_TEST
...@@ -407,6 +404,7 @@ source "drivers/clk/keystone/Kconfig" ...@@ -407,6 +404,7 @@ source "drivers/clk/keystone/Kconfig"
source "drivers/clk/mediatek/Kconfig" source "drivers/clk/mediatek/Kconfig"
source "drivers/clk/meson/Kconfig" source "drivers/clk/meson/Kconfig"
source "drivers/clk/mstar/Kconfig" source "drivers/clk/mstar/Kconfig"
source "drivers/clk/microchip/Kconfig"
source "drivers/clk/mvebu/Kconfig" source "drivers/clk/mvebu/Kconfig"
source "drivers/clk/pistachio/Kconfig" source "drivers/clk/pistachio/Kconfig"
source "drivers/clk/qcom/Kconfig" source "drivers/clk/qcom/Kconfig"
......
...@@ -91,7 +91,7 @@ obj-$(CONFIG_ARCH_KEYSTONE) += keystone/ ...@@ -91,7 +91,7 @@ obj-$(CONFIG_ARCH_KEYSTONE) += keystone/
obj-$(CONFIG_MACH_LOONGSON32) += loongson1/ obj-$(CONFIG_MACH_LOONGSON32) += loongson1/
obj-y += mediatek/ obj-y += mediatek/
obj-$(CONFIG_ARCH_MESON) += meson/ obj-$(CONFIG_ARCH_MESON) += meson/
obj-$(CONFIG_MACH_PIC32) += microchip/ obj-y += microchip/
ifeq ($(CONFIG_COMMON_CLK), y) ifeq ($(CONFIG_COMMON_CLK), y)
obj-$(CONFIG_ARCH_MMP) += mmp/ obj-$(CONFIG_ARCH_MMP) += mmp/
endif endif
......
...@@ -143,8 +143,7 @@ static void __init at91rm9200_pmc_setup(struct device_node *np) ...@@ -143,8 +143,7 @@ static void __init at91rm9200_pmc_setup(struct device_node *np)
parent_names, parent_names,
&at91rm9200_master_layout, &at91rm9200_master_layout,
&rm9200_mck_characteristics, &rm9200_mck_characteristics,
&rm9200_mck_lock, CLK_SET_RATE_GATE, &rm9200_mck_lock);
INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -419,8 +419,7 @@ static void __init at91sam926x_pmc_setup(struct device_node *np, ...@@ -419,8 +419,7 @@ static void __init at91sam926x_pmc_setup(struct device_node *np,
parent_names, parent_names,
&at91rm9200_master_layout, &at91rm9200_master_layout,
data->mck_characteristics, data->mck_characteristics,
&at91sam9260_mck_lock, &at91sam9260_mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -154,8 +154,7 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np) ...@@ -154,8 +154,7 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np)
parent_names, parent_names,
&at91rm9200_master_layout, &at91rm9200_master_layout,
&mck_characteristics, &mck_characteristics,
&at91sam9g45_mck_lock, &at91sam9g45_mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -181,8 +181,7 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np) ...@@ -181,8 +181,7 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np)
parent_names, parent_names,
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics, &mck_characteristics,
&at91sam9n12_mck_lock, &at91sam9n12_mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -123,8 +123,7 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np) ...@@ -123,8 +123,7 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np)
parent_names, parent_names,
&at91rm9200_master_layout, &at91rm9200_master_layout,
&sam9rl_mck_characteristics, &sam9rl_mck_characteristics,
&sam9rl_mck_lock, CLK_SET_RATE_GATE, &sam9rl_mck_lock);
INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -201,8 +201,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np, ...@@ -201,8 +201,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np,
hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names, parent_names,
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics, &mck_lock, &mck_characteristics, &mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -374,85 +374,6 @@ static void clk_sama7g5_master_best_diff(struct clk_rate_request *req, ...@@ -374,85 +374,6 @@ static void clk_sama7g5_master_best_diff(struct clk_rate_request *req,
} }
} }
static int clk_master_pres_determine_rate(struct clk_hw *hw,
struct clk_rate_request *req)
{
struct clk_master *master = to_clk_master(hw);
struct clk_rate_request req_parent = *req;
const struct clk_master_characteristics *characteristics =
master->characteristics;
struct clk_hw *parent;
long best_rate = LONG_MIN, best_diff = LONG_MIN;
u32 pres;
int i;
if (master->chg_pid < 0)
return -EOPNOTSUPP;
parent = clk_hw_get_parent_by_index(hw, master->chg_pid);
if (!parent)
return -EOPNOTSUPP;
for (i = 0; i <= MASTER_PRES_MAX; i++) {
if (characteristics->have_div3_pres && i == MASTER_PRES_MAX)
pres = 3;
else
pres = 1 << i;
req_parent.rate = req->rate * pres;
if (__clk_determine_rate(parent, &req_parent))
continue;
clk_sama7g5_master_best_diff(req, parent, req_parent.rate,
&best_diff, &best_rate, pres);
if (!best_diff)
break;
}
return 0;
}
static int clk_master_pres_set_rate(struct clk_hw *hw, unsigned long rate,
unsigned long parent_rate)
{
struct clk_master *master = to_clk_master(hw);
unsigned long flags;
unsigned int pres, mckr, tmp;
int ret;
pres = DIV_ROUND_CLOSEST(parent_rate, rate);
if (pres > MASTER_PRES_MAX)
return -EINVAL;
else if (pres == 3)
pres = MASTER_PRES_MAX;
else if (pres)
pres = ffs(pres) - 1;
spin_lock_irqsave(master->lock, flags);
ret = regmap_read(master->regmap, master->layout->offset, &mckr);
if (ret)
goto unlock;
mckr &= master->layout->mask;
tmp = (mckr >> master->layout->pres_shift) & MASTER_PRES_MASK;
if (pres == tmp)
goto unlock;
mckr &= ~(MASTER_PRES_MASK << master->layout->pres_shift);
mckr |= (pres << master->layout->pres_shift);
ret = regmap_write(master->regmap, master->layout->offset, mckr);
if (ret)
goto unlock;
while (!clk_master_ready(master))
cpu_relax();
unlock:
spin_unlock_irqrestore(master->lock, flags);
return ret;
}
static unsigned long clk_master_pres_recalc_rate(struct clk_hw *hw, static unsigned long clk_master_pres_recalc_rate(struct clk_hw *hw,
unsigned long parent_rate) unsigned long parent_rate)
{ {
...@@ -539,13 +460,6 @@ static void clk_master_pres_restore_context(struct clk_hw *hw) ...@@ -539,13 +460,6 @@ static void clk_master_pres_restore_context(struct clk_hw *hw)
pr_warn("MCKR PRES was not configured properly by firmware!\n"); pr_warn("MCKR PRES was not configured properly by firmware!\n");
} }
static void clk_master_pres_restore_context_chg(struct clk_hw *hw)
{
struct clk_master *master = to_clk_master(hw);
clk_master_pres_set_rate(hw, master->pms.rate, master->pms.parent_rate);
}
static const struct clk_ops master_pres_ops = { static const struct clk_ops master_pres_ops = {
.prepare = clk_master_prepare, .prepare = clk_master_prepare,
.is_prepared = clk_master_is_prepared, .is_prepared = clk_master_is_prepared,
...@@ -555,25 +469,13 @@ static const struct clk_ops master_pres_ops = { ...@@ -555,25 +469,13 @@ static const struct clk_ops master_pres_ops = {
.restore_context = clk_master_pres_restore_context, .restore_context = clk_master_pres_restore_context,
}; };
static const struct clk_ops master_pres_ops_chg = {
.prepare = clk_master_prepare,
.is_prepared = clk_master_is_prepared,
.determine_rate = clk_master_pres_determine_rate,
.recalc_rate = clk_master_pres_recalc_rate,
.get_parent = clk_master_pres_get_parent,
.set_rate = clk_master_pres_set_rate,
.save_context = clk_master_pres_save_context,
.restore_context = clk_master_pres_restore_context_chg,
};
static struct clk_hw * __init static struct clk_hw * __init
at91_clk_register_master_internal(struct regmap *regmap, at91_clk_register_master_internal(struct regmap *regmap,
const char *name, int num_parents, const char *name, int num_parents,
const char **parent_names, const char **parent_names,
const struct clk_master_layout *layout, const struct clk_master_layout *layout,
const struct clk_master_characteristics *characteristics, const struct clk_master_characteristics *characteristics,
const struct clk_ops *ops, spinlock_t *lock, u32 flags, const struct clk_ops *ops, spinlock_t *lock, u32 flags)
int chg_pid)
{ {
struct clk_master *master; struct clk_master *master;
struct clk_init_data init; struct clk_init_data init;
...@@ -599,7 +501,6 @@ at91_clk_register_master_internal(struct regmap *regmap, ...@@ -599,7 +501,6 @@ at91_clk_register_master_internal(struct regmap *regmap,
master->layout = layout; master->layout = layout;
master->characteristics = characteristics; master->characteristics = characteristics;
master->regmap = regmap; master->regmap = regmap;
master->chg_pid = chg_pid;
master->lock = lock; master->lock = lock;
if (ops == &master_div_ops_chg) { if (ops == &master_div_ops_chg) {
...@@ -628,19 +529,13 @@ at91_clk_register_master_pres(struct regmap *regmap, ...@@ -628,19 +529,13 @@ at91_clk_register_master_pres(struct regmap *regmap,
const char **parent_names, const char **parent_names,
const struct clk_master_layout *layout, const struct clk_master_layout *layout,
const struct clk_master_characteristics *characteristics, const struct clk_master_characteristics *characteristics,
spinlock_t *lock, u32 flags, int chg_pid) spinlock_t *lock)
{ {
const struct clk_ops *ops;
if (flags & CLK_SET_RATE_GATE)
ops = &master_pres_ops;
else
ops = &master_pres_ops_chg;
return at91_clk_register_master_internal(regmap, name, num_parents, return at91_clk_register_master_internal(regmap, name, num_parents,
parent_names, layout, parent_names, layout,
characteristics, ops, characteristics,
lock, flags, chg_pid); &master_pres_ops,
lock, CLK_SET_RATE_GATE);
} }
struct clk_hw * __init struct clk_hw * __init
...@@ -661,7 +556,7 @@ at91_clk_register_master_div(struct regmap *regmap, ...@@ -661,7 +556,7 @@ at91_clk_register_master_div(struct regmap *regmap,
hw = at91_clk_register_master_internal(regmap, name, 1, hw = at91_clk_register_master_internal(regmap, name, 1,
&parent_name, layout, &parent_name, layout,
characteristics, ops, characteristics, ops,
lock, flags, -EINVAL); lock, flags);
if (!IS_ERR(hw) && safe_div) { if (!IS_ERR(hw) && safe_div) {
master_div = to_clk_master(hw); master_div = to_clk_master(hw);
......
...@@ -392,8 +392,7 @@ of_at91_clk_master_setup(struct device_node *np, ...@@ -392,8 +392,7 @@ of_at91_clk_master_setup(struct device_node *np,
hw = at91_clk_register_master_pres(regmap, "masterck_pres", num_parents, hw = at91_clk_register_master_pres(regmap, "masterck_pres", num_parents,
parent_names, layout, parent_names, layout,
characteristics, &mck_lock, characteristics, &mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto out_free_characteristics; goto out_free_characteristics;
......
...@@ -175,7 +175,7 @@ at91_clk_register_master_pres(struct regmap *regmap, const char *name, ...@@ -175,7 +175,7 @@ at91_clk_register_master_pres(struct regmap *regmap, const char *name,
int num_parents, const char **parent_names, int num_parents, const char **parent_names,
const struct clk_master_layout *layout, const struct clk_master_layout *layout,
const struct clk_master_characteristics *characteristics, const struct clk_master_characteristics *characteristics,
spinlock_t *lock, u32 flags, int chg_pid); spinlock_t *lock);
struct clk_hw * __init struct clk_hw * __init
at91_clk_register_master_div(struct regmap *regmap, const char *name, at91_clk_register_master_div(struct regmap *regmap, const char *name,
......
...@@ -271,8 +271,7 @@ static void __init sam9x60_pmc_setup(struct device_node *np) ...@@ -271,8 +271,7 @@ static void __init sam9x60_pmc_setup(struct device_node *np)
parent_names[2] = "pllack_divck"; parent_names[2] = "pllack_divck";
hw = at91_clk_register_master_pres(regmap, "masterck_pres", 3, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 3,
parent_names, &sam9x60_master_layout, parent_names, &sam9x60_master_layout,
&mck_characteristics, &mck_lock, &mck_characteristics, &mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -168,7 +168,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -168,7 +168,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
if (IS_ERR(regmap)) if (IS_ERR(regmap))
return; return;
sama5d2_pmc = pmc_data_allocate(PMC_AUDIOPLLCK + 1, sama5d2_pmc = pmc_data_allocate(PMC_AUDIOPINCK + 1,
nck(sama5d2_systemck), nck(sama5d2_systemck),
nck(sama5d2_periph32ck), nck(sama5d2_periph32ck),
nck(sama5d2_gck), 3); nck(sama5d2_gck), 3);
...@@ -216,6 +216,8 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -216,6 +216,8 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
sama5d2_pmc->chws[PMC_AUDIOPINCK] = hw;
hw = at91_clk_register_audio_pll_pmc(regmap, "audiopll_pmcck", hw = at91_clk_register_audio_pll_pmc(regmap, "audiopll_pmcck",
"audiopll_fracck"); "audiopll_fracck");
if (IS_ERR(hw)) if (IS_ERR(hw))
...@@ -240,8 +242,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -240,8 +242,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names, parent_names,
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics, &mck_lock, &mck_characteristics, &mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -175,8 +175,7 @@ static void __init sama5d3_pmc_setup(struct device_node *np) ...@@ -175,8 +175,7 @@ static void __init sama5d3_pmc_setup(struct device_node *np)
hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names, parent_names,
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics, &mck_lock, &mck_characteristics, &mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -190,8 +190,7 @@ static void __init sama5d4_pmc_setup(struct device_node *np) ...@@ -190,8 +190,7 @@ static void __init sama5d4_pmc_setup(struct device_node *np)
hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names, parent_names,
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics, &mck_lock, &mck_characteristics, &mck_lock);
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -302,6 +302,7 @@ static const struct { ...@@ -302,6 +302,7 @@ static const struct {
* @ep_count: extra parents count * @ep_count: extra parents count
* @ep_mux_table: mux table for extra parents * @ep_mux_table: mux table for extra parents
* @id: clock id * @id: clock id
* @eid: export index in sama7g5->chws[] array
* @c: true if clock is critical and cannot be disabled * @c: true if clock is critical and cannot be disabled
*/ */
static const struct { static const struct {
...@@ -311,6 +312,7 @@ static const struct { ...@@ -311,6 +312,7 @@ static const struct {
u8 ep_count; u8 ep_count;
u8 ep_mux_table[4]; u8 ep_mux_table[4];
u8 id; u8 id;
u8 eid;
u8 c; u8 c;
} sama7g5_mckx[] = { } sama7g5_mckx[] = {
{ .n = "mck1", { .n = "mck1",
...@@ -319,6 +321,7 @@ static const struct { ...@@ -319,6 +321,7 @@ static const struct {
.ep_mux_table = { 5, }, .ep_mux_table = { 5, },
.ep_count = 1, .ep_count = 1,
.ep_chg_id = INT_MIN, .ep_chg_id = INT_MIN,
.eid = PMC_MCK1,
.c = 1, }, .c = 1, },
{ .n = "mck2", { .n = "mck2",
...@@ -696,16 +699,16 @@ static const struct { ...@@ -696,16 +699,16 @@ static const struct {
{ .n = "pdmc0_gclk", { .n = "pdmc0_gclk",
.id = 68, .id = 68,
.r = { .max = 50000000 }, .r = { .max = 50000000 },
.pp = { "syspll_divpmcck", "baudpll_divpmcck", }, .pp = { "syspll_divpmcck", "audiopll_divpmcck", },
.pp_mux_table = { 5, 8, }, .pp_mux_table = { 5, 9, },
.pp_count = 2, .pp_count = 2,
.pp_chg_id = INT_MIN, }, .pp_chg_id = INT_MIN, },
{ .n = "pdmc1_gclk", { .n = "pdmc1_gclk",
.id = 69, .id = 69,
.r = { .max = 50000000, }, .r = { .max = 50000000, },
.pp = { "syspll_divpmcck", "baudpll_divpmcck", }, .pp = { "syspll_divpmcck", "audiopll_divpmcck", },
.pp_mux_table = { 5, 8, }, .pp_mux_table = { 5, 9, },
.pp_count = 2, .pp_count = 2,
.pp_chg_id = INT_MIN, }, .pp_chg_id = INT_MIN, },
...@@ -913,7 +916,7 @@ static void __init sama7g5_pmc_setup(struct device_node *np) ...@@ -913,7 +916,7 @@ static void __init sama7g5_pmc_setup(struct device_node *np)
if (IS_ERR(regmap)) if (IS_ERR(regmap))
return; return;
sama7g5_pmc = pmc_data_allocate(PMC_CPU + 1, sama7g5_pmc = pmc_data_allocate(PMC_MCK1 + 1,
nck(sama7g5_systemck), nck(sama7g5_systemck),
nck(sama7g5_periphck), nck(sama7g5_periphck),
nck(sama7g5_gck), 8); nck(sama7g5_gck), 8);
...@@ -1027,6 +1030,9 @@ static void __init sama7g5_pmc_setup(struct device_node *np) ...@@ -1027,6 +1030,9 @@ static void __init sama7g5_pmc_setup(struct device_node *np)
goto err_free; goto err_free;
alloc_mem[alloc_mem_size++] = mux_table; alloc_mem[alloc_mem_size++] = mux_table;
if (sama7g5_mckx[i].eid)
sama7g5_pmc->chws[sama7g5_mckx[i].eid] = hw;
} }
hw = at91_clk_sama7g5_register_utmi(regmap, "utmick", "main_xtal"); hw = at91_clk_sama7g5_register_utmi(regmap, "utmick", "main_xtal");
......
...@@ -798,6 +798,15 @@ static unsigned long si5341_output_clk_recalc_rate(struct clk_hw *hw, ...@@ -798,6 +798,15 @@ static unsigned long si5341_output_clk_recalc_rate(struct clk_hw *hw,
u32 r_divider; u32 r_divider;
u8 r[3]; u8 r[3];
err = regmap_read(output->data->regmap,
SI5341_OUT_CONFIG(output), &val);
if (err < 0)
return err;
/* If SI5341_OUT_CFG_RDIV_FORCE2 is set, r_divider is 2 */
if (val & SI5341_OUT_CFG_RDIV_FORCE2)
return parent_rate / 2;
err = regmap_bulk_read(output->data->regmap, err = regmap_bulk_read(output->data->regmap,
SI5341_OUT_R_REG(output), r, 3); SI5341_OUT_R_REG(output), r, 3);
if (err < 0) if (err < 0)
...@@ -814,13 +823,6 @@ static unsigned long si5341_output_clk_recalc_rate(struct clk_hw *hw, ...@@ -814,13 +823,6 @@ static unsigned long si5341_output_clk_recalc_rate(struct clk_hw *hw,
r_divider += 1; r_divider += 1;
r_divider <<= 1; r_divider <<= 1;
err = regmap_read(output->data->regmap,
SI5341_OUT_CONFIG(output), &val);
if (err < 0)
return err;
if (val & SI5341_OUT_CFG_RDIV_FORCE2)
r_divider = 2;
return parent_rate / r_divider; return parent_rate / r_divider;
} }
...@@ -1468,7 +1470,7 @@ static ssize_t input_present_show(struct device *dev, ...@@ -1468,7 +1470,7 @@ static ssize_t input_present_show(struct device *dev,
if (res < 0) if (res < 0)
return res; return res;
res = !(status & SI5341_STATUS_LOSREF); res = !(status & SI5341_STATUS_LOSREF);
return snprintf(buf, PAGE_SIZE, "%d\n", res); return sysfs_emit(buf, "%d\n", res);
} }
static DEVICE_ATTR_RO(input_present); static DEVICE_ATTR_RO(input_present);
...@@ -1483,7 +1485,7 @@ static ssize_t input_present_sticky_show(struct device *dev, ...@@ -1483,7 +1485,7 @@ static ssize_t input_present_sticky_show(struct device *dev,
if (res < 0) if (res < 0)
return res; return res;
res = !(status & SI5341_STATUS_LOSREF); res = !(status & SI5341_STATUS_LOSREF);
return snprintf(buf, PAGE_SIZE, "%d\n", res); return sysfs_emit(buf, "%d\n", res);
} }
static DEVICE_ATTR_RO(input_present_sticky); static DEVICE_ATTR_RO(input_present_sticky);
...@@ -1498,7 +1500,7 @@ static ssize_t pll_locked_show(struct device *dev, ...@@ -1498,7 +1500,7 @@ static ssize_t pll_locked_show(struct device *dev,
if (res < 0) if (res < 0)
return res; return res;
res = !(status & SI5341_STATUS_LOL); res = !(status & SI5341_STATUS_LOL);
return snprintf(buf, PAGE_SIZE, "%d\n", res); return sysfs_emit(buf, "%d\n", res);
} }
static DEVICE_ATTR_RO(pll_locked); static DEVICE_ATTR_RO(pll_locked);
...@@ -1513,7 +1515,7 @@ static ssize_t pll_locked_sticky_show(struct device *dev, ...@@ -1513,7 +1515,7 @@ static ssize_t pll_locked_sticky_show(struct device *dev,
if (res < 0) if (res < 0)
return res; return res;
res = !(status & SI5341_STATUS_LOL); res = !(status & SI5341_STATUS_LOL);
return snprintf(buf, PAGE_SIZE, "%d\n", res); return sysfs_emit(buf, "%d\n", res);
} }
static DEVICE_ATTR_RO(pll_locked_sticky); static DEVICE_ATTR_RO(pll_locked_sticky);
......
...@@ -155,6 +155,10 @@ static const char * const eth_src[] = { ...@@ -155,6 +155,10 @@ static const char * const eth_src[] = {
"pll4_p", "pll3_q" "pll4_p", "pll3_q"
}; };
const struct clk_parent_data ethrx_src[] = {
{ .name = "ethck_k", .fw_name = "ETH_RX_CLK/ETH_REF_CLK" },
};
static const char * const rng_src[] = { static const char * const rng_src[] = {
"ck_csi", "pll4_r", "ck_lse", "ck_lsi" "ck_csi", "pll4_r", "ck_lse", "ck_lsi"
}; };
...@@ -317,6 +321,7 @@ struct clock_config { ...@@ -317,6 +321,7 @@ struct clock_config {
const char *name; const char *name;
const char *parent_name; const char *parent_name;
const char * const *parent_names; const char * const *parent_names;
const struct clk_parent_data *parent_data;
int num_parents; int num_parents;
unsigned long flags; unsigned long flags;
void *cfg; void *cfg;
...@@ -576,6 +581,7 @@ static struct clk_hw * ...@@ -576,6 +581,7 @@ static struct clk_hw *
clk_stm32_register_gate_ops(struct device *dev, clk_stm32_register_gate_ops(struct device *dev,
const char *name, const char *name,
const char *parent_name, const char *parent_name,
const struct clk_parent_data *parent_data,
unsigned long flags, unsigned long flags,
void __iomem *base, void __iomem *base,
const struct stm32_gate_cfg *cfg, const struct stm32_gate_cfg *cfg,
...@@ -586,7 +592,10 @@ clk_stm32_register_gate_ops(struct device *dev, ...@@ -586,7 +592,10 @@ clk_stm32_register_gate_ops(struct device *dev,
int ret; int ret;
init.name = name; init.name = name;
if (parent_name)
init.parent_names = &parent_name; init.parent_names = &parent_name;
if (parent_data)
init.parent_data = parent_data;
init.num_parents = 1; init.num_parents = 1;
init.flags = flags; init.flags = flags;
...@@ -611,6 +620,7 @@ clk_stm32_register_gate_ops(struct device *dev, ...@@ -611,6 +620,7 @@ clk_stm32_register_gate_ops(struct device *dev,
static struct clk_hw * static struct clk_hw *
clk_stm32_register_composite(struct device *dev, clk_stm32_register_composite(struct device *dev,
const char *name, const char * const *parent_names, const char *name, const char * const *parent_names,
const struct clk_parent_data *parent_data,
int num_parents, void __iomem *base, int num_parents, void __iomem *base,
const struct stm32_composite_cfg *cfg, const struct stm32_composite_cfg *cfg,
unsigned long flags, spinlock_t *lock) unsigned long flags, spinlock_t *lock)
...@@ -1135,6 +1145,7 @@ _clk_stm32_register_gate(struct device *dev, ...@@ -1135,6 +1145,7 @@ _clk_stm32_register_gate(struct device *dev,
return clk_stm32_register_gate_ops(dev, return clk_stm32_register_gate_ops(dev,
cfg->name, cfg->name,
cfg->parent_name, cfg->parent_name,
cfg->parent_data,
cfg->flags, cfg->flags,
base, base,
cfg->cfg, cfg->cfg,
...@@ -1148,8 +1159,8 @@ _clk_stm32_register_composite(struct device *dev, ...@@ -1148,8 +1159,8 @@ _clk_stm32_register_composite(struct device *dev,
const struct clock_config *cfg) const struct clock_config *cfg)
{ {
return clk_stm32_register_composite(dev, cfg->name, cfg->parent_names, return clk_stm32_register_composite(dev, cfg->name, cfg->parent_names,
cfg->num_parents, base, cfg->cfg, cfg->parent_data, cfg->num_parents,
cfg->flags, lock); base, cfg->cfg, cfg->flags, lock);
} }
#define GATE(_id, _name, _parent, _flags, _offset, _bit_idx, _gate_flags)\ #define GATE(_id, _name, _parent, _flags, _offset, _bit_idx, _gate_flags)\
...@@ -1258,6 +1269,16 @@ _clk_stm32_register_composite(struct device *dev, ...@@ -1258,6 +1269,16 @@ _clk_stm32_register_composite(struct device *dev,
.func = _clk_stm32_register_gate,\ .func = _clk_stm32_register_gate,\
} }
#define STM32_GATE_PDATA(_id, _name, _parent, _flags, _gate)\
{\
.id = _id,\
.name = _name,\
.parent_data = _parent,\
.flags = _flags,\
.cfg = (struct stm32_gate_cfg *) {_gate},\
.func = _clk_stm32_register_gate,\
}
#define _STM32_GATE(_gate_offset, _gate_bit_idx, _gate_flags, _mgate, _ops)\ #define _STM32_GATE(_gate_offset, _gate_bit_idx, _gate_flags, _mgate, _ops)\
(&(struct stm32_gate_cfg) {\ (&(struct stm32_gate_cfg) {\
&(struct gate_cfg) {\ &(struct gate_cfg) {\
...@@ -1291,6 +1312,10 @@ _clk_stm32_register_composite(struct device *dev, ...@@ -1291,6 +1312,10 @@ _clk_stm32_register_composite(struct device *dev,
STM32_GATE(_id, _name, _parent, _flags,\ STM32_GATE(_id, _name, _parent, _flags,\
_STM32_MGATE(_mgate)) _STM32_MGATE(_mgate))
#define MGATE_MP1_PDATA(_id, _name, _parent, _flags, _mgate)\
STM32_GATE_PDATA(_id, _name, _parent, _flags,\
_STM32_MGATE(_mgate))
#define _STM32_DIV(_div_offset, _div_shift, _div_width,\ #define _STM32_DIV(_div_offset, _div_shift, _div_width,\
_div_flags, _div_table, _ops)\ _div_flags, _div_table, _ops)\
.div = &(struct stm32_div_cfg) {\ .div = &(struct stm32_div_cfg) {\
...@@ -1354,6 +1379,9 @@ _clk_stm32_register_composite(struct device *dev, ...@@ -1354,6 +1379,9 @@ _clk_stm32_register_composite(struct device *dev,
#define PCLK(_id, _name, _parent, _flags, _mgate)\ #define PCLK(_id, _name, _parent, _flags, _mgate)\
MGATE_MP1(_id, _name, _parent, _flags, _mgate) MGATE_MP1(_id, _name, _parent, _flags, _mgate)
#define PCLK_PDATA(_id, _name, _parent, _flags, _mgate)\
MGATE_MP1_PDATA(_id, _name, _parent, _flags, _mgate)
#define KCLK(_id, _name, _parents, _flags, _mgate, _mmux)\ #define KCLK(_id, _name, _parents, _flags, _mgate, _mmux)\
COMPOSITE(_id, _name, _parents, CLK_OPS_PARENT_ENABLE |\ COMPOSITE(_id, _name, _parents, CLK_OPS_PARENT_ENABLE |\
CLK_SET_RATE_NO_REPARENT | _flags,\ CLK_SET_RATE_NO_REPARENT | _flags,\
...@@ -1951,7 +1979,7 @@ static const struct clock_config stm32mp1_clock_cfg[] = { ...@@ -1951,7 +1979,7 @@ static const struct clock_config stm32mp1_clock_cfg[] = {
PCLK(MDMA, "mdma", "ck_axi", 0, G_MDMA), PCLK(MDMA, "mdma", "ck_axi", 0, G_MDMA),
PCLK(GPU, "gpu", "ck_axi", 0, G_GPU), PCLK(GPU, "gpu", "ck_axi", 0, G_GPU),
PCLK(ETHTX, "ethtx", "ck_axi", 0, G_ETHTX), PCLK(ETHTX, "ethtx", "ck_axi", 0, G_ETHTX),
PCLK(ETHRX, "ethrx", "ck_axi", 0, G_ETHRX), PCLK_PDATA(ETHRX, "ethrx", ethrx_src, 0, G_ETHRX),
PCLK(ETHMAC, "ethmac", "ck_axi", 0, G_ETHMAC), PCLK(ETHMAC, "ethmac", "ck_axi", 0, G_ETHMAC),
PCLK(FMC, "fmc", "ck_axi", CLK_IGNORE_UNUSED, G_FMC), PCLK(FMC, "fmc", "ck_axi", CLK_IGNORE_UNUSED, G_FMC),
PCLK(QSPI, "qspi", "ck_axi", CLK_IGNORE_UNUSED, G_QSPI), PCLK(QSPI, "qspi", "ck_axi", CLK_IGNORE_UNUSED, G_QSPI),
...@@ -2008,7 +2036,6 @@ static const struct clock_config stm32mp1_clock_cfg[] = { ...@@ -2008,7 +2036,6 @@ static const struct clock_config stm32mp1_clock_cfg[] = {
KCLK(DSI_K, "dsi_k", dsi_src, 0, G_DSI, M_DSI), KCLK(DSI_K, "dsi_k", dsi_src, 0, G_DSI, M_DSI),
KCLK(ADFSDM_K, "adfsdm_k", sai_src, 0, G_ADFSDM, M_SAI1), KCLK(ADFSDM_K, "adfsdm_k", sai_src, 0, G_ADFSDM, M_SAI1),
KCLK(USBO_K, "usbo_k", usbo_src, 0, G_USBO, M_USBO), KCLK(USBO_K, "usbo_k", usbo_src, 0, G_USBO, M_USBO),
KCLK(ETHCK_K, "ethck_k", eth_src, 0, G_ETHCK, M_ETHCK),
/* Particulary Kernel Clocks (no mux or no gate) */ /* Particulary Kernel Clocks (no mux or no gate) */
MGATE_MP1(DFSDM_K, "dfsdm_k", "ck_mcu", 0, G_DFSDM), MGATE_MP1(DFSDM_K, "dfsdm_k", "ck_mcu", 0, G_DFSDM),
...@@ -2017,11 +2044,16 @@ static const struct clock_config stm32mp1_clock_cfg[] = { ...@@ -2017,11 +2044,16 @@ static const struct clock_config stm32mp1_clock_cfg[] = {
MGATE_MP1(GPU_K, "gpu_k", "pll2_q", 0, G_GPU), MGATE_MP1(GPU_K, "gpu_k", "pll2_q", 0, G_GPU),
MGATE_MP1(DAC12_K, "dac12_k", "ck_lsi", 0, G_DAC12), MGATE_MP1(DAC12_K, "dac12_k", "ck_lsi", 0, G_DAC12),
COMPOSITE(ETHPTP_K, "ethptp_k", eth_src, CLK_OPS_PARENT_ENABLE | COMPOSITE(NO_ID, "ck_ker_eth", eth_src, CLK_OPS_PARENT_ENABLE |
CLK_SET_RATE_NO_REPARENT, CLK_SET_RATE_NO_REPARENT,
_NO_GATE, _NO_GATE,
_MMUX(M_ETHCK), _MMUX(M_ETHCK),
_DIV(RCC_ETHCKSELR, 4, 4, 0, NULL)), _NO_DIV),
MGATE_MP1(ETHCK_K, "ethck_k", "ck_ker_eth", 0, G_ETHCK),
DIV(ETHPTP_K, "ethptp_k", "ck_ker_eth", CLK_OPS_PARENT_ENABLE |
CLK_SET_RATE_NO_REPARENT, RCC_ETHCKSELR, 4, 4, 0),
/* RTC clock */ /* RTC clock */
COMPOSITE(RTC, "ck_rtc", rtc_src, CLK_OPS_PARENT_ENABLE, COMPOSITE(RTC, "ck_rtc", rtc_src, CLK_OPS_PARENT_ENABLE,
......
...@@ -92,7 +92,7 @@ struct clk * __init mtk_clk_register_ref2usb_tx(const char *name, ...@@ -92,7 +92,7 @@ struct clk * __init mtk_clk_register_ref2usb_tx(const char *name,
clk = clk_register(NULL, &tx->hw); clk = clk_register(NULL, &tx->hw);
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
pr_err("Failed to register clk %s: %ld\n", name, PTR_ERR(clk)); pr_err("Failed to register clk %s: %pe\n", name, clk);
kfree(tx); kfree(tx);
} }
......
...@@ -5,13 +5,24 @@ ...@@ -5,13 +5,24 @@
*/ */
#include <linux/clk-provider.h> #include <linux/clk-provider.h>
#include <linux/container_of.h>
#include <linux/err.h>
#include <linux/mfd/syscon.h> #include <linux/mfd/syscon.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/regmap.h>
#include <linux/slab.h> #include <linux/slab.h>
#include "clk-mtk.h" #include "clk-mtk.h"
#include "clk-cpumux.h" #include "clk-cpumux.h"
struct mtk_clk_cpumux {
struct clk_hw hw;
struct regmap *regmap;
u32 reg;
u32 mask;
u8 shift;
};
static inline struct mtk_clk_cpumux *to_mtk_clk_cpumux(struct clk_hw *_hw) static inline struct mtk_clk_cpumux *to_mtk_clk_cpumux(struct clk_hw *_hw)
{ {
return container_of(_hw, struct mtk_clk_cpumux, hw); return container_of(_hw, struct mtk_clk_cpumux, hw);
...@@ -77,6 +88,21 @@ mtk_clk_register_cpumux(const struct mtk_composite *mux, ...@@ -77,6 +88,21 @@ mtk_clk_register_cpumux(const struct mtk_composite *mux,
return clk; return clk;
} }
static void mtk_clk_unregister_cpumux(struct clk *clk)
{
struct mtk_clk_cpumux *cpumux;
struct clk_hw *hw;
hw = __clk_get_hw(clk);
if (!hw)
return;
cpumux = to_mtk_clk_cpumux(hw);
clk_unregister(clk);
kfree(cpumux);
}
int mtk_clk_register_cpumuxes(struct device_node *node, int mtk_clk_register_cpumuxes(struct device_node *node,
const struct mtk_composite *clks, int num, const struct mtk_composite *clks, int num,
struct clk_onecell_data *clk_data) struct clk_onecell_data *clk_data)
...@@ -87,25 +113,58 @@ int mtk_clk_register_cpumuxes(struct device_node *node, ...@@ -87,25 +113,58 @@ int mtk_clk_register_cpumuxes(struct device_node *node,
regmap = device_node_to_regmap(node); regmap = device_node_to_regmap(node);
if (IS_ERR(regmap)) { if (IS_ERR(regmap)) {
pr_err("Cannot find regmap for %pOF: %ld\n", node, pr_err("Cannot find regmap for %pOF: %pe\n", node, regmap);
PTR_ERR(regmap));
return PTR_ERR(regmap); return PTR_ERR(regmap);
} }
for (i = 0; i < num; i++) { for (i = 0; i < num; i++) {
const struct mtk_composite *mux = &clks[i]; const struct mtk_composite *mux = &clks[i];
if (!IS_ERR_OR_NULL(clk_data->clks[mux->id])) {
pr_warn("%pOF: Trying to register duplicate clock ID: %d\n",
node, mux->id);
continue;
}
clk = mtk_clk_register_cpumux(mux, regmap); clk = mtk_clk_register_cpumux(mux, regmap);
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
pr_err("Failed to register clk %s: %ld\n", pr_err("Failed to register clk %s: %pe\n", mux->name, clk);
mux->name, PTR_ERR(clk)); goto err;
continue;
} }
clk_data->clks[mux->id] = clk; clk_data->clks[mux->id] = clk;
} }
return 0; return 0;
err:
while (--i >= 0) {
const struct mtk_composite *mux = &clks[i];
if (IS_ERR_OR_NULL(clk_data->clks[mux->id]))
continue;
mtk_clk_unregister_cpumux(clk_data->clks[mux->id]);
clk_data->clks[mux->id] = ERR_PTR(-ENOENT);
}
return PTR_ERR(clk);
}
void mtk_clk_unregister_cpumuxes(const struct mtk_composite *clks, int num,
struct clk_onecell_data *clk_data)
{
int i;
for (i = num; i > 0; i--) {
const struct mtk_composite *mux = &clks[i - 1];
if (IS_ERR_OR_NULL(clk_data->clks[mux->id]))
continue;
mtk_clk_unregister_cpumux(clk_data->clks[mux->id]);
clk_data->clks[mux->id] = ERR_PTR(-ENOENT);
}
} }
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
...@@ -7,16 +7,15 @@ ...@@ -7,16 +7,15 @@
#ifndef __DRV_CLK_CPUMUX_H #ifndef __DRV_CLK_CPUMUX_H
#define __DRV_CLK_CPUMUX_H #define __DRV_CLK_CPUMUX_H
struct mtk_clk_cpumux { struct clk_onecell_data;
struct clk_hw hw; struct device_node;
struct regmap *regmap; struct mtk_composite;
u32 reg;
u32 mask;
u8 shift;
};
int mtk_clk_register_cpumuxes(struct device_node *node, int mtk_clk_register_cpumuxes(struct device_node *node,
const struct mtk_composite *clks, int num, const struct mtk_composite *clks, int num,
struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
void mtk_clk_unregister_cpumuxes(const struct mtk_composite *clks, int num,
struct clk_onecell_data *clk_data);
#endif /* __DRV_CLK_CPUMUX_H */ #endif /* __DRV_CLK_CPUMUX_H */
...@@ -4,18 +4,30 @@ ...@@ -4,18 +4,30 @@
* Author: James Liao <jamesjj.liao@mediatek.com> * Author: James Liao <jamesjj.liao@mediatek.com>
*/ */
#include <linux/of.h> #include <linux/clk-provider.h>
#include <linux/of_address.h> #include <linux/mfd/syscon.h>
#include <linux/io.h>
#include <linux/slab.h>
#include <linux/delay.h>
#include <linux/clkdev.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/printk.h>
#include <linux/regmap.h>
#include <linux/slab.h>
#include <linux/types.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
struct mtk_clk_gate {
struct clk_hw hw;
struct regmap *regmap;
int set_ofs;
int clr_ofs;
int sta_ofs;
u8 bit;
};
static inline struct mtk_clk_gate *to_mtk_clk_gate(struct clk_hw *hw)
{
return container_of(hw, struct mtk_clk_gate, hw);
}
static u32 mtk_get_clockgating(struct clk_hw *hw) static u32 mtk_get_clockgating(struct clk_hw *hw)
{ {
struct mtk_clk_gate *cg = to_mtk_clk_gate(hw); struct mtk_clk_gate *cg = to_mtk_clk_gate(hw);
...@@ -140,17 +152,12 @@ const struct clk_ops mtk_clk_gate_ops_no_setclr_inv = { ...@@ -140,17 +152,12 @@ const struct clk_ops mtk_clk_gate_ops_no_setclr_inv = {
}; };
EXPORT_SYMBOL_GPL(mtk_clk_gate_ops_no_setclr_inv); EXPORT_SYMBOL_GPL(mtk_clk_gate_ops_no_setclr_inv);
struct clk *mtk_clk_register_gate( static struct clk *mtk_clk_register_gate(const char *name,
const char *name,
const char *parent_name, const char *parent_name,
struct regmap *regmap, struct regmap *regmap, int set_ofs,
int set_ofs, int clr_ofs, int sta_ofs, u8 bit,
int clr_ofs,
int sta_ofs,
u8 bit,
const struct clk_ops *ops, const struct clk_ops *ops,
unsigned long flags, unsigned long flags, struct device *dev)
struct device *dev)
{ {
struct mtk_clk_gate *cg; struct mtk_clk_gate *cg;
struct clk *clk; struct clk *clk;
...@@ -180,6 +187,107 @@ struct clk *mtk_clk_register_gate( ...@@ -180,6 +187,107 @@ struct clk *mtk_clk_register_gate(
return clk; return clk;
} }
EXPORT_SYMBOL_GPL(mtk_clk_register_gate);
static void mtk_clk_unregister_gate(struct clk *clk)
{
struct mtk_clk_gate *cg;
struct clk_hw *hw;
hw = __clk_get_hw(clk);
if (!hw)
return;
cg = to_mtk_clk_gate(hw);
clk_unregister(clk);
kfree(cg);
}
int mtk_clk_register_gates_with_dev(struct device_node *node,
const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data,
struct device *dev)
{
int i;
struct clk *clk;
struct regmap *regmap;
if (!clk_data)
return -ENOMEM;
regmap = device_node_to_regmap(node);
if (IS_ERR(regmap)) {
pr_err("Cannot find regmap for %pOF: %pe\n", node, regmap);
return PTR_ERR(regmap);
}
for (i = 0; i < num; i++) {
const struct mtk_gate *gate = &clks[i];
if (!IS_ERR_OR_NULL(clk_data->clks[gate->id])) {
pr_warn("%pOF: Trying to register duplicate clock ID: %d\n",
node, gate->id);
continue;
}
clk = mtk_clk_register_gate(gate->name, gate->parent_name,
regmap,
gate->regs->set_ofs,
gate->regs->clr_ofs,
gate->regs->sta_ofs,
gate->shift, gate->ops,
gate->flags, dev);
if (IS_ERR(clk)) {
pr_err("Failed to register clk %s: %pe\n", gate->name, clk);
goto err;
}
clk_data->clks[gate->id] = clk;
}
return 0;
err:
while (--i >= 0) {
const struct mtk_gate *gate = &clks[i];
if (IS_ERR_OR_NULL(clk_data->clks[gate->id]))
continue;
mtk_clk_unregister_gate(clk_data->clks[gate->id]);
clk_data->clks[gate->id] = ERR_PTR(-ENOENT);
}
return PTR_ERR(clk);
}
int mtk_clk_register_gates(struct device_node *node,
const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data)
{
return mtk_clk_register_gates_with_dev(node, clks, num, clk_data, NULL);
}
EXPORT_SYMBOL_GPL(mtk_clk_register_gates);
void mtk_clk_unregister_gates(const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data)
{
int i;
if (!clk_data)
return;
for (i = num; i > 0; i--) {
const struct mtk_gate *gate = &clks[i - 1];
if (IS_ERR_OR_NULL(clk_data->clks[gate->id]))
continue;
mtk_clk_unregister_gate(clk_data->clks[gate->id]);
clk_data->clks[gate->id] = ERR_PTR(-ENOENT);
}
}
EXPORT_SYMBOL_GPL(mtk_clk_unregister_gates);
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
...@@ -7,41 +7,34 @@ ...@@ -7,41 +7,34 @@
#ifndef __DRV_CLK_GATE_H #ifndef __DRV_CLK_GATE_H
#define __DRV_CLK_GATE_H #define __DRV_CLK_GATE_H
#include <linux/regmap.h> #include <linux/types.h>
#include <linux/clk-provider.h>
struct clk; struct clk;
struct clk_onecell_data;
struct mtk_clk_gate { struct clk_ops;
struct clk_hw hw; struct device;
struct regmap *regmap; struct device_node;
int set_ofs;
int clr_ofs;
int sta_ofs;
u8 bit;
};
static inline struct mtk_clk_gate *to_mtk_clk_gate(struct clk_hw *hw)
{
return container_of(hw, struct mtk_clk_gate, hw);
}
extern const struct clk_ops mtk_clk_gate_ops_setclr; extern const struct clk_ops mtk_clk_gate_ops_setclr;
extern const struct clk_ops mtk_clk_gate_ops_setclr_inv; extern const struct clk_ops mtk_clk_gate_ops_setclr_inv;
extern const struct clk_ops mtk_clk_gate_ops_no_setclr; extern const struct clk_ops mtk_clk_gate_ops_no_setclr;
extern const struct clk_ops mtk_clk_gate_ops_no_setclr_inv; extern const struct clk_ops mtk_clk_gate_ops_no_setclr_inv;
struct clk *mtk_clk_register_gate( struct mtk_gate_regs {
const char *name, u32 sta_ofs;
const char *parent_name, u32 clr_ofs;
struct regmap *regmap, u32 set_ofs;
int set_ofs, };
int clr_ofs,
int sta_ofs, struct mtk_gate {
u8 bit, int id;
const struct clk_ops *ops, const char *name;
unsigned long flags, const char *parent_name;
struct device *dev); const struct mtk_gate_regs *regs;
int shift;
const struct clk_ops *ops;
unsigned long flags;
};
#define GATE_MTK_FLAGS(_id, _name, _parent, _regs, _shift, \ #define GATE_MTK_FLAGS(_id, _name, _parent, _regs, _shift, \
_ops, _flags) { \ _ops, _flags) { \
...@@ -57,4 +50,16 @@ struct clk *mtk_clk_register_gate( ...@@ -57,4 +50,16 @@ struct clk *mtk_clk_register_gate(
#define GATE_MTK(_id, _name, _parent, _regs, _shift, _ops) \ #define GATE_MTK(_id, _name, _parent, _regs, _shift, _ops) \
GATE_MTK_FLAGS(_id, _name, _parent, _regs, _shift, _ops, 0) GATE_MTK_FLAGS(_id, _name, _parent, _regs, _shift, _ops, 0)
int mtk_clk_register_gates(struct device_node *node,
const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data);
int mtk_clk_register_gates_with_dev(struct device_node *node,
const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data,
struct device *dev);
void mtk_clk_unregister_gates(const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data);
#endif /* __DRV_CLK_GATE_H */ #endif /* __DRV_CLK_GATE_H */
...@@ -10,9 +10,10 @@ ...@@ -10,9 +10,10 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-mtk.h"
#include "clk-gate.h"
#include "clk-cpumux.h" #include "clk-cpumux.h"
#include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt2701-clk.h> #include <dt-bindings/clock/mt2701-clk.h>
......
...@@ -13,8 +13,9 @@ ...@@ -13,8 +13,9 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/slab.h> #include <linux/slab.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-pll.h"
#include "clk-mtk.h"
#include <dt-bindings/clock/mt2712-clk.h> #include <dt-bindings/clock/mt2712-clk.h>
......
...@@ -12,9 +12,10 @@ ...@@ -12,9 +12,10 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-mux.h" #include "clk-mux.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt6765-clk.h> #include <dt-bindings/clock/mt6765-clk.h>
......
...@@ -10,9 +10,10 @@ ...@@ -10,9 +10,10 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-gate.h"
#include "clk-mtk.h" #include "clk-mtk.h"
#include "clk-mux.h" #include "clk-mux.h"
#include "clk-gate.h" #include "clk-pll.h"
#include <dt-bindings/clock/mt6779-clk.h> #include <dt-bindings/clock/mt6779-clk.h>
......
...@@ -9,8 +9,9 @@ ...@@ -9,8 +9,9 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt6797-clk.h> #include <dt-bindings/clock/mt6797-clk.h>
......
...@@ -11,9 +11,10 @@ ...@@ -11,9 +11,10 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-mtk.h"
#include "clk-gate.h"
#include "clk-cpumux.h" #include "clk-cpumux.h"
#include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt7622-clk.h> #include <dt-bindings/clock/mt7622-clk.h>
#include <linux/clk.h> /* for consumer */ #include <linux/clk.h> /* for consumer */
......
...@@ -12,9 +12,10 @@ ...@@ -12,9 +12,10 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-mtk.h"
#include "clk-gate.h"
#include "clk-cpumux.h" #include "clk-cpumux.h"
#include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt7629-clk.h> #include <dt-bindings/clock/mt7629-clk.h>
......
...@@ -10,9 +10,11 @@ ...@@ -10,9 +10,11 @@
#include <linux/of_address.h> #include <linux/of_address.h>
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-mux.h" #include "clk-mux.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt7986-clk.h> #include <dt-bindings/clock/mt7986-clk.h>
#include <linux/clk.h> #include <linux/clk.h>
......
...@@ -11,8 +11,9 @@ ...@@ -11,8 +11,9 @@
#include <linux/mfd/syscon.h> #include <linux/mfd/syscon.h>
#include <dt-bindings/clock/mt8135-clk.h> #include <dt-bindings/clock/mt8135-clk.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
static DEFINE_SPINLOCK(mt8135_clk_lock); static DEFINE_SPINLOCK(mt8135_clk_lock);
......
...@@ -12,8 +12,9 @@ ...@@ -12,8 +12,9 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/mfd/syscon.h> #include <linux/mfd/syscon.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt8167-clk.h> #include <dt-bindings/clock/mt8167-clk.h>
......
...@@ -8,9 +8,10 @@ ...@@ -8,9 +8,10 @@
#include <linux/of.h> #include <linux/of.h>
#include <linux/of_address.h> #include <linux/of_address.h>
#include "clk-mtk.h"
#include "clk-gate.h"
#include "clk-cpumux.h" #include "clk-cpumux.h"
#include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt8173-clk.h> #include <dt-bindings/clock/mt8173-clk.h>
......
...@@ -11,9 +11,10 @@ ...@@ -11,9 +11,10 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/slab.h> #include <linux/slab.h>
#include "clk-gate.h"
#include "clk-mtk.h" #include "clk-mtk.h"
#include "clk-mux.h" #include "clk-mux.h"
#include "clk-gate.h" #include "clk-pll.h"
#include <dt-bindings/clock/mt8183-clk.h> #include <dt-bindings/clock/mt8183-clk.h>
......
...@@ -12,9 +12,10 @@ ...@@ -12,9 +12,10 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/slab.h> #include <linux/slab.h>
#include "clk-gate.h"
#include "clk-mtk.h" #include "clk-mtk.h"
#include "clk-mux.h" #include "clk-mux.h"
#include "clk-gate.h" #include "clk-pll.h"
#include <dt-bindings/clock/mt8192-clk.h> #include <dt-bindings/clock/mt8192-clk.h>
...@@ -1236,9 +1237,17 @@ static int clk_mt8192_infra_probe(struct platform_device *pdev) ...@@ -1236,9 +1237,17 @@ static int clk_mt8192_infra_probe(struct platform_device *pdev)
r = mtk_clk_register_gates(node, infra_clks, ARRAY_SIZE(infra_clks), clk_data); r = mtk_clk_register_gates(node, infra_clks, ARRAY_SIZE(infra_clks), clk_data);
if (r) if (r)
goto free_clk_data;
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r)
goto free_clk_data;
return r; return r;
return of_clk_add_provider(node, of_clk_src_onecell_get, clk_data); free_clk_data:
mtk_free_clk_data(clk_data);
return r;
} }
static int clk_mt8192_peri_probe(struct platform_device *pdev) static int clk_mt8192_peri_probe(struct platform_device *pdev)
...@@ -1253,9 +1262,17 @@ static int clk_mt8192_peri_probe(struct platform_device *pdev) ...@@ -1253,9 +1262,17 @@ static int clk_mt8192_peri_probe(struct platform_device *pdev)
r = mtk_clk_register_gates(node, peri_clks, ARRAY_SIZE(peri_clks), clk_data); r = mtk_clk_register_gates(node, peri_clks, ARRAY_SIZE(peri_clks), clk_data);
if (r) if (r)
goto free_clk_data;
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r)
goto free_clk_data;
return r; return r;
return of_clk_add_provider(node, of_clk_src_onecell_get, clk_data); free_clk_data:
mtk_free_clk_data(clk_data);
return r;
} }
static int clk_mt8192_apmixed_probe(struct platform_device *pdev) static int clk_mt8192_apmixed_probe(struct platform_device *pdev)
...@@ -1271,9 +1288,17 @@ static int clk_mt8192_apmixed_probe(struct platform_device *pdev) ...@@ -1271,9 +1288,17 @@ static int clk_mt8192_apmixed_probe(struct platform_device *pdev)
mtk_clk_register_plls(node, plls, ARRAY_SIZE(plls), clk_data); mtk_clk_register_plls(node, plls, ARRAY_SIZE(plls), clk_data);
r = mtk_clk_register_gates(node, apmixed_clks, ARRAY_SIZE(apmixed_clks), clk_data); r = mtk_clk_register_gates(node, apmixed_clks, ARRAY_SIZE(apmixed_clks), clk_data);
if (r) if (r)
goto free_clk_data;
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r)
goto free_clk_data;
return r; return r;
return of_clk_add_provider(node, of_clk_src_onecell_get, clk_data); free_clk_data:
mtk_free_clk_data(clk_data);
return r;
} }
static const struct of_device_id of_match_clk_mt8192[] = { static const struct of_device_id of_match_clk_mt8192[] = {
......
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h" #include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt8195-clk.h> #include <dt-bindings/clock/mt8195-clk.h>
#include <linux/of_device.h> #include <linux/of_device.h>
...@@ -119,24 +120,47 @@ static int clk_mt8195_apmixed_probe(struct platform_device *pdev) ...@@ -119,24 +120,47 @@ static int clk_mt8195_apmixed_probe(struct platform_device *pdev)
if (!clk_data) if (!clk_data)
return -ENOMEM; return -ENOMEM;
mtk_clk_register_plls(node, plls, ARRAY_SIZE(plls), clk_data); r = mtk_clk_register_plls(node, plls, ARRAY_SIZE(plls), clk_data);
r = mtk_clk_register_gates(node, apmixed_clks, ARRAY_SIZE(apmixed_clks), clk_data);
if (r) if (r)
goto free_apmixed_data; goto free_apmixed_data;
r = mtk_clk_register_gates(node, apmixed_clks, ARRAY_SIZE(apmixed_clks), clk_data);
if (r)
goto unregister_plls;
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data); r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r) if (r)
goto free_apmixed_data; goto unregister_gates;
platform_set_drvdata(pdev, clk_data);
return r; return r;
unregister_gates:
mtk_clk_unregister_gates(apmixed_clks, ARRAY_SIZE(apmixed_clks), clk_data);
unregister_plls:
mtk_clk_unregister_plls(plls, ARRAY_SIZE(plls), clk_data);
free_apmixed_data: free_apmixed_data:
mtk_free_clk_data(clk_data); mtk_free_clk_data(clk_data);
return r; return r;
} }
static int clk_mt8195_apmixed_remove(struct platform_device *pdev)
{
struct device_node *node = pdev->dev.of_node;
struct clk_onecell_data *clk_data = platform_get_drvdata(pdev);
of_clk_del_provider(node);
mtk_clk_unregister_gates(apmixed_clks, ARRAY_SIZE(apmixed_clks), clk_data);
mtk_clk_unregister_plls(plls, ARRAY_SIZE(plls), clk_data);
mtk_free_clk_data(clk_data);
return 0;
}
static struct platform_driver clk_mt8195_apmixed_drv = { static struct platform_driver clk_mt8195_apmixed_drv = {
.probe = clk_mt8195_apmixed_probe, .probe = clk_mt8195_apmixed_probe,
.remove = clk_mt8195_apmixed_remove,
.driver = { .driver = {
.name = "clk-mt8195-apmixed", .name = "clk-mt8195-apmixed",
.of_match_table = of_match_clk_mt8195_apmixed, .of_match_table = of_match_clk_mt8195_apmixed,
......
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
// Author: Chun-Jie Chen <chun-jie.chen@mediatek.com> // Author: Chun-Jie Chen <chun-jie.chen@mediatek.com>
#include "clk-mtk.h" #include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt8195-clk.h> #include <dt-bindings/clock/mt8195-clk.h>
#include <linux/clk-provider.h> #include <linux/clk-provider.h>
...@@ -65,18 +66,37 @@ static int clk_mt8195_apusys_pll_probe(struct platform_device *pdev) ...@@ -65,18 +66,37 @@ static int clk_mt8195_apusys_pll_probe(struct platform_device *pdev)
if (!clk_data) if (!clk_data)
return -ENOMEM; return -ENOMEM;
mtk_clk_register_plls(node, apusys_plls, ARRAY_SIZE(apusys_plls), clk_data); r = mtk_clk_register_plls(node, apusys_plls, ARRAY_SIZE(apusys_plls), clk_data);
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r) if (r)
goto free_apusys_pll_data; goto free_apusys_pll_data;
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r)
goto unregister_plls;
platform_set_drvdata(pdev, clk_data);
return r; return r;
unregister_plls:
mtk_clk_unregister_plls(apusys_plls, ARRAY_SIZE(apusys_plls), clk_data);
free_apusys_pll_data: free_apusys_pll_data:
mtk_free_clk_data(clk_data); mtk_free_clk_data(clk_data);
return r; return r;
} }
static int clk_mt8195_apusys_pll_remove(struct platform_device *pdev)
{
struct clk_onecell_data *clk_data = platform_get_drvdata(pdev);
struct device_node *node = pdev->dev.of_node;
of_clk_del_provider(node);
mtk_clk_unregister_plls(apusys_plls, ARRAY_SIZE(apusys_plls), clk_data);
mtk_free_clk_data(clk_data);
return 0;
}
static const struct of_device_id of_match_clk_mt8195_apusys_pll[] = { static const struct of_device_id of_match_clk_mt8195_apusys_pll[] = {
{ .compatible = "mediatek,mt8195-apusys_pll", }, { .compatible = "mediatek,mt8195-apusys_pll", },
{} {}
...@@ -84,6 +104,7 @@ static const struct of_device_id of_match_clk_mt8195_apusys_pll[] = { ...@@ -84,6 +104,7 @@ static const struct of_device_id of_match_clk_mt8195_apusys_pll[] = {
static struct platform_driver clk_mt8195_apusys_pll_drv = { static struct platform_driver clk_mt8195_apusys_pll_drv = {
.probe = clk_mt8195_apusys_pll_probe, .probe = clk_mt8195_apusys_pll_probe,
.remove = clk_mt8195_apusys_pll_remove,
.driver = { .driver = {
.name = "clk-mt8195-apusys_pll", .name = "clk-mt8195-apusys_pll",
.of_match_table = of_match_clk_mt8195_apusys_pll, .of_match_table = of_match_clk_mt8195_apusys_pll,
......
...@@ -134,6 +134,7 @@ static const struct of_device_id of_match_clk_mt8195_cam[] = { ...@@ -134,6 +134,7 @@ static const struct of_device_id of_match_clk_mt8195_cam[] = {
static struct platform_driver clk_mt8195_cam_drv = { static struct platform_driver clk_mt8195_cam_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-cam", .name = "clk-mt8195-cam",
.of_match_table = of_match_clk_mt8195_cam, .of_match_table = of_match_clk_mt8195_cam,
......
...@@ -42,6 +42,7 @@ static const struct of_device_id of_match_clk_mt8195_ccu[] = { ...@@ -42,6 +42,7 @@ static const struct of_device_id of_match_clk_mt8195_ccu[] = {
static struct platform_driver clk_mt8195_ccu_drv = { static struct platform_driver clk_mt8195_ccu_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-ccu", .name = "clk-mt8195-ccu",
.of_match_table = of_match_clk_mt8195_ccu, .of_match_table = of_match_clk_mt8195_ccu,
......
...@@ -88,6 +88,7 @@ static const struct of_device_id of_match_clk_mt8195_img[] = { ...@@ -88,6 +88,7 @@ static const struct of_device_id of_match_clk_mt8195_img[] = {
static struct platform_driver clk_mt8195_img_drv = { static struct platform_driver clk_mt8195_img_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-img", .name = "clk-mt8195-img",
.of_match_table = of_match_clk_mt8195_img, .of_match_table = of_match_clk_mt8195_img,
......
...@@ -58,6 +58,7 @@ static const struct of_device_id of_match_clk_mt8195_imp_iic_wrap[] = { ...@@ -58,6 +58,7 @@ static const struct of_device_id of_match_clk_mt8195_imp_iic_wrap[] = {
static struct platform_driver clk_mt8195_imp_iic_wrap_drv = { static struct platform_driver clk_mt8195_imp_iic_wrap_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-imp_iic_wrap", .name = "clk-mt8195-imp_iic_wrap",
.of_match_table = of_match_clk_mt8195_imp_iic_wrap, .of_match_table = of_match_clk_mt8195_imp_iic_wrap,
......
...@@ -198,6 +198,7 @@ static const struct of_device_id of_match_clk_mt8195_infra_ao[] = { ...@@ -198,6 +198,7 @@ static const struct of_device_id of_match_clk_mt8195_infra_ao[] = {
static struct platform_driver clk_mt8195_infra_ao_drv = { static struct platform_driver clk_mt8195_infra_ao_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-infra_ao", .name = "clk-mt8195-infra_ao",
.of_match_table = of_match_clk_mt8195_infra_ao, .of_match_table = of_match_clk_mt8195_infra_ao,
......
...@@ -43,6 +43,7 @@ static const struct of_device_id of_match_clk_mt8195_ipe[] = { ...@@ -43,6 +43,7 @@ static const struct of_device_id of_match_clk_mt8195_ipe[] = {
static struct platform_driver clk_mt8195_ipe_drv = { static struct platform_driver clk_mt8195_ipe_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-ipe", .name = "clk-mt8195-ipe",
.of_match_table = of_match_clk_mt8195_ipe, .of_match_table = of_match_clk_mt8195_ipe,
......
...@@ -39,6 +39,7 @@ static const struct of_device_id of_match_clk_mt8195_mfg[] = { ...@@ -39,6 +39,7 @@ static const struct of_device_id of_match_clk_mt8195_mfg[] = {
static struct platform_driver clk_mt8195_mfg_drv = { static struct platform_driver clk_mt8195_mfg_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-mfg", .name = "clk-mt8195-mfg",
.of_match_table = of_match_clk_mt8195_mfg, .of_match_table = of_match_clk_mt8195_mfg,
......
...@@ -54,6 +54,7 @@ static const struct of_device_id of_match_clk_mt8195_peri_ao[] = { ...@@ -54,6 +54,7 @@ static const struct of_device_id of_match_clk_mt8195_peri_ao[] = {
static struct platform_driver clk_mt8195_peri_ao_drv = { static struct platform_driver clk_mt8195_peri_ao_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-peri_ao", .name = "clk-mt8195-peri_ao",
.of_match_table = of_match_clk_mt8195_peri_ao, .of_match_table = of_match_clk_mt8195_peri_ao,
......
...@@ -39,6 +39,7 @@ static const struct of_device_id of_match_clk_mt8195_scp_adsp[] = { ...@@ -39,6 +39,7 @@ static const struct of_device_id of_match_clk_mt8195_scp_adsp[] = {
static struct platform_driver clk_mt8195_scp_adsp_drv = { static struct platform_driver clk_mt8195_scp_adsp_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-scp_adsp", .name = "clk-mt8195-scp_adsp",
.of_match_table = of_match_clk_mt8195_scp_adsp, .of_match_table = of_match_clk_mt8195_scp_adsp,
......
...@@ -1239,32 +1239,79 @@ static int clk_mt8195_topck_probe(struct platform_device *pdev) ...@@ -1239,32 +1239,79 @@ static int clk_mt8195_topck_probe(struct platform_device *pdev)
goto free_top_data; goto free_top_data;
} }
mtk_clk_register_fixed_clks(top_fixed_clks, ARRAY_SIZE(top_fixed_clks), r = mtk_clk_register_fixed_clks(top_fixed_clks, ARRAY_SIZE(top_fixed_clks),
top_clk_data); top_clk_data);
mtk_clk_register_factors(top_divs, ARRAY_SIZE(top_divs), top_clk_data); if (r)
mtk_clk_register_muxes(top_mtk_muxes, ARRAY_SIZE(top_mtk_muxes), node, goto free_top_data;
r = mtk_clk_register_factors(top_divs, ARRAY_SIZE(top_divs), top_clk_data);
if (r)
goto unregister_fixed_clks;
r = mtk_clk_register_muxes(top_mtk_muxes, ARRAY_SIZE(top_mtk_muxes), node,
&mt8195_clk_lock, top_clk_data); &mt8195_clk_lock, top_clk_data);
mtk_clk_register_composites(top_muxes, ARRAY_SIZE(top_muxes), base, if (r)
goto unregister_factors;
r = mtk_clk_register_composites(top_muxes, ARRAY_SIZE(top_muxes), base,
&mt8195_clk_lock, top_clk_data); &mt8195_clk_lock, top_clk_data);
mtk_clk_register_composites(top_adj_divs, ARRAY_SIZE(top_adj_divs), base, if (r)
goto unregister_muxes;
r = mtk_clk_register_composites(top_adj_divs, ARRAY_SIZE(top_adj_divs), base,
&mt8195_clk_lock, top_clk_data); &mt8195_clk_lock, top_clk_data);
if (r)
goto unregister_composite_muxes;
r = mtk_clk_register_gates(node, top_clks, ARRAY_SIZE(top_clks), top_clk_data); r = mtk_clk_register_gates(node, top_clks, ARRAY_SIZE(top_clks), top_clk_data);
if (r) if (r)
goto free_top_data; goto unregister_composite_divs;
r = of_clk_add_provider(node, of_clk_src_onecell_get, top_clk_data); r = of_clk_add_provider(node, of_clk_src_onecell_get, top_clk_data);
if (r) if (r)
goto free_top_data; goto unregister_gates;
platform_set_drvdata(pdev, top_clk_data);
return r; return r;
unregister_gates:
mtk_clk_unregister_gates(top_clks, ARRAY_SIZE(top_clks), top_clk_data);
unregister_composite_divs:
mtk_clk_unregister_composites(top_adj_divs, ARRAY_SIZE(top_adj_divs), top_clk_data);
unregister_composite_muxes:
mtk_clk_unregister_composites(top_muxes, ARRAY_SIZE(top_muxes), top_clk_data);
unregister_muxes:
mtk_clk_unregister_muxes(top_mtk_muxes, ARRAY_SIZE(top_mtk_muxes), top_clk_data);
unregister_factors:
mtk_clk_unregister_factors(top_divs, ARRAY_SIZE(top_divs), top_clk_data);
unregister_fixed_clks:
mtk_clk_unregister_fixed_clks(top_fixed_clks, ARRAY_SIZE(top_fixed_clks), top_clk_data);
free_top_data: free_top_data:
mtk_free_clk_data(top_clk_data); mtk_free_clk_data(top_clk_data);
return r; return r;
} }
static int clk_mt8195_topck_remove(struct platform_device *pdev)
{
struct clk_onecell_data *top_clk_data = platform_get_drvdata(pdev);
struct device_node *node = pdev->dev.of_node;
of_clk_del_provider(node);
mtk_clk_unregister_gates(top_clks, ARRAY_SIZE(top_clks), top_clk_data);
mtk_clk_unregister_composites(top_adj_divs, ARRAY_SIZE(top_adj_divs), top_clk_data);
mtk_clk_unregister_composites(top_muxes, ARRAY_SIZE(top_muxes), top_clk_data);
mtk_clk_unregister_muxes(top_mtk_muxes, ARRAY_SIZE(top_mtk_muxes), top_clk_data);
mtk_clk_unregister_factors(top_divs, ARRAY_SIZE(top_divs), top_clk_data);
mtk_clk_unregister_fixed_clks(top_fixed_clks, ARRAY_SIZE(top_fixed_clks), top_clk_data);
mtk_free_clk_data(top_clk_data);
return 0;
}
static struct platform_driver clk_mt8195_topck_drv = { static struct platform_driver clk_mt8195_topck_drv = {
.probe = clk_mt8195_topck_probe, .probe = clk_mt8195_topck_probe,
.remove = clk_mt8195_topck_remove,
.driver = { .driver = {
.name = "clk-mt8195-topck", .name = "clk-mt8195-topck",
.of_match_table = of_match_clk_mt8195_topck, .of_match_table = of_match_clk_mt8195_topck,
......
...@@ -96,6 +96,7 @@ static const struct of_device_id of_match_clk_mt8195_vdec[] = { ...@@ -96,6 +96,7 @@ static const struct of_device_id of_match_clk_mt8195_vdec[] = {
static struct platform_driver clk_mt8195_vdec_drv = { static struct platform_driver clk_mt8195_vdec_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-vdec", .name = "clk-mt8195-vdec",
.of_match_table = of_match_clk_mt8195_vdec, .of_match_table = of_match_clk_mt8195_vdec,
......
...@@ -105,17 +105,35 @@ static int clk_mt8195_vdo0_probe(struct platform_device *pdev) ...@@ -105,17 +105,35 @@ static int clk_mt8195_vdo0_probe(struct platform_device *pdev)
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data); r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r) if (r)
goto free_vdo0_data; goto unregister_gates;
platform_set_drvdata(pdev, clk_data);
return r; return r;
unregister_gates:
mtk_clk_unregister_gates(vdo0_clks, ARRAY_SIZE(vdo0_clks), clk_data);
free_vdo0_data: free_vdo0_data:
mtk_free_clk_data(clk_data); mtk_free_clk_data(clk_data);
return r; return r;
} }
static int clk_mt8195_vdo0_remove(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct device_node *node = dev->parent->of_node;
struct clk_onecell_data *clk_data = platform_get_drvdata(pdev);
of_clk_del_provider(node);
mtk_clk_unregister_gates(vdo0_clks, ARRAY_SIZE(vdo0_clks), clk_data);
mtk_free_clk_data(clk_data);
return 0;
}
static struct platform_driver clk_mt8195_vdo0_drv = { static struct platform_driver clk_mt8195_vdo0_drv = {
.probe = clk_mt8195_vdo0_probe, .probe = clk_mt8195_vdo0_probe,
.remove = clk_mt8195_vdo0_remove,
.driver = { .driver = {
.name = "clk-mt8195-vdo0", .name = "clk-mt8195-vdo0",
}, },
......
...@@ -122,17 +122,35 @@ static int clk_mt8195_vdo1_probe(struct platform_device *pdev) ...@@ -122,17 +122,35 @@ static int clk_mt8195_vdo1_probe(struct platform_device *pdev)
r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data); r = of_clk_add_provider(node, of_clk_src_onecell_get, clk_data);
if (r) if (r)
goto free_vdo1_data; goto unregister_gates;
platform_set_drvdata(pdev, clk_data);
return r; return r;
unregister_gates:
mtk_clk_unregister_gates(vdo1_clks, ARRAY_SIZE(vdo1_clks), clk_data);
free_vdo1_data: free_vdo1_data:
mtk_free_clk_data(clk_data); mtk_free_clk_data(clk_data);
return r; return r;
} }
static int clk_mt8195_vdo1_remove(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct device_node *node = dev->parent->of_node;
struct clk_onecell_data *clk_data = platform_get_drvdata(pdev);
of_clk_del_provider(node);
mtk_clk_unregister_gates(vdo1_clks, ARRAY_SIZE(vdo1_clks), clk_data);
mtk_free_clk_data(clk_data);
return 0;
}
static struct platform_driver clk_mt8195_vdo1_drv = { static struct platform_driver clk_mt8195_vdo1_drv = {
.probe = clk_mt8195_vdo1_probe, .probe = clk_mt8195_vdo1_probe,
.remove = clk_mt8195_vdo1_remove,
.driver = { .driver = {
.name = "clk-mt8195-vdo1", .name = "clk-mt8195-vdo1",
}, },
......
...@@ -61,6 +61,7 @@ static const struct of_device_id of_match_clk_mt8195_venc[] = { ...@@ -61,6 +61,7 @@ static const struct of_device_id of_match_clk_mt8195_venc[] = {
static struct platform_driver clk_mt8195_venc_drv = { static struct platform_driver clk_mt8195_venc_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-venc", .name = "clk-mt8195-venc",
.of_match_table = of_match_clk_mt8195_venc, .of_match_table = of_match_clk_mt8195_venc,
......
...@@ -102,6 +102,7 @@ static const struct of_device_id of_match_clk_mt8195_vpp0[] = { ...@@ -102,6 +102,7 @@ static const struct of_device_id of_match_clk_mt8195_vpp0[] = {
static struct platform_driver clk_mt8195_vpp0_drv = { static struct platform_driver clk_mt8195_vpp0_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-vpp0", .name = "clk-mt8195-vpp0",
.of_match_table = of_match_clk_mt8195_vpp0, .of_match_table = of_match_clk_mt8195_vpp0,
......
...@@ -100,6 +100,7 @@ static const struct of_device_id of_match_clk_mt8195_vpp1[] = { ...@@ -100,6 +100,7 @@ static const struct of_device_id of_match_clk_mt8195_vpp1[] = {
static struct platform_driver clk_mt8195_vpp1_drv = { static struct platform_driver clk_mt8195_vpp1_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-vpp1", .name = "clk-mt8195-vpp1",
.of_match_table = of_match_clk_mt8195_vpp1, .of_match_table = of_match_clk_mt8195_vpp1,
......
...@@ -135,6 +135,7 @@ static const struct of_device_id of_match_clk_mt8195_wpe[] = { ...@@ -135,6 +135,7 @@ static const struct of_device_id of_match_clk_mt8195_wpe[] = {
static struct platform_driver clk_mt8195_wpe_drv = { static struct platform_driver clk_mt8195_wpe_drv = {
.probe = mtk_clk_simple_probe, .probe = mtk_clk_simple_probe,
.remove = mtk_clk_simple_remove,
.driver = { .driver = {
.name = "clk-mt8195-wpe", .name = "clk-mt8195-wpe",
.of_match_table = of_match_clk_mt8195_wpe, .of_match_table = of_match_clk_mt8195_wpe,
......
...@@ -11,8 +11,9 @@ ...@@ -11,8 +11,9 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/mfd/syscon.h> #include <linux/mfd/syscon.h>
#include "clk-mtk.h"
#include "clk-gate.h" #include "clk-gate.h"
#include "clk-mtk.h"
#include "clk-pll.h"
#include <dt-bindings/clock/mt8516-clk.h> #include <dt-bindings/clock/mt8516-clk.h>
......
This diff is collapsed.
...@@ -7,19 +7,19 @@ ...@@ -7,19 +7,19 @@
#ifndef __DRV_CLK_MTK_H #ifndef __DRV_CLK_MTK_H
#define __DRV_CLK_MTK_H #define __DRV_CLK_MTK_H
#include <linux/regmap.h>
#include <linux/bitops.h>
#include <linux/clk-provider.h> #include <linux/clk-provider.h>
#include <linux/platform_device.h> #include <linux/io.h>
#include <linux/kernel.h>
struct clk; #include <linux/spinlock.h>
struct clk_onecell_data; #include <linux/types.h>
#define MAX_MUX_GATE_BIT 31 #define MAX_MUX_GATE_BIT 31
#define INVALID_MUX_GATE_BIT (MAX_MUX_GATE_BIT + 1) #define INVALID_MUX_GATE_BIT (MAX_MUX_GATE_BIT + 1)
#define MHZ (1000 * 1000) #define MHZ (1000 * 1000)
struct platform_device;
struct mtk_fixed_clk { struct mtk_fixed_clk {
int id; int id;
const char *name; const char *name;
...@@ -34,8 +34,10 @@ struct mtk_fixed_clk { ...@@ -34,8 +34,10 @@ struct mtk_fixed_clk {
.rate = _rate, \ .rate = _rate, \
} }
void mtk_clk_register_fixed_clks(const struct mtk_fixed_clk *clks, int mtk_clk_register_fixed_clks(const struct mtk_fixed_clk *clks, int num,
int num, struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
void mtk_clk_unregister_fixed_clks(const struct mtk_fixed_clk *clks, int num,
struct clk_onecell_data *clk_data);
struct mtk_fixed_factor { struct mtk_fixed_factor {
int id; int id;
...@@ -53,8 +55,10 @@ struct mtk_fixed_factor { ...@@ -53,8 +55,10 @@ struct mtk_fixed_factor {
.div = _div, \ .div = _div, \
} }
void mtk_clk_register_factors(const struct mtk_fixed_factor *clks, int mtk_clk_register_factors(const struct mtk_fixed_factor *clks, int num,
int num, struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
void mtk_clk_unregister_factors(const struct mtk_fixed_factor *clks, int num,
struct clk_onecell_data *clk_data);
struct mtk_composite { struct mtk_composite {
int id; int id;
...@@ -146,35 +150,12 @@ struct mtk_composite { ...@@ -146,35 +150,12 @@ struct mtk_composite {
struct clk *mtk_clk_register_composite(const struct mtk_composite *mc, struct clk *mtk_clk_register_composite(const struct mtk_composite *mc,
void __iomem *base, spinlock_t *lock); void __iomem *base, spinlock_t *lock);
void mtk_clk_register_composites(const struct mtk_composite *mcs, int mtk_clk_register_composites(const struct mtk_composite *mcs, int num,
int num, void __iomem *base, spinlock_t *lock, void __iomem *base, spinlock_t *lock,
struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
void mtk_clk_unregister_composites(const struct mtk_composite *mcs, int num,
struct mtk_gate_regs {
u32 sta_ofs;
u32 clr_ofs;
u32 set_ofs;
};
struct mtk_gate {
int id;
const char *name;
const char *parent_name;
const struct mtk_gate_regs *regs;
int shift;
const struct clk_ops *ops;
unsigned long flags;
};
int mtk_clk_register_gates(struct device_node *node,
const struct mtk_gate *clks, int num,
struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
int mtk_clk_register_gates_with_dev(struct device_node *node,
const struct mtk_gate *clks,
int num, struct clk_onecell_data *clk_data,
struct device *dev);
struct mtk_clk_divider { struct mtk_clk_divider {
int id; int id;
const char *name; const char *name;
...@@ -197,52 +178,15 @@ struct mtk_clk_divider { ...@@ -197,52 +178,15 @@ struct mtk_clk_divider {
.div_width = _width, \ .div_width = _width, \
} }
void mtk_clk_register_dividers(const struct mtk_clk_divider *mcds, int mtk_clk_register_dividers(const struct mtk_clk_divider *mcds, int num,
int num, void __iomem *base, spinlock_t *lock, void __iomem *base, spinlock_t *lock,
struct clk_onecell_data *clk_data);
void mtk_clk_unregister_dividers(const struct mtk_clk_divider *mcds, int num,
struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
struct clk_onecell_data *mtk_alloc_clk_data(unsigned int clk_num); struct clk_onecell_data *mtk_alloc_clk_data(unsigned int clk_num);
void mtk_free_clk_data(struct clk_onecell_data *clk_data); void mtk_free_clk_data(struct clk_onecell_data *clk_data);
#define HAVE_RST_BAR BIT(0)
#define PLL_AO BIT(1)
struct mtk_pll_div_table {
u32 div;
unsigned long freq;
};
struct mtk_pll_data {
int id;
const char *name;
u32 reg;
u32 pwr_reg;
u32 en_mask;
u32 pd_reg;
u32 tuner_reg;
u32 tuner_en_reg;
u8 tuner_en_bit;
int pd_shift;
unsigned int flags;
const struct clk_ops *ops;
u32 rst_bar_mask;
unsigned long fmin;
unsigned long fmax;
int pcwbits;
int pcwibits;
u32 pcw_reg;
int pcw_shift;
u32 pcw_chg_reg;
const struct mtk_pll_div_table *div_table;
const char *parent_name;
u32 en_reg;
u8 pll_en_bit; /* Assume 0, indicates BIT(0) by default */
};
void mtk_clk_register_plls(struct device_node *node,
const struct mtk_pll_data *plls, int num_plls,
struct clk_onecell_data *clk_data);
struct clk *mtk_clk_register_ref2usb_tx(const char *name, struct clk *mtk_clk_register_ref2usb_tx(const char *name,
const char *parent_name, void __iomem *reg); const char *parent_name, void __iomem *reg);
...@@ -258,5 +202,6 @@ struct mtk_clk_desc { ...@@ -258,5 +202,6 @@ struct mtk_clk_desc {
}; };
int mtk_clk_simple_probe(struct platform_device *pdev); int mtk_clk_simple_probe(struct platform_device *pdev);
int mtk_clk_simple_remove(struct platform_device *pdev);
#endif /* __DRV_CLK_MTK_H */ #endif /* __DRV_CLK_MTK_H */
...@@ -4,15 +4,26 @@ ...@@ -4,15 +4,26 @@
* Author: Owen Chen <owen.chen@mediatek.com> * Author: Owen Chen <owen.chen@mediatek.com>
*/ */
#include <linux/of.h> #include <linux/clk-provider.h>
#include <linux/of_address.h> #include <linux/compiler_types.h>
#include <linux/slab.h> #include <linux/container_of.h>
#include <linux/err.h>
#include <linux/mfd/syscon.h> #include <linux/mfd/syscon.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/regmap.h>
#include <linux/spinlock.h>
#include <linux/slab.h>
#include "clk-mtk.h"
#include "clk-mux.h" #include "clk-mux.h"
struct mtk_clk_mux {
struct clk_hw hw;
struct regmap *regmap;
const struct mtk_mux *data;
spinlock_t *lock;
bool reparent;
};
static inline struct mtk_clk_mux *to_mtk_clk_mux(struct clk_hw *hw) static inline struct mtk_clk_mux *to_mtk_clk_mux(struct clk_hw *hw)
{ {
return container_of(hw, struct mtk_clk_mux, hw); return container_of(hw, struct mtk_clk_mux, hw);
...@@ -164,6 +175,21 @@ static struct clk *mtk_clk_register_mux(const struct mtk_mux *mux, ...@@ -164,6 +175,21 @@ static struct clk *mtk_clk_register_mux(const struct mtk_mux *mux,
return clk; return clk;
} }
static void mtk_clk_unregister_mux(struct clk *clk)
{
struct mtk_clk_mux *mux;
struct clk_hw *hw;
hw = __clk_get_hw(clk);
if (!hw)
return;
mux = to_mtk_clk_mux(hw);
clk_unregister(clk);
kfree(mux);
}
int mtk_clk_register_muxes(const struct mtk_mux *muxes, int mtk_clk_register_muxes(const struct mtk_mux *muxes,
int num, struct device_node *node, int num, struct device_node *node,
spinlock_t *lock, spinlock_t *lock,
...@@ -175,29 +201,64 @@ int mtk_clk_register_muxes(const struct mtk_mux *muxes, ...@@ -175,29 +201,64 @@ int mtk_clk_register_muxes(const struct mtk_mux *muxes,
regmap = device_node_to_regmap(node); regmap = device_node_to_regmap(node);
if (IS_ERR(regmap)) { if (IS_ERR(regmap)) {
pr_err("Cannot find regmap for %pOF: %ld\n", node, pr_err("Cannot find regmap for %pOF: %pe\n", node, regmap);
PTR_ERR(regmap));
return PTR_ERR(regmap); return PTR_ERR(regmap);
} }
for (i = 0; i < num; i++) { for (i = 0; i < num; i++) {
const struct mtk_mux *mux = &muxes[i]; const struct mtk_mux *mux = &muxes[i];
if (IS_ERR_OR_NULL(clk_data->clks[mux->id])) { if (!IS_ERR_OR_NULL(clk_data->clks[mux->id])) {
pr_warn("%pOF: Trying to register duplicate clock ID: %d\n",
node, mux->id);
continue;
}
clk = mtk_clk_register_mux(mux, regmap, lock); clk = mtk_clk_register_mux(mux, regmap, lock);
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
pr_err("Failed to register clk %s: %ld\n", pr_err("Failed to register clk %s: %pe\n", mux->name, clk);
mux->name, PTR_ERR(clk)); goto err;
continue;
} }
clk_data->clks[mux->id] = clk; clk_data->clks[mux->id] = clk;
} }
}
return 0; return 0;
err:
while (--i >= 0) {
const struct mtk_mux *mux = &muxes[i];
if (IS_ERR_OR_NULL(clk_data->clks[mux->id]))
continue;
mtk_clk_unregister_mux(clk_data->clks[mux->id]);
clk_data->clks[mux->id] = ERR_PTR(-ENOENT);
}
return PTR_ERR(clk);
} }
EXPORT_SYMBOL_GPL(mtk_clk_register_muxes); EXPORT_SYMBOL_GPL(mtk_clk_register_muxes);
void mtk_clk_unregister_muxes(const struct mtk_mux *muxes, int num,
struct clk_onecell_data *clk_data)
{
int i;
if (!clk_data)
return;
for (i = num; i > 0; i--) {
const struct mtk_mux *mux = &muxes[i - 1];
if (IS_ERR_OR_NULL(clk_data->clks[mux->id]))
continue;
mtk_clk_unregister_mux(clk_data->clks[mux->id]);
clk_data->clks[mux->id] = ERR_PTR(-ENOENT);
}
}
EXPORT_SYMBOL_GPL(mtk_clk_unregister_muxes);
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
...@@ -7,15 +7,13 @@ ...@@ -7,15 +7,13 @@
#ifndef __DRV_CLK_MTK_MUX_H #ifndef __DRV_CLK_MTK_MUX_H
#define __DRV_CLK_MTK_MUX_H #define __DRV_CLK_MTK_MUX_H
#include <linux/clk-provider.h> #include <linux/spinlock.h>
#include <linux/types.h>
struct mtk_clk_mux { struct clk;
struct clk_hw hw; struct clk_onecell_data;
struct regmap *regmap; struct clk_ops;
const struct mtk_mux *data; struct device_node;
spinlock_t *lock;
bool reparent;
};
struct mtk_mux { struct mtk_mux {
int id; int id;
...@@ -88,4 +86,7 @@ int mtk_clk_register_muxes(const struct mtk_mux *muxes, ...@@ -88,4 +86,7 @@ int mtk_clk_register_muxes(const struct mtk_mux *muxes,
spinlock_t *lock, spinlock_t *lock,
struct clk_onecell_data *clk_data); struct clk_onecell_data *clk_data);
void mtk_clk_unregister_muxes(const struct mtk_mux *muxes, int num,
struct clk_onecell_data *clk_data);
#endif /* __DRV_CLK_MTK_MUX_H */ #endif /* __DRV_CLK_MTK_MUX_H */
...@@ -4,15 +4,18 @@ ...@@ -4,15 +4,18 @@
* Author: James Liao <jamesjj.liao@mediatek.com> * Author: James Liao <jamesjj.liao@mediatek.com>
*/ */
#include <linux/of.h> #include <linux/clk-provider.h>
#include <linux/of_address.h> #include <linux/container_of.h>
#include <linux/delay.h>
#include <linux/err.h>
#include <linux/io.h> #include <linux/io.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/of_address.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/clkdev.h>
#include <linux/delay.h>
#include "clk-mtk.h" #include "clk-pll.h"
#define MHZ (1000 * 1000)
#define REG_CON0 0 #define REG_CON0 0
#define REG_CON1 4 #define REG_CON1 4
...@@ -359,8 +362,24 @@ static struct clk *mtk_clk_register_pll(const struct mtk_pll_data *data, ...@@ -359,8 +362,24 @@ static struct clk *mtk_clk_register_pll(const struct mtk_pll_data *data,
return clk; return clk;
} }
void mtk_clk_register_plls(struct device_node *node, static void mtk_clk_unregister_pll(struct clk *clk)
const struct mtk_pll_data *plls, int num_plls, struct clk_onecell_data *clk_data) {
struct clk_hw *hw;
struct mtk_clk_pll *pll;
hw = __clk_get_hw(clk);
if (!hw)
return;
pll = to_mtk_clk_pll(hw);
clk_unregister(clk);
kfree(pll);
}
int mtk_clk_register_plls(struct device_node *node,
const struct mtk_pll_data *plls, int num_plls,
struct clk_onecell_data *clk_data)
{ {
void __iomem *base; void __iomem *base;
int i; int i;
...@@ -369,23 +388,82 @@ void mtk_clk_register_plls(struct device_node *node, ...@@ -369,23 +388,82 @@ void mtk_clk_register_plls(struct device_node *node,
base = of_iomap(node, 0); base = of_iomap(node, 0);
if (!base) { if (!base) {
pr_err("%s(): ioremap failed\n", __func__); pr_err("%s(): ioremap failed\n", __func__);
return; return -EINVAL;
} }
for (i = 0; i < num_plls; i++) { for (i = 0; i < num_plls; i++) {
const struct mtk_pll_data *pll = &plls[i]; const struct mtk_pll_data *pll = &plls[i];
if (!IS_ERR_OR_NULL(clk_data->clks[pll->id])) {
pr_warn("%pOF: Trying to register duplicate clock ID: %d\n",
node, pll->id);
continue;
}
clk = mtk_clk_register_pll(pll, base); clk = mtk_clk_register_pll(pll, base);
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
pr_err("Failed to register clk %s: %ld\n", pr_err("Failed to register clk %s: %pe\n", pll->name, clk);
pll->name, PTR_ERR(clk)); goto err;
continue;
} }
clk_data->clks[pll->id] = clk; clk_data->clks[pll->id] = clk;
} }
return 0;
err:
while (--i >= 0) {
const struct mtk_pll_data *pll = &plls[i];
mtk_clk_unregister_pll(clk_data->clks[pll->id]);
clk_data->clks[pll->id] = ERR_PTR(-ENOENT);
}
iounmap(base);
return PTR_ERR(clk);
} }
EXPORT_SYMBOL_GPL(mtk_clk_register_plls); EXPORT_SYMBOL_GPL(mtk_clk_register_plls);
static __iomem void *mtk_clk_pll_get_base(struct clk *clk,
const struct mtk_pll_data *data)
{
struct clk_hw *hw = __clk_get_hw(clk);
struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
return pll->base_addr - data->reg;
}
void mtk_clk_unregister_plls(const struct mtk_pll_data *plls, int num_plls,
struct clk_onecell_data *clk_data)
{
__iomem void *base = NULL;
int i;
if (!clk_data)
return;
for (i = num_plls; i > 0; i--) {
const struct mtk_pll_data *pll = &plls[i - 1];
if (IS_ERR_OR_NULL(clk_data->clks[pll->id]))
continue;
/*
* This is quite ugly but unfortunately the clks don't have
* any device tied to them, so there's no place to store the
* pointer to the I/O region base address. We have to fetch
* it from one of the registered clks.
*/
base = mtk_clk_pll_get_base(clk_data->clks[pll->id], pll);
mtk_clk_unregister_pll(clk_data->clks[pll->id]);
clk_data->clks[pll->id] = ERR_PTR(-ENOENT);
}
iounmap(base);
}
EXPORT_SYMBOL_GPL(mtk_clk_unregister_plls);
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
/* SPDX-License-Identifier: GPL-2.0-only */
/*
* Copyright (c) 2014 MediaTek Inc.
* Author: James Liao <jamesjj.liao@mediatek.com>
*/
#ifndef __DRV_CLK_MTK_PLL_H
#define __DRV_CLK_MTK_PLL_H
#include <linux/types.h>
struct clk_ops;
struct clk_onecell_data;
struct device_node;
struct mtk_pll_div_table {
u32 div;
unsigned long freq;
};
#define HAVE_RST_BAR BIT(0)
#define PLL_AO BIT(1)
struct mtk_pll_data {
int id;
const char *name;
u32 reg;
u32 pwr_reg;
u32 en_mask;
u32 pd_reg;
u32 tuner_reg;
u32 tuner_en_reg;
u8 tuner_en_bit;
int pd_shift;
unsigned int flags;
const struct clk_ops *ops;
u32 rst_bar_mask;
unsigned long fmin;
unsigned long fmax;
int pcwbits;
int pcwibits;
u32 pcw_reg;
int pcw_shift;
u32 pcw_chg_reg;
const struct mtk_pll_div_table *div_table;
const char *parent_name;
u32 en_reg;
u8 pll_en_bit; /* Assume 0, indicates BIT(0) by default */
};
int mtk_clk_register_plls(struct device_node *node,
const struct mtk_pll_data *plls, int num_plls,
struct clk_onecell_data *clk_data);
void mtk_clk_unregister_plls(const struct mtk_pll_data *plls, int num_plls,
struct clk_onecell_data *clk_data);
#endif /* __DRV_CLK_MTK_PLL_H */
...@@ -100,8 +100,7 @@ static void mtk_register_reset_controller_common(struct device_node *np, ...@@ -100,8 +100,7 @@ static void mtk_register_reset_controller_common(struct device_node *np,
regmap = device_node_to_regmap(np); regmap = device_node_to_regmap(np);
if (IS_ERR(regmap)) { if (IS_ERR(regmap)) {
pr_err("Cannot find regmap for %pOF: %ld\n", np, pr_err("Cannot find regmap for %pOF: %pe\n", np, regmap);
PTR_ERR(regmap));
return; return;
} }
......
# SPDX-License-Identifier: GPL-2.0
config COMMON_CLK_PIC32
def_bool COMMON_CLK && MACH_PIC32
config MCHP_CLK_MPFS
bool "Clk driver for PolarFire SoC"
depends on (RISCV && SOC_MICROCHIP_POLARFIRE) || COMPILE_TEST
help
Supports Clock Configuration for PolarFire SoC
# SPDX-License-Identifier: GPL-2.0-only # SPDX-License-Identifier: GPL-2.0-only
obj-$(CONFIG_COMMON_CLK_PIC32) += clk-core.o obj-$(CONFIG_COMMON_CLK_PIC32) += clk-core.o
obj-$(CONFIG_PIC32MZDA) += clk-pic32mzda.o obj-$(CONFIG_PIC32MZDA) += clk-pic32mzda.o
obj-$(CONFIG_MCHP_CLK_MPFS) += clk-mpfs.o
This diff is collapsed.
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
#define PMC_PLLACK 7 #define PMC_PLLACK 7
#define PMC_PLLBCK 8 #define PMC_PLLBCK 8
#define PMC_AUDIOPLLCK 9 #define PMC_AUDIOPLLCK 9
#define PMC_AUDIOPINCK 10
/* SAMA7G5 */ /* SAMA7G5 */
#define PMC_CPUPLL (PMC_MAIN + 1) #define PMC_CPUPLL (PMC_MAIN + 1)
...@@ -35,6 +36,7 @@ ...@@ -35,6 +36,7 @@
#define PMC_AUDIOIOPLL (PMC_MAIN + 7) #define PMC_AUDIOIOPLL (PMC_MAIN + 7)
#define PMC_ETHPLL (PMC_MAIN + 8) #define PMC_ETHPLL (PMC_MAIN + 8)
#define PMC_CPU (PMC_MAIN + 9) #define PMC_CPU (PMC_MAIN + 9)
#define PMC_MCK1 (PMC_MAIN + 10)
#ifndef AT91_PMC_MOSCS #ifndef AT91_PMC_MOSCS
#define AT91_PMC_MOSCS 0 /* MOSCS Flag */ #define AT91_PMC_MOSCS 0 /* MOSCS Flag */
......
/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */
/*
* Daire McNamara,<daire.mcnamara@microchip.com>
* Copyright (C) 2020 Microchip Technology Inc. All rights reserved.
*/
#ifndef _DT_BINDINGS_CLK_MICROCHIP_MPFS_H_
#define _DT_BINDINGS_CLK_MICROCHIP_MPFS_H_
#define CLK_CPU 0
#define CLK_AXI 1
#define CLK_AHB 2
#define CLK_ENVM 3
#define CLK_MAC0 4
#define CLK_MAC1 5
#define CLK_MMC 6
#define CLK_TIMER 7
#define CLK_MMUART0 8
#define CLK_MMUART1 9
#define CLK_MMUART2 10
#define CLK_MMUART3 11
#define CLK_MMUART4 12
#define CLK_SPI0 13
#define CLK_SPI1 14
#define CLK_I2C0 15
#define CLK_I2C1 16
#define CLK_CAN0 17
#define CLK_CAN1 18
#define CLK_USB 19
#define CLK_RESERVED 20
#define CLK_RTC 21
#define CLK_QSPI 22
#define CLK_GPIO0 23
#define CLK_GPIO1 24
#define CLK_GPIO2 25
#define CLK_DDRC 26
#define CLK_FIC0 27
#define CLK_FIC1 28
#define CLK_FIC2 29
#define CLK_FIC3 30
#define CLK_ATHENA 31
#define CLK_CFM 32
#endif /* _DT_BINDINGS_CLK_MICROCHIP_MPFS_H_ */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment