atomicops.s
上传用户:zhongxx05
上传日期:2007-06-06
资源大小:33641k
文件大小:7k
源码类别:

Symbian

开发平台:

C/C++

  1. // ***** BEGIN LICENSE BLOCK *****
  2. // Source last modified: $Id: atomicops.s,v 1.1 2003/08/03 21:55:06 dcollins Exp $
  3. //
  4. // Portions Copyright (c) 1995-2003 RealNetworks, Inc. All Rights Reserved.
  5. //
  6. // The contents of this file, and the files included with this file,
  7. // are subject to the current version of the RealNetworks Public
  8. // Source License (the "RPSL") available at
  9. // http://www.helixcommunity.org/content/rpsl unless you have licensed
  10. // the file under the current version of the RealNetworks Community
  11. // Source License (the "RCSL") available at
  12. // http://www.helixcommunity.org/content/rcsl, in which case the RCSL
  13. // will apply. You may also obtain the license terms directly from
  14. // RealNetworks.  You may not use this file except in compliance with
  15. // the RPSL or, if you have a valid RCSL with RealNetworks applicable
  16. // to this file, the RCSL.  Please see the applicable RPSL or RCSL for
  17. // the rights, obligations and limitations governing use of the
  18. // contents of the file.
  19. //
  20. // This file is part of the Helix DNA Technology. RealNetworks is the
  21. // developer of the Original Code and owns the copyrights in the
  22. // portions it created.
  23. //
  24. // This file, and the files included with this file, is distributed
  25. // and made available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY
  26. // KIND, EITHER EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS
  27. // ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES
  28. // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET
  29. // ENJOYMENT OR NON-INFRINGEMENT.
  30. //
  31. // Technology Compatibility Kit Test Suite(s) Location:
  32. //    http://www.helixcommunity.org/content/tck
  33. //
  34. // Contributor(s):
  35. //
  36. // ***** END LICENSE BLOCK ***** */
  37. //////////////////////////////////////////////////////////////////////
  38. //
  39. // atomicops.s
  40. //      Several atomic operators, implemented in IA64 (Itanium) assembly
  41. //
  42. // Implementation Notes:
  43. //      A work-in-progress....
  44. //
  45. //////////////////////////////////////////////////////////////////////
  46. //////////////////////////////////////////////////////////////////////
  47. // Misc. Assembler Directives
  48. //////////////////////////////////////////////////////////////////////
  49.         .radix  C       // C-style numeric constants
  50.         .section .text = "ax", "progbits" // read-only object code
  51. //////////////////////////////////////////////////////////////////////
  52. //
  53. // _HXAtomicIncRetUINT32
  54. //      Atomically increment by 1 and return new value
  55. //
  56. // Interface:
  57. //   extern "C" UINT32 _HXAtomicIncRetUINT32(UINT32* pNum);
  58. //
  59. // Inputs:
  60. //   Paramaters:
  61. //     in0 :  UINT32* pNum - pointer to integer to modify
  62. //
  63. // Outputs:
  64. //   Atomically modifies memory at *pNum:
  65. //     *pNum = *pNum + 1
  66. //
  67. //   Return value:
  68. //     r8 : UINT32 - new value of *pNum
  69. //
  70. //////////////////////////////////////////////////////////////////////
  71.         .proc   _HXAtomicIncRetUINT32
  72.         .global _HXAtomicIncRetUINT32
  73. _HXAtomicIncRetUINT32:
  74.         alloc           loc0 = ar.pfs, 1, 1, 0, 0 //one input, one local
  75.         fetchadd4.acq   r8 = [in0], 1;;         //increment *plock
  76.         add             r8 = 1, r8;;            //update local copy
  77. br.ret.dptk.few rp;;                    //return new value
  78.         .endp   _HXAtomicIncRetUINT32
  79. //////////////////////////////////////////////////////////////////////
  80. //
  81. // _HXAtomicDecRetUINT32
  82. //      Atomically decrement by 1 and return new value
  83. //
  84. // Interface:
  85. //   extern "C" UINT32 _HXAtomicDecRetUINT32(UINT32* pNum);
  86. //
  87. // Inputs:
  88. //   Paramaters:
  89. //     in0 :  UINT32* pNum - pointer to integer to modify
  90. //
  91. // Outputs:
  92. //   Atomically modifies memory at *pNum:
  93. //     *pNum = *pNum - 1
  94. //
  95. //   Return value:
  96. //     r8 : UINT32 - new value of *pNum
  97. //
  98. //////////////////////////////////////////////////////////////////////
  99.         .proc   _HXAtomicDecRetUINT32
  100.         .global _HXAtomicDecRetUINT32
  101. _HXAtomicDecRetUINT32:
  102.         alloc           loc0 = ar.pfs, 1, 1, 0, 0 //one input, one local
  103.         fetchadd4.acq   r8 = [in0], -1;;        //decrement *pLock
  104.         add             r8 = -1, r8;;           //update local copy
  105. br.ret.dptk.few rp;;                    //return new value
  106.         .endp   _HXAtomicDecRetUINT32
  107. //////////////////////////////////////////////////////////////////////
  108. //
  109. // _HXAtomicAddRetUINT32
  110. //      Atomically add n and return new value
  111. //
  112. // Interface:
  113. //   extern "C" UINT32 _HXAtomicAddRetUINT32(UINT32* pNum, UINT32 ulNum);
  114. //
  115. // Inputs:
  116. //   Paramaters:
  117. //     in0 :  UINT32* pNum  - pointer to integer to modify
  118. //     in1 :  UINT32  ulNum - amount to increment by
  119. //
  120. // Outputs:
  121. //   Atomically modifies memory at *pNum:
  122. //     *pNum = *pNum + ulNum
  123. //
  124. //   Return value:
  125. //     r8 : UINT32 - new value of *pNum
  126. //
  127. //////////////////////////////////////////////////////////////////////
  128.         .proc   _HXAtomicAddRetUINT32
  129.         .global _HXAtomicAddRetUINT32
  130. _HXAtomicAddRetUINT32:
  131.         alloc           loc4 = ar.pfs, 2, 5, 0, 0 // two inputs, five locals
  132.         mov             loc3 = ar.ccv;;         //save the CCV register
  133. atomic_back1:                                   //
  134.         ld4             loc0 = [in0];;          //loc0 = current value of *pNum
  135.         mov             ar.ccv = loc0;;         //copy it to the CCV register
  136.         add             loc1 = loc0, in1;;      //compute new value
  137.         cmpxchg4.acq    r8 = [in0], loc1;;      //write new val if *pLock == CCV
  138.         cmp4.eq         p1, p2 = loc0, r8;;     //check success of cmpxchg4
  139.    (p2) br.cond.dpnt.few atomic_back1;;         //if p2=1 (it failed) try again
  140.         add             r8 = r8, in1;;          //update local copy of *pNum
  141.         mov             ar.ccv = loc3;;         //restore the CCV register
  142. br.ret.dptk.few rp;;                    //return new value of *pNum
  143.         .endp   _HXAtomicAddRetUINT32
  144. //////////////////////////////////////////////////////////////////////
  145. //
  146. // _HXAtomicSubRetUINT32
  147. //      Atomically subtract n and return new value
  148. //
  149. // Interface:
  150. //   extern "C" UINT32 _HXAtomicSubRetUINT32(UINT32* pNum, UINT32 ulNum);
  151. //
  152. // Inputs:
  153. //   Paramaters:
  154. //     in0 :  UINT32* pNum  - pointer to integer to modify
  155. //     in1 :  UINT32  ulNum - amount to increment by
  156. //
  157. // Outputs:
  158. //   Atomically modifies memory at *pNum:
  159. //     *pNum = *pNum + ulNum
  160. //
  161. //   Return value:
  162. //     r8 : UINT32 - new value of *pNum
  163. //
  164. //////////////////////////////////////////////////////////////////////
  165.         .proc   _HXAtomicSubRetUINT32
  166.         .global _HXAtomicSubRetUINT32
  167. _HXAtomicSubRetUINT32:
  168.         alloc           loc4 = ar.pfs, 2, 5, 0, 0 // two inputs, five locals
  169.         mov             loc3 = ar.ccv;;         //save the CCV register
  170. atomic_back2:                                   //
  171.         ld4             loc0 = [in0];;          //loc0 = current value of *pNum
  172.         mov             ar.ccv = loc0;;         //copy it to the CCV register
  173.         sub             loc1 = loc0, in1;;      //compute new value
  174.         cmpxchg4.acq    r8 = [in0], loc1;;      //write new val if *pLock == CCV
  175.         cmp4.eq         p1, p2 = loc0, r8;;     //check success of cmpxchg4
  176.    (p2) br.cond.dpnt.few atomic_back2;;         //if p2=1 (it failed) try again
  177.         sub             r8 = r8, in1;;          //update local copy of *pNum
  178.         mov             ar.ccv = loc3;;         //restore the CCV register
  179. br.ret.dptk.few rp;;                    //return new value of *pNum
  180.         .endp   _HXAtomicSubRetUINT32