diff options
Diffstat (limited to 'arch/riscv/lib/sifive_clint.c')
| -rw-r--r-- | arch/riscv/lib/sifive_clint.c | 62 | 
1 files changed, 34 insertions, 28 deletions
| diff --git a/arch/riscv/lib/sifive_clint.c b/arch/riscv/lib/sifive_clint.c index b9a2c649cc4..c9704c596fa 100644 --- a/arch/riscv/lib/sifive_clint.c +++ b/arch/riscv/lib/sifive_clint.c @@ -8,9 +8,9 @@   */  #include <common.h> +#include <clk.h>  #include <dm.h> -#include <regmap.h> -#include <syscon.h> +#include <timer.h>  #include <asm/io.h>  #include <asm/syscon.h>  #include <linux/err.h> @@ -24,68 +24,74 @@  DECLARE_GLOBAL_DATA_PTR; -int riscv_get_time(u64 *time) +int riscv_init_ipi(void)  { -	/* ensure timer register base has a sane value */ -	riscv_init_ipi(); +	int ret; +	struct udevice *dev; + +	ret = uclass_get_device_by_driver(UCLASS_TIMER, +					  DM_GET_DRIVER(sifive_clint), &dev); +	if (ret) +		return ret; -	*time = readq((void __iomem *)MTIME_REG(gd->arch.clint)); +	gd->arch.clint = dev_read_addr_ptr(dev); +	if (!gd->arch.clint) +		return -EINVAL;  	return 0;  } -int riscv_set_timecmp(int hart, u64 cmp) +int riscv_send_ipi(int hart)  { -	/* ensure timer register base has a sane value */ -	riscv_init_ipi(); - -	writeq(cmp, (void __iomem *)MTIMECMP_REG(gd->arch.clint, hart)); +	writel(1, (void __iomem *)MSIP_REG(gd->arch.clint, hart));  	return 0;  } -int riscv_init_ipi(void) +int riscv_clear_ipi(int hart)  { -	if (!gd->arch.clint) { -		long *ret = syscon_get_first_range(RISCV_SYSCON_CLINT); - -		if (IS_ERR(ret)) -			return PTR_ERR(ret); -		gd->arch.clint = ret; -	} +	writel(0, (void __iomem *)MSIP_REG(gd->arch.clint, hart));  	return 0;  } -int riscv_send_ipi(int hart) +int riscv_get_ipi(int hart, int *pending)  { -	writel(1, (void __iomem *)MSIP_REG(gd->arch.clint, hart)); +	*pending = readl((void __iomem *)MSIP_REG(gd->arch.clint, hart));  	return 0;  } -int riscv_clear_ipi(int hart) +static int sifive_clint_get_count(struct udevice *dev, u64 *count)  { -	writel(0, (void __iomem *)MSIP_REG(gd->arch.clint, hart)); +	*count = readq((void __iomem *)MTIME_REG(dev->priv));  	return 0;  } -int riscv_get_ipi(int hart, int *pending) +static const struct timer_ops sifive_clint_ops = { +	.get_count = sifive_clint_get_count, +}; + +static int sifive_clint_probe(struct udevice *dev)  { -	*pending = readl((void __iomem *)MSIP_REG(gd->arch.clint, hart)); +	dev->priv = dev_read_addr_ptr(dev); +	if (!dev->priv) +		return -EINVAL; -	return 0; +	return timer_timebase_fallback(dev);  }  static const struct udevice_id sifive_clint_ids[] = { -	{ .compatible = "riscv,clint0", .data = RISCV_SYSCON_CLINT }, +	{ .compatible = "riscv,clint0" },  	{ }  };  U_BOOT_DRIVER(sifive_clint) = {  	.name		= "sifive_clint", -	.id		= UCLASS_SYSCON, +	.id		= UCLASS_TIMER,  	.of_match	= sifive_clint_ids, +	.probe		= sifive_clint_probe, +	.ops		= &sifive_clint_ops,  	.flags		= DM_FLAG_PRE_RELOC,  }; | 
