mirror of
https://github.com/reactos/wine.git
synced 2024-11-26 21:20:25 +00:00
401710d757
Fri Sep 3 11:52:18 1993 Bob Amstadt * [windows/timer.c] Changed to use CallWindowProc() rather directly calling callback. * [windows/event.c] Implemented SetCapture() and ReleaseCapture() * [windows/keyboard.c] Created stub for GetKeyState() * [objects/linedda.c] Created stub for LineDDA() * [if1632/callback.c] Created callback handler for LineDDA callback procedure. * [if1632/callback.c] Created FreeProcInstance() Fri Sep 3 08:36:52 1993 David Metcalfe * [loader/signal.c] Patch to and code for INT 1A Thu Sep 2 00:31:54 1993 Alexandre Julliard * [objects/font.c] [objects/text.c] More text support: implemented justification and underlining. * [windows/clipping.c] [objects/clipping.c] Moved low-level clipping functions to objects/clipping.c. * [windows/clipping.c] [windows/event.c] [windows/message.c] Implemented window update regions. * [windows/dc.c] [objects/dcvalues.c] Moved some device-independent DC functions to objects/dcvalues.c. * [windows/graphics.c] Implemented InvertRect() and GetPixel(). Sat Aug 28 08:40:23 1993 Eric Youngdale * [include/neexe.h] [loader/wine.c] Added code to handle relocation type 4. * [loader/signal.h] [loader/wine.c] [loader/selector.c] Added support for dos interrupts. Thu 26 Aug 19:15:00 1993 Eric Youngdale * [loader/selector.c] Fixed bug dealing with loading DLLs. Thu Aug 26 19:22:40 1993 Alexandre Julliard * [include/gdi.h] [objects/font.c] [windows/dc.c] Beginning of real font support. * [windows/graphics.c] Implemented PatBlt(). * [memory/global.c] Corrected a bug with linked list handling in GlobalAlloc(). * [objects/bitmap.c] Corrected a bug in BITMAP_SelectObject(). Tue Aug 24 19:22:40 1993 David Metcalfe * [controls/Command*] [controls/Label*] [controls[MenuButto*] [controls/SmeMenuButt*] Change code to support & as a special character in menu item text. Tue Aug 24 19:22:40 1993 Alexandre Julliard * [include/gdi.h] [windows/dc.c] Heavily modified the DC structure for better device-independence. * [objects/bitmap.c] Implemented bitmap dimensions. * [windows/dc.c] [windows/dce.c] Implemented DC state saving and restoring. * [windows/dc.c] Implemented ROP mode. * [windows/graphics.c] Implemented FillRect(). Mon Aug 23 22:08:34 1993 Bob Amstadt (bob at pooh) * [misc/xt.c] Fixed bug in InvalidateRect(). Solitaire attempted to clear window before it was realized. * [loader/resource.c] Began rewrite of LoadBitmap(). * [loader/wine.c] Fixed code which set Argv and Argc global variables. * [loader/selector.c] Added code to set up command line arguments. * [include/neexe.h] Fixed error in PSP structure. Tue Aug 17 20:41:12 1993 Alexandre Julliard * [include/gdi.h] [windows/dc.c] Implemented device capabilities. * [objects/region.c] Implemented EqualRgn() and CombineRgn(). * [windows/clipping.c] Implemented Save/RestoreVisRgn(). * [windows/graphics.c] Implemented PaintRgn() and FillRgn(). * [windows/mapping.c] Implemented mapping modes. Tue Aug 10 14:07:38 1993 Alexandre Julliard * [if1632/user.spec] [misc/rect.c] Implemented rectangle API functions. * [if1632/gdi.spec] [include/gdi.h] [objects/region.c] Implemented regions. * [windows/class.c] Corrected a typo in UnregisterClass(). * [windows/clipping.c] [windows/dc.c] Implemented DC clipping and visible region. Tue Aug 10 20:57:56 1993 Bob Amstadt (bob at pooh) * [controls/menu.c] [windows/win.c] SetMenu(), GetMenu(), CheckMenuItem() implemented Thu Aug 5 22:33:22 1993 Bob Amstadt (bob at pooh) * [controls/menu.c] [windows/win.c] Many improvements menus. LoadMenu() should work. Wed Aug 4 14:55:36 1993 Alexandre Julliard * [objects/dib.c] Started the implementation of device-independent bitmaps. * [objects/bitmap.c] Added support for multiple bitmap depths. * [objects/brush.c] Implemented pattern brushes. * [windows/dc.c] [windows/graphics.c] Implemented some GDI graphics primitives. Tue Aug 3 21:16:47 1993 Bob Amstadt (bob at pooh) * [controls/menu.c] [windows/win.c] [include/menu.h] Code to load class menus from executable file. * [if1632/user.spec] Fixed specification of SendMessage() and PostMessage. Mon Jul 26 21:53:24 1993 Alexandre Julliard * [if1632/call.S] Corrected a bug in KERNEL_InitTask(). * [include/windows.h] Added a lot of constants. * [loader/selector.c] Corrected a bug in segment allocation in CreateSelectors(). * [objects/bitmap.c] Implemented SelectObject() for bitmaps. * [objects/brush.c] Implemented hatched brushes and SelectObject(). * [objects/gdiobj.c] Removed linked list (not needed). * [objects/palette.c] Implemented system palette creation and misc. palette API functions. * [windows/timer.c] Implemented timers. * [windows/dc.c] Implemented memory device contexts. Tue Jul 20 10:38:59 1993 Bob Amstadt (bob at pooh) * [dos.c] Split DOS3Call() out of kernel.c. Added support for get date and time functions. * [call.S] Added function ReturnFromRegisterFunc() to allow DOS calls to return values in registers. * [regfunc.h] Macros to access registers saved on stack. Tue Jul 20 10:38:59 1993 Alexandre Julliard * [win.c] Corrected allocation of the WM_CREATE data structure. * [dce.c] [dce.h] Implemented DCE handling. * [bitmap.c] [brush.c] [dc.c] [font.c] [gdi.h] [gdi.spec] [gdiobj.c] [palette.c] [pen.c] Implemented the GDI objects data structures and allocation. * [windows.h] Added several structures and constants for GDI objects. Mon Jul 19 12:51:10 1993 Bob Amstadt (bob at pooh) * [ldtlib.c] Modified system calls to match Linus' new interface for the LDT modification. * [win.c] Fixed bug with WM_CREATE message. * [heap.c] [kernel.spec] Completed local heap allocation functions. * [global.c] Created function GlobalQuickAlloc() for easy allocation from DLLs
431 lines
8.9 KiB
C
431 lines
8.9 KiB
C
static char RCSId[] = "$Id: heap.c,v 1.3 1993/07/04 04:04:21 root Exp root $";
|
||
static char Copyright[] = "Copyright Robert J. Amstadt, 1993";
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include "prototypes.h"
|
||
#include "segmem.h"
|
||
#include "heap.h"
|
||
|
||
MDESC *LOCAL_FreeList;
|
||
|
||
/**********************************************************************
|
||
* HEAP_Init
|
||
*/
|
||
void
|
||
HEAP_Init(MDESC **free_list, void *start, int length)
|
||
{
|
||
*free_list = (MDESC *) start;
|
||
(*free_list)->prev = NULL;
|
||
(*free_list)->next = NULL;
|
||
(*free_list)->length = length - sizeof(MDESC);
|
||
}
|
||
|
||
/**********************************************************************
|
||
* HEAP_Alloc
|
||
*/
|
||
void *
|
||
HEAP_Alloc(MDESC **free_list, int flags, int bytes)
|
||
{
|
||
MDESC *m, *m_new;
|
||
|
||
#ifdef DEBUG_HEAP
|
||
printf("HeapAlloc: free_list %08x, flags %x, bytes %d\n",
|
||
free_list, flags, bytes);
|
||
#endif
|
||
|
||
/*
|
||
* Find free block big enough.
|
||
*/
|
||
for (m = *free_list; m != NULL; m = m->next)
|
||
{
|
||
if (m->length >= bytes && m->length < bytes + 4 * sizeof(MDESC))
|
||
{
|
||
break;
|
||
}
|
||
else if (m->length > bytes)
|
||
{
|
||
m_new = m + (bytes / sizeof(MDESC)) + 2;
|
||
if (m->prev == NULL)
|
||
*free_list = m_new;
|
||
else
|
||
m->prev->next = m_new;
|
||
|
||
if (m->next != NULL)
|
||
m->next->prev = m_new;
|
||
|
||
m_new->next = m->next;
|
||
m_new->prev = m->prev;
|
||
m_new->length = m->length - ((int) m_new - (int) m);
|
||
m->length -= (m_new->length + sizeof(MDESC));
|
||
|
||
m->prev = m;
|
||
m->next = m;
|
||
m->lock = 0;
|
||
m->flags = 0;
|
||
if (flags & GLOBAL_FLAGS_ZEROINIT)
|
||
memset(m + 1, 0, bytes);
|
||
#ifdef DEBUG_HEAP
|
||
printf("HeapAlloc: returning %08x\n", (m + 1));
|
||
#endif
|
||
return (void *) (m + 1);
|
||
}
|
||
}
|
||
|
||
if (m != NULL)
|
||
{
|
||
if (m->prev == NULL)
|
||
*free_list = m->next;
|
||
else
|
||
m->prev->next = m->next;
|
||
|
||
if (m->next != NULL)
|
||
m->next->prev = m->prev;
|
||
|
||
m->prev = m;
|
||
m->next = m;
|
||
m->lock = 0;
|
||
m->flags = 0;
|
||
if (flags & GLOBAL_FLAGS_ZEROINIT)
|
||
memset(m + 1, 0, bytes);
|
||
#ifdef DEBUG_HEAP
|
||
printf("HeapAlloc: returning %08x\n", (m + 1));
|
||
#endif
|
||
return (void *) (m + 1);
|
||
}
|
||
|
||
#ifdef DEBUG_HEAP
|
||
printf("HeapAlloc: returning %08x\n", 0);
|
||
#endif
|
||
return 0;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* HEAP_ReAlloc
|
||
*/
|
||
void *
|
||
HEAP_ReAlloc(MDESC **free_list, void *old_block,
|
||
int new_size, unsigned int flags)
|
||
{
|
||
MDESC *m_free;
|
||
MDESC *m;
|
||
|
||
/*
|
||
* Check validity of block
|
||
*/
|
||
m = (MDESC *) old_block - 1;
|
||
if (m->prev != m || m->next != m ||
|
||
((int) m & 0xffff0000) != ((int) *free_list & 0xffff0000))
|
||
{
|
||
#ifdef DEBUG_HEAP
|
||
printf("Attempt to resize bad pointer, m = %08x, *free_list = %08x\n",
|
||
m, free_list);
|
||
#endif
|
||
return NULL;
|
||
}
|
||
|
||
/*
|
||
* Check for grow block
|
||
*/
|
||
if (new_size > m->length)
|
||
{
|
||
m_free = m + 1 + m->length / sizeof(MDESC);
|
||
if (m_free->next != m_free ||
|
||
m_free->prev != m_free ||
|
||
m_free->length + sizeof(MDESC) < new_size)
|
||
{
|
||
void *new_p = HEAP_Alloc(free_list, flags, new_size);
|
||
|
||
if (new_p ==NULL)
|
||
return NULL;
|
||
memcpy(new_p, old_block, m->length);
|
||
HEAP_Free(free_list, old_block);
|
||
return new_p;
|
||
}
|
||
|
||
if (m_free->prev == NULL)
|
||
*free_list = m_free->next;
|
||
else
|
||
m_free->prev->next = m_free->next;
|
||
|
||
if (m_free->next != NULL)
|
||
m_free->next->prev = m_free->prev;
|
||
|
||
m->length += sizeof(MDESC) + m_free->length;
|
||
if (flags & GLOBAL_FLAGS_ZEROINIT)
|
||
memset(m_free, '\0', sizeof(MDESC) + m_free->length);
|
||
}
|
||
|
||
/*
|
||
* Check for shrink block.
|
||
*/
|
||
if (new_size < m->length - 4 * sizeof(MDESC))
|
||
{
|
||
m_free = m + new_size / sizeof(MDESC) + 2;
|
||
m_free->next = m_free;
|
||
m_free->prev = m_free;
|
||
m_free->length = m->length - (int) m_free - (int) m;
|
||
m->length = (int) m_free - (int) (m + 1);
|
||
HEAP_Free(free_list, m_free + 1);
|
||
}
|
||
|
||
return old_block;
|
||
}
|
||
|
||
|
||
/**********************************************************************
|
||
* HEAP_Free
|
||
*/
|
||
int
|
||
HEAP_Free(MDESC **free_list, void *block)
|
||
{
|
||
MDESC *m_free;
|
||
MDESC *m;
|
||
MDESC *m_prev;
|
||
|
||
/*
|
||
* Validate pointer.
|
||
*/
|
||
m_free = (MDESC *) block - 1;
|
||
if (m_free->prev != m_free || m_free->next != m_free ||
|
||
((int) m_free & 0xffff0000) != ((int) *free_list & 0xffff0000))
|
||
{
|
||
#ifdef DEBUG_HEAP
|
||
printf("Attempt to free bad pointer,"
|
||
"m_free = %08x, *free_list = %08x\n",
|
||
m_free, free_list);
|
||
#endif
|
||
return -1;
|
||
}
|
||
|
||
/*
|
||
* Find location in free list.
|
||
*/
|
||
m_prev = NULL;
|
||
for (m = *free_list; m != NULL && m < m_free; m = m->next)
|
||
m_prev = m;
|
||
|
||
if (m_prev != NULL && (int) m_prev + m_prev->length > (int) m_free)
|
||
{
|
||
#ifdef DEBUG_HEAP
|
||
printf("Attempt to free bad pointer,"
|
||
"m_free = %08x, m_prev = %08x (length %x)\n",
|
||
m_free, m_prev, m_prev->length);
|
||
#endif
|
||
return -1;
|
||
}
|
||
|
||
if ((m != NULL && (int) m_free + m_free->length > (int) m) ||
|
||
(int) m_free + m_free->length > ((int) m_free | 0xffff))
|
||
{
|
||
#ifdef DEBUG_HEAP
|
||
printf("Attempt to free bad pointer,"
|
||
"m_free = %08x (length %x), m = %08x\n",
|
||
m_free, m_free->length, m);
|
||
#endif
|
||
return -1;
|
||
}
|
||
|
||
/*
|
||
* Put block back in free list.
|
||
* Does it merge with the previos block?
|
||
*/
|
||
if (m_prev != NULL)
|
||
{
|
||
if ((int) m_prev + m_prev->length == (int) m_free)
|
||
{
|
||
m_prev->length += sizeof(MDESC) + m_free->length;
|
||
m_free = m_prev;
|
||
}
|
||
else
|
||
{
|
||
m_prev->next = m_free;
|
||
m_free->prev = m_prev;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
*free_list = m_free;
|
||
m_free->prev = NULL;
|
||
}
|
||
|
||
/*
|
||
* Does it merge with the next block?
|
||
*/
|
||
if (m != NULL)
|
||
{
|
||
if ((int) m_free + m_free->length == (int) m)
|
||
{
|
||
m_free->length += sizeof(MDESC) + m->length;
|
||
m_free->next = m->next;
|
||
}
|
||
else
|
||
{
|
||
m->prev = m_free;
|
||
m_free->next = m;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
m_free->next = NULL;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* HEAP_LocalInit
|
||
*/
|
||
void
|
||
HEAP_LocalInit(void *start, int length)
|
||
{
|
||
HEAP_Init(&LOCAL_FreeList, start, length);
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalAlloc
|
||
*/
|
||
void *
|
||
LocalAlloc(int flags, int bytes)
|
||
{
|
||
void *m;
|
||
|
||
#ifdef DEBUG_HEAP
|
||
printf("LocalAlloc: flags %x, bytes %d\n", flags, bytes);
|
||
#endif
|
||
|
||
m = HEAP_Alloc(&LOCAL_FreeList, flags, bytes);
|
||
|
||
#ifdef DEBUG_HEAP
|
||
printf("LocalAlloc: returning %x\n", (int) m);
|
||
#endif
|
||
return m;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalCompact
|
||
*/
|
||
int
|
||
LocalCompact(int min_free)
|
||
{
|
||
MDESC *m;
|
||
int max_block;
|
||
|
||
max_block = 0;
|
||
for (m = LOCAL_FreeList; m != NULL; m = m->next)
|
||
if (m->length > max_block)
|
||
max_block = m->length;
|
||
|
||
return max_block;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalFlags
|
||
*/
|
||
unsigned int
|
||
LocalFlags(unsigned int handle)
|
||
{
|
||
MDESC *m;
|
||
|
||
m = (MDESC *) (((int) LOCAL_FreeList & 0xffff0000) |
|
||
(handle & 0xffff)) - 1;
|
||
if (m->next != m || m->prev != m)
|
||
return 0;
|
||
|
||
return m->lock;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalFree
|
||
*/
|
||
unsigned int
|
||
LocalFree(unsigned int handle)
|
||
{
|
||
unsigned int addr;
|
||
|
||
addr = ((int) LOCAL_FreeList & 0xffff0000) | (handle & 0xffff);
|
||
if (HEAP_Free(&LOCAL_FreeList, (void *) addr) < 0)
|
||
return handle;
|
||
else
|
||
return 0;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalInit
|
||
*/
|
||
unsigned int
|
||
LocalInit(unsigned int segment, unsigned int start, unsigned int end)
|
||
{
|
||
HEAP_Init(&LOCAL_FreeList,
|
||
(void *) ((segment << 16) | start), end - start + 1);
|
||
|
||
return segment;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalLock
|
||
*/
|
||
void *
|
||
LocalLock(unsigned int handle)
|
||
{
|
||
MDESC *m;
|
||
|
||
m = (MDESC *) (((int) LOCAL_FreeList & 0xffff0000) |
|
||
(handle & 0xffff)) - 1;
|
||
if (m->next != m || m->prev != m)
|
||
return 0;
|
||
|
||
m->lock++;
|
||
return (void *) (m + 1);
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalReAlloc
|
||
*/
|
||
void *
|
||
LocalReAlloc(unsigned int handle, int flags, int bytes)
|
||
{
|
||
void *m;
|
||
|
||
m = HEAP_ReAlloc(&LOCAL_FreeList, (void *)
|
||
(((int) LOCAL_FreeList & 0xffff0000) | (handle & 0xffff)),
|
||
bytes, flags);
|
||
|
||
return m;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalSize
|
||
*/
|
||
unsigned int
|
||
LocalSize(unsigned int handle)
|
||
{
|
||
MDESC *m;
|
||
|
||
m = (MDESC *) (((int) LOCAL_FreeList & 0xffff0000) |
|
||
(handle & 0xffff)) - 1;
|
||
if (m->next != m || m->prev != m)
|
||
return 0;
|
||
|
||
return m->length;
|
||
}
|
||
|
||
/**********************************************************************
|
||
* LocalUnlock
|
||
*/
|
||
unsigned int
|
||
LocalUnlock(unsigned int handle)
|
||
{
|
||
MDESC *m;
|
||
|
||
m = (MDESC *) (((int) LOCAL_FreeList & 0xffff0000) |
|
||
(handle & 0xffff)) - 1;
|
||
if (m->next != m || m->prev != m)
|
||
return 1;
|
||
|
||
if (m->lock > 0)
|
||
m->lock--;
|
||
|
||
return 0;
|
||
}
|