mirror of
git://source.winehq.org/git/wine.git
synced 2024-10-31 11:43:31 +00:00
d37eb3656c
Sat Jul 19 13:03:01 1997 Alexandre Julliard <julliard@lrc.epfl.ch> * [tools/build.c] [include/stackframe.h] Save the complete %ebp register in CallFrom16; fixes a crash with LabView reported by Andreas Mohr. * [loader/main.c] Avoid executing a built-in DLL. * [controls/static.c] Converted static window procedure to Win32. * [windows/message.c] [windows/queue.c] [include/queue.h] Hacked SendMessage functions to support inter-task messages with SendMessage32A/W. Sun Jul 13 16:55:35 1997 Bernhard Rosenkraenzer <bero@bero-online.ml.org> * [ipc/bit_array.c] Don't use bitops.h in Linux 2.1.x (these versions do not return the previous state for clear_bit and set_bit) * [ipc/shm_main_blk.c] Adapt to GLIBC's ipc_perm structure. * [memory/ldt.c] Include <asm/unistd.h> on Linux/GLIBC systems (required for _syscall3). Wed Jul 9 23:53:19 1997 David A. Cuthbert <dacut@henry.ece.cmu.edu> * [include/options.h] [files/profile.c] Added PROFILE_GetWineIniBool and PROFILE_EnumerateWineIniSection. * [include/sysmetrics.h] [include/windows.h] [windows/sysmetrics.c] All sysmetrics moved to array (no more constant macros). Added MOUSEWHEELPRESENT metric. * [include/bitmap.h] [objects/oembitmap.c] Added OBM_Init() (see also loader/main.c) and more support for Win95 bitmaps; added size info to OEM bitmaps. * [include/graphics.h] [windows/graphics.h] Added GRAPH_DrawGenericReliefRect. * [loader/main.c] Added TWEAK_Init() and TWEAK_CheckConfiguration() calls (the latter checks for invalid entries in wine.conf). * [include/debug.h] [include/stddebug.h] [include/nonclient.h] [include/tweak.h] [controls/menu.c] [misc/tweak.c] [objects/gdiobj.c] [windows/syscolor.c] [windows/nonclient.c] [BUGS] [documentation/win95look] Added tweaks for Windows 95 interface support. See documentation/win95look for more information. * [controls/edit.c] Fixed EDIT_MoveHome bug. * [misc/ver.c] Changed name of dprintf_ver_string to ver_dstring to fix problem with tools/make_debug utility. Wed Jul 9 21:31:54 1997 Marcus Meissner <msmeissn@cip.informatik.uni-erlangen.de> * [objects/dib.c] Don't use palettes with dibs with biBitCount > 8. * [misc/ole2nls.c][misc/ver.c] IsValidLocale, EnumSystemLocales fixed (winhlp32.exe works) Some VerLanguage coded moved to ole2nls.c, some cleanups. * [multimedia/mcistring.c] Fixed "capabilities <dev> device type" crash (cool.exe). * [misc/main.c] SystemParametersInfo*: added stub option 41 (GETNONCLIENTMETRICS), duplicated some stuff away from SPI16 that writes 32bit vars.(one COMCTL32.DLL crash, freecell.exe) Tue Jul 8 22:40:53 1997 Morten Welinder <terra@diku.dk> * [if1632/shell32.spec] Use Windows 95's ordinals. Help wanted, inquire within. Mon Jul 7 11:20:36 1997 Philippe De Muyter <phdm@info.ucl.ac.be> * [if1632/relay.c] [if1632/user.spec] [if1632/kernel.spec] [tools/build-spec.txt] [tools/build.c] Added type 'segstr' (segmented pointer to null-terminated string) to .spec files. * [windows/user.c] [if1632/user.spec] ExitWindowsExec stub function added. Mon Jul 7 01:18:25 1997 U. Bonnes <bon@elektron.ikp.physik.th-darmstadt.de> * [files/file.c] [include/winbase.h] [if1632/kernel32.spec] Implement MoveFileEx32, some enhancement for Movefile32. Sat Jul 5 18:13:48 1997 Bruce Milner <Bruce.Milner@genetics.utah.edu. * [files/file.c] [if1632/kernel32.spec] [include/winerror.h] [msdos/int21.c] [win32/file.c] Add LockFile/UnlockFile implementation. Add back in int21 func(0x5c) Record locking functions. * [files/file.c] Fixed bug with OF_REOPEN in FILE_DoOpenFile. Fri Jul 4 12:00:00 1997 Henrik Olsen <Henrik.Olsen@iaeste.dk> * [misc/ole2nls.c] [programs/progman/Da.rc] [programs/winhelp/Da.rc] [resources/sysres_Da.rc] Added/updated Danish language support. Thu Jul 3 13:04:20 1997 Claus Fischer <fischer@iue.tuwien.ac.at> * [files/dos_fs.c] Properly implemented DOSFS_UnixTimeToFileTime and DOSFS_FileTimeToUnixTime. * [documentation/wine.texinfo] First version of texinfo documentation.
284 lines
6.3 KiB
C
284 lines
6.3 KiB
C
/***************************************************************************
|
|
* Copyright 1995, Technion, Israel Institute of Technology
|
|
* Electrical Eng, Software Lab.
|
|
* Author: Michael Veksler.
|
|
***************************************************************************
|
|
* File: bit_array.c
|
|
* Purpose : manipulate array of bits
|
|
* Portability: This is not completely portable, non CISC arcitectures
|
|
* Might not have atomic Clear/Set/Toggle bit. On those
|
|
* architectures semaphores should be used.
|
|
* Big Endian Concerns: This code is big endian compatible,
|
|
* but the byte order will be different (i.e. bit 0 will be
|
|
* located in byte 3).
|
|
***************************************************************************
|
|
*/
|
|
|
|
#ifdef CONFIG_IPC
|
|
|
|
/*
|
|
** uncoment the following line to disable assertions,
|
|
** this may boost performance by up to 50%
|
|
*/
|
|
/* #define NDEBUG */
|
|
|
|
#if defined(linux) && !defined(NO_ASM)
|
|
#include <linux/version.h>
|
|
#if LINUX_VERSION_CODE <= 131328 /* Linux 2.1.x doesn't return values with clear_bit and set_bit */
|
|
#define HAS_BITOPS
|
|
#endif
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "bit_array.h"
|
|
#ifdef HAS_BITOPS
|
|
#define inline __inline__ /* So we can compile with -ansi */
|
|
#include <asm/bitops.h>
|
|
#else
|
|
static __inline__ int clear_bit(int bit, int *mem);
|
|
static __inline__ int set_bit(int bit, int *mem);
|
|
#endif /* HAS_BITOPS */
|
|
|
|
|
|
#define INT_NR(bit_nr) ((bit_nr) >> INT_LOG2)
|
|
#define INT_COUNT(bit_count) INT_NR( bit_count + BITS_PER_INT - 1 )
|
|
#define BIT_IN_INT(bit_nr) ((bit_nr) & (BITS_PER_INT - 1))
|
|
|
|
#if !defined(HAS_BITOPS)
|
|
|
|
/* first_zero maps bytes value to the index of first zero bit */
|
|
static char first_zero[256];
|
|
static int arrays_initialized=0;
|
|
|
|
|
|
/*
|
|
** initialize static arrays used for bit operations speedup.
|
|
** Currently initialized: first_zero[256]
|
|
** set "arrays_initialized" to inidate that arrays where initialized
|
|
*/
|
|
|
|
static void initialize_arrays()
|
|
{
|
|
int i;
|
|
int bit;
|
|
|
|
for (i=0 ; i<256 ; i++) {
|
|
/* find the first zero bit in `i' */
|
|
for (bit=0 ; bit < BITS_PER_BYTE ; bit++)
|
|
/* break if the bit is zero */
|
|
if ( ( (1 << bit) & i )
|
|
== 0)
|
|
break;
|
|
first_zero[i]= bit;
|
|
}
|
|
arrays_initialized=1;
|
|
}
|
|
|
|
/*
|
|
** Find first zero bit in the integer.
|
|
** Assume there is at least one zero.
|
|
*/
|
|
static __inline__ int find_zbit_in_integer(unsigned int integer)
|
|
{
|
|
int i;
|
|
|
|
/* find the zero bit */
|
|
for (i=0 ; i < sizeof(int) ; i++, integer>>=8) {
|
|
int byte= integer & 0xff;
|
|
|
|
if (byte != 0xff)
|
|
return ( first_zero[ byte ]
|
|
+ (i << BYTE_LOG2) );
|
|
}
|
|
assert(0); /* never reached */
|
|
return 0;
|
|
}
|
|
|
|
/* return -1 on failure */
|
|
static __inline__ int find_first_zero_bit(unsigned *array, int bits)
|
|
{
|
|
unsigned int integer;
|
|
int i;
|
|
int bytes=INT_COUNT(bits);
|
|
|
|
if (!arrays_initialized)
|
|
initialize_arrays();
|
|
|
|
for ( i=bytes ; i ; i--, array++) {
|
|
integer= *array;
|
|
|
|
/* test if integer contains a zero bit */
|
|
if (integer != ~0U)
|
|
return ( find_zbit_in_integer(integer)
|
|
+ ((bytes-i) << INT_LOG2) );
|
|
}
|
|
|
|
/* indicate failure */
|
|
return -1;
|
|
}
|
|
|
|
static __inline__ int test_bit(int pos, unsigned *array)
|
|
{
|
|
unsigned int integer;
|
|
int bit = BIT_IN_INT(pos);
|
|
|
|
integer= array[ pos >> INT_LOG2 ];
|
|
|
|
return ( (integer & (1 << bit)) != 0
|
|
? 1
|
|
: 0 ) ;
|
|
}
|
|
|
|
/*
|
|
** The following two functions are x86 specific ,
|
|
** other processors will need porting
|
|
*/
|
|
|
|
/* inputs: bit number and memory address (32 bit) */
|
|
/* output: Value of the bit before modification */
|
|
static __inline__ int clear_bit(int bit, int *mem)
|
|
{
|
|
int ret;
|
|
|
|
__asm__("xor %1,%1
|
|
btrl %2,%0
|
|
adcl %1,%1"
|
|
:"=m" (*mem), "=&r" (ret)
|
|
:"r" (bit));
|
|
return (ret);
|
|
}
|
|
|
|
static __inline__ int set_bit(int bit, int *mem)
|
|
{
|
|
int ret;
|
|
__asm__("xor %1,%1
|
|
btsl %2,%0
|
|
adcl %1,%1"
|
|
:"=m" (*mem), "=&r" (ret)
|
|
:"r" (bit));
|
|
return (ret);
|
|
}
|
|
|
|
#endif /* !deined(HAS_BITOPS) */
|
|
|
|
|
|
/* AssembleArray: assemble an array object using existing data */
|
|
bit_array *AssembleArray(bit_array *new_array, unsigned int *buff, int bits)
|
|
{
|
|
assert(new_array!=NULL);
|
|
assert(buff!=NULL);
|
|
assert(bits>0);
|
|
assert((1 << INT_LOG2) == BITS_PER_INT); /* if fails, redefine INT_LOG2 */
|
|
|
|
new_array->bits=bits;
|
|
new_array->array=buff;
|
|
return new_array;
|
|
}
|
|
|
|
/* ResetArray: reset the bit array to zeros */
|
|
int ResetArray(bit_array *bits)
|
|
{
|
|
int i;
|
|
int *p;
|
|
|
|
assert(bits!=NULL);
|
|
assert(bits->array!=NULL);
|
|
|
|
for(i= INT_COUNT(bits->bits), p=bits->array; i ; p++, i--)
|
|
*p=0;
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* VacantBit: find a vacant (zero) bit in the array,
|
|
* Return: Bit index on success, -1 on failure.
|
|
*/
|
|
int VacantBit(bit_array *bits)
|
|
{
|
|
int bit;
|
|
|
|
assert(bits!=NULL);
|
|
assert(bits->array!=NULL);
|
|
|
|
bit= find_first_zero_bit(bits->array, bits->bits);
|
|
|
|
if (bit >= bits->bits) /* failed? */
|
|
return -1;
|
|
|
|
return bit;
|
|
}
|
|
|
|
int SampleBit(bit_array *bits, int i)
|
|
{
|
|
assert(bits != NULL);
|
|
assert(bits->array != NULL);
|
|
assert(i >= 0 && i < bits->bits);
|
|
|
|
return ( test_bit(i,bits->array) != 0
|
|
? 1
|
|
: 0
|
|
);
|
|
}
|
|
|
|
|
|
/*
|
|
** Use "compare and exchange" mechanism to make sure
|
|
** that bits are not modified while "integer" value
|
|
** is calculated.
|
|
**
|
|
** This may be the slowest technique, but it is the most portable
|
|
** (Since most architectures have compare and exchange command)
|
|
*/
|
|
int AssignBit(bit_array *bits, int bit_nr, int val)
|
|
{
|
|
int ret;
|
|
|
|
assert(bits != NULL);
|
|
assert(bits->array != NULL);
|
|
assert(val==0 || val==1);
|
|
assert(bit_nr >= 0 && bit_nr < bits->bits);
|
|
|
|
if (val==0)
|
|
ret= clear_bit(BIT_IN_INT(bit_nr), &bits->array[ INT_NR(bit_nr) ]);
|
|
else
|
|
ret= set_bit(BIT_IN_INT(bit_nr), &bits->array[ INT_NR(bit_nr) ]);
|
|
|
|
return ( (ret!=0) ? 1 : 0);
|
|
}
|
|
|
|
/*
|
|
** Allocate a free bit (==0) and make it used (==1).
|
|
** This operation is guaranteed to resemble an atomic instruction.
|
|
**
|
|
** Return: allocated bit index, or -1 on failure.
|
|
**
|
|
** There is a crack between locating free bit, and allocating it.
|
|
** We assign 1 to the bit, test it was not '1' before the assignment.
|
|
** If it was, restart the seek and assign cycle.
|
|
**
|
|
*/
|
|
|
|
int AllocateBit(bit_array *bits)
|
|
{
|
|
int bit_nr;
|
|
int orig_bit;
|
|
|
|
assert(bits != NULL);
|
|
assert(bits->array != NULL);
|
|
|
|
do {
|
|
bit_nr= VacantBit(bits);
|
|
|
|
if (bit_nr == -1) /* No vacant bit ? */
|
|
return -1;
|
|
|
|
orig_bit = AssignBit(bits, bit_nr, 1);
|
|
} while (orig_bit != 0); /* it got assigned before we tried */
|
|
|
|
return bit_nr;
|
|
}
|
|
|
|
#endif /* CONFIG_IPC */
|