| /* |
| * Copyright (C) 2015 Thomas Chou <thomas@wytron.com.tw> |
| * |
| * SPDX-License-Identifier: GPL-2.0+ |
| */ |
| |
| #include <common.h> |
| #include <dm.h> |
| #include <dm/lists.h> |
| #include <dm/device-internal.h> |
| #include <clk.h> |
| #include <errno.h> |
| #include <timer.h> |
| |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| /* |
| * Implement a timer uclass to work with lib/time.c. The timer is usually |
| * a 32/64 bits free-running up counter. The get_rate() method is used to get |
| * the input clock frequency of the timer. The get_count() method is used |
| * to get the current 64 bits count value. If the hardware is counting down, |
| * the value should be inversed inside the method. There may be no real |
| * tick, and no timer interrupt. |
| */ |
| |
| int notrace timer_get_count(struct udevice *dev, u64 *count) |
| { |
| const struct timer_ops *ops = device_get_ops(dev); |
| |
| if (!ops->get_count) |
| return -ENOSYS; |
| |
| return ops->get_count(dev, count); |
| } |
| |
| unsigned long notrace timer_get_rate(struct udevice *dev) |
| { |
| struct timer_dev_priv *uc_priv = dev->uclass_priv; |
| |
| return uc_priv->clock_rate; |
| } |
| |
| static int timer_pre_probe(struct udevice *dev) |
| { |
| #if !CONFIG_IS_ENABLED(OF_PLATDATA) |
| struct timer_dev_priv *uc_priv = dev_get_uclass_priv(dev); |
| struct clk timer_clk; |
| int err; |
| ulong ret; |
| |
| err = clk_get_by_index(dev, 0, &timer_clk); |
| if (!err) { |
| ret = clk_get_rate(&timer_clk); |
| if (IS_ERR_VALUE(ret)) |
| return ret; |
| uc_priv->clock_rate = ret; |
| } else |
| uc_priv->clock_rate = fdtdec_get_int(gd->fdt_blob, |
| dev_of_offset(dev), "clock-frequency", 0); |
| #endif |
| |
| return 0; |
| } |
| |
| static int timer_post_probe(struct udevice *dev) |
| { |
| struct timer_dev_priv *uc_priv = dev_get_uclass_priv(dev); |
| |
| if (!uc_priv->clock_rate) |
| return -EINVAL; |
| |
| return 0; |
| } |
| |
| u64 timer_conv_64(u32 count) |
| { |
| /* increment tbh if tbl has rolled over */ |
| if (count < gd->timebase_l) |
| gd->timebase_h++; |
| gd->timebase_l = count; |
| return ((u64)gd->timebase_h << 32) | gd->timebase_l; |
| } |
| |
| int notrace dm_timer_init(void) |
| { |
| __maybe_unused const void *blob = gd->fdt_blob; |
| struct udevice *dev = NULL; |
| int node = -ENOENT; |
| int ret; |
| |
| if (gd->timer) |
| return 0; |
| |
| #if !CONFIG_IS_ENABLED(OF_PLATDATA) |
| /* Check for a chosen timer to be used for tick */ |
| node = fdtdec_get_chosen_node(blob, "tick-timer"); |
| #endif |
| if (node < 0) { |
| /* No chosen timer, trying first available timer */ |
| ret = uclass_first_device_err(UCLASS_TIMER, &dev); |
| if (ret) |
| return ret; |
| } else { |
| if (uclass_get_device_by_of_offset(UCLASS_TIMER, node, &dev)) { |
| /* |
| * If the timer is not marked to be bound before |
| * relocation, bind it anyway. |
| */ |
| if (node > 0 && |
| !lists_bind_fdt(gd->dm_root, offset_to_ofnode(node), |
| &dev)) { |
| ret = device_probe(dev); |
| if (ret) |
| return ret; |
| } |
| } |
| } |
| |
| if (dev) { |
| gd->timer = dev; |
| return 0; |
| } |
| |
| return -ENODEV; |
| } |
| |
| UCLASS_DRIVER(timer) = { |
| .id = UCLASS_TIMER, |
| .name = "timer", |
| .pre_probe = timer_pre_probe, |
| .flags = DM_UC_FLAG_SEQ_ALIAS, |
| .post_probe = timer_post_probe, |
| .per_device_auto_alloc_size = sizeof(struct timer_dev_priv), |
| }; |