blob: 3627ed79b8196b11d129cea6872a1bafbb3139af [file] [log] [blame]
Bin Meng60262cd02018-12-12 06:12:27 -08001// SPDX-License-Identifier: GPL-2.0+
2/*
Sean Andersonc33efaf2020-09-28 10:52:21 -04003 * Copyright (C) 2020, Sean Anderson <seanga2@gmail.com>
Bin Meng60262cd02018-12-12 06:12:27 -08004 * Copyright (C) 2018, Bin Meng <bmeng.cn@gmail.com>
Sean Andersonc33efaf2020-09-28 10:52:21 -04005 * Copyright (C) 2018, Anup Patel <anup@brainfault.org>
6 * Copyright (C) 2012 Regents of the University of California
Bin Meng60262cd02018-12-12 06:12:27 -08007 *
Sean Andersonc33efaf2020-09-28 10:52:21 -04008 * RISC-V architecturally-defined generic timer driver
Bin Meng60262cd02018-12-12 06:12:27 -08009 *
Sean Andersonc33efaf2020-09-28 10:52:21 -040010 * This driver provides generic timer support for S-mode U-Boot.
Bin Meng60262cd02018-12-12 06:12:27 -080011 */
12
13#include <common.h>
14#include <dm.h>
15#include <errno.h>
16#include <timer.h>
Sean Andersonc33efaf2020-09-28 10:52:21 -040017#include <asm/csr.h>
Bin Meng60262cd02018-12-12 06:12:27 -080018
Pragnesh Patelbc8d12b2021-01-17 18:11:25 +053019static u64 notrace riscv_timer_get_count(struct udevice *dev)
Bin Meng60262cd02018-12-12 06:12:27 -080020{
Sean Anderson8af7bb92020-10-07 14:37:44 -040021 __maybe_unused u32 hi, lo;
Sean Andersonc33efaf2020-09-28 10:52:21 -040022
Sean Anderson8af7bb92020-10-07 14:37:44 -040023 if (IS_ENABLED(CONFIG_64BIT))
24 return csr_read(CSR_TIME);
Sean Andersonc33efaf2020-09-28 10:52:21 -040025
Sean Anderson8af7bb92020-10-07 14:37:44 -040026 do {
27 hi = csr_read(CSR_TIMEH);
28 lo = csr_read(CSR_TIME);
29 } while (hi != csr_read(CSR_TIMEH));
Sean Andersonc33efaf2020-09-28 10:52:21 -040030
Sean Anderson8af7bb92020-10-07 14:37:44 -040031 return ((u64)hi << 32) | lo;
Bin Meng60262cd02018-12-12 06:12:27 -080032}
33
Pragnesh Patelbc8d12b2021-01-17 18:11:25 +053034#if CONFIG_IS_ENABLED(RISCV_SMODE) && IS_ENABLED(CONFIG_TIMER_EARLY)
35/**
36 * timer_early_get_rate() - Get the timer rate before driver model
37 */
38unsigned long notrace timer_early_get_rate(void)
39{
40 return RISCV_SMODE_TIMER_FREQ;
41}
42
43/**
44 * timer_early_get_count() - Get the timer count before driver model
45 *
46 */
47u64 notrace timer_early_get_count(void)
48{
49 return riscv_timer_get_count(NULL);
50}
51#endif
52
Bin Meng60262cd02018-12-12 06:12:27 -080053static int riscv_timer_probe(struct udevice *dev)
54{
55 struct timer_dev_priv *uc_priv = dev_get_uclass_priv(dev);
56
57 /* clock frequency was passed from the cpu driver as driver data */
58 uc_priv->clock_rate = dev->driver_data;
59
60 return 0;
61}
62
63static const struct timer_ops riscv_timer_ops = {
64 .get_count = riscv_timer_get_count,
65};
66
67U_BOOT_DRIVER(riscv_timer) = {
68 .name = "riscv_timer",
69 .id = UCLASS_TIMER,
70 .probe = riscv_timer_probe,
71 .ops = &riscv_timer_ops,
72 .flags = DM_FLAG_PRE_RELOC,
73};