atom.s 1018 B

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. /*
  2. * R4000 user-level atomic operations
  3. */
  4. #define LL(base, rt) WORD $((060<<26)|((base)<<21)|((rt)<<16))
  5. #define SC(base, rt) WORD $((070<<26)|((base)<<21)|((rt)<<16))
  6. #define NOOP WORD $0x27
  7. TEXT ainc(SB), 1, $-4 /* long ainc(long *); */
  8. TEXT _xinc(SB), 1, $-4 /* void _xinc(long *); */
  9. MOVW R1, R2 /* address of counter */
  10. loop: MOVW $1, R3
  11. LL(2, 1)
  12. NOOP
  13. ADD R1,R3,R3
  14. SC(2, 3)
  15. NOOP
  16. BEQ R3,loop
  17. RET
  18. TEXT adec(SB), 1, $-4 /* long adec(long*); */
  19. TEXT _xdec(SB), 1, $-4 /* long _xdec(long *); */
  20. MOVW R1, R2 /* address of counter */
  21. loop1: MOVW $-1, R3
  22. LL(2, 1)
  23. NOOP
  24. ADD R1,R3,R3
  25. MOVW R3, R1
  26. SC(2, 3)
  27. NOOP
  28. BEQ R3,loop1
  29. RET
  30. /*
  31. * int cas(uint* p, int ov, int nv);
  32. */
  33. TEXT cas(SB), 1, $-4
  34. MOVW ov+4(FP), R2
  35. MOVW nv+8(FP), R3
  36. spincas:
  37. LL(1, 4) /* R4 = *R1 */
  38. NOOP
  39. BNE R2, R4, fail
  40. SC(1, 3) /* *R1 = R3 */
  41. NOOP
  42. BEQ R3, spincas /* R3 == 0 means store failed */
  43. MOVW $1, R1
  44. RET
  45. fail:
  46. MOVW $0, R1
  47. RET
  48. /* general-purpose abort */
  49. _trap:
  50. MOVD $0, R0
  51. MOVD 0(R0), R0
  52. RET