dw-apb-timer.c 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Designware APB Timer driver
  4. *
  5. * Copyright (C) 2018 Marek Vasut <marex@denx.de>
  6. */
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <clk.h>
  10. #include <timer.h>
  11. #include <asm/io.h>
  12. #include <asm/arch/timer.h>
  13. #define DW_APB_LOAD_VAL 0x0
  14. #define DW_APB_CURR_VAL 0x4
  15. #define DW_APB_CTRL 0x8
  16. DECLARE_GLOBAL_DATA_PTR;
  17. struct dw_apb_timer_priv {
  18. fdt_addr_t regs;
  19. };
  20. static int dw_apb_timer_get_count(struct udevice *dev, u64 *count)
  21. {
  22. struct dw_apb_timer_priv *priv = dev_get_priv(dev);
  23. /*
  24. * The DW APB counter counts down, but this function
  25. * requires the count to be incrementing. Invert the
  26. * result.
  27. */
  28. *count = ~readl(priv->regs + DW_APB_CURR_VAL);
  29. return 0;
  30. }
  31. static int dw_apb_timer_probe(struct udevice *dev)
  32. {
  33. struct timer_dev_priv *uc_priv = dev_get_uclass_priv(dev);
  34. struct dw_apb_timer_priv *priv = dev_get_priv(dev);
  35. struct clk clk;
  36. int ret;
  37. ret = clk_get_by_index(dev, 0, &clk);
  38. if (ret)
  39. return ret;
  40. uc_priv->clock_rate = clk_get_rate(&clk);
  41. clk_free(&clk);
  42. /* init timer */
  43. writel(0xffffffff, priv->regs + DW_APB_LOAD_VAL);
  44. writel(0xffffffff, priv->regs + DW_APB_CURR_VAL);
  45. setbits_le32(priv->regs + DW_APB_CTRL, 0x3);
  46. return 0;
  47. }
  48. static int dw_apb_timer_ofdata_to_platdata(struct udevice *dev)
  49. {
  50. struct dw_apb_timer_priv *priv = dev_get_priv(dev);
  51. priv->regs = dev_read_addr(dev);
  52. return 0;
  53. }
  54. static const struct timer_ops dw_apb_timer_ops = {
  55. .get_count = dw_apb_timer_get_count,
  56. };
  57. static const struct udevice_id dw_apb_timer_ids[] = {
  58. { .compatible = "snps,dw-apb-timer" },
  59. {}
  60. };
  61. U_BOOT_DRIVER(dw_apb_timer) = {
  62. .name = "dw_apb_timer",
  63. .id = UCLASS_TIMER,
  64. .ops = &dw_apb_timer_ops,
  65. .probe = dw_apb_timer_probe,
  66. .of_match = dw_apb_timer_ids,
  67. .ofdata_to_platdata = dw_apb_timer_ofdata_to_platdata,
  68. .priv_auto_alloc_size = sizeof(struct dw_apb_timer_priv),
  69. };