clocksource: sh_tmu: Add DT support
Document DT bindings and parse them in the TMU driver. Signed-off-by: Laurent Pinchart <laurent.pinchart+renesas@ideasonboard.com> Tested-by: Simon Horman <horms+renesas@verge.net.au>
This commit is contained in:
parent
1768aa2f4c
commit
3e29b5543f
2 changed files with 80 additions and 10 deletions
39
Documentation/devicetree/bindings/timer/renesas,tmu.txt
Normal file
39
Documentation/devicetree/bindings/timer/renesas,tmu.txt
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
* Renesas R-Car Timer Unit (TMU)
|
||||||
|
|
||||||
|
The TMU is a 32-bit timer/counter with configurable clock inputs and
|
||||||
|
programmable compare match.
|
||||||
|
|
||||||
|
Channels share hardware resources but their counter and compare match value
|
||||||
|
are independent. The TMU hardware supports up to three channels.
|
||||||
|
|
||||||
|
Required Properties:
|
||||||
|
|
||||||
|
- compatible: must contain "renesas,tmu"
|
||||||
|
|
||||||
|
- reg: base address and length of the registers block for the timer module.
|
||||||
|
|
||||||
|
- interrupts: interrupt-specifier for the timer, one per channel.
|
||||||
|
|
||||||
|
- clocks: a list of phandle + clock-specifier pairs, one for each entry
|
||||||
|
in clock-names.
|
||||||
|
- clock-names: must contain "fck" for the functional clock.
|
||||||
|
|
||||||
|
Optional Properties:
|
||||||
|
|
||||||
|
- #renesas,channels: number of channels implemented by the timer, must be 2
|
||||||
|
or 3 (if not specified the value defaults to 3).
|
||||||
|
|
||||||
|
|
||||||
|
Example: R8A7779 (R-Car H1) TMU0 node
|
||||||
|
|
||||||
|
tmu0: timer@ffd80000 {
|
||||||
|
compatible = "renesas,tmu";
|
||||||
|
reg = <0xffd80000 0x30>;
|
||||||
|
interrupts = <0 32 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<0 33 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<0 34 IRQ_TYPE_LEVEL_HIGH>;
|
||||||
|
clocks = <&mstp0_clks R8A7779_CLK_TMU0>;
|
||||||
|
clock-names = "fck";
|
||||||
|
|
||||||
|
#renesas,channels = <3>;
|
||||||
|
};
|
|
@ -24,6 +24,7 @@
|
||||||
#include <linux/ioport.h>
|
#include <linux/ioport.h>
|
||||||
#include <linux/irq.h>
|
#include <linux/irq.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
|
#include <linux/of.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
#include <linux/pm_domain.h>
|
#include <linux/pm_domain.h>
|
||||||
#include <linux/pm_runtime.h>
|
#include <linux/pm_runtime.h>
|
||||||
|
@ -509,23 +510,48 @@ static int sh_tmu_map_memory(struct sh_tmu_device *tmu)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int sh_tmu_parse_dt(struct sh_tmu_device *tmu)
|
||||||
|
{
|
||||||
|
struct device_node *np = tmu->pdev->dev.of_node;
|
||||||
|
|
||||||
|
tmu->model = SH_TMU;
|
||||||
|
tmu->num_channels = 3;
|
||||||
|
|
||||||
|
of_property_read_u32(np, "#renesas,channels", &tmu->num_channels);
|
||||||
|
|
||||||
|
if (tmu->num_channels != 2 && tmu->num_channels != 3) {
|
||||||
|
dev_err(&tmu->pdev->dev, "invalid number of channels %u\n",
|
||||||
|
tmu->num_channels);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int sh_tmu_setup(struct sh_tmu_device *tmu, struct platform_device *pdev)
|
static int sh_tmu_setup(struct sh_tmu_device *tmu, struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct sh_timer_config *cfg = pdev->dev.platform_data;
|
|
||||||
const struct platform_device_id *id = pdev->id_entry;
|
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (!cfg) {
|
tmu->pdev = pdev;
|
||||||
|
|
||||||
|
raw_spin_lock_init(&tmu->lock);
|
||||||
|
|
||||||
|
if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) {
|
||||||
|
ret = sh_tmu_parse_dt(tmu);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
} else if (pdev->dev.platform_data) {
|
||||||
|
const struct platform_device_id *id = pdev->id_entry;
|
||||||
|
struct sh_timer_config *cfg = pdev->dev.platform_data;
|
||||||
|
|
||||||
|
tmu->model = id->driver_data;
|
||||||
|
tmu->num_channels = hweight8(cfg->channels_mask);
|
||||||
|
} else {
|
||||||
dev_err(&tmu->pdev->dev, "missing platform data\n");
|
dev_err(&tmu->pdev->dev, "missing platform data\n");
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
tmu->pdev = pdev;
|
|
||||||
tmu->model = id->driver_data;
|
|
||||||
|
|
||||||
raw_spin_lock_init(&tmu->lock);
|
|
||||||
|
|
||||||
/* Get hold of clock. */
|
/* Get hold of clock. */
|
||||||
tmu->clk = clk_get(&tmu->pdev->dev, "fck");
|
tmu->clk = clk_get(&tmu->pdev->dev, "fck");
|
||||||
if (IS_ERR(tmu->clk)) {
|
if (IS_ERR(tmu->clk)) {
|
||||||
|
@ -545,8 +571,6 @@ static int sh_tmu_setup(struct sh_tmu_device *tmu, struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate and setup the channels. */
|
/* Allocate and setup the channels. */
|
||||||
tmu->num_channels = hweight8(cfg->channels_mask);
|
|
||||||
|
|
||||||
tmu->channels = kzalloc(sizeof(*tmu->channels) * tmu->num_channels,
|
tmu->channels = kzalloc(sizeof(*tmu->channels) * tmu->num_channels,
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (tmu->channels == NULL) {
|
if (tmu->channels == NULL) {
|
||||||
|
@ -628,11 +652,18 @@ static const struct platform_device_id sh_tmu_id_table[] = {
|
||||||
};
|
};
|
||||||
MODULE_DEVICE_TABLE(platform, sh_tmu_id_table);
|
MODULE_DEVICE_TABLE(platform, sh_tmu_id_table);
|
||||||
|
|
||||||
|
static const struct of_device_id sh_tmu_of_table[] __maybe_unused = {
|
||||||
|
{ .compatible = "renesas,tmu" },
|
||||||
|
{ }
|
||||||
|
};
|
||||||
|
MODULE_DEVICE_TABLE(of, sh_tmu_of_table);
|
||||||
|
|
||||||
static struct platform_driver sh_tmu_device_driver = {
|
static struct platform_driver sh_tmu_device_driver = {
|
||||||
.probe = sh_tmu_probe,
|
.probe = sh_tmu_probe,
|
||||||
.remove = sh_tmu_remove,
|
.remove = sh_tmu_remove,
|
||||||
.driver = {
|
.driver = {
|
||||||
.name = "sh_tmu",
|
.name = "sh_tmu",
|
||||||
|
.of_match_table = of_match_ptr(sh_tmu_of_table),
|
||||||
},
|
},
|
||||||
.id_table = sh_tmu_id_table,
|
.id_table = sh_tmu_id_table,
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Reference in a new issue