123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590 |
- //
- // 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
- /***************************************************************************
- ** 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 : ALLOC.CPP *
- * *
- * Programmer : Joe L. Bostic *
- * *
- * Start Date : February 1, 1992 *
- * *
- * Last Update : March 9, 1995 [JLB] *
- * *
- *-------------------------------------------------------------------------*
- * Functions: *
- * Alloc -- Allocates system RAM. *
- * Ram_Free -- Determines the largest free chunk of RAM. *
- * Free -- Free an Alloc'ed block of RAM. *
- * Resize_Alloc -- Change the size of an allocated block. *
- * Heap_Size -- Size of the heap we have. *
- * Total_Ram_Free -- Total amount of free RAM. *
- * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
- #if (0)
- #include <malloc.h>
- #include <string.h>
- #include <stdlib.h>
- #include <dos.h>
- #include <bios.h>
- #include <stdio.h>
- #ifndef WWMEM_H
- #include "wwmem.h"
- #endif
- extern "C" unsigned long Largest_Mem_Block ( void ) ;
- //
- // use double-word alignment for allocs
- //
- #define LONG_ALIGNMENT 1
- /*
- ** Define the equates necessary to call a DPMI interrupt.
- */
- #define DPMI_INT 0x0031
- #define DPMI_LOCK_MEM 0x0600
- #define DPMI_UNLOCK_MEM 0x0601
- #define LOGGING FALSE
- /*=========================================================================*/
- /* The following PRIVATE functions are in this file: */
- /*=========================================================================*/
- /*= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =*/
- unsigned long MinRam=0L; // Record of least memory at worst case.
- unsigned long MaxRam=0L; // Record of total allocated at worst case.
- static unsigned long TotalRam = 0L;
- static unsigned long Memory_Calls = 0L;
- static unsigned long RequestedSystemRam = 8*1024*1024;
- static unsigned long LargestRamBlock = 0L;
- void (*Memory_Error)(void) = NULL;
- void (*Memory_Error_Exit)(char *string) = NULL;
- /***************************************************************************
- * DPMI_LOCK -- handles locking a block of DPMI memory *
- * *
- * INPUT: *
- * *
- * OUTPUT: *
- * *
- * WARNINGS: *
- * *
- * HISTORY: *
- * 06/23/1995 PWG : Created. *
- *=========================================================================*/
- #include"mono.h"
- void DPMI_Lock(VOID const *ptr, long const size)
- {
- union REGS regs;
- struct SREGS sregs;
- /*
- ** Lock memory
- ** AX = 0x600
- ** BX:CX = starting linear address of memory to lock
- ** SI:DI = size of region to lock (in bytes)
- ** - If Failure, carry flag is set.
- */
- memset (®s, 0 ,sizeof(regs));
- segread (&sregs);
- regs.x.eax = DPMI_LOCK_MEM;
- regs.x.ebx = ((long)ptr & 0xffff0000) >> 16;
- regs.x.ecx = ((long)ptr & 0x0000ffff);
- regs.x.esi = ((long)size & 0xffff0000) >> 16;
- regs.x.edi = ((long)size & 0x0000ffff);
- int386x (DPMI_INT, ®s, ®s, &sregs); // call DPMI
- // if (regs.x.cflag) {
- // }
- #if(0)
- char *temp = (char *)ptr;
- char hold;
- for (int lp = 0; lp < size; lp += 2048) {
- hold = *temp;
- temp += 2048;
- }
- #endif
- }
- /***************************************************************************
- * DPMI_UNLOCK -- Handles unlocking a locked block of DPMI *
- * *
- * INPUT: *
- * *
- * OUTPUT: *
- * *
- * WARNINGS: *
- * *
- * HISTORY: *
- * 06/23/1995 PWG : Created. *
- *=========================================================================*/
- void DPMI_Unlock(void const *ptr, long const size)
- {
- union REGS regs;
- struct SREGS sregs;
- /*
- ** Unlock the memory
- */
- memset (®s, 0 ,sizeof(regs));
- segread (&sregs);
- regs.x.eax = DPMI_UNLOCK_MEM; // DPMI function to call
- regs.x.ebx = ((long)ptr & 0xffff0000) >> 16;
- regs.x.ecx = ((long)ptr & 0x0000ffff);
- regs.x.esi = ((long)size & 0xffff0000) >> 16;
- regs.x.edi = ((long)size & 0x0000ffff);
- int386x (DPMI_INT, ®s, ®s, &sregs); // call DPMI
- // if (regs.x.cflag) {
- // }
- }
- /***************************************************************************
- * Alloc -- Allocates system RAM. *
- * *
- * This is the basic RAM allocation function. It is used for all *
- * memory allocations needed by the system or the main program. *
- * *
- * INPUT: bytes_to_alloc -- LONG value of the number of bytes to alloc. *
- * *
- * flags -- Memory allocation control flags. *
- * MEM_NORMAL: No special flags. *
- * MEM_CLEAR: Zero out memory block. *
- * MEM_NEW: Called by a new. *
- * *
- * OUTPUT: Returns with pointer to allocated block. If NULL was returned *
- * it indicates a failure to allocate. Note: NULL will never be *
- * returned if the standard library allocation error routine is *
- * used. *
- * *
- * WARNINGS: If you replace the standard memory allocation error routine *
- * and make it so that Alloc CAN return with a NULL, be sure *
- * and check for this in your code. *
- * *
- * HISTORY: *
- * 09/03/1991 JLB : Documented. *
- * 08/09/1993 JLB : Updated with EMS memory support. *
- * 04/28/1994 JAW : Updated to 32bit Protected mode. *
- * 03/09/1995 JLB : Fixed *
- *=========================================================================*/
- void *Alloc(unsigned long bytes_to_alloc, MemoryFlagType flags)
- {
- union REGS regs ;
- struct SREGS sregs ;
- unsigned char *retval=NULL; // Pointer to allocated block.
- unsigned long original_size; // Original allocation size.
- unsigned long bytesfree; // Number of free bytes.
- long *longptr=NULL; // Pointer used to store selector
- static unsigned char _allocinit=0;
- //
- // Init memory system by finding largest block to alloc
- // then allocate it to get one large heap and free it.
- // There may be more memory available from DPMI but we only are
- // for now allocating and freeing the first largest block.
- //
- if ( !_allocinit ) {
- unsigned long largestblock = Largest_Mem_Block();
- largestblock -= 1024; // subtract for heap header and misc
- largestblock &= 0xffff0000; // forcing to 64K boundary
- if ( largestblock ) {
- LargestRamBlock = MIN( largestblock, RequestedSystemRam );
- unsigned char *lptr = (unsigned char *)malloc( LargestRamBlock );
- if ( lptr ) {
- free( (void *)lptr );
- }
- }
- /*
- ** Initialize the total ram available value.
- */
- TotalRam = Total_Ram_Free(MEM_NORMAL);
- _allocinit = 1;
- }
- /*
- ** Save the original allocated space size so that we can clear the
- ** exact amount of RAM if they specified MEM_CLEAR.
- */
- original_size = bytes_to_alloc;
- /*
- ** Reserve one byte for the header of the memory we allocated.
- ** We will store the flags variable there for later use.
- */
- #if (LONG_ALIGNMENT)
- bytes_to_alloc += (flags & MEM_LOCK) ? 8 : 4;
- #else
- bytes_to_alloc += (flags & MEM_LOCK) ? 5 : 1;
- #endif
- // Try to allocate the memory out of the protected mode memory
- // chain if we did not require a real mode allocation. If this
- // fails we will have to try to allocate it out of real mode memory.
- // Real mode memory is a last resort because some types of applications
- // require real mode memory.
- if (!(flags & MEM_REAL)) {
- retval = (unsigned char*)malloc(bytes_to_alloc);
- }
- // Try to allocate the memory out of the real mode memory using DPMI
- // service 0x100. Note that retval will be null if we are requesting
- // real mode memory so that we do not have to explicitly check for the
- // real mode flag. Remember we need to reserve room for the dos
- // selector value at the beginning of our allocated block so rather than
- // adding fifteen and rounding, we need to add 19 and round.
- if (!retval) {
- flags = (MemoryFlagType)(flags | MEM_REAL);
- regs.x.eax = 0x100;
- regs.x.ebx = (bytes_to_alloc + 19) >> 4;
- if (regs.x.ebx & 0xFFFF0000) {
- retval = NULL;
- } else {
- segread ( & sregs ) ;
- int386x ( 0x31 , & regs, & regs , & sregs ) ;
- if (regs.x.cflag)
- retval = NULL;
- else {
- #if (LONG_ALIGNMENT)
- longptr = (long *)(((regs.x.eax & 0xFFFF) << 4)+ 4);
- #else
- longptr = (long *)(((regs.x.eax & 0xFFFF) << 4)+ 1);
- #endif
- *longptr++ = regs.x.edx & 0xFFFF;
- retval = (unsigned char *)longptr;
- }
- }
- }
- // If the alloc failed then we need to signify a memory error.
- if (retval == NULL) {
- if (Memory_Error != NULL)
- Memory_Error();
- return NULL;
- }
- // If the memory needs to be DPMI locked then we should store the
- // original size in the header before we store the flags.
- if (flags & MEM_LOCK) {
- longptr = (long *)retval;
- *longptr++ = original_size;
- retval = (unsigned char *)longptr;
- }
- // Now that we know the alloc was sucessful (and for an extra byte
- // more than the user wanted) we need to stick in the memory flags.
- #if (LONG_ALIGNMENT)
- if ( !(flags & (MEM_LOCK|MEM_REAL)) ) {
- //
- // WARNING!!!!!!!!!!
- // USE this only with the WATCOM malloc ALLOCATION!!!!!!!!!
- // it reads the actual block size before the ptr returned.
- // then eors and uses the upper word for a validation later on free.
- //
- longptr = (long *)retval;
- *longptr = ((*(longptr - 1)) ^ 0xffffffff) & 0xffff0000;
- *retval++ = flags;
- *retval++ = (unsigned char)(flags ^ 0xff);
- retval += 2;
- }
- else {
- *retval++ = flags;
- *retval++ = (unsigned char)(flags ^ 0xff);
- *retval++ = 0;
- *retval++ = 0;
- }
- #else
- *retval++ = (unsigned char)(flags | (((flags ^ 0x07) & 0x07) << 5));
- #endif
- // If the memory needed to be DPMI locked then set it up so it
- // is locked.
- if (flags & MEM_LOCK) {
- DPMI_Lock(retval, original_size);
- }
- /* Clear the space if they wanted it clear */
- if (flags & MEM_CLEAR) {
- unsigned char *ptr; // Working memory block pointer.
- ptr = retval;
- memset(ptr, '\0', original_size);
- }
- bytesfree = Total_Ram_Free(MEM_NORMAL);
- if (bytesfree < MinRam) {
- MinRam = bytesfree;
- }
- if (TotalRam-bytesfree > MaxRam) {
- MaxRam = TotalRam-bytesfree;
- }
- Memory_Calls++;
- #if(LOGGING)
- int val = _heapchk();
- FILE *file = fopen("mem.txt","at");
- fprintf(file, "%P Alloc size = %d, Actual Size = %d, flags = %d, heap = %d\n",
- retval,
- original_size,
- bytes_to_alloc,
- flags,
- val);
- fclose(file);
- #endif
- return(retval);
- }
- /***************************************************************************
- * Free -- Free an Alloc'ed block of RAM. *
- * *
- * FUNCTION: *
- * *
- * INPUT: A pointer to a block of RAM from Alloc. *
- * *
- * OUTPUT: None. *
- * *
- * WARNINGS: Don't use this for an Alloc_Block'ed RAM block. *
- * *
- * HISTORY: *
- * 05/25/1990 : Created. *
- ***************************************************************************/
- void Free(void const *pointer)
- {
- union REGS regs ;
- struct SREGS sregs ;
- void const *original = pointer;
- char string[80];
- if (pointer) {
- /*
- ** Get a pointer to the flags that we stored off.
- */
- #if (LONG_ALIGNMENT)
- unsigned char *byteptr = ((unsigned char *)pointer) - 4;
- //
- // validate the flags with and eor of the flags
- //
- if ( *byteptr != ((*(byteptr + 1)) ^ 0xff) ) {
- if (Memory_Error_Exit != NULL) {
- sprintf( string, "Error freeing pointer %p. Header invalid!!!\n", pointer );
- Memory_Error_Exit( string );
- }
- }
- else {
- if ( !(*byteptr & (MEM_LOCK|MEM_REAL)) ) {
- unsigned short *wordptr = (unsigned short *)(byteptr - 2);
- //
- // WARNING!!!!!!!!!!
- // USE this only with the WATCOM malloc ALLOCATION!!!!!!!!!
- // it reads the actual block size before the ptr to be freed.
- // then compares with the EOR to the value stored during allocation.
- //
- if ( *wordptr != ((*(wordptr + 2)) ^ 0xffff) ) {
- if (Memory_Error_Exit != NULL) {
- sprintf( string, "Error freeing pointer %p. Header invalid!!!\n", pointer );
- Memory_Error_Exit( string );
- }
- }
- }
- else if ( *(byteptr + 2) || *(byteptr + 3) ) {
- if (Memory_Error_Exit != NULL) {
- sprintf( string, "Error freeing pointer %p. Header invalid!!!\n", pointer );
- Memory_Error_Exit( string );
- }
- }
- }
- // if ( *byteptr != (*(byteptr + 1) ^ 0xff) ||
- // *(byteptr + 2) || *(byteptr + 3) ) {
- // if (Memory_Error_Exit != NULL) {
- // sprintf( string, "Error freeing pointer %p. Header invalid!!!\n", pointer );
- // Memory_Error_Exit( string );
- // }
- // }
- #else
- unsigned char *byteptr = ((unsigned char *)pointer) - 1;
- if ( (*byteptr & 0xe0) != (((*byteptr ^ 0x07) & 0x07) << 5) ) {
- if (Memory_Error_Exit != NULL) {
- sprintf( string, "Error freeing pointer %p. Header invalid!!!\n", pointer );
- Memory_Error_Exit( string );
- }
- }
- #endif
- /*
- ** Check to see if this was locked me and if it was unlock it.
- */
- if (*byteptr & MEM_LOCK) {
- long *longptr = ((long *)byteptr) - 1;
- DPMI_Unlock(pointer, *longptr);
- pointer = (void *)longptr;
- } else
- pointer = (void *)byteptr;
- #if(LOGGING)
- int val = _heapchk();
- FILE *file = fopen("mem.txt","at");
- fprintf(file, "%P Free flags = %d, Heap = %d\n",
- original,
- *byteptr,
- val);
- fclose(file);
- #endif
- // If the pointer is a real mode pointer than it will point to the
- // first megabyte of system memory. If it does than we need to
- // use DPMI to free it.
- if (*byteptr & MEM_REAL) {
- regs.x.eax = 0x101;
- regs.x.edx = *(((long *)pointer) - 1);
- segread ( & sregs ) ;
- int386x(0x31, ®s, ®s, &sregs);
- } else {
- free((void *)pointer);
- }
- Memory_Calls--;
- }
- }
- /***************************************************************************
- * Resize_Alloc -- Change the size of an allocated block. *
- * *
- * This routine will take a previously allocated block and change its *
- * size without unnecessarily altering its contents. *
- * *
- * INPUT: pointer -- Pointer to the original memory allocation. *
- * *
- * new_size -- Size in bytes that it will be converted to. *
- * *
- * OUTPUT: Returns with a pointer to the new allocation. *
- * *
- * WARNINGS: ??? *
- * *
- * HISTORY: *
- * 02/01/1992 JLB : Commented. *
- *=========================================================================*/
- void *Resize_Alloc(void *original_ptr, unsigned long new_size_in_bytes)
- {
- unsigned long *temp;
- // unsigned long diff, flags;
- temp = (unsigned long*)original_ptr;
- /* ReAlloc the space */
- temp = (unsigned long *)realloc(temp, new_size_in_bytes);
- if (temp == NULL) {
- if (Memory_Error != NULL)
- Memory_Error();
- return NULL;
- }
- return(temp);
- }
- /***************************************************************************
- * Ram_Free -- Determines the largest free chunk of RAM. *
- * *
- * Use this routine to determine the largest free chunk of available *
- * RAM for allocation. It also performs a check of the memory chain. *
- * *
- * INPUT: none *
- * *
- * OUTPUT: Returns with the size of the largest free chunk of RAM. *
- * *
- * WARNINGS: This does not return the TOTAL memory free, only the *
- * largest free chunk. *
- * *
- * HISTORY: *
- * 09/03/1991 JLB : Commented. *
- *=========================================================================*/
- long Ram_Free(MemoryFlagType)
- {
- return(_memmax());
- // return Largest_Mem_Block();
- }
- /***************************************************************************
- * Heap_Size -- Size of the heap we have. *
- * *
- * *
- * *
- * INPUT: *
- * *
- * OUTPUT: *
- * *
- * WARNINGS: *
- * *
- * HISTORY: *
- * 06/21/1994 SKB : Created. *
- *=========================================================================*/
- long Heap_Size(MemoryFlagType )
- {
- if (!TotalRam) {
- TotalRam = Total_Ram_Free(MEM_NORMAL);
- }
- return(TotalRam);
- }
- /***************************************************************************
- * Total_Ram_Free -- Total amount of free RAM. *
- * *
- * *
- * *
- * INPUT: *
- * *
- * OUTPUT: *
- * *
- * WARNINGS: *
- * *
- * HISTORY: *
- * 06/21/1994 SKB : Created. *
- * 03/09/1995 JLB : Uses prerecorded heap size maximum. *
- *=========================================================================*/
- long Total_Ram_Free(MemoryFlagType )
- {
- return(_memavl());
- // return Largest_Mem_Block () ;
- }
- #endif
|