// // Copyright 2020 Electronic Arts Inc. // // TiberianDawn.DLL and RedAlert.dll and corresponding source code is free // software: you can redistribute it and/or modify it under the terms of // the GNU General Public License as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // TiberianDawn.DLL and RedAlert.dll and corresponding source code is distributed // in the hope that it will be useful, but with permitted additional restrictions // under Section 7 of the GPL. See the GNU General Public License in LICENSE.TXT // distributed with this program. You should have received a copy of the // GNU General Public License along with permitted additional restrictions // with this program. If not, see https://github.com/electronicarts/CnC_Remastered_Collection /* ** ** Misc. assembly code moved from headers ** ** ** ** ** */ #include "FUNCTION.H" extern "C" void __cdecl Mem_Copy(void const *source, void *dest, unsigned long bytes_to_copy) { memcpy(dest, source, bytes_to_copy); } /*********************************************************************************************** * Distance -- Determines the lepton distance between two coordinates. * * * * This routine is used to determine the distance between two coordinates. It uses the * * Dragon Strike method of distance determination and thus it is very fast. * * * * INPUT: coord1 -- First coordinate. * * * * coord2 -- Second coordinate. * * * * OUTPUT: Returns the lepton distance between the two coordinates. * * * * WARNINGS: none * * * * HISTORY: * * 05/27/1994 JLB : Created. * *=============================================================================================*/ int Distance_Coord(COORDINATE coord1, COORDINATE coord2) { __asm { mov eax,[coord1] mov ebx,[coord2] mov dx,ax sub dx,bx jg okx neg dx okx: shr eax,16 shr ebx,16 sub ax,bx jg oky neg ax oky: cmp ax,dx jg ok xchg ax,dx ok: shr dx,1 add ax,dx } } /* ;*************************************************************************** ;* DESIRED_FACING16 -- Converts coordinates into a facing number. * ;* * ;* This converts coordinates into a desired facing number that ranges * ;* from 0 to 15 (0 equals North and going clockwise). * ;* * ;* INPUT: x1,y1 -- Position of origin point. * ;* * ;* x2,y2 -- Position of target. * ;* * ;* OUTPUT: Returns desired facing as a number from 0 to 255 but * ;* accurate to 22.5 degree increments. * ;* * ;* WARNINGS: If the two coordinates are the same, then -1 will be * ;* returned. It is up to you to handle this case. * ;* * ;* HISTORY: * ;* 08/14/1991 JLB : Created. * ;*=========================================================================* */ long __cdecl Desired_Facing16(long x1, long y1, long x2, long y2) { static const char _new_facing16[] = { 3, 2, 4,-1, 1, 2,0,-1, 13,14,12,-1,15,14,0,-1, 5, 6, 4,-1, 7, 6,8,-1, 11,10,12,-1, 9,10,8,-1 }; __asm { xor ebx,ebx //; Index byte (built). //; Determine Y axis difference. mov edx,[y1] mov ecx,[y2] sub edx,ecx //; DX = Y axis (signed). jns short absy inc ebx //; Set the signed bit. neg edx //; ABS(y) absy: //; Determine X axis difference. shl ebx,1 mov eax,[x1] mov ecx,[x2] sub ecx,eax //; CX = X axis (signed). jns short absx inc ebx //; Set the signed bit. neg ecx //; ABS(x) absx: //; Determine the greater axis. cmp ecx,edx jb short dxisbig xchg ecx,edx dxisbig: rcl ebx,1 //; Y > X flag bit. //; Determine the closeness or farness of lesser axis. mov eax,edx inc eax //; Round up. shr eax,1 inc eax //; Round up. shr eax,1 //; 1/4 of greater axis. cmp ecx,eax rcl ebx,1 //; Very close to major axis bit. sub edx,eax cmp edx,ecx rcl ebx,1 //; Very far from major axis bit. xor eax,eax mov al,[_new_facing16+ebx] //; Normalize to 0..FF range. shl eax,4 // ret } } /* ;*************************************************************************** ;* Desired_Facing256 -- Desired facing algorithm 0..255 resolution. * ;* * ;* This is a desired facing algorithm that has a resolution of 0 * ;* through 255. * ;* * ;* INPUT: srcx,srcy -- Source coordinate. * ;* * ;* dstx,dsty -- Destination coordinate. * ;* * ;* OUTPUT: Returns with the desired facing to face the destination * ;* coordinate from the position of the source coordinate. North * ;* is 0, East is 64, etc. * ;* * ;* WARNINGS: This routine is slower than the other forms of desired * ;* facing calculation. Use this routine when accuracy is * ;* required. * ;* * ;* HISTORY: * ;* 12/24/1991 JLB : Adapted. * ;*=========================================================================*/ int __cdecl Desired_Facing256(LONG srcx, LONG srcy, LONG dstx, LONG dsty) { __asm { xor ebx,ebx //; Facing number. ////; Determine absolute X delta and left/right direction. mov ecx,[dstx] sub ecx,[srcx] jge short xnotneg neg ecx mov ebx,11000000b //; Set bit 7 and 6 for leftward. xnotneg: //; Determine absolute Y delta and top/bottom direction. mov eax,[srcy] sub eax,[dsty] jge short ynotneg xor ebx,01000000b //; Complement bit 6 for downward. neg eax ynotneg: //; Set DX=64 for quadrants 0 and 2. mov edx,ebx and edx,01000000b xor edx,01000000b //; Determine if the direction is closer to the Y axis and make sure that //; CX holds the larger of the two deltas. This is in preparation for the //; divide. cmp eax,ecx jb short gotaxis xchg eax,ecx xor edx,01000000b //; Closer to Y axis so make DX=64 for quad 0 and 2. gotaxis: //; If closer to the X axis then add 64 for quadrants 0 and 2. If //; closer to the Y axis then add 64 for quadrants 1 and 3. Determined //; add value is in DX and save on stack. push edx //; Make sure that the division won't overflow. Reduce precision until //; the larger number is less than 256 if it appears that an overflow //; will occur. If the high byte of the divisor is not zero, then this //; guarantees no overflow, so just abort shift operation. test eax,0FFFFFF00h jnz short nooverflow again: test ecx,0FFFFFF00h jz short nooverflow shr ecx,1 shr eax,1 jmp short again nooverflow: //; Make sure that the division won't underflow (divide by zero). If //; this would occur, then set the quotient to $FF and skip divide. or ecx,ecx jnz short nounderflow mov eax,0FFFFFFFFh jmp short divcomplete //; Derive a pseudo angle number for the octant. The angle is based //; on $00 = angle matches long axis, $00 = angle matches $FF degrees. nounderflow: xor edx,edx shld edx,eax,8 //; shift high byte of eax into dl shl eax,8 div ecx divcomplete: //; Integrate the 5 most significant bits into the angle index. If DX //; is not zero, then it is 64. This means that the dividend must be negated //; before it is added into the final angle value. shr eax,3 pop edx or edx,edx je short noneg dec edx neg eax noneg: add eax,edx add eax,ebx and eax,0FFH // ret } } /* ;*************************************************************************** ;** C O N F I D E N T I A L --- W E S T W O O D A S S O C I A T E S ** ;*************************************************************************** ;* * ;* Project Name : Support Library * ;* * ;* File Name : FACING8.ASM * ;* * ;* Programmer : Joe L. Bostic * ;* * ;* Start Date : May 8, 1991 * ;* * ;* Last Update : February 6, 1995 [BWG] * ;* * ;*-------------------------------------------------------------------------* ;* Functions: * ;* Desired_Facing8 -- Determines facing to reach a position. * ;* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * IDEAL P386 MODEL USE32 FLAT GLOBAL C Desired_Facing8 :NEAR ; INCLUDE "wwlib.i" DATASEG ; 8 direction desired facing lookup table. Build the index according ; to the following bits: ; ; bit 3 = Is y2 < y1? ; bit 2 = Is x2 < x1? ; bit 1 = Is the ABS(x2-x1) < ABS(y2-y1)? ; bit 0 = Is the facing closer to a major axis? //NewFacing8 DB 1,2,1,0,7,6,7,0,3,2,3,4,5,6,5,4 // CODESEG */ /* ;*************************************************************************** ;* DESIRED_FACING8 -- Determines facing to reach a position. * ;* * ;* This routine will return with the most desirable facing to reach * ;* one position from another. It is accurate to a resolution of 0 to * ;* 7. * ;* * ;* INPUT: x1,y1 -- Position of origin point. * ;* * ;* x2,y2 -- Position of target. * ;* * ;* OUTPUT: Returns desired facing as a number from 0..255 with an * ;* accuracy of 32 degree increments. * ;* * ;* WARNINGS: If the two coordinates are the same, then -1 will be * ;* returned. It is up to you to handle this case. * ;* * ;* HISTORY: * ;* 07/15/1991 JLB : Documented. * ;* 08/08/1991 JLB : Same position check. * ;* 08/14/1991 JLB : New algorithm * ;* 02/06/1995 BWG : Convert to 32-bit * ;*=========================================================================* */ int __cdecl Desired_Facing8(long x1, long y1, long x2, long y2) { static const char _new_facing8[] = {1,2,1,0,7,6,7,0,3,2,3,4,5,6,5,4}; __asm { xor ebx,ebx //; Index byte (built). //; Determine Y axis difference. mov edx,[y1] mov ecx,[y2] sub edx,ecx //; DX = Y axis (signed). jns short absy inc ebx //; Set the signed bit. neg edx //; ABS(y) absy: //; Determine X axis difference. shl ebx,1 mov eax,[x1] mov ecx,[x2] sub ecx,eax //; CX = X axis (signed). jns short absx inc ebx //; Set the signed bit. neg ecx //; ABS(x) absx: //; Determine the greater axis. cmp ecx,edx jb short dxisbig xchg ecx,edx dxisbig: rcl ebx,1 //; Y > X flag bit. //; Determine the closeness or farness of lesser axis. mov eax,edx inc eax //; Round up. shr eax,1 cmp ecx,eax rcl ebx,1 //; Close to major axis bit. xor eax,eax mov al,[_new_facing8+ebx] //; Normalize to 0..FF range. shl eax,5 // ret } } #if (0) /* ; $Header: //depot/Projects/Mobius/QA/Project/Run/SOURCECODE/TIBERIANDAWN/MiscAsm.cpp#138 $ ;*************************************************************************** ;** C O N F I D E N T I A L --- W E S T W O O D A S S O C I A T E S ** ;*************************************************************************** ;* * ;* Project Name : Support Library * ;* * ;* File Name : FACING16.ASM * ;* * ;* Programmer : Joe L. Bostic * ;* * ;* Start Date : May 8, 1991 * ;* * ;* Last Update : February 6, 1995 [BWG] * ;* * ;*-------------------------------------------------------------------------* ;* Functions: * ;* Desired_Facing16 -- Converts coordinates into a facing number. * ;* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * IDEAL P386 MODEL USE32 FLAT GLOBAL C Desired_Facing16 :NEAR ; INCLUDE "wwlib.i" DATASEG ; 16 direction desired facing lookup table. Build the index according ; to the following bits: ; ; bit 4 = Is y2 < y1? ; bit 3 = Is x2 < x1? ; bit 2 = Is the ABS(x2-x1) < ABS(y2-y1)? ; bit 1 = Is the lesser absolute difference very close to zero? ; bit 0 = Is the lesser absolute difference very close to the greater dist? NewFacing16 DB 3, 2, 4,-1, 1, 2,0,-1 DB 13,14,12,-1,15,14,0,-1 DB 5, 6, 4,-1, 7, 6,8,-1 DB 11,10,12,-1, 9,10,8,-1 CODESEG ;*************************************************************************** ;* DESIRED_FACING16 -- Converts coordinates into a facing number. * ;* * ;* This converts coordinates into a desired facing number that ranges * ;* from 0 to 15 (0 equals North and going clockwise). * ;* * ;* INPUT: x1,y1 -- Position of origin point. * ;* * ;* x2,y2 -- Position of target. * ;* * ;* OUTPUT: Returns desired facing as a number from 0 to 255 but * ;* accurate to 22.5 degree increments. * ;* * ;* WARNINGS: If the two coordinates are the same, then -1 will be * ;* returned. It is up to you to handle this case. * ;* * ;* HISTORY: * ;* 08/14/1991 JLB : Created. * ;*=========================================================================* */ long __cdecl Desired_Facing16(long x1, long y1, long x2, long y2) { __asm { xor ebx,ebx ; Index byte (built). ; Determine Y axis difference. mov edx,[y1] mov ecx,[y2] sub edx,ecx //; DX = Y axis (signed). jns short absy inc ebx //; Set the signed bit. neg edx //; ABS(y) absy: //; Determine X axis difference. shl ebx,1 mov eax,[x1] mov ecx,[x2] sub ecx,eax //; CX = X axis (signed). jns short absx inc ebx //; Set the signed bit. neg ecx //; ABS(x) absx: //; Determine the greater axis. cmp ecx,edx jb short dxisbig xchg ecx,edx dxisbig: rcl ebx,1 //; Y > X flag bit. //; Determine the closeness or farness of lesser axis. mov eax,edx inc eax //; Round up. shr eax,1 inc eax //; Round up. shr eax,1 //; 1/4 of greater axis. cmp ecx,eax rcl ebx,1 //; Very close to major axis bit. sub edx,eax cmp edx,ecx rcl ebx,1 //; Very far from major axis bit. xor eax,eax mov al,[NewFacing16+ebx] //; Normalize to 0..FF range. shl eax,4 // ret } } #if (0) PROC Desired_Facing16 C near USES ebx, ecx, edx ARG x1:DWORD ARG y1:DWORD ARG x2:DWORD ARG y2:DWORD xor ebx,ebx ; Index byte (built). ; Determine Y axis difference. mov edx,[y1] mov ecx,[y2] sub edx,ecx ; DX = Y axis (signed). jns short ??absy inc ebx ; Set the signed bit. neg edx ; ABS(y) ??absy: ; Determine X axis difference. shl ebx,1 mov eax,[x1] mov ecx,[x2] sub ecx,eax ; CX = X axis (signed). jns short ??absx inc ebx ; Set the signed bit. neg ecx ; ABS(x) ??absx: ; Determine the greater axis. cmp ecx,edx jb short ??dxisbig xchg ecx,edx ??dxisbig: rcl ebx,1 ; Y > X flag bit. ; Determine the closeness or farness of lesser axis. mov eax,edx inc eax ; Round up. shr eax,1 inc eax ; Round up. shr eax,1 ; 1/4 of greater axis. cmp ecx,eax rcl ebx,1 ; Very close to major axis bit. sub edx,eax cmp edx,ecx rcl ebx,1 ; Very far from major axis bit. xor eax,eax mov al,[NewFacing16+ebx] ; Normalize to 0..FF range. shl eax,4 ret ENDP Desired_Facing16 END #endif #endif /* ;*********************************************************************************************** ;* Cardinal_To_Fixed -- Converts cardinal numbers into a fixed point number. * ;* * ;* This utility function will convert cardinal numbers into a fixed point fraction. The * ;* use of fixed point numbers occurs throughout the product -- since it is a convenient * ;* tool. The fixed point number is based on the formula: * ;* * ;* result = cardinal / base * ;* * ;* The accuracy of the fixed point number is limited to 1/256 as the lowest and up to * ;* 256 as the largest. * ;* * ;* INPUT: base -- The key number to base the fraction about. * ;* * ;* cardinal -- The other number (hey -- what do you call it?) * ;* * ;* OUTPUT: Returns with the fixed point number of the "cardinal" parameter as it relates * ;* to the "base" parameter. * ;* * ;* WARNINGS: none * ;* * ;* HISTORY: * ;* 02/17/1995 BWG : Created. * ;*=============================================================================================*/ unsigned int __cdecl Cardinal_To_Fixed(unsigned base, unsigned cardinal) { __asm { mov eax,0FFFFh //; establish default return value mov ebx,[base] or ebx, ebx jz retneg1 //; if base==0, return 65535 mov eax,[cardinal] //; otherwise, return (cardinal*256)/base shl eax,8 xor edx,edx div ebx retneg1: //ret } } #if (0) PROC Cardinal_To_Fixed C near USES ebx, edx ARG base:DWORD ARG cardinal:DWORD mov eax,0FFFFh ; establish default return value mov ebx,[base] or ebx,ebx jz near ??retneg1 ; if base==0, return 65535 mov eax,[cardinal] ; otherwise, return (cardinal*256)/base shl eax,8 xor edx,edx div ebx ??retneg1: ret ENDP Cardinal_To_Fixed #endif /* ;*********************************************************************************************** ;* Fixed_To_Cardinal -- Converts a fixed point number into a cardinal number. * ;* * ;* Use this routine to convert a fixed point number into a cardinal number. * ;* * ;* INPUT: base -- The base number that the original fixed point number was created from. * ;* * ;* fixed -- The fixed point number to convert. * ;* * ;* OUTPUT: Returns with the reconverted number. * ;* * ;* WARNINGS: none * ;* * ;* HISTORY: * ;* 02/17/1995 BWG : Created. * ;*=============================================================================================*/ unsigned int __cdecl Fixed_To_Cardinal(unsigned base, unsigned fixed) { // PROC Fixed_To_Cardinal C near // USES edx // ARG base:DWORD // ARG fixed:DWORD __asm { mov eax,[base] mul [fixed] add eax,080h //; eax = (base * fixed) + 0x80 test eax,0FF000000h //; if high byte set, return FFFF jnz rneg1 shr eax,8 //; else, return eax/256 jmp all_done rneg1: mov eax,0FFFFh //; establish default return value all_done: //ret } #if (0) mov eax,[base] mul [fixed] add eax,080h ; eax = (base * fixed) + 0x80 test eax,0FF000000h ; if high byte set, return FFFF jnz ??rneg1 shr eax,8 ; else, return eax/256 ret ??rneg1 : mov eax,0FFFFh ; establish default return value ret ENDP Fixed_To_Cardinal END #endif } void __cdecl Set_Bit(void * array, int bit, int value) { __asm { mov ecx, [bit] mov eax, [value] mov esi, [array] mov ebx,ecx shr ebx,5 and ecx,01Fh btr [esi+ebx*4],ecx or eax,eax jz ok bts [esi+ebx*4],ecx ok: } } int __cdecl Get_Bit(void const * array, int bit) { __asm { mov eax, [bit] mov esi, [array] mov ebx,eax shr ebx,5 and eax,01Fh bt [esi+ebx*4],eax setc al } } int __cdecl First_True_Bit(void const * array) { __asm { mov esi, [array] mov eax,-32 again: add eax,32 mov ebx,[esi] add esi,4 bsf ebx,ebx jz again add eax,ebx } } int __cdecl First_False_Bit(void const * array) { __asm { mov esi, [array] mov eax,-32 again: add eax,32 mov ebx,[esi] not ebx add esi,4 bsf ebx,ebx jz again add eax,ebx } } int __cdecl Bound(int original, int min, int max) { __asm { mov eax,[original] mov ebx,[min] mov ecx,[max] cmp ebx,ecx jl okorder xchg ebx,ecx okorder: cmp eax,ebx jg okmin mov eax,ebx okmin: cmp eax,ecx jl okmax mov eax,ecx okmax: } } CELL __cdecl Coord_Cell(COORDINATE coord) { __asm { mov eax, coord mov ebx,eax shr eax,010h xor al,al shr eax,2 or al,bh } } /* ;*********************************************************** ; SHAKE_SCREEN ; ; VOID Shake_Screen(int shakes); ; ; This routine shakes the screen the number of times indicated. ; ; Bounds Checking: None ; ;* */ void __cdecl Shake_Screen(int shakes) { // PG_TO_FIX // Need a different solution for shaking the screen shakes; } #if (0) GLOBAL C Shake_Screen :NEAR CODESEG ;*********************************************************** ; SHAKE_SCREEN ; ; VOID Shake_Screen(int shakes); ; ; This routine shakes the screen the number of times indicated. ; ; Bounds Checking: None ; ;* PROC Shake_Screen C near USES ecx, edx ARG shakes:DWORD ret mov ecx,[shakes] ;;; push es ;;; mov ax,40h ;;; mov es,ax ;;; mov dx,[es:63h] ;;; pop es mov eax,[0463h] ; get CRTC I/O port mov dx,ax add dl,6 ; video status port ??top_loop: ??start_retrace: in al,dx test al,8 jz ??start_retrace ??end_retrace: in al,dx test al,8 jnz ??end_retrace cli sub dl,6 ; dx = 3B4H or 3D4H mov ah,01 ; top word of start address mov al,0Ch out dx,al xchg ah,al inc dx out dx,al xchg ah,al dec dx mov ah,040h ; bottom word = 40 (140h) inc al out dx,al xchg ah,al inc dx out dx,al xchg ah,al sti add dl,5 ??start_retrace2: in al,dx test al,8 jz ??start_retrace2 ??end_retrace2: in al,dx test al,8 jnz ??end_retrace2 ??start_retrace3: in al,dx test al,8 jz ??start_retrace3 ??end_retrace3: in al,dx test al,8 jnz ??end_retrace3 cli sub dl,6 ; dx = 3B4H or 3D4H mov ah,0 mov al,0Ch out dx,al xchg ah,al inc dx out dx,al xchg ah,al dec dx mov ah,0 inc al out dx,al xchg ah,al inc dx out dx,al xchg ah,al sti add dl,5 loop ??top_loop ret ENDP Shake_Screen ;*********************************************************** END #endif /* ;*************************************************************************** ;* Conquer_Build_Fading_Table -- Builds custom shadow/light fading table. * ;* * ;* This routine is used to build a special fading table for C&C. There * ;* are certain colors that get faded to and cannot be faded again. * ;* With this rule, it is possible to draw a shadow multiple times and * ;* not have it get any lighter or darker. * ;* * ;* INPUT: palette -- Pointer to the 768 byte IBM palette to build from. * ;* * ;* dest -- Pointer to the 256 byte remap table. * ;* * ;* color -- Color index of the color to "fade to". * ;* * ;* frac -- The fraction to fade to the specified color * ;* * ;* OUTPUT: Returns with pointer to the remap table. * ;* * ;* WARNINGS: none * ;* * ;* HISTORY: * ;* 10/07/1992 JLB : Created. * ;*=========================================================================*/ void * __cdecl Conquer_Build_Fading_Table(void const *palette, void *dest, int color, int frac) { /* global C Conquer_Build_Fading_Table : NEAR PROC Conquer_Build_Fading_Table C near USES ebx, ecx, edi, esi ARG palette:DWORD ARG dest:DWORD ARG color:DWORD ARG frac:DWORD LOCAL matchvalue:DWORD ; Last recorded match value. LOCAL targetred:BYTE ; Target gun red. LOCAL targetgreen:BYTE ; Target gun green. LOCAL targetblue:BYTE ; Target gun blue. LOCAL idealred:BYTE LOCAL idealgreen:BYTE LOCAL idealblue:BYTE LOCAL matchcolor:BYTE ; Tentative match color. ALLOWED_COUNT EQU 16 ALLOWED_START EQU 256-ALLOWED_COUNT */ #define ALLOWED_COUNT 16 #define ALLOWED_START 256-ALLOWED_COUNT int matchvalue = 0; //:DWORD ; Last recorded match value. unsigned char targetred = 0; //BYTE ; Target gun red. unsigned char targetgreen = 0; //BYTE ; Target gun green. unsigned char targetblue = 0; //BYTE ; Target gun blue. unsigned char idealred = 0; //BYTE unsigned char idealgreen = 0; //BYTE unsigned char idealblue = 0; //BYTE unsigned char matchcolor = 0; //:BYTE ; Tentative match color. __asm { cld ; If the source palette is NULL, then just return with current fading table pointer. cmp [palette],0 je fini1 cmp [dest],0 je fini1 ; Fractions above 255 become 255. mov eax,[frac] cmp eax,0100h jb short ok mov [frac],0FFh ok: ; Record the target gun values. mov esi,[palette] mov ebx,[color] add esi,ebx add esi,ebx add esi,ebx lodsb mov [targetred],al lodsb mov [targetgreen],al lodsb mov [targetblue],al ; Main loop. xor ebx,ebx ; Remap table index. ; Transparent black never gets remapped. mov edi,[dest] mov [edi],bl inc edi ; EBX = source palette logical number (1..255). ; EDI = running pointer into dest remap table. mainloop: inc ebx mov esi,[palette] add esi,ebx add esi,ebx add esi,ebx mov edx,[frac] shr edx,1 ; new = orig - ((orig-target) * fraction); lodsb ; orig mov dh,al ; preserve it for later. sub al,[targetred] ; al = (orig-target) imul dl ; ax = (orig-target)*fraction shl eax,1 sub dh,ah ; dh = orig - ((orig-target) * fraction) mov [idealred],dh ; preserve ideal color gun value. lodsb ; orig mov dh,al ; preserve it for later. sub al,[targetgreen] ; al = (orig-target) imul dl ; ax = (orig-target)*fraction shl eax,1 sub dh,ah ; dh = orig - ((orig-target) * fraction) mov [idealgreen],dh ; preserve ideal color gun value. lodsb ; orig mov dh,al ; preserve it for later. sub al,[targetblue] ; al = (orig-target) imul dl ; ax = (orig-target)*fraction shl eax,1 sub dh,ah ; dh = orig - ((orig-target) * fraction) mov [idealblue],dh ; preserve ideal color gun value. ; Sweep through a limited set of existing colors to find the closest ; matching color. mov eax,[color] mov [matchcolor],al ; Default color (self). mov [matchvalue],-1 ; Ridiculous match value init. mov ecx,ALLOWED_COUNT mov esi,[palette] ; Pointer to original palette. add esi,(ALLOWED_START)*3 ; BH = color index. mov bh,ALLOWED_START innerloop: xor edx,edx ; Comparison value starts null. ; Build the comparison value based on the sum of the differences of the color ; guns squared. lodsb sub al,[idealred] mov ah,al imul ah add edx,eax lodsb sub al,[idealgreen] mov ah,al imul ah add edx,eax lodsb sub al,[idealblue] mov ah,al imul ah add edx,eax jz short perfect ; If perfect match found then quit early. cmp edx,[matchvalue] jae short notclose mov [matchvalue],edx ; Record new possible color. mov [matchcolor],bh notclose: inc bh ; Checking color index. loop innerloop mov bh,[matchcolor] perfect: mov [matchcolor],bh xor bh,bh ; Make BX valid main index again. ; When the loop exits, we have found the closest match. mov al,[matchcolor] stosb cmp ebx,ALLOWED_START-1 jne mainloop ; Fill the remainder of the remap table with values ; that will remap the color to itself. mov ecx,ALLOWED_COUNT fillerloop: inc bl mov al,bl stosb loop fillerloop fini1: mov esi,[dest] mov eax,esi //ret } } extern "C" long __cdecl Reverse_Long(long number) { __asm { mov eax,dword ptr [number] xchg al,ah ror eax,16 xchg al,ah } } extern "C" short __cdecl Reverse_Short(short number) { __asm { mov ax,[number] xchg ah,al } } extern "C" long __cdecl Swap_Long(long number) { __asm { mov eax,dword ptr [number] ror eax,16 } } /* ;*************************************************************************** ;* strtrim -- Remove the trailing white space from a string. * ;* * ;* Use this routine to remove white space characters from the beginning * ;* and end of the string. The string is modified in place by * ;* this routine. * ;* * ;* INPUT: buffer -- Pointer to the string to modify. * ;* * ;* OUTPUT: none * ;* * ;* WARNINGS: none * ;* * ;* HISTORY: * ;* 10/07/1992 JLB : Created. * ;*=========================================================================* ; VOID cdecl strtrim(BYTE *buffer); global C strtrim :NEAR PROC strtrim C near USES ax, edi, esi ARG buffer:DWORD ; Pointer to string to modify. */ void __cdecl strtrim(char *buffer) { __asm { cmp [buffer],0 je short fini ; Prepare for string scanning by loading pointers. cld mov esi,[buffer] mov edi,esi ; Strip white space from the start of the string. looper: lodsb cmp al,20h ; Space je short looper cmp al,9 ; TAB je short looper stosb ; Copy the rest of the string. gruntloop: lodsb stosb or al,al jnz short gruntloop dec edi ; Strip the white space from the end of the string. looper2: mov [edi],al dec edi mov ah,[edi] cmp ah,20h je short looper2 cmp ah,9 je short looper2 fini: //ret } } /* ;*************************************************************************** ;* Fat_Put_Pixel -- Draws a fat pixel. * ;* * ;* Use this routine to draw a "pixel" that is bigger than 1 pixel * ;* across. This routine is faster than drawing a similar small shape * ;* and faster than calling Fill_Rect. * ;* * ;* INPUT: x,y -- Screen coordinates to draw the pixel's upper * ;* left corner. * ;* * ;* color -- The color to render the pixel in. * ;* * ;* size -- The number of pixels width of the big "pixel". * ;* * ;* page -- The pointer to a GraphicBuffer class or something * ;* * ;* OUTPUT: none * ;* * ;* WARNINGS: none * ;* * ;* HISTORY: * ;* 03/17/1994 JLB : Created. * ;*=========================================================================* ; VOID cdecl Fat_Put_Pixel(long x, long y, long color, long size, void *page) global C Fat_Put_Pixel:NEAR PROC Fat_Put_Pixel C near USES eax, ebx, ecx, edx, edi, esi ARG x:DWORD ; X coordinate of upper left pixel corner. ARG y:DWORD ; Y coordinate of upper left pixel corner. ARG color:DWORD ; Color to use for the "pixel". ARG siz:DWORD ; Size of "pixel" to plot (square). ARG gpage:DWORD ; graphic page address to plot onto */ void __cdecl Fat_Put_Pixel(int x, int y, int color, int siz, GraphicViewPortClass &gpage) { __asm { cmp [siz],0 je short exit_label ; Set EDI to point to start of logical page memory. ;*=================================================================== ; Get the viewport information and put bytes per row in ecx ;*=================================================================== mov ebx,[gpage] ; get a pointer to viewport mov edi,[ebx]GraphicViewPortClass.Offset ; get the correct offset ; Verify the the Y pixel offset is legal. mov eax,[y] cmp eax,[ebx]GraphicViewPortClass.Height ;YPIXEL_MAX jae short exit_label mov ecx,[ebx]GraphicViewPortClass.Width add ecx,[ebx]GraphicViewPortClass.XAdd add ecx,[ebx]GraphicViewPortClass.Pitch mul ecx add edi,eax ; Verify the the X pixel offset is legal. mov edx,[ebx]GraphicViewPortClass.Width cmp edx,[x] mov edx,ecx jbe short exit_label add edi,[x] ; Write the pixel to the screen. mov ebx,[siz] ; Copy of pixel size. sub edx,ebx ; Modulo to reach start of next row. mov eax,[color] again: mov ecx,ebx rep stosb add edi,edx ; EDI points to start of next row. dec [siz] jnz short again exit_label: //ret } } /* ;*************************************************************************** ;** C O N F I D E N T I A L --- W E S T W O O D A S S O C I A T E S ** ;*************************************************************************** ;* * ;* Project Name : Westwood Library * ;* * ;* File Name : CRC.ASM * ;* * ;* Programmer : Joe L. Bostic * ;* * ;* Start Date : June 12, 1992 * ;* * ;* Last Update : February 10, 1995 [BWG] * ;* * ;*-------------------------------------------------------------------------* ;* Functions: * ;* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * IDEAL P386 MODEL USE32 FLAT GLOBAL C Calculate_CRC :NEAR CODESEG */ extern "C" long __cdecl Calculate_CRC(void *buffer, long length) { unsigned long crc; unsigned long local_length = (unsigned long) length; __asm { ; Load pointer to data block. mov [crc],0 pushad mov esi,[buffer] cld ; Clear CRC to default (NULL) value. xor ebx,ebx //; Fetch the length of the data block to CRC. mov ecx,[local_length] jecxz short fini ; Prepare the length counters. mov edx,ecx and dl,011b shr ecx,2 ; Perform the bulk of the CRC scanning. jecxz short remainder2 accumloop: lodsd rol ebx,1 add ebx,eax loop accumloop ; Handle the remainder bytes. remainder2: or dl,dl jz short fini mov ecx,edx xor eax,eax and ecx,0FFFFh push ecx nextbyte: lodsb ror eax,8 loop nextbyte pop ecx neg ecx add ecx,4 shl ecx,3 ror eax,cl ;nextbyte: ; shl eax,8 ; lodsb ; loop nextbyte rol ebx,1 add ebx,eax fini: mov [crc],ebx popad mov eax,[crc] //ret } } extern "C" void __cdecl Set_Palette_Range(void *palette) { if (palette == NULL) { return; } memcpy(CurrentPalette, palette, 768); Set_DD_Palette(palette); }