IB_AT.asm 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. ; Seven Kingdoms: Ancient Adversaries
  2. ;
  3. ; Copyright 1997,1998 Enlight Software Ltd.
  4. ;
  5. ; This program is free software: you can redistribute it and/or modify
  6. ; it under the terms of the GNU General Public License as published by
  7. ; the Free Software Foundation, either version 2 of the License, or
  8. ; (at your option) any later version.
  9. ;
  10. ; This program is distributed in the hope that it will be useful,
  11. ; but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. ; GNU General Public License for more details.
  14. ;
  15. ; You should have received a copy of the GNU General Public License
  16. ; along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. ;
  18. ;Filename : IB_AT.ASM
  19. ;Description : Blt a bitmap to the display surface buffer without color key transparency handling
  20. INCLUDE IMGFUN.inc
  21. INCLUDE COLCODE.inc
  22. .CODE
  23. ;----------- BEGIN OF FUNCTION IMGbltAreaTrans ------------
  24. ;
  25. ; Put an non-compressed bitmap on image buffer.
  26. ; It does not handle color key transparency.
  27. ; It can blt a specific area of the source image to the
  28. ; destination buffer instead of the whole source image.
  29. ;
  30. ; Syntax : IMGbltAreaTrans( imageBuf, pitch, desX, desY, bitmapBuf, srcX1, srcY1, srcX2, srcY2 )
  31. ;
  32. ; char *imageBuf - the pointer to the display surface buffer
  33. ; int pitch - pitch of the display surface buffer
  34. ; int desX, desY - where to put the area on the surface buffer
  35. ; char *bitmapPtr - the pointer to the bitmap buffer
  36. ; int srcX1, srcY1 - where to get the area on the source buffer
  37. ; srcX2, srcY2
  38. ;
  39. ;-------------------------------------------------
  40. ;
  41. ; Format of the bitmap data :
  42. ;
  43. ; <short> width
  44. ; <short> height
  45. ; <char..> bitmap image
  46. ;
  47. ;-------------------------------------------------
  48. PUBLIC IMGbltAreaTrans
  49. IMGbltAreaTrans PROC imageBuf, pitch, desX, desY, bitmapPtr, srcX1, srcY1, srcX2, srcY2
  50. LOCAL srcLineDiff
  51. STARTPROC
  52. MOV EAX, imageBuf ; store the address of the image buffer to a variable
  53. MOV image_buf, EAX
  54. ;------ get the bitmap width and height -----;
  55. MOV AX , DS
  56. MOV ES , AX
  57. MOV ESI, bitmapPtr
  58. XOR EAX, EAX
  59. LODSW ; get bitmap width
  60. MOV EBX, EAX
  61. ADD ESI, 2 ; by pass the bitmap height, we don't need it. srcY2 and srcY1 will give us the data we need
  62. MUL srcY1 ; calculate the source starting address
  63. ADD EAX, srcX1 ; bitmap width * srcY1 + srcX1
  64. ADD ESI, EAX
  65. MOV EAX, srcX2 ; srcLineDiff = bitmap width - (srcX2-srcX1+1)
  66. SUB EAX, srcX1
  67. INC EAX
  68. MOV srcLineDiff, EBX
  69. SUB srcLineDiff, EAX
  70. MOV EBX, EAX ; EBX - line pixel copy count
  71. MOV EDX, pitch ; EDX = lineDiff
  72. SUB EDX, EAX ; lineDiff = image_width - (srcX2-srcX1+1)
  73. MOV ECX, srcY2 ; blt lines = srcY2-srcY1+1
  74. SUB ECX, srcY1
  75. INC ECX
  76. CLD ; clear direction flag for MOVSB
  77. ;---------- pixels copying loop ----------;
  78. CALC_ADDR_2 EDI, desX, desY, srcX1, srcY1, pitch ; Get the address to the destination buffer
  79. @@moreLines:
  80. PUSH ECX
  81. MOV ECX, EBX
  82. SHR ECX, 2
  83. JZ SHORT @@nextScan
  84. ;-----------------------------------------------------------------------//
  85. ; The idea here is to not branch very often so we unroll the loop by four
  86. ; and try to not branch when a whole run of pixels is either transparent
  87. ; or not transparent.
  88. ;
  89. ; There are two loops. One loop is for a run of pixels equal to the
  90. ; transparent color, the other is for runs of pixels we need to store.
  91. ;
  92. ; When we detect a "bad" pixel we jump to the same position in the
  93. ; other loop.
  94. ;
  95. ; Here is the loop we will stay in as long as we encounter a "transparent"
  96. ; pixel in the source.
  97. ;-----------------------------------------------------------------------//
  98. align 4
  99. @@same:
  100. mov al, ds:[esi]
  101. cmp al, TRANSPARENT_CODE
  102. jne short @@diff0
  103. @@same0:
  104. mov al, ds:[esi+1]
  105. cmp al, TRANSPARENT_CODE
  106. jne short @@diff1
  107. @@same1:
  108. mov al, ds:[esi+2]
  109. cmp al, TRANSPARENT_CODE
  110. jne short @@diff2
  111. @@same2:
  112. mov al, ds:[esi+3]
  113. cmp al, TRANSPARENT_CODE
  114. jne short @@diff3
  115. @@same3:
  116. add edi,4
  117. add esi,4
  118. dec ecx
  119. jnz short @@same
  120. jz short @@nextScan
  121. ;-----------------------------------------------------------------------//
  122. ; Here is the loop we will stay in as long as
  123. ; we encounter a "non transparent" pixel in the source.
  124. ;-----------------------------------------------------------------------//
  125. align 4
  126. @@diff:
  127. mov al, ds:[esi]
  128. cmp al, TRANSPARENT_CODE
  129. je short @@same0
  130. @@diff0:
  131. mov es:[edi],al
  132. mov al, ds:[esi+1]
  133. cmp al, TRANSPARENT_CODE
  134. je short @@same1
  135. @@diff1:
  136. mov es:[edi+1],al
  137. mov al, ds:[esi+2]
  138. cmp al, TRANSPARENT_CODE
  139. je short @@same2
  140. @@diff2:
  141. mov es:[edi+2],al
  142. mov al, ds:[esi+3]
  143. cmp al, TRANSPARENT_CODE
  144. je short @@same3
  145. @@diff3:
  146. mov es:[edi+3],al
  147. add edi,4
  148. add esi,4
  149. dec ecx
  150. jnz short @@diff
  151. jz short @@nextScan
  152. ;-----------------------------------------------------------------------//
  153. ; We are at the end of a scan, check for odd leftover pixels to do
  154. ; and go to the next scan.
  155. ;-----------------------------------------------------------------------//
  156. align 4
  157. @@nextScan:
  158. mov ecx,ebx ; ebx = bitmap width
  159. and ecx,11b ; if its pixel count is an odd number
  160. jnz short @@oddStuff
  161. ;-----------------------------------------------------------------------//
  162. ; move on to the start of the next line
  163. ;-----------------------------------------------------------------------//
  164. @@nextScan1:
  165. add edi, edx ; edx = lineDiff
  166. add esi, srcLineDiff
  167. pop ecx
  168. loop @@moreLines
  169. jmp short @@end
  170. ;-----------------------------------------------------------------------//
  171. ; If the width is not a multiple of 4 we will come here to clean up
  172. ; the last few pixels
  173. ;-----------------------------------------------------------------------//
  174. @@oddStuff:
  175. inc ecx
  176. @@oddLoop:
  177. dec ecx
  178. jz short @@nextScan1
  179. mov al, ds:[esi]
  180. inc esi
  181. inc edi
  182. cmp al, TRANSPARENT_CODE
  183. je short @@oddLoop
  184. mov es:[edi-1],al
  185. jmp short @@oddLoop
  186. @@end: ENDPROC
  187. IMGbltAreaTrans ENDP
  188. ;----------- END OF FUNCTION IMGbltAreaTrans ----------
  189. END