helpers.h 1.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556
  1. /* SPDX-License-Identifier: GPL-2.0 */
  2. #ifndef __VDSO_HELPERS_H
  3. #define __VDSO_HELPERS_H
  4. #ifndef __ASSEMBLY__
  5. #include <vdso/datapage.h>
  6. static __always_inline u32 vdso_read_begin(const struct vdso_data *vd)
  7. {
  8. u32 seq;
  9. while (unlikely((seq = READ_ONCE(vd->seq)) & 1))
  10. cpu_relax();
  11. smp_rmb();
  12. return seq;
  13. }
  14. static __always_inline u32 vdso_read_retry(const struct vdso_data *vd,
  15. u32 start)
  16. {
  17. u32 seq;
  18. smp_rmb();
  19. seq = READ_ONCE(vd->seq);
  20. return seq != start;
  21. }
  22. static __always_inline void vdso_write_begin(struct vdso_data *vd)
  23. {
  24. /*
  25. * WRITE_ONCE it is required otherwise the compiler can validly tear
  26. * updates to vd[x].seq and it is possible that the value seen by the
  27. * reader it is inconsistent.
  28. */
  29. WRITE_ONCE(vd[CS_HRES_COARSE].seq, vd[CS_HRES_COARSE].seq + 1);
  30. WRITE_ONCE(vd[CS_RAW].seq, vd[CS_RAW].seq + 1);
  31. smp_wmb();
  32. }
  33. static __always_inline void vdso_write_end(struct vdso_data *vd)
  34. {
  35. smp_wmb();
  36. /*
  37. * WRITE_ONCE it is required otherwise the compiler can validly tear
  38. * updates to vd[x].seq and it is possible that the value seen by the
  39. * reader it is inconsistent.
  40. */
  41. WRITE_ONCE(vd[CS_HRES_COARSE].seq, vd[CS_HRES_COARSE].seq + 1);
  42. WRITE_ONCE(vd[CS_RAW].seq, vd[CS_RAW].seq + 1);
  43. }
  44. #endif /* !__ASSEMBLY__ */
  45. #endif /* __VDSO_HELPERS_H */