Release 950727
Sat Jul 22 22:39:09 IDT 1995 Michael Veksler <e1678223@tochnapc2.technion.ac.il>
* [ipc/*]
New directory. This directory contains the new inter-wine
communications support. It enables DDE protocols between two wine
instances. Currently it is limited to DDE, but can be enhanced to
support OLE between 2 different wine instances. This is very
important for libwine.a DDE/OLE support.
* [tools/ipcl]
A script to delete garbage IPC handles (shared memory, semaphores
and message queues). The current inter-wine communication is not
perfect, and sometimes leaves garbage behind.
* [if1632/relay.c] [include/atom.h] [include/global.h]
[loader/selector.c] [loader/task.c] [loader/module.c]
[loader/signal.c] [memory/global.c] [misc/atom.c]
[windows/class.c] [windows/message.c] [windows/win.c]
[Imakefile]
Hooks for inter-wine DDE support, current Global.*Atom functions
renamed to Local.*Atom since Global.*Atom are used for Inter-Wine
DDE communication. (The first call to these functions sets up the
IPC structures - which otherwise cause unneeded overhead.
Mon Jul 17 19:55:21 1995 Alexandre Julliard <julliard@sunsite.unc.edu>
* [controls/menu.c]
Don't crash if a NULL string is passed to menu functions.
* [memory/selector.c]
We now use a bit in ldt_flags_copy to indicate free LDT entries.
Fixed a bug in SELECTOR_ReallocBlock that could cause it to
overwrite valid LDT entries when growing a block.
* [miscemu/instr.c]
Emulate int xx instruction by storing the interrupt vector in
CS:IP and returning directly. This allows a program to install an
interrupt vector.
* [windows/win.c]
Added function WIN_GetTopParent to get the top-level parent of a
window.
Sun Jul 16 18:17:17 1995 Gregory Trubetskoy <grisha@mira.com>
* [loader/resource.c]
Added LoadIconHandler. It doesn't do anything yet, but now you
can use borland help files with winhelp.exe.
Sun Jul 16 11:58:45 1995 Anand Kumria <akumria@ozemail.com.au>
* [misc/main.c]
Fixed to return 386 Enhanced mode correctly. Also return the same
type of CPU, for both Enhanced and Standard mode, namely a 386.
Sun Jul 16 00:02:04 1995 Martin von Loewis <loewis@informatik.hu-berlin.de>
* [Configure] [include/options.h] [include/wineopts.h]
[misc/main.c][misc/spy.c]
Removed support of spy file. Redirected spy messages to stddeb.
Removed -spy option. Added -debugmsg +spy option.
* [debugger/dbg.y][debugger/debug.l]
Enabled segmented addresses (seg:offs) for break and x commands.
* [if1632/gdi.spec] [objects/region.c] [windows/graphics.c]
[include/region.h]
FrameRgn, REGION_FrameRgn: New functions
* [if1632/kernel.spec]
IsWinOldApTask: Return false
* [if1632/mouse.spec]
CplApplet: Removed
* [if1632/user.spec] [windows/win.c]
ShowOwnedPopups: New function
* [if1632/winsock.spec] [misc/winsocket.c]
inet_addr, select: New prototypes in relay code
Fixed memory layout for netdb functions (getXbyY).
WINSOCK_ioctlsocket: Translated FIONREAD, FIONBIO, and FIOASYNC
* [objects/clipping.c]
RectVisible: Fixed call to LPToDP
* [rc/winerc.c]
main: Removed extra argument to getopt for Linux.
Tue Jul 11 00:14:41 1995 Bernd Schmidt <crux@pool.informatik.rwth-aachen.de>
* [controls/listbox.c]
Yet another fix for ListBoxDirectory().
* [loader/module.c] [if1632/kernel.spec]
Make GetModuleHandle() accept instance handles as parameter.
* [if1632/relay.c] [loader/task.c]
Put a magic cookie at the bottom of the 32 bit stack, and check on
each return from a 32 bit function whether it's still there. Complain
if it's not.
* [if1632/user.spec]
Wrong entry for CloseDriver().
* [misc/dos_fs.c] [loader/task.c] [include/dos_fs.h] [misc/file.c]
[miscemu/int21.c]
Large parts of dos_fs.c simplified. Changed it to use one
current drive/directory per task, which is set to the module path on
task creation.
Prevent CorelPaint from closing stdin.
open() with O_CREAT set must be passed three parameters.
DOS FindFirst()/FindNext() could crash when FA_LABEL was set. Fixed,
it's in DOS_readdir() now.
* [misc/profile.c]
Some badly written software (Lotus Freelance Graphics) passes a bogus
size parameter that caused Wine to write off the end of a segment.
Fixed. (It's probably too paranoid now.)
* [multimedia/mmsystem.c] [multimedia/time.c] [multimedia/joystick.c]
[multimedia/Imakefile] [if1632/winprocs.spec]
16 bit entry point for MMSysTimeCallback.
Split off time.c and joystick.c from mmsystem.c.
* [objects/dib.c]
GetDIBits(): call XGetImage() via CallTo32_LargeStack.
* [windows/cursor.c]
DestroyCursor(): do nothing for builtin cursors.
* [windows/mdi.c]
Half of WM_MDISETMENU implemented.
* [windows/win.c]
EnumWindows() and EnumTaskWindows() never enumerated any windows.
Fixed.
* [windows/*.c]
Fixed GetParent() to return correct values for owned windows.
* [windows/message.c]
Don't try to activate disabled top-level windows.
* [windows/nonclient.c]
Work around a bug in gcc-2.7.0.
* [tools/build.c] [include/stackframe.h] [memory/global.c]
[loader/task.c] [memory/selector.c]
Some Visual Basic programs (and possibly others, too) expect ES to be
preserved by a call to an API function, so we have to save it.
In GlobalFree() and FreeSelector(), we must clear CURRENT_STACK16->es
to prevent segfaults if ES contained the selector to be freed.
Sun Jul 9 20:21:20 1995 Jon Tombs <jon@gtex02.us.es>
* [*/*]
Added missing prototypes to header files and relevant includes
to reduce compile time warnings.
Sun Jul 9 18:32:56 1995 Michael Patra <micky@marie.physik.tu-berlin.de>
* [configure.in] [include/config.h] [*/Makefile.in]
New configuration scheme based on autoconf.
Sat Jul 8 14:12:45 1995 Morten Welinder <terra+@cs.cmu.edu>
* [miscemu/ioports.c]
Revamp to have only one in- and one out- variant, both really
implemented.
* [miscemu/instr.c]
INSTR_EmulateInstruction: Use new ioport interface. Implement
string io. Correct instruction pointer for 32-bit code.
* [include/miscemu.h]
Update port function prototypes.
* [include/registers.h]
Defined FS and GS.
Sat Jul 8 13:38:54 1995 Hans de Graaff <graaff@twi72.twi.tudelft.nl>
* [misc/dos_fs.c]
ChopOffSlash(): A path consisting off a single slash is left
intact, and multiple slashes are all removed.
1995-07-29 15:09:43 +02:00
|
|
|
/***************************************************************************
|
|
|
|
* Copyright 1995 Michael Veksler. mveksler@vnet.ibm.com
|
|
|
|
***************************************************************************
|
|
|
|
* File: generic_hash.c
|
|
|
|
* Purpose : dynamically growing hash, may use shared or local memory.
|
|
|
|
***************************************************************************
|
|
|
|
*/
|
Release 950901
Thu Aug 31 17:19:57 1995 Alexandre Julliard <julliard@sunsite.unc.edu>
* [Configure]
Added compile-time option for IPC.
* [configure.in]
Added command-line options for language, IPC and malloc
debugging.
* [controls/menu.c]
WM_MENUSELECT was sometimes sent to the wrong window.
* [debugger/break.c]
For the 'next' command, only step over instruction that require
it. This allows 'next' to do the right thing with jmp and ret
instructions.
* [ipc/*.c] [memory/atom.c] [memory/global.c]
IPC can now be configured out at compile-time.
* [loader/task.c]
Bug fix in TASK_Reschedule() that could cause a task to be deleted
twice.
* [miscemu/dosmem.c] (New file)
Partial emulation of the BIOS data segment.
* [miscemu/instr.c]
Trap attempts to access selector 0x40 and remap the access to
segment __0040H.
* [tools/build.c]
Fixed bug in CallTo32_LargeStack() that caused problems when
compiling Wine with the -fomit-frame-pointer option.
* [windows/message.c]
Fixed bug in hardware event handling that could cause some events
to get ignored.
Sat Aug 26 13:12:59 IST 1995 Michael Veksler <mveksler@vnet.ibm.com>
* [ipc/README] [ipc/dde.tex]
LaTeX documentation for the ipc and DDE stuff.
Wed Aug 23 22:01:23 GMT 1995 Michael Veksler <mveksler@vnet.ibm.com>
* [ipc/Imakefile] [ipc/wine_test_stub.c]
Fixed IPC testing. Now it can be compiled with "make tests"
Wed Aug 23 21:04:14 1995 Fons Botman <botman@wab-tis.rabobank.nl>
* [if1632/kernel.spec] [include/windows.h] [misc/main.c]
Added GetWinDebugInfo/SetWinDebugInfo stub for player.exe
Sun Aug 20 13:49:42 1995 Marcus Meissner <msmeissn@faui01.informatik.uni-erlangen.de>
* [miscemu/int21.c]
Misc fix to int21,ah=40 (write) to match _lwrite().
AX=0x440A (check if handle is remote) added.
* [multimedia/mmsystem.c]
Moved mciSendString to mcistring.c.
* [multimedia/mcistring.c]
New file, string interface for MCI (not complete, not thoroughly
tested).
* [multimedia/audio.c]
IOCTL prints errors; one paranoid check disabled.
* [misc/file.c]
Misc operator precedence fixes.
* [if1632/gdi.spec] [objects/bitblt.c]
Stub for FastWindowFrame (parameters not correct).
Sat Aug 19 01:31:23 1995 Graham Menhennitt <gfm@werple.mira.net.au>
* [loader/ne_image.c]
Preliminary support for iterated segments.
Sat Aug 19 00:43:04 1995 Andrew Taylor (andrew@riscan.com)
* [windows/mapping.c]
In function MAPPING_FixIsotropic(), VportExt[XY] is multiplied by
the absolute value of (ydim / xdim) or (xdim / ydim).
Thu Aug 15 23:00:16 Gregory Trubetskoy <grisha@mira.com>
* [objects/oembitmap.c]
Added some includes for Windows 95.
* [include/sysmetrics.h]
Added some sysmetrics for Windows 95.
* [include/bitmaps/*95]
New files: obm_close_95, obm_closed_95, obm_reduce_95, obm_reduced_95
obm_zoom_95, obm_zoomd_95 - these are some pixmaps for Windows 95.
Thu Aug 10 12:00:00 1995 Jan Willamowius (jan@janhh.shnet.org)
* [misc/shell.c] [rc/sysres*.rc]
The caption of the ShellAbout dialog box is language specific and
should be defined in the resources.
1995-09-01 17:57:28 +02:00
|
|
|
#ifdef CONFIG_IPC
|
|
|
|
|
Release 950817
Thu Aug 17 19:30:14 1995 Alexandre Julliard <julliard@sunsite.unc.edu>
* [*/Makefile.in]
Removed winelibclean target, as it doesn't work anyway.
* [controls/button.c]
Avoid drawing the focus rectangle outside of the button.
* [controls/widgets.c]
Fixed bug with the size of the reserved bytes for the Edit
control (caused Eudora to crash).
* [debugger/*] [include/debugger.h]
Unified debugger address handling. Segmented and linear addresses
are no grouped in a single type DBG_ADDR.
All commands now accept seg:off addresses.
Module entry points are now loaded upon first entry to the
debugger, so that entry points of the loaded executable also
appear in the symbol table.
* [include/registers.h] [miscemu/*.c]
Register macros are now of the form 'AX_reg(context)' instead of 'AX'.
This makes code less readable, but will prevent a lot of name
clashes with other definitions. It also avoids a hidden reference
to the 'context' variable.
* [ipc/dde_atom.c] [misc/atom.c]
All *AddAtom and *FindAtom functions now take a SEGPTR parameter,
to allow supporting integer atoms.
Moved atom.c to memory/ directory.
* [loader/task.c]
Fixed environment allocation to compute the size dynamically.
Added 'windir' environment variable.
Fixed GetDOSEnvironment() to return the current task environment.
* [windows/message.c]
Fixed bug in MSG_GetWindowForEvent().
Wed Aug 9 11:40:43 1995 Marcus Meissner <msmeissn@faui01.informatik.uni-erlangen.de>
* [include/ole.h]
Added a lot of structures from my Borland Manual. Neither complete,
nor 100% right (check please)
* [misc/shell.c]
Fixed some of the Reg* functions.
Enhanced ShellExecute.
Please test: wine "regedit.exe /v" mplayer.exe soundrec.exe
Do YOU know the format of \WINDOWS\REG.DAT? Mail me please :)
* [misc/dos_fs.c]
Make umsdos mounted windows dirs work again.
* [miscemu/emulate.c]
Added some comments, preimplementation of subfunction 7.
* [multimedia/mmsystem.c]
Implemented mciSendString. not complete, not clean, not
necessarily working (only checked with a program which uses
'cdaudio' (one working program is cool.exe, a shareware waveditor
with cdaudio play facilities.)
* [multimedia/mcicda.c]
Segptr fixes in DriverProc
Default cdrom drive in Linux is /dev/cdrom ... usually a symbolic
link to your real cdrom device.
Tue Aug 8 19:41:50 CDT 1995 Daniel Schepler <dks2@cec.wustl.edu>
* [loader/resource.c]
Don't crash in a LoadString to NULL
* [loader/resource.c]
Fixed accelerators to work with modifiers. (ALT-x modifiers still
won't work unless the ALT keypress exited the menu.)
* [misc/file.c]
Expand a file to the current offset with an _lwrite of size zero.
* [misc/file.c]
Set a newly created file to read-write instead of write-only.
Sun Aug 6 20:28:35 1995 Anand Kumria <akumria@ozemail.com.au>
* [misc/main.c] [include/msdos.h]
Fixed to return DOS version 6.22, and the correct byte order
for Windows programs.
Wed Aug 2 12:36:33 1995 Bernd Schmidt <crux@pool.informatik.rwth-aachen.de>
* [include/options.h] [memory/global.c] [misc/main.c]
Make the new IPC run-time selectible, disabling it by default.
(I think it's only useful for libwine, anyway.)
* [loader/task.c] [memory/selector.c]
In FreeSelector(), walk up the stack and fix the frames.
* [objects/dib.c]
Missing break statement in DIB_SetImageBits_RLE8().
In GetDIBits(), set the compression flag in the bitmap info to zero.
* [windows/dialog.c]
GetNextDlgGroupItem() needs to treat the first child as if it had
an implicit WS_GROUP bit set.
Mon Jul 31 15:44:47 EDT 1995 Louis-D. Dubeau <ldd@step.polymtl.ca>
* [misc/dos_fs.c]
Quick'n dirty fix for the initialisation of the Z: information
structure.
1995-08-17 19:11:36 +02:00
|
|
|
#include <sys/types.h>
|
Release 950727
Sat Jul 22 22:39:09 IDT 1995 Michael Veksler <e1678223@tochnapc2.technion.ac.il>
* [ipc/*]
New directory. This directory contains the new inter-wine
communications support. It enables DDE protocols between two wine
instances. Currently it is limited to DDE, but can be enhanced to
support OLE between 2 different wine instances. This is very
important for libwine.a DDE/OLE support.
* [tools/ipcl]
A script to delete garbage IPC handles (shared memory, semaphores
and message queues). The current inter-wine communication is not
perfect, and sometimes leaves garbage behind.
* [if1632/relay.c] [include/atom.h] [include/global.h]
[loader/selector.c] [loader/task.c] [loader/module.c]
[loader/signal.c] [memory/global.c] [misc/atom.c]
[windows/class.c] [windows/message.c] [windows/win.c]
[Imakefile]
Hooks for inter-wine DDE support, current Global.*Atom functions
renamed to Local.*Atom since Global.*Atom are used for Inter-Wine
DDE communication. (The first call to these functions sets up the
IPC structures - which otherwise cause unneeded overhead.
Mon Jul 17 19:55:21 1995 Alexandre Julliard <julliard@sunsite.unc.edu>
* [controls/menu.c]
Don't crash if a NULL string is passed to menu functions.
* [memory/selector.c]
We now use a bit in ldt_flags_copy to indicate free LDT entries.
Fixed a bug in SELECTOR_ReallocBlock that could cause it to
overwrite valid LDT entries when growing a block.
* [miscemu/instr.c]
Emulate int xx instruction by storing the interrupt vector in
CS:IP and returning directly. This allows a program to install an
interrupt vector.
* [windows/win.c]
Added function WIN_GetTopParent to get the top-level parent of a
window.
Sun Jul 16 18:17:17 1995 Gregory Trubetskoy <grisha@mira.com>
* [loader/resource.c]
Added LoadIconHandler. It doesn't do anything yet, but now you
can use borland help files with winhelp.exe.
Sun Jul 16 11:58:45 1995 Anand Kumria <akumria@ozemail.com.au>
* [misc/main.c]
Fixed to return 386 Enhanced mode correctly. Also return the same
type of CPU, for both Enhanced and Standard mode, namely a 386.
Sun Jul 16 00:02:04 1995 Martin von Loewis <loewis@informatik.hu-berlin.de>
* [Configure] [include/options.h] [include/wineopts.h]
[misc/main.c][misc/spy.c]
Removed support of spy file. Redirected spy messages to stddeb.
Removed -spy option. Added -debugmsg +spy option.
* [debugger/dbg.y][debugger/debug.l]
Enabled segmented addresses (seg:offs) for break and x commands.
* [if1632/gdi.spec] [objects/region.c] [windows/graphics.c]
[include/region.h]
FrameRgn, REGION_FrameRgn: New functions
* [if1632/kernel.spec]
IsWinOldApTask: Return false
* [if1632/mouse.spec]
CplApplet: Removed
* [if1632/user.spec] [windows/win.c]
ShowOwnedPopups: New function
* [if1632/winsock.spec] [misc/winsocket.c]
inet_addr, select: New prototypes in relay code
Fixed memory layout for netdb functions (getXbyY).
WINSOCK_ioctlsocket: Translated FIONREAD, FIONBIO, and FIOASYNC
* [objects/clipping.c]
RectVisible: Fixed call to LPToDP
* [rc/winerc.c]
main: Removed extra argument to getopt for Linux.
Tue Jul 11 00:14:41 1995 Bernd Schmidt <crux@pool.informatik.rwth-aachen.de>
* [controls/listbox.c]
Yet another fix for ListBoxDirectory().
* [loader/module.c] [if1632/kernel.spec]
Make GetModuleHandle() accept instance handles as parameter.
* [if1632/relay.c] [loader/task.c]
Put a magic cookie at the bottom of the 32 bit stack, and check on
each return from a 32 bit function whether it's still there. Complain
if it's not.
* [if1632/user.spec]
Wrong entry for CloseDriver().
* [misc/dos_fs.c] [loader/task.c] [include/dos_fs.h] [misc/file.c]
[miscemu/int21.c]
Large parts of dos_fs.c simplified. Changed it to use one
current drive/directory per task, which is set to the module path on
task creation.
Prevent CorelPaint from closing stdin.
open() with O_CREAT set must be passed three parameters.
DOS FindFirst()/FindNext() could crash when FA_LABEL was set. Fixed,
it's in DOS_readdir() now.
* [misc/profile.c]
Some badly written software (Lotus Freelance Graphics) passes a bogus
size parameter that caused Wine to write off the end of a segment.
Fixed. (It's probably too paranoid now.)
* [multimedia/mmsystem.c] [multimedia/time.c] [multimedia/joystick.c]
[multimedia/Imakefile] [if1632/winprocs.spec]
16 bit entry point for MMSysTimeCallback.
Split off time.c and joystick.c from mmsystem.c.
* [objects/dib.c]
GetDIBits(): call XGetImage() via CallTo32_LargeStack.
* [windows/cursor.c]
DestroyCursor(): do nothing for builtin cursors.
* [windows/mdi.c]
Half of WM_MDISETMENU implemented.
* [windows/win.c]
EnumWindows() and EnumTaskWindows() never enumerated any windows.
Fixed.
* [windows/*.c]
Fixed GetParent() to return correct values for owned windows.
* [windows/message.c]
Don't try to activate disabled top-level windows.
* [windows/nonclient.c]
Work around a bug in gcc-2.7.0.
* [tools/build.c] [include/stackframe.h] [memory/global.c]
[loader/task.c] [memory/selector.c]
Some Visual Basic programs (and possibly others, too) expect ES to be
preserved by a call to an API function, so we have to save it.
In GlobalFree() and FreeSelector(), we must clear CURRENT_STACK16->es
to prevent segfaults if ES contained the selector to be freed.
Sun Jul 9 20:21:20 1995 Jon Tombs <jon@gtex02.us.es>
* [*/*]
Added missing prototypes to header files and relevant includes
to reduce compile time warnings.
Sun Jul 9 18:32:56 1995 Michael Patra <micky@marie.physik.tu-berlin.de>
* [configure.in] [include/config.h] [*/Makefile.in]
New configuration scheme based on autoconf.
Sat Jul 8 14:12:45 1995 Morten Welinder <terra+@cs.cmu.edu>
* [miscemu/ioports.c]
Revamp to have only one in- and one out- variant, both really
implemented.
* [miscemu/instr.c]
INSTR_EmulateInstruction: Use new ioport interface. Implement
string io. Correct instruction pointer for 32-bit code.
* [include/miscemu.h]
Update port function prototypes.
* [include/registers.h]
Defined FS and GS.
Sat Jul 8 13:38:54 1995 Hans de Graaff <graaff@twi72.twi.tudelft.nl>
* [misc/dos_fs.c]
ChopOffSlash(): A path consisting off a single slash is left
intact, and multiple slashes are all removed.
1995-07-29 15:09:43 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "generic_hash.h"
|
|
|
|
|
|
|
|
#define ROUND_UP4(num) (( (num)+3) & ~3)
|
|
|
|
|
|
|
|
#define FREE_ENTRY 0
|
|
|
|
#define DELETED_ENTRY ((DWORD)-1)
|
|
|
|
|
|
|
|
#define NO_OF_PRIMES 512
|
|
|
|
#define GET_ITEM(items,size,i)\
|
|
|
|
(*(HASH_ITEM*) \
|
|
|
|
( ((char *)(items))+ \
|
|
|
|
(i)*(size)) )
|
|
|
|
|
|
|
|
static HASH_ITEM *locate_entry(HASH_CONTAINER* hash, DWORD key,
|
|
|
|
HASH_VAL *seeked_data, BOOL skip_deleted);
|
|
|
|
|
|
|
|
static void copy_hash_items(HASH_CONTAINER *hash, HASH_ITEM *old_items,
|
|
|
|
int old_n_items);
|
|
|
|
|
|
|
|
static BOOL arrays_initialized = FALSE;
|
|
|
|
static int primes[NO_OF_PRIMES];
|
|
|
|
static int best_primes[NO_OF_PRIMES];
|
|
|
|
static int no_of_primes;
|
|
|
|
static int no_of_best_primes;
|
|
|
|
static int max_num;
|
|
|
|
|
|
|
|
/* binary search for `num' in the `primes' array */
|
|
|
|
static BOOL prime_binary_search_found(int num)
|
|
|
|
{
|
|
|
|
int min_idx, max_idx, idx;
|
|
|
|
|
|
|
|
min_idx=0;
|
|
|
|
max_idx=no_of_primes-1;
|
|
|
|
|
|
|
|
while (min_idx <= max_idx) {
|
|
|
|
idx = (max_idx + min_idx) >> 1;
|
|
|
|
if (num == primes[idx])
|
|
|
|
return TRUE;
|
|
|
|
if (num < primes[idx])
|
|
|
|
max_idx = idx-1;
|
|
|
|
else
|
|
|
|
min_idx = idx+1;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL is_prime(int num)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
if ((num & 0x1) == 0) /* can be divided by 2 */
|
|
|
|
if (num == 2)
|
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (num <= primes[no_of_primes-1])
|
|
|
|
return prime_binary_search_found(num);
|
|
|
|
|
|
|
|
for (i=0 ; i < no_of_primes ; i++) {
|
|
|
|
if (num % primes[i] == 0)
|
|
|
|
return FALSE;
|
|
|
|
if (num < primes[i] * primes[i])
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setup_primes()
|
|
|
|
{
|
|
|
|
int num;
|
|
|
|
|
|
|
|
primes[0]=2;
|
|
|
|
primes[1]=3;
|
|
|
|
no_of_primes=2;
|
|
|
|
|
|
|
|
/* count in modulo 6 to avoid numbers that divide by 2 or 3 */
|
|
|
|
for (num=5 ; ; num+=6) {
|
|
|
|
if (is_prime(num)) {
|
|
|
|
primes[no_of_primes++]=num;
|
|
|
|
if (no_of_primes >= NO_OF_PRIMES)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (is_prime(num+2)) {
|
|
|
|
primes[no_of_primes++]=num+2;
|
|
|
|
if (no_of_primes >= NO_OF_PRIMES)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
max_num= primes[no_of_primes-1] * primes[no_of_primes-1];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Find primes which are far "enough" from powers of two */
|
|
|
|
|
|
|
|
void setup_best_primes()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int num;
|
|
|
|
int pow2before, pow2after;
|
|
|
|
int min_range, max_range;
|
|
|
|
|
|
|
|
min_range=3;
|
|
|
|
max_range=3;
|
|
|
|
pow2before= 2;
|
|
|
|
pow2after= 4;
|
|
|
|
|
|
|
|
no_of_best_primes= 0;
|
|
|
|
for (i=0 ; i < no_of_primes ; i++){
|
|
|
|
num= primes[i];
|
|
|
|
|
|
|
|
if (num > pow2after) {
|
|
|
|
pow2before= pow2after;
|
|
|
|
pow2after <<=1;
|
|
|
|
min_range= pow2before+ (pow2before >> 3);
|
|
|
|
max_range= pow2after- (pow2before >> 2);
|
|
|
|
}
|
|
|
|
if (num > min_range && num < max_range)
|
|
|
|
best_primes[no_of_best_primes++]=num;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* binary search for `num' in the `best_primes' array,
|
|
|
|
* Return smallest best_prime >= num.
|
|
|
|
*/
|
|
|
|
static int best_prime_binary_search(int num)
|
|
|
|
{
|
|
|
|
int min_idx, max_idx, idx;
|
|
|
|
|
|
|
|
min_idx=0;
|
|
|
|
max_idx=no_of_best_primes-1;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
idx = (max_idx + min_idx) >> 1;
|
|
|
|
if (num == best_primes[idx])
|
|
|
|
return num;
|
|
|
|
if (num < best_primes[idx]) {
|
|
|
|
max_idx = idx-1;
|
|
|
|
if (max_idx <= min_idx)
|
|
|
|
return best_primes[idx];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
min_idx = idx+1;
|
|
|
|
if (min_idx >= max_idx)
|
|
|
|
return best_primes[max_idx];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find the best prime, near `num' (which can be any number) */
|
|
|
|
static int best_prime(int num)
|
|
|
|
{
|
|
|
|
int log2;
|
|
|
|
int pow2less, pow2more;
|
|
|
|
int min_range, max_range;
|
|
|
|
|
|
|
|
if (num < 11)
|
|
|
|
return 11;
|
|
|
|
|
|
|
|
if (num <= best_primes[no_of_best_primes-1])
|
|
|
|
return best_prime_binary_search(num);
|
|
|
|
|
|
|
|
assert( num < max_num );
|
|
|
|
|
|
|
|
for (log2=0 ; num >> log2 ; log2++)
|
|
|
|
;
|
|
|
|
|
|
|
|
pow2less= 1 << log2;
|
|
|
|
pow2more= 1 << (log2+1);
|
|
|
|
min_range= pow2less + (pow2less >> 3);
|
|
|
|
max_range= pow2more - (pow2more >> 3);
|
|
|
|
|
|
|
|
if (num < min_range)
|
|
|
|
num= min_range;
|
|
|
|
|
|
|
|
num |= 1; /* make sure num can't be divided by 2 */
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
if (num >= max_range) {
|
|
|
|
pow2less<<= 1;
|
|
|
|
pow2more<<= 1;
|
|
|
|
min_range= pow2less + (pow2less >> 3);
|
|
|
|
max_range= pow2more - (pow2more >> 3);
|
|
|
|
num= min_range | 1; /* make sure num can't be divided by 2 */
|
|
|
|
}
|
|
|
|
/* num should be here in the range: (min_range, max_range) */
|
|
|
|
if (is_prime(num))
|
|
|
|
return num;
|
|
|
|
num+=2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: This can be done before compiling. (uning a script)*/
|
|
|
|
static void setup_arrays()
|
|
|
|
{
|
|
|
|
setup_primes();
|
|
|
|
setup_best_primes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Discard all DELETED_ENTRYs moving the data to it's correct location.
|
|
|
|
* Done without a temporary buffer.
|
|
|
|
* May require some efficiency improvements ( currently it's o(N^2)
|
|
|
|
* or is it o(N^3) in the worst case ? In the avarege it seems to be
|
|
|
|
* something like o(N log (N)))
|
|
|
|
*/
|
|
|
|
static void static_collect_garbge(HASH_CONTAINER *hash)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
BOOL change;
|
|
|
|
HASH_ITEM *items;
|
|
|
|
HASH_ITEM *located;
|
|
|
|
HASH_ITEM *item;
|
|
|
|
int key;
|
|
|
|
|
|
|
|
items= hash->items;
|
|
|
|
|
|
|
|
do {
|
|
|
|
change= FALSE;
|
|
|
|
for (i=hash->shared->total_items-1 ; i >= 0 ; i--) {
|
|
|
|
item= &GET_ITEM(items,hash->bytes_per_item,i);
|
|
|
|
key= item->key;
|
|
|
|
if (key != DELETED_ENTRY && key != FREE_ENTRY) {
|
|
|
|
/* try to place the entry in a deleted location */
|
|
|
|
located= locate_entry(hash, key, &item->data,
|
|
|
|
0 /* no skip_deleted */);
|
|
|
|
if (located->key == DELETED_ENTRY) {
|
|
|
|
change= TRUE;
|
|
|
|
memcpy(&located, &item,
|
|
|
|
hash->bytes_per_item);
|
|
|
|
item->key= DELETED_ENTRY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (change);
|
|
|
|
|
|
|
|
/* No change means that there is no need to go through a DELETED_ENTRY
|
|
|
|
* in order to reach an item, so DELETED_ENTRY looses it's special
|
|
|
|
* meaning, and it is the same as FREE_ENTRY.
|
|
|
|
*/
|
|
|
|
for (i=hash->shared->total_items-1 ; i >= 0 ; i--)
|
|
|
|
if (GET_ITEM(items,hash->bytes_per_item,i).key == DELETED_ENTRY)
|
|
|
|
GET_ITEM(items,hash->bytes_per_item,i).key = FREE_ENTRY;
|
|
|
|
hash->shared->deleted_items=0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void collect_garbge(HASH_CONTAINER *hash)
|
|
|
|
{
|
|
|
|
HASH_SHARED *shared= hash->shared;
|
|
|
|
HASH_ITEM *temp_items;
|
|
|
|
int size;
|
|
|
|
|
|
|
|
size= shared->total_items * hash->bytes_per_item;
|
|
|
|
temp_items= (HASH_ITEM*)malloc(size);
|
|
|
|
if (temp_items==NULL) {
|
|
|
|
static_collect_garbge(hash);
|
|
|
|
} else {
|
|
|
|
memcpy(temp_items, hash->items, size);
|
|
|
|
copy_hash_items(hash, temp_items, shared->total_items);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void copy_hash_items(HASH_CONTAINER *hash, HASH_ITEM *old_items,
|
|
|
|
int old_n_items)
|
|
|
|
{
|
|
|
|
HASH_SHARED *shared= hash->shared;
|
|
|
|
HASH_ITEM *item;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
shared->deleted_items = 0;
|
|
|
|
shared->free_items= shared->total_items;
|
|
|
|
|
|
|
|
/* make all items free */
|
|
|
|
for (i= shared->total_items-1 ; i>=0 ; i--)
|
|
|
|
GET_ITEM(hash->items, hash->bytes_per_item, i).key = FREE_ENTRY;
|
|
|
|
|
|
|
|
/* copy items */
|
|
|
|
for (i=0 ; i <= old_n_items; i++) {
|
|
|
|
item= &GET_ITEM(old_items, hash->bytes_per_item,i);
|
|
|
|
if (item->key != FREE_ENTRY && item->key != DELETED_ENTRY)
|
|
|
|
hash_add_item(hash, item->key, &item->data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void reorder_hash(HASH_CONTAINER *hash)
|
|
|
|
{
|
|
|
|
HASH_SHARED *shared= hash->shared;
|
|
|
|
HASH_ITEM *items, *old_items;
|
|
|
|
HASH_PTR shared_items, old_shared_items;
|
|
|
|
int n_items, old_n_items;
|
|
|
|
int size;
|
|
|
|
|
|
|
|
if (shared->deleted_items > hash->min_free_items) {
|
|
|
|
collect_garbge(hash);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
n_items= best_prime(shared->total_items * HASH_REALLOC_JUMPS);
|
|
|
|
|
|
|
|
size= n_items *
|
|
|
|
(sizeof(items[0]) - sizeof(items[0].data) + hash->bytes_per_item);
|
|
|
|
|
|
|
|
shared_items= hash->allocate_mem(size);
|
|
|
|
items= hash->access_mem(shared_items);
|
|
|
|
|
|
|
|
if (items == NULL) {
|
|
|
|
collect_garbge(hash);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
old_shared_items = shared->items;
|
|
|
|
old_n_items= shared->total_items;
|
|
|
|
old_items= hash->items;
|
|
|
|
|
|
|
|
/* setup a new clean hash based on the parameters of the original one */
|
|
|
|
hash->items= items;
|
|
|
|
shared->total_items = n_items;
|
|
|
|
shared->items= shared_items;
|
|
|
|
set_hash_parameters(hash, hash->maximum_load);
|
|
|
|
copy_hash_items(hash, old_items, old_n_items);
|
|
|
|
|
|
|
|
hash->free_mem(old_shared_items);
|
|
|
|
hash->last_ptr_update= ++shared->ptr_updates;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* low level: attach hash existing hash items, no checks are performed
|
|
|
|
* No complex calculations done.
|
|
|
|
*/
|
|
|
|
static HASH_CONTAINER *attach_no_check(HASH_ITEM *items, int bytes_per_datum)
|
|
|
|
{
|
|
|
|
HASH_CONTAINER *hash;
|
|
|
|
int bytes_per_item;
|
|
|
|
HASH_ITEM dummy_item;
|
|
|
|
|
|
|
|
hash= (HASH_CONTAINER*) malloc(sizeof(HASH_CONTAINER) );
|
|
|
|
if (hash == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
bytes_per_item= bytes_per_datum+
|
|
|
|
sizeof(dummy_item)-sizeof(dummy_item.data);
|
|
|
|
hash->bytes_per_item= ROUND_UP4(bytes_per_item);
|
|
|
|
hash->items= items;
|
|
|
|
hash->is_correct_item= NULL;
|
|
|
|
hash->allocate_mem= HASH_MEM_ALLOC;
|
|
|
|
hash->access_mem= HASH_MEM_ACCESS;
|
|
|
|
hash->free_mem= HASH_MEM_FREE;
|
|
|
|
set_hash_parameters(hash, HASH_LOAD);
|
|
|
|
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Attach existing & running remote (i.e. shared) hash.
|
|
|
|
* Attach the items using the data stored in "shared"
|
|
|
|
*/
|
|
|
|
HASH_CONTAINER *attach_remote_hash(HASH_SHARED *shared, int bytes_per_datum,
|
|
|
|
HASH_ITEM *(*access_mem)(HASH_PTR))
|
|
|
|
{
|
|
|
|
HASH_CONTAINER *hash;
|
|
|
|
HASH_ITEM *items;
|
|
|
|
|
|
|
|
assert(access_mem != NULL);
|
|
|
|
if (! arrays_initialized)
|
|
|
|
setup_arrays();
|
|
|
|
|
|
|
|
items=access_mem(shared->items);
|
|
|
|
hash= attach_no_check(items, bytes_per_datum);
|
|
|
|
if (hash == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
hash->shared_was_malloced = FALSE;
|
|
|
|
hash->shared= shared;
|
|
|
|
return (hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HASH_CONTAINER *create_remote_hash(HASH_SHARED *shared,
|
|
|
|
int bytes_per_datum,
|
|
|
|
int total_items,
|
|
|
|
HASH_PTR (*allocate_mem)(int size),
|
|
|
|
HASH_ITEM *(*access_mem)(HASH_PTR))
|
|
|
|
{
|
|
|
|
HASH_CONTAINER *hash;
|
|
|
|
int size;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
assert(total_items >= 1);
|
|
|
|
assert(bytes_per_datum >=1);
|
|
|
|
assert(access_mem != NULL);
|
|
|
|
assert(allocate_mem != NULL);
|
|
|
|
assert(shared != NULL);
|
|
|
|
if (! arrays_initialized)
|
|
|
|
setup_arrays();
|
|
|
|
|
|
|
|
if (total_items < MIN_HASH)
|
|
|
|
total_items= MIN_HASH;
|
|
|
|
else
|
|
|
|
total_items= best_prime(total_items);
|
|
|
|
|
|
|
|
hash= attach_no_check(NULL, bytes_per_datum);
|
|
|
|
|
|
|
|
if (hash==NULL) {
|
|
|
|
free(hash);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
shared->total_items= total_items;
|
|
|
|
hash->shared= shared;
|
|
|
|
hash->shared_was_malloced = FALSE;
|
|
|
|
|
|
|
|
size= total_items * hash->bytes_per_item;
|
|
|
|
|
|
|
|
shared->items = allocate_mem(size);
|
|
|
|
hash->items= access_mem(shared->items);
|
|
|
|
|
|
|
|
if (hash->items == NULL ) {
|
|
|
|
free(hash);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
shared->items.ptr= hash->items;
|
|
|
|
|
|
|
|
/* make all items free */
|
|
|
|
for (i=0 ; i < total_items ; i++)
|
|
|
|
GET_ITEM(hash->items,hash->bytes_per_item,i).key = FREE_ENTRY;
|
|
|
|
|
|
|
|
shared->deleted_items= 0;
|
|
|
|
shared->free_items= total_items;
|
|
|
|
shared->ptr_updates= 0;
|
|
|
|
return hash;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* hash constructor: create brand new hash */
|
|
|
|
HASH_CONTAINER *create_hash(int bytes_per_datum, int total_items)
|
|
|
|
{
|
|
|
|
HASH_CONTAINER *hash;
|
|
|
|
HASH_SHARED *shared;
|
|
|
|
|
|
|
|
|
|
|
|
shared= (HASH_SHARED*)malloc(sizeof(HASH_SHARED));
|
|
|
|
if (shared == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
hash= create_remote_hash(shared, bytes_per_datum, total_items,
|
|
|
|
HASH_MEM_ALLOC, HASH_MEM_ACCESS);
|
|
|
|
|
|
|
|
if (hash == NULL) {
|
|
|
|
free(shared);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
hash->shared_was_malloced = TRUE;
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the extra handlers to non default values */
|
|
|
|
void set_hash_handlers(HASH_CONTAINER *hash,
|
|
|
|
HASH_ITEM_TEST *is_correct_item,
|
|
|
|
HASH_PTR (*allocate_mem)(int size),
|
|
|
|
void (*free_mem)(HASH_PTR),
|
|
|
|
HASH_ITEM *(*access_mem)(HASH_PTR))
|
|
|
|
{
|
|
|
|
assert(hash);
|
|
|
|
assert(allocate_mem);
|
|
|
|
assert(free_mem);
|
|
|
|
|
|
|
|
hash->free_mem = free_mem;
|
|
|
|
hash->allocate_mem = allocate_mem;
|
|
|
|
hash->access_mem = access_mem;
|
|
|
|
hash->is_correct_item = is_correct_item;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* set extra parameters */
|
|
|
|
void set_hash_parameters(HASH_CONTAINER *hash, int load)
|
|
|
|
{
|
|
|
|
assert(hash);
|
|
|
|
assert(load>30); /* no sence to realloc with less than */
|
|
|
|
/* 50% load, limiting to 30% to be on */
|
|
|
|
/* the safe size */
|
|
|
|
assert(load<=100);
|
|
|
|
|
|
|
|
hash->maximum_load= load;
|
|
|
|
hash->min_free_items= (1.0 - load/100.0) * hash->shared->total_items + 1 ;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* hash destructor: destroy anything related to the hash */
|
|
|
|
void destroy_hash(HASH_CONTAINER *hash)
|
|
|
|
{
|
|
|
|
assert(hash);
|
|
|
|
hash->free_mem(hash->shared->items);
|
|
|
|
if (hash->shared_was_malloced)
|
|
|
|
free(hash->shared);
|
|
|
|
free(hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* hash destructor: just detach hash, without destroing it (makes */
|
|
|
|
/* sence in shared memory environment) */
|
|
|
|
void detach_hash(HASH_CONTAINER *hash)
|
|
|
|
{
|
|
|
|
assert(hash);
|
|
|
|
free(hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/********** Hash usage *************/
|
|
|
|
static __inline__ BOOL
|
|
|
|
correct_entry(HASH_ITEM *item, int key, HASH_VAL *seeked_data,
|
|
|
|
HASH_ITEM_TEST *is_correct_item, BOOL skip_deleted)
|
|
|
|
{
|
|
|
|
switch(item->key) {
|
|
|
|
case FREE_ENTRY:
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
case DELETED_ENTRY:
|
|
|
|
return skip_deleted ? FALSE : TRUE;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (item->key != key)
|
|
|
|
return FALSE;
|
|
|
|
if (is_correct_item != NULL)
|
|
|
|
return is_correct_item(&item->data, seeked_data);
|
|
|
|
else
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The algorithm of the hash (one of the 2 standard hash implementations):
|
|
|
|
* Iterate through the hash table until
|
|
|
|
* 1. The entry has been found.
|
|
|
|
* 2. A FREE entry has been found.
|
|
|
|
* 3. For insert operations only- A DELETED entry has been found.
|
|
|
|
* The difference between DELETED and FREE entires is that
|
|
|
|
* DELETED entry was one occupied, while FREE was never allocated.
|
|
|
|
* The idea behind this structure to keep other entries reachable.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static HASH_ITEM *locate_entry(HASH_CONTAINER* hash, DWORD key,
|
|
|
|
HASH_VAL *seeked_data, BOOL skip_deleted)
|
|
|
|
{
|
|
|
|
DWORD hash_idx, hash_leaps;
|
|
|
|
HASH_ITEM *item;
|
|
|
|
int i;
|
|
|
|
int total_items;
|
|
|
|
|
|
|
|
assert(hash);
|
|
|
|
|
|
|
|
total_items= hash->shared->total_items;
|
|
|
|
hash_idx= key % total_items;
|
|
|
|
|
|
|
|
item= &GET_ITEM(hash->items, hash->bytes_per_item, hash_idx);
|
|
|
|
|
|
|
|
if ( correct_entry( item, key, seeked_data,
|
|
|
|
hash->is_correct_item, skip_deleted) )
|
|
|
|
return item;
|
|
|
|
|
|
|
|
/* get the WORDs in different order in this DWORD to avoid clustering */
|
|
|
|
hash_leaps=((DWORD)MAKELONG(HIWORD(key), LOWORD(key))
|
|
|
|
% (total_items-1)) +1;
|
|
|
|
|
|
|
|
/* interate through the hash table using hash_leaps */
|
|
|
|
for (i= total_items ; i ; i--) {
|
|
|
|
hash_idx+= hash_leaps;
|
|
|
|
if (hash_idx > total_items)
|
|
|
|
hash_idx -= total_items;
|
|
|
|
|
|
|
|
item= &GET_ITEM(hash->items,hash->bytes_per_item, hash_idx);
|
|
|
|
if ( correct_entry( item, key, seeked_data,
|
|
|
|
hash->is_correct_item, skip_deleted) )
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline__ void sync_shared_hash(HASH_CONTAINER *hash)
|
|
|
|
{
|
|
|
|
HASH_SHARED *shared= hash->shared;
|
|
|
|
|
|
|
|
if (shared->ptr_updates == hash->last_ptr_update)
|
|
|
|
return;
|
|
|
|
|
|
|
|
assert(shared->ptr_updates >= hash->last_ptr_update);
|
|
|
|
hash->last_ptr_update= shared->ptr_updates;
|
|
|
|
hash->min_free_items= (1.0 - hash->maximum_load/100.0) *
|
|
|
|
shared->total_items + 1 ;
|
|
|
|
|
|
|
|
hash->items= hash->access_mem(shared->items);
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_VAL *hash_locate_item(HASH_CONTAINER* hash,
|
|
|
|
int key, HASH_VAL *seeked_data)
|
|
|
|
{
|
|
|
|
HASH_ITEM *item;
|
|
|
|
|
|
|
|
assert(hash != NULL);
|
|
|
|
sync_shared_hash(hash);
|
|
|
|
|
|
|
|
item= locate_entry(hash, key, seeked_data, 1 /* skip_deleted */);
|
|
|
|
if (item == NULL)
|
|
|
|
return NULL;
|
|
|
|
if (item->key == FREE_ENTRY )
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return &item->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOL hash_add_item(HASH_CONTAINER* hash, int key, HASH_VAL *data)
|
|
|
|
{
|
|
|
|
HASH_SHARED *shared;
|
|
|
|
HASH_ITEM *item;
|
|
|
|
|
|
|
|
assert(hash != NULL);
|
|
|
|
|
|
|
|
sync_shared_hash(hash);
|
|
|
|
shared= hash->shared;
|
|
|
|
|
|
|
|
item=locate_entry(hash, key, data, 0 /* no skip_deleted */);
|
|
|
|
assert(item != NULL);
|
|
|
|
if (item->key == key)
|
|
|
|
return FALSE;
|
|
|
|
if (item->key == FREE_ENTRY)
|
|
|
|
shared->free_items--;
|
|
|
|
else
|
|
|
|
shared->deleted_items--;
|
|
|
|
|
|
|
|
item->key= key;
|
|
|
|
memcpy(&item->data, data, hash->bytes_per_item-sizeof(key));
|
|
|
|
|
|
|
|
if (shared->free_items < hash->min_free_items ||
|
|
|
|
shared->deleted_items > hash->min_free_items)
|
|
|
|
reorder_hash(hash);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOL hash_delete_item(HASH_CONTAINER* hash, int key, HASH_VAL *seeked_data)
|
|
|
|
{
|
|
|
|
HASH_ITEM *item;
|
|
|
|
|
|
|
|
assert(hash != NULL);
|
|
|
|
sync_shared_hash(hash);
|
|
|
|
|
|
|
|
item=locate_entry(hash, key, seeked_data, 1 /* skip_deleted */);
|
|
|
|
if (item == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (item->key == FREE_ENTRY)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
item->key = DELETED_ENTRY;
|
|
|
|
hash->shared->deleted_items++;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *ret_null()
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HASH_ITEM *access_local_hash(HASH_PTR ptr)
|
|
|
|
{
|
|
|
|
return ptr.ptr;
|
|
|
|
}
|
Release 950901
Thu Aug 31 17:19:57 1995 Alexandre Julliard <julliard@sunsite.unc.edu>
* [Configure]
Added compile-time option for IPC.
* [configure.in]
Added command-line options for language, IPC and malloc
debugging.
* [controls/menu.c]
WM_MENUSELECT was sometimes sent to the wrong window.
* [debugger/break.c]
For the 'next' command, only step over instruction that require
it. This allows 'next' to do the right thing with jmp and ret
instructions.
* [ipc/*.c] [memory/atom.c] [memory/global.c]
IPC can now be configured out at compile-time.
* [loader/task.c]
Bug fix in TASK_Reschedule() that could cause a task to be deleted
twice.
* [miscemu/dosmem.c] (New file)
Partial emulation of the BIOS data segment.
* [miscemu/instr.c]
Trap attempts to access selector 0x40 and remap the access to
segment __0040H.
* [tools/build.c]
Fixed bug in CallTo32_LargeStack() that caused problems when
compiling Wine with the -fomit-frame-pointer option.
* [windows/message.c]
Fixed bug in hardware event handling that could cause some events
to get ignored.
Sat Aug 26 13:12:59 IST 1995 Michael Veksler <mveksler@vnet.ibm.com>
* [ipc/README] [ipc/dde.tex]
LaTeX documentation for the ipc and DDE stuff.
Wed Aug 23 22:01:23 GMT 1995 Michael Veksler <mveksler@vnet.ibm.com>
* [ipc/Imakefile] [ipc/wine_test_stub.c]
Fixed IPC testing. Now it can be compiled with "make tests"
Wed Aug 23 21:04:14 1995 Fons Botman <botman@wab-tis.rabobank.nl>
* [if1632/kernel.spec] [include/windows.h] [misc/main.c]
Added GetWinDebugInfo/SetWinDebugInfo stub for player.exe
Sun Aug 20 13:49:42 1995 Marcus Meissner <msmeissn@faui01.informatik.uni-erlangen.de>
* [miscemu/int21.c]
Misc fix to int21,ah=40 (write) to match _lwrite().
AX=0x440A (check if handle is remote) added.
* [multimedia/mmsystem.c]
Moved mciSendString to mcistring.c.
* [multimedia/mcistring.c]
New file, string interface for MCI (not complete, not thoroughly
tested).
* [multimedia/audio.c]
IOCTL prints errors; one paranoid check disabled.
* [misc/file.c]
Misc operator precedence fixes.
* [if1632/gdi.spec] [objects/bitblt.c]
Stub for FastWindowFrame (parameters not correct).
Sat Aug 19 01:31:23 1995 Graham Menhennitt <gfm@werple.mira.net.au>
* [loader/ne_image.c]
Preliminary support for iterated segments.
Sat Aug 19 00:43:04 1995 Andrew Taylor (andrew@riscan.com)
* [windows/mapping.c]
In function MAPPING_FixIsotropic(), VportExt[XY] is multiplied by
the absolute value of (ydim / xdim) or (xdim / ydim).
Thu Aug 15 23:00:16 Gregory Trubetskoy <grisha@mira.com>
* [objects/oembitmap.c]
Added some includes for Windows 95.
* [include/sysmetrics.h]
Added some sysmetrics for Windows 95.
* [include/bitmaps/*95]
New files: obm_close_95, obm_closed_95, obm_reduce_95, obm_reduced_95
obm_zoom_95, obm_zoomd_95 - these are some pixmaps for Windows 95.
Thu Aug 10 12:00:00 1995 Jan Willamowius (jan@janhh.shnet.org)
* [misc/shell.c] [rc/sysres*.rc]
The caption of the ShellAbout dialog box is language specific and
should be defined in the resources.
1995-09-01 17:57:28 +02:00
|
|
|
|
|
|
|
#endif /* CONFIG_IPC */
|