riscv_io.h 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. /*
  2. * SPDX-License-Identifier: BSD-2-Clause
  3. *
  4. * Copyright (c) 2019 Western Digital Corporation or its affiliates.
  5. *
  6. * Authors:
  7. * Anup Patel <anup.patel@wdc.com>
  8. */
  9. #ifndef __RISCV_IO_H__
  10. #define __RISCV_IO_H__
  11. static inline uint32_t __raw_hartid(void)
  12. {
  13. #ifdef RISCV_S_MODE
  14. extern int boot_hartid;
  15. return boot_hartid;
  16. #else
  17. uint32_t x;
  18. asm volatile("csrr %0, mhartid" : "=r" (x) );
  19. return x;
  20. #endif
  21. }
  22. static inline void __raw_writeb(rt_uint8_t val, volatile void *addr)
  23. {
  24. asm volatile("sb %0, 0(%1)"
  25. :
  26. : "r"(val), "r"(addr));
  27. }
  28. static inline void __raw_writew(rt_uint16_t val, volatile void *addr)
  29. {
  30. asm volatile("sh %0, 0(%1)"
  31. :
  32. : "r"(val), "r"(addr));
  33. }
  34. static inline void __raw_writel(rt_uint32_t val, volatile void *addr)
  35. {
  36. asm volatile("sw %0, 0(%1)"
  37. :
  38. : "r"(val), "r"(addr));
  39. }
  40. #if __riscv_xlen != 32
  41. static inline void __raw_writeq(rt_uint64_t val, volatile void *addr)
  42. {
  43. asm volatile("sd %0, 0(%1)"
  44. :
  45. : "r"(val), "r"(addr));
  46. }
  47. #endif
  48. static inline rt_uint8_t __raw_readb(const volatile void *addr)
  49. {
  50. rt_uint8_t val;
  51. asm volatile("lb %0, 0(%1)"
  52. : "=r"(val)
  53. : "r"(addr));
  54. return val;
  55. }
  56. static inline rt_uint16_t __raw_readw(const volatile void *addr)
  57. {
  58. rt_uint16_t val;
  59. asm volatile("lh %0, 0(%1)"
  60. : "=r"(val)
  61. : "r"(addr));
  62. return val;
  63. }
  64. static inline rt_uint32_t __raw_readl(const volatile void *addr)
  65. {
  66. rt_uint32_t val;
  67. asm volatile("lw %0, 0(%1)"
  68. : "=r"(val)
  69. : "r"(addr));
  70. return val;
  71. }
  72. #if __riscv_xlen != 32
  73. static inline rt_uint64_t __raw_readq(const volatile void *addr)
  74. {
  75. rt_uint64_t val;
  76. asm volatile("ld %0, 0(%1)"
  77. : "=r"(val)
  78. : "r"(addr));
  79. return val;
  80. }
  81. #endif
  82. /* FIXME: These are now the same as asm-generic */
  83. /* clang-format off */
  84. #define __io_rbr() do {} while (0)
  85. #define __io_rar() do {} while (0)
  86. #define __io_rbw() do {} while (0)
  87. #define __io_raw() do {} while (0)
  88. #define readb_relaxed(c) ({ rt_uint8_t __v; __io_rbr(); __v = __raw_readb(c); __io_rar(); __v; })
  89. #define readw_relaxed(c) ({ rt_uint16_t __v; __io_rbr(); __v = __raw_readw(c); __io_rar(); __v; })
  90. #define readl_relaxed(c) ({ rt_uint32_t __v; __io_rbr(); __v = __raw_readl(c); __io_rar(); __v; })
  91. #define writeb_relaxed(v,c) ({ __io_rbw(); __raw_writeb((v),(c)); __io_raw(); })
  92. #define writew_relaxed(v,c) ({ __io_rbw(); __raw_writew((v),(c)); __io_raw(); })
  93. #define writel_relaxed(v,c) ({ __io_rbw(); __raw_writel((v),(c)); __io_raw(); })
  94. #if __riscv_xlen != 32
  95. #define readq_relaxed(c) ({ rt_uint64_t __v; __io_rbr(); __v = __raw_readq(c); __io_rar(); __v; })
  96. #define writeq_relaxed(v,c) ({ __io_rbw(); __raw_writeq((v),(c)); __io_raw(); })
  97. #endif
  98. #define __io_br() do {} while (0)
  99. #define __io_ar() __asm__ __volatile__ ("fence i,r" : : : "memory");
  100. #define __io_bw() __asm__ __volatile__ ("fence w,o" : : : "memory");
  101. #define __io_aw() do {} while (0)
  102. #define readb(c) ({ rt_uint8_t __v; __io_br(); __v = __raw_readb(c); __io_ar(); __v; })
  103. #define readw(c) ({ rt_uint16_t __v; __io_br(); __v = __raw_readw(c); __io_ar(); __v; })
  104. #define readl(c) ({ rt_uint32_t __v; __io_br(); __v = __raw_readl(c); __io_ar(); __v; })
  105. #define writeb(v,c) ({ __io_bw(); __raw_writeb((v),(c)); __io_aw(); })
  106. #define writew(v,c) ({ __io_bw(); __raw_writew((v),(c)); __io_aw(); })
  107. #define writel(v,c) ({ __io_bw(); __raw_writel((v),(c)); __io_aw(); })
  108. #if __riscv_xlen != 32
  109. #define readq(c) ({ rt_uint64_t __v; __io_br(); __v = __raw_readq(c); __io_ar(); __v; })
  110. #define writeq(v,c) ({ __io_bw(); __raw_writeq((v),(c)); __io_aw(); })
  111. #endif
  112. #endif