-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge tag 'zynqmp-soc-clk-for-v4.20' of https://github.com/Xilinx/lin…
…ux-xlnx into next/drivers arm64: zynqmp: SoC CLK changes for v4.20 This patchset adds CCF compliant clock driver for ZynqMP. Clock driver queries supported clock information from firmware and regiters pll and output clocks with CCF. * tag 'zynqmp-soc-clk-for-v4.20' of https://github.com/Xilinx/linux-xlnx: drivers: clk: Add ZynqMP clock driver dt-bindings: clock: Add bindings for ZynqMP clock driver firmware: xilinx: Add zynqmp IOCTL API for device control Documentation: xilinx: Add documentation for eemi APIs Signed-off-by: Arnd Bergmann <arnd@arndb.de>
- Loading branch information
Showing
15 changed files
with
1,919 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,67 @@ | ||
--------------------------------------------------------------------- | ||
Xilinx Zynq MPSoC EEMI Documentation | ||
--------------------------------------------------------------------- | ||
|
||
Xilinx Zynq MPSoC Firmware Interface | ||
------------------------------------- | ||
The zynqmp-firmware node describes the interface to platform firmware. | ||
ZynqMP has an interface to communicate with secure firmware. Firmware | ||
driver provides an interface to firmware APIs. Interface APIs can be | ||
used by any driver to communicate with PMC(Platform Management Controller). | ||
|
||
Embedded Energy Management Interface (EEMI) | ||
---------------------------------------------- | ||
The embedded energy management interface is used to allow software | ||
components running across different processing clusters on a chip or | ||
device to communicate with a power management controller (PMC) on a | ||
device to issue or respond to power management requests. | ||
|
||
EEMI ops is a structure containing all eemi APIs supported by Zynq MPSoC. | ||
The zynqmp-firmware driver maintain all EEMI APIs in zynqmp_eemi_ops | ||
structure. Any driver who want to communicate with PMC using EEMI APIs | ||
can call zynqmp_pm_get_eemi_ops(). | ||
|
||
Example of EEMI ops: | ||
|
||
/* zynqmp-firmware driver maintain all EEMI APIs */ | ||
struct zynqmp_eemi_ops { | ||
int (*get_api_version)(u32 *version); | ||
int (*query_data)(struct zynqmp_pm_query_data qdata, u32 *out); | ||
}; | ||
|
||
static const struct zynqmp_eemi_ops eemi_ops = { | ||
.get_api_version = zynqmp_pm_get_api_version, | ||
.query_data = zynqmp_pm_query_data, | ||
}; | ||
|
||
Example of EEMI ops usage: | ||
|
||
static const struct zynqmp_eemi_ops *eemi_ops; | ||
u32 ret_payload[PAYLOAD_ARG_CNT]; | ||
int ret; | ||
|
||
eemi_ops = zynqmp_pm_get_eemi_ops(); | ||
if (!eemi_ops) | ||
return -ENXIO; | ||
|
||
ret = eemi_ops->query_data(qdata, ret_payload); | ||
|
||
IOCTL | ||
------ | ||
IOCTL API is for device control and configuration. It is not a system | ||
IOCTL but it is an EEMI API. This API can be used by master to control | ||
any device specific configuration. IOCTL definitions can be platform | ||
specific. This API also manage shared device configuration. | ||
|
||
The following IOCTL IDs are valid for device control: | ||
- IOCTL_SET_PLL_FRAC_MODE 8 | ||
- IOCTL_GET_PLL_FRAC_MODE 9 | ||
- IOCTL_SET_PLL_FRAC_DATA 10 | ||
- IOCTL_GET_PLL_FRAC_DATA 11 | ||
|
||
Refer EEMI API guide [0] for IOCTL specific parameters and other EEMI APIs. | ||
|
||
References | ||
---------- | ||
[0] Embedded Energy Management Interface (EEMI) API guide: | ||
https://www.xilinx.com/support/documentation/user_guides/ug1200-eemi-api.pdf |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
# SPDX-License-Identifier: GPL-2.0 | ||
|
||
config COMMON_CLK_ZYNQMP | ||
bool "Support for Xilinx ZynqMP Ultrascale+ clock controllers" | ||
depends on ARCH_ZYNQMP || COMPILE_TEST | ||
depends on ZYNQMP_FIRMWARE | ||
help | ||
Support for the Zynqmp Ultrascale clock controller. | ||
It has a dependency on the PMU firmware. | ||
Say Y if you want to include clock support. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
# SPDX-License-Identifier: GPL-2.0 | ||
# Zynq Ultrascale+ MPSoC clock specific Makefile | ||
|
||
obj-$(CONFIG_ARCH_ZYNQMP) += pll.o clk-gate-zynqmp.o divider.o clk-mux-zynqmp.o clkc.o |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,144 @@ | ||
// SPDX-License-Identifier: GPL-2.0 | ||
/* | ||
* Zynq UltraScale+ MPSoC clock controller | ||
* | ||
* Copyright (C) 2016-2018 Xilinx | ||
* | ||
* Gated clock implementation | ||
*/ | ||
|
||
#include <linux/clk-provider.h> | ||
#include <linux/slab.h> | ||
#include "clk-zynqmp.h" | ||
|
||
/** | ||
* struct clk_gate - gating clock | ||
* @hw: handle between common and hardware-specific interfaces | ||
* @flags: hardware-specific flags | ||
* @clk_id: Id of clock | ||
*/ | ||
struct zynqmp_clk_gate { | ||
struct clk_hw hw; | ||
u8 flags; | ||
u32 clk_id; | ||
}; | ||
|
||
#define to_zynqmp_clk_gate(_hw) container_of(_hw, struct zynqmp_clk_gate, hw) | ||
|
||
/** | ||
* zynqmp_clk_gate_enable() - Enable clock | ||
* @hw: handle between common and hardware-specific interfaces | ||
* | ||
* Return: 0 on success else error code | ||
*/ | ||
static int zynqmp_clk_gate_enable(struct clk_hw *hw) | ||
{ | ||
struct zynqmp_clk_gate *gate = to_zynqmp_clk_gate(hw); | ||
const char *clk_name = clk_hw_get_name(hw); | ||
u32 clk_id = gate->clk_id; | ||
int ret; | ||
const struct zynqmp_eemi_ops *eemi_ops = zynqmp_pm_get_eemi_ops(); | ||
|
||
ret = eemi_ops->clock_enable(clk_id); | ||
|
||
if (ret) | ||
pr_warn_once("%s() clock enabled failed for %s, ret = %d\n", | ||
__func__, clk_name, ret); | ||
|
||
return ret; | ||
} | ||
|
||
/* | ||
* zynqmp_clk_gate_disable() - Disable clock | ||
* @hw: handle between common and hardware-specific interfaces | ||
*/ | ||
static void zynqmp_clk_gate_disable(struct clk_hw *hw) | ||
{ | ||
struct zynqmp_clk_gate *gate = to_zynqmp_clk_gate(hw); | ||
const char *clk_name = clk_hw_get_name(hw); | ||
u32 clk_id = gate->clk_id; | ||
int ret; | ||
const struct zynqmp_eemi_ops *eemi_ops = zynqmp_pm_get_eemi_ops(); | ||
|
||
ret = eemi_ops->clock_disable(clk_id); | ||
|
||
if (ret) | ||
pr_warn_once("%s() clock disable failed for %s, ret = %d\n", | ||
__func__, clk_name, ret); | ||
} | ||
|
||
/** | ||
* zynqmp_clk_gate_is_enable() - Check clock state | ||
* @hw: handle between common and hardware-specific interfaces | ||
* | ||
* Return: 1 if enabled, 0 if disabled else error code | ||
*/ | ||
static int zynqmp_clk_gate_is_enabled(struct clk_hw *hw) | ||
{ | ||
struct zynqmp_clk_gate *gate = to_zynqmp_clk_gate(hw); | ||
const char *clk_name = clk_hw_get_name(hw); | ||
u32 clk_id = gate->clk_id; | ||
int state, ret; | ||
const struct zynqmp_eemi_ops *eemi_ops = zynqmp_pm_get_eemi_ops(); | ||
|
||
ret = eemi_ops->clock_getstate(clk_id, &state); | ||
if (ret) { | ||
pr_warn_once("%s() clock get state failed for %s, ret = %d\n", | ||
__func__, clk_name, ret); | ||
return -EIO; | ||
} | ||
|
||
return state ? 1 : 0; | ||
} | ||
|
||
static const struct clk_ops zynqmp_clk_gate_ops = { | ||
.enable = zynqmp_clk_gate_enable, | ||
.disable = zynqmp_clk_gate_disable, | ||
.is_enabled = zynqmp_clk_gate_is_enabled, | ||
}; | ||
|
||
/** | ||
* zynqmp_clk_register_gate() - Register a gate clock with the clock framework | ||
* @name: Name of this clock | ||
* @clk_id: Id of this clock | ||
* @parents: Name of this clock's parents | ||
* @num_parents: Number of parents | ||
* @nodes: Clock topology node | ||
* | ||
* Return: clock hardware of the registered clock gate | ||
*/ | ||
struct clk_hw *zynqmp_clk_register_gate(const char *name, u32 clk_id, | ||
const char * const *parents, | ||
u8 num_parents, | ||
const struct clock_topology *nodes) | ||
{ | ||
struct zynqmp_clk_gate *gate; | ||
struct clk_hw *hw; | ||
int ret; | ||
struct clk_init_data init; | ||
|
||
/* allocate the gate */ | ||
gate = kzalloc(sizeof(*gate), GFP_KERNEL); | ||
if (!gate) | ||
return ERR_PTR(-ENOMEM); | ||
|
||
init.name = name; | ||
init.ops = &zynqmp_clk_gate_ops; | ||
init.flags = nodes->flag; | ||
init.parent_names = parents; | ||
init.num_parents = 1; | ||
|
||
/* struct clk_gate assignments */ | ||
gate->flags = nodes->type_flag; | ||
gate->hw.init = &init; | ||
gate->clk_id = clk_id; | ||
|
||
hw = &gate->hw; | ||
ret = clk_hw_register(NULL, hw); | ||
if (ret) { | ||
kfree(gate); | ||
hw = ERR_PTR(ret); | ||
} | ||
|
||
return hw; | ||
} |
Oops, something went wrong.