home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
OS/2 Shareware BBS: 10 Tools
/
10-Tools.zip
/
mitsch75.zip
/
scheme-7_5_17-src.zip
/
scheme-7.5.17
/
src
/
microcode
/
bitstr.c
< prev
next >
Wrap
C/C++ Source or Header
|
2000-12-05
|
26KB
|
873 lines
/* -*-C-*-
$Id: bitstr.c,v 9.63 2000/12/05 21:23:43 cph Exp $
Copyright (c) 1987-2000 Massachusetts Institute of Technology
This program 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 2 of the License, or (at
your option) any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/* Bit string primitives.
Conversions between nonnegative integers and bit strings are
implemented here; they use the standard binary encoding, in which
each index selects the bit corresponding to that power of 2. Thus
bit 0 is the LSB. */
#include "scheme.h"
#include "prims.h"
#include "bitstr.h"
static void EXFUN
(copy_bits, (SCHEME_OBJECT *, long, SCHEME_OBJECT *, long, long));
static SCHEME_OBJECT
DEFUN (allocate_bit_string, (length), long length)
{
long total_pointers;
SCHEME_OBJECT result;
total_pointers = (1 + (BIT_STRING_LENGTH_TO_GC_LENGTH (length)));
result = (allocate_non_marked_vector (TC_BIT_STRING, total_pointers, true));
FAST_MEMORY_SET (result, BIT_STRING_LENGTH_OFFSET, length);
return (result);
}
/* (BIT-STRING-ALLOCATE length)
Returns an uninitialized bit string of the given length. */
DEFINE_PRIMITIVE ("BIT-STRING-ALLOCATE", Prim_bit_string_allocate, 1, 1, 0)
{
PRIMITIVE_HEADER (1);
PRIMITIVE_RETURN (allocate_bit_string (arg_nonnegative_integer (1)));
}
/* (BIT-STRING? object)
Returns #T iff object is a bit string. */
DEFINE_PRIMITIVE ("BIT-STRING?", Prim_bit_string_p, 1, 1, 0)
{
fast SCHEME_OBJECT object;
PRIMITIVE_HEADER (1);
TOUCH_IN_PRIMITIVE ((ARG_REF (1)), object);
PRIMITIVE_RETURN (BOOLEAN_TO_OBJECT (BIT_STRING_P (object)));
}
void
DEFUN (fill_bit_string, (bit_string, sense),
SCHEME_OBJECT bit_string AND
int sense)
{
SCHEME_OBJECT *scanner;
SCHEME_OBJECT filler;
long i;
filler = ((SCHEME_OBJECT) (sense ? (~ 0) : 0));
scanner = BIT_STRING_HIGH_PTR (bit_string);
for (i = BIT_STRING_LENGTH_TO_GC_LENGTH (BIT_STRING_LENGTH (bit_string));
(i > 0); i -= 1)
(* (DEC_BIT_STRING_PTR (scanner))) = filler;
}
extern void EXFUN (clear_bit_string, (SCHEME_OBJECT));
void
DEFUN (clear_bit_string, (bit_string), SCHEME_OBJECT bit_string)
{
SCHEME_OBJECT *scanner;
long i;
scanner = BIT_STRING_HIGH_PTR (bit_string);
for (i = BIT_STRING_LENGTH_TO_GC_LENGTH (BIT_STRING_LENGTH (bit_string));
(i > 0); i -= 1)
(* (DEC_BIT_STRING_PTR (scanner))) = 0;
}
DEFINE_PRIMITIVE ("MAKE-BIT-STRING", Prim_make_bit_string, 2, 2,
"(SIZE INITIALIZATION)\n\
Returns a bit string of the specified size with all the bits\n\
set to zero if the initialization is false, one otherwise.")
{
SCHEME_OBJECT result;
PRIMITIVE_HEADER (2);
result = allocate_bit_string (arg_nonnegative_integer (1));
fill_bit_string (result, (OBJECT_TO_BOOLEAN (ARG_REF (2))));
PRIMITIVE_RETURN (result);
}
DEFINE_PRIMITIVE ("BIT-STRING-FILL!", Prim_bit_string_fill_x, 2, 2,
"(BIT-STRING INITIALIZATION)\n\
Fills the bit string with zeros if the initialization is false, \
otherwise fills it with ones.")
{
PRIMITIVE_HEADER (2);
CHECK_ARG (1, BIT_STRING_P);
fill_bit_string ((ARG_REF (1)), (OBJECT_TO_BOOLEAN (ARG_REF (2))));
PRIMITIVE_RETURN (UNSPECIFIC);
}
/* */
DEFINE_PRIMITIVE ("BIT-STRING-LENGTH", Prim_bit_string_length, 1, 1,
"(BIT-STRING)\n\
Returns the number of bits in BIT-STRING.")
{
PRIMITIVE_HEADER (1);
CHECK_ARG (1, BIT_STRING_P);
PRIMITIVE_RETURN (LONG_TO_UNSIGNED_FIXNUM (BIT_STRING_LENGTH (ARG_REF (1))));
}
#define REF_INITIALIZATION() \
fast SCHEME_OBJECT bit_string; \
fast long index; \
fast SCHEME_OBJECT *ptr; \
fast long mask; \
PRIMITIVE_HEADER (2); \
\
CHECK_ARG (1, BIT_STRING_P); \
bit_string = (ARG_REF (1)); \
index = (arg_nonnegative_integer (2)); \
if (index >= (BIT_STRING_LENGTH (bit_string))) \
error_bad_range_arg (2); \
\
ptr = \
(MEMORY_LOC \
(bit_string, (BIT_STRING_INDEX_TO_WORD (bit_string, index)))); \
mask = (1L << (index % OBJECT_LENGTH))
DEFINE_PRIMITIVE ("BIT-STRING-REF", Prim_bit_string_ref, 2, 2,
"(BIT-STRING INDEX)\n\
Returns the boolean value of the indexed bit.")
{
REF_INITIALIZATION ();
PRIMITIVE_RETURN (BOOLEAN_TO_OBJECT (((BIT_STRING_WORD (ptr)) & mask) != 0));
}
/* */
DEFINE_PRIMITIVE ("BIT-STRING-CLEAR!", Prim_bit_string_clear_x, 2, 2,
"(BIT-STRING INDEX)\n\
Sets the indexed bit to zero, returning its previous value as a boolean.")
{
REF_INITIALIZATION ();
if (((BIT_STRING_WORD (ptr)) & mask) == 0)
PRIMITIVE_RETURN (SHARP_F);
(BIT_STRING_WORD (ptr)) &= ~mask;
PRIMITIVE_RETURN (SHARP_T);
}
DEFINE_PRIMITIVE ("BIT-STRING-SET!", Prim_bit_string_set_x, 2, 2,
"(BIT-STRING INDEX)\n\
Sets the indexed bit to one, returning its previous value as a boolean.")
{
REF_INITIALIZATION ();
if (((BIT_STRING_WORD (ptr)) & mask) != 0)
PRIMITIVE_RETURN (SHARP_T);
(BIT_STRING_WORD (ptr)) |= mask;
PRIMITIVE_RETURN (SHARP_F);
}
#define ZERO_SECTION_P() \
{ \
for (i = (length / OBJECT_LENGTH); (i > 0); i -= 1) \
if ((* (DEC_BIT_STRING_PTR (scan))) != 0) \
PRIMITIVE_RETURN (SHARP_F); \
PRIMITIVE_RETURN (SHARP_T); \
}
DEFINE_PRIMITIVE ("BIT-STRING-ZERO?", Prim_bit_string_zero_p, 1, 1,
"(BIT-STRING)\n\
Returns true the argument has no \"set\" bits.")
{
fast SCHEME_OBJECT bit_string;
fast SCHEME_OBJECT *scan;
fast long i;
long length, odd_bits;
PRIMITIVE_HEADER (1);
CHECK_ARG (1, BIT_STRING_P);
bit_string = (ARG_REF (1));
length = (BIT_STRING_LENGTH (bit_string));
odd_bits = (length % OBJECT_LENGTH);
scan = (BIT_STRING_HIGH_PTR (bit_string));
if (odd_bits == 0)
{
ZERO_SECTION_P ();
}
else if (((BIT_STRING_WORD (scan)) & (LOW_MASK (odd_bits))) != 0)
PRIMITIVE_RETURN (SHARP_F);
else
{
DEC_BIT_STRING_PTR (scan);
ZERO_SECTION_P ();
}
}
#define EQUAL_SECTIONS_P() \
{ \
for (i = (length / OBJECT_LENGTH); (i > 0); i -= 1) \
if ((* (DEC_BIT_STRING_PTR (scan1))) != \
(* (DEC_BIT_STRING_PTR (scan2)))) \
PRIMITIVE_RETURN (SHARP_F); \
PRIMITIVE_RETURN (SHARP_T); \
}
DEFINE_PRIMITIVE ("BIT-STRING=?", Prim_bit_string_equal_p, 2, 2,
"(BIT-STRING-1 BIT-STRING-2)\n\
Returns true iff the two bit strings contain the same bits.")
{
SCHEME_OBJECT bit_string_1, bit_string_2;
long length;
fast SCHEME_OBJECT *scan1, *scan2;
fast long i;
long odd_bits;
PRIMITIVE_HEADER (2);
CHECK_ARG (1, BIT_STRING_P);
CHECK_ARG (2, BIT_STRING_P);
bit_string_1 = (ARG_REF (1));
bit_string_2 = (ARG_REF (2));
length = BIT_STRING_LENGTH (bit_string_1);
if (length != BIT_STRING_LENGTH (bit_string_2))
PRIMITIVE_RETURN (SHARP_F);
scan1 = (BIT_STRING_HIGH_PTR (bit_string_1));
scan2 = (BIT_STRING_HIGH_PTR (bit_string_2));
odd_bits = (length % OBJECT_LENGTH);
if (odd_bits == 0)
{
EQUAL_SECTIONS_P ();
}
else
{
long mask;
mask = (LOW_MASK (odd_bits));
if (((BIT_STRING_MSW (bit_string_1)) & mask) !=
((BIT_STRING_MSW (bit_string_2)) & mask))
PRIMITIVE_RETURN (SHARP_F);
else
{
DEC_BIT_STRING_PTR (scan1);
DEC_BIT_STRING_PTR (scan2);
EQUAL_SECTIONS_P ();
}
}
}
/* (BIT-STRING-OPERATION! destination source)
Modifies destination to be the result of using OPERATION bitwise on
destination and source. */
#define BITWISE_OP(action) \
{ \
SCHEME_OBJECT bit_string_1, bit_string_2; \
fast long i; \
fast SCHEME_OBJECT *scan1, *scan2; \
PRIMITIVE_HEADER (2); \
bit_string_1 = (ARG_REF (1)); \
bit_string_2 = (ARG_REF (2)); \
if ((BIT_STRING_LENGTH (bit_string_1)) != \
(BIT_STRING_LENGTH (bit_string_2))) \
error_bad_range_arg (1); \
scan1 = (BIT_STRING_HIGH_PTR (bit_string_1)); \
scan2 = (BIT_STRING_HIGH_PTR (bit_string_2)); \
for (i = ((VECTOR_LENGTH (bit_string_1)) - 1); (i > 0); i -= 1) \
(* (DEC_BIT_STRING_PTR (scan1))) action \
(* (DEC_BIT_STRING_PTR (scan2))); \
PRIMITIVE_RETURN (UNSPECIFIC); \
}
DEFINE_PRIMITIVE ("BIT-STRING-MOVE!", Prim_bit_string_move_x, 2, 2, 0)
BITWISE_OP (=)
DEFINE_PRIMITIVE ("BIT-STRING-MOVEC!", Prim_bit_string_movec_x, 2, 2, 0)
BITWISE_OP (=~)
DEFINE_PRIMITIVE ("BIT-STRING-OR!", Prim_bit_string_or_x, 2, 2, 0)
BITWISE_OP (|=)
DEFINE_PRIMITIVE ("BIT-STRING-AND!", Prim_bit_string_and_x, 2, 2, 0)
BITWISE_OP (&=)
DEFINE_PRIMITIVE ("BIT-STRING-ANDC!", Prim_bit_string_andc_x, 2, 2, 0)
BITWISE_OP (&=~)
DEFINE_PRIMITIVE ("BIT-STRING-XOR!", Prim_bit_string_xor_x, 2, 2, 0)
BITWISE_OP (^=)
DEFINE_PRIMITIVE ("BIT-SUBSTRING-MOVE-RIGHT!", Prim_bit_substring_move_right_x, 5, 5,
"(SOURCE START1 END1 DESTINATION START2)\n\
Destructively copies the substring of SOURCE between START1 and \
END1 into DESTINATION at START2. The copying is done from the \
MSB to the LSB (which only matters when SOURCE and DESTINATION \
are the same).")
{
fast SCHEME_OBJECT bit_string_1, bit_string_2;
long start1, end1, start2, end2, nbits;
long end1_mod, end2_mod;
PRIMITIVE_HEADER (5);
CHECK_ARG (1, BIT_STRING_P);
bit_string_1 = (ARG_REF (1));
start1 = (arg_nonnegative_integer (2));
end1 = (arg_nonnegative_integer (3));
CHECK_ARG (4, BIT_STRING_P);
bit_string_2 = (ARG_REF (4));
start2 = (arg_nonnegative_integer (5));
nbits = (end1 - start1);
end2 = (start2 + nbits);
if ((start1 < 0) || (start1 > end1))
error_bad_range_arg (2);
if (end1 > (BIT_STRING_LENGTH (bit_string_1)))
error_bad_range_arg (3);
if ((start2 < 0) || (end2 > (BIT_STRING_LENGTH (bit_string_2))))
error_bad_range_arg (5);
end1_mod = (end1 % OBJECT_LENGTH);
end2_mod = (end2 % OBJECT_LENGTH);
/* Using `BIT_STRING_INDEX_TO_WORD' here with -1 offset will work in every
case except when the `end' is 0. In this case the result of
the expression `(-1 / OBJECT_LENGTH)' is either 0 or -1, at
the discretion of the C compiler being used. This doesn't
matter because if `end' is zero, then no bits will be moved. */
copy_bits ((MEMORY_LOC
(bit_string_1,
(BIT_STRING_INDEX_TO_WORD (bit_string_1, (end1 - 1))))),
((end1_mod == 0) ? 0 : (OBJECT_LENGTH - end1_mod)),
(MEMORY_LOC
(bit_string_2,
(BIT_STRING_INDEX_TO_WORD (bit_string_2, (end2 - 1))))),
((end2_mod == 0) ? 0 : (OBJECT_LENGTH - end2_mod)),
nbits);
PRIMITIVE_RETURN (UNSPECIFIC);
}
#define MASKED_TRANSFER(source, destination, nbits, offset) do \
{ \
long mask = (ANY_MASK (nbits, offset)); \
(BIT_STRING_WORD (destination)) \
= (((BIT_STRING_WORD (source)) & mask) \
| ((BIT_STRING_WORD (destination)) &~ mask)); \
} while (0)
/* This procedure copies bits from one place to another.
The offsets are measured from the MSB of the first SCHEME_OBJECT of
each of the arguments SOURCE and DESTINATION. It copies the bits
starting with the MSB of a bit string and moving down. */
static void
DEFUN (copy_bits,
(source, source_offset, destination, destination_offset, nbits),
SCHEME_OBJECT * source AND
long source_offset AND
SCHEME_OBJECT * destination AND
long destination_offset AND
long nbits)
{
if (nbits == 0)
return;
/* This common case can be done very quickly, by splitting the
bit string into three parts. Since the source and destination are
aligned relative to one another, the main body of bits can be
transferred as SCHEME_OBJECTs, and only the `head' and `tail' need be
treated specially. */
if (source_offset == destination_offset)
{
if (source_offset != 0)
{
long head = (OBJECT_LENGTH - source_offset);
if (nbits <= head)
{
MASKED_TRANSFER (source, destination, nbits, (head - nbits));
nbits = 0;
}
else
{
long mask = (LOW_MASK (head));
SCHEME_OBJECT temp = (BIT_STRING_WORD (destination));
(* (DEC_BIT_STRING_PTR (destination)))
= (((* (DEC_BIT_STRING_PTR (source))) & mask)
| (temp &~ mask));
nbits -= head;
}
}
while (nbits >= OBJECT_LENGTH)
{
(* (DEC_BIT_STRING_PTR (destination)))
= (* (DEC_BIT_STRING_PTR (source)));
nbits -= OBJECT_LENGTH;
}
if (nbits > 0)
MASKED_TRANSFER (source, destination, nbits, (OBJECT_LENGTH - nbits));
}
else if (source_offset < destination_offset)
{
long offset1 = (destination_offset - source_offset);
long offset2 = (OBJECT_LENGTH - offset1);
long head = (OBJECT_LENGTH - destination_offset);
if (nbits <= head)
{
long mask = (ANY_MASK (nbits, (head - nbits)));
(BIT_STRING_WORD (destination))
= ((((BIT_STRING_WORD (source)) >> offset1) & mask)
| ((BIT_STRING_WORD (destination)) &~ mask));
}
else
{
long mask1 = (LOW_MASK (offset1));
long mask2 = (LOW_MASK (offset2));
{
long mask = (LOW_MASK (head));
SCHEME_OBJECT temp = (BIT_STRING_WORD (destination));
(* (DEC_BIT_STRING_PTR (destination)))
= ((((BIT_STRING_WORD (source)) >> offset1) & mask)
| (temp &~ mask));
}
nbits -= head;
while (nbits >= OBJECT_LENGTH)
{
long i
= (((* (DEC_BIT_STRING_PTR (source))) & mask1) << offset2);
(* (DEC_BIT_STRING_PTR (destination)))
= ((((BIT_STRING_WORD (source)) >> offset1) & mask2) | i);
nbits -= OBJECT_LENGTH;
}
if (nbits > 0)
{
long dest_tail
= ((BIT_STRING_WORD (destination))
& (LOW_MASK (OBJECT_LENGTH - nbits)));
if (nbits <= offset1)
(BIT_STRING_WORD (destination))
= ((((BIT_STRING_WORD (source))
& (ANY_MASK (nbits, (offset1 - nbits))))
<< offset2)
| dest_tail);
else
{
long i
= (((* (DEC_BIT_STRING_PTR (source))) & mask1) << offset2);
long j = (nbits - offset1);
(BIT_STRING_WORD (destination))
= ((((BIT_STRING_WORD (source))
&
(ANY_MASK (j, (OBJECT_LENGTH - j))))
>> offset1)
| i
| dest_tail);
}
}
}
}
else /* if (source_offset > destination_offset) */
{
long offset1 = (source_offset - destination_offset);
long offset2 = (OBJECT_LENGTH - offset1);
long head = (OBJECT_LENGTH - source_offset);
if (nbits <= head)
{
long mask = (ANY_MASK (nbits, (offset1 + (head - nbits))));
(BIT_STRING_WORD (destination))
= ((((BIT_STRING_WORD (source)) << offset1) & mask)
| ((BIT_STRING_WORD (destination)) &~ mask));
}
else
{
long mask1 = (LOW_MASK (offset1));
long dest_buffer
= (((head + offset1) < OBJECT_LENGTH)
? ((BIT_STRING_WORD (destination))
&~ (LOW_MASK (head + offset1)))
: 0);
dest_buffer
|= (((* (DEC_BIT_STRING_PTR (source))) & (LOW_MASK (head)))
<< offset1);
nbits -= head;
while (nbits >= OBJECT_LENGTH)
{
(* (DEC_BIT_STRING_PTR (destination)))
= (dest_buffer
| (((BIT_STRING_WORD (source)) >> offset2) & mask1));
dest_buffer = ((* (DEC_BIT_STRING_PTR (source))) << offset1);
nbits -= OBJECT_LENGTH;
}
if (nbits <= offset1)
(BIT_STRING_WORD (destination))
= (dest_buffer
| ((BIT_STRING_WORD (destination))
& (LOW_MASK (offset1 - nbits)))
| (((BIT_STRING_WORD (source)) >> offset2)
& (ANY_MASK (nbits, (offset1 - nbits)))));
else
{
(* (DEC_BIT_STRING_PTR (destination)))
= (dest_buffer
| (((BIT_STRING_WORD (source)) >> offset2) & mask1));
nbits -= offset1;
{
long mask = (LOW_MASK (OBJECT_LENGTH - nbits));
(BIT_STRING_WORD (destination))
= (((BIT_STRING_WORD (destination)) & mask)
| (((BIT_STRING_WORD (source)) << offset1) &~ mask));
}
}
}
}
}
/* Integer <-> Bit-string Conversions */
long
DEFUN (count_significant_bits, (number, start), long number AND long start)
{
long significant_bits, i;
significant_bits = start;
for (i = (1L << (start - 1)); (i >= 0); i >>= 1)
{
if (number >= i)
break;
significant_bits -= 1;
}
return (significant_bits);
}
long
DEFUN (long_significant_bits, (number), long number)
{
return
((number < 0)
? ((sizeof (long)) * CHAR_BIT)
: (count_significant_bits (number, (((sizeof (long)) * CHAR_BIT) - 1))));
}
SCHEME_OBJECT
DEFUN (zero_to_bit_string, (length), long length)
{
SCHEME_OBJECT result;
result = (allocate_bit_string (length));
clear_bit_string (result);
return (result);
}
SCHEME_OBJECT
DEFUN (long_to_bit_string, (length, number), long length AND long number)
{
if (number < 0)
error_bad_range_arg (2);
if (number == 0)
{
return (zero_to_bit_string (length));
}
else
{
SCHEME_OBJECT result;
if (length < (long_significant_bits (number)))
error_bad_range_arg (2);
result = (zero_to_bit_string (length));
(BIT_STRING_LSW (result)) = number;
return (result);
}
}
static void
DEFUN (btbs_consumer, (result_ptr, digit),
PTR result_ptr
AND long digit)
{
(* (INC_BIT_STRING_PTR (* ((unsigned char **) result_ptr))))
= ((unsigned char) digit);
}
SCHEME_OBJECT
DEFUN (bignum_to_bit_string, (length, bignum),
long length AND SCHEME_OBJECT bignum)
{
switch (bignum_test (bignum))
{
case bignum_comparison_equal:
return (zero_to_bit_string (length));
case bignum_comparison_less:
error_bad_range_arg (2);
case bignum_comparison_greater:
if (! (bignum_fits_in_word_p (bignum, length, 0)))
error_bad_range_arg (2);
{
SCHEME_OBJECT result = (zero_to_bit_string (length));
unsigned char * result_ptr =
((unsigned char *) (BIT_STRING_LOW_PTR (result)));
bignum_to_digit_stream
(bignum, (1L << CHAR_BIT), btbs_consumer, (&result_ptr));
return (result);
}
default:
/*NOTREACHED*/
return (0);
}
}
struct bitstr_to_bignm_context
{
unsigned char *source_ptr;
unsigned int mask;
};
static unsigned int
DEFUN (bstb_producer, (context), PTR context)
{
struct bitstr_to_bignm_context * c = context;
unsigned int result = (c->mask & (BIT_STRING_WORD (c->source_ptr)));
c->mask = (LOW_MASK (CHAR_BIT));
DEC_BIT_STRING_PTR (c->source_ptr);
return (result);
}
SCHEME_OBJECT
DEFUN (bit_string_to_bignum, (nbits, bitstr),
long nbits AND SCHEME_OBJECT bitstr)
{
struct bitstr_to_bignm_context context;
int ndigits, skip;
ndigits = ((nbits + (CHAR_BIT - 1)) / CHAR_BIT);
context.mask = (LOW_MASK (((nbits - 1) % (CHAR_BIT)) + 1));
context.source_ptr =
((unsigned char *)
(MEMORY_LOC (bitstr, (BIT_STRING_INDEX_TO_WORD (bitstr, (nbits - 1))))));
if (ndigits != 0)
{
skip = ((sizeof (SCHEME_OBJECT)) -
(((ndigits - 1) % (sizeof (SCHEME_OBJECT))) + 1));
while ((--skip) >= 0)
{
DEC_BIT_STRING_PTR (context.source_ptr);
}
}
return
(digit_stream_to_bignum (ndigits, bstb_producer,
(&context), (1L << CHAR_BIT),
0));
}
DEFINE_PRIMITIVE ("UNSIGNED-INTEGER->BIT-STRING", Prim_unsigned_to_bit_string, 2, 2,
"(LENGTH INTEGER)\n\
INTEGER, which must be a non-negative integer, is converted to \
a bit-string of length LENGTH. If INTEGER is too large, an \
error is signalled.")
{
fast long length;
fast SCHEME_OBJECT object;
PRIMITIVE_HEADER (2);
length = (arg_nonnegative_integer (1));
object = (ARG_REF (2));
if (FIXNUM_P (object))
{
if (FIXNUM_NEGATIVE_P (object))
error_bad_range_arg (2);
PRIMITIVE_RETURN
(long_to_bit_string
(length, (UNSIGNED_FIXNUM_TO_LONG (object))));
}
if (BIGNUM_P (object))
PRIMITIVE_RETURN (bignum_to_bit_string (length, object));
error_wrong_type_arg (2);
/*NOTREACHED*/
return (0);
}
/* */
DEFINE_PRIMITIVE ("BIT-STRING->UNSIGNED-INTEGER", Prim_bit_string_to_unsigned, 1, 1,
"(BIT-STRING)\n\
BIT-STRING is converted to the appropriate non-negative integer. \
This operation is the inverse of `unsigned-integer->bit-string'.")
{
fast SCHEME_OBJECT bit_string, *scan;
long nwords, nbits, word;
PRIMITIVE_HEADER (1);
CHECK_ARG (1, BIT_STRING_P);
bit_string = (ARG_REF (1));
/* Count the number of significant bits.*/
scan = (BIT_STRING_HIGH_PTR (bit_string));
nbits = ((BIT_STRING_LENGTH (bit_string)) % OBJECT_LENGTH);
word =
((nbits > 0)
? ((* (DEC_BIT_STRING_PTR (scan))) & (LOW_MASK (nbits)))
: (* (DEC_BIT_STRING_PTR (scan))));
for (nwords = ((VECTOR_LENGTH (bit_string)) - 1); (nwords > 0); nwords -= 1)
{
if (word != 0)
break;
word = (* (DEC_BIT_STRING_PTR (scan)));
}
if (nwords == 0)
PRIMITIVE_RETURN (LONG_TO_UNSIGNED_FIXNUM (0));
nbits = (((nwords - 1) * OBJECT_LENGTH) + (long_significant_bits (word)));
PRIMITIVE_RETURN
((nbits <= FIXNUM_LENGTH)
? (LONG_TO_UNSIGNED_FIXNUM (word))
: (bit_string_to_bignum (nbits, bit_string)));
}
#define READ_BITS_INITIALIZE() \
SCHEME_OBJECT bit_string; \
long end, end_mod, offset; \
SCHEME_OBJECT *start; \
PRIMITIVE_HEADER (3); \
CHECK_ARG (3, BIT_STRING_P); \
bit_string = (ARG_REF (3)); \
end = (BIT_STRING_LENGTH (bit_string)); \
end_mod = (end % OBJECT_LENGTH); \
offset = (arg_nonnegative_integer (2)); \
start = (READ_BITS_PTR ((ARG_REF (1)), offset, end)); \
COMPUTE_READ_BITS_OFFSET (offset, end)
DEFINE_PRIMITIVE ("READ-BITS!", Prim_read_bits_x, 3, 3,
"(POINTER OFFSET BIT-STRING)\n\
Read the contents of memory at the address (POINTER,OFFSET) into BIT-STRING.")
{
READ_BITS_INITIALIZE ();
copy_bits (start,
offset,
(MEMORY_LOC
(bit_string,
(BIT_STRING_INDEX_TO_WORD (bit_string, (end - 1))))),
((end_mod == 0) ? 0 : (OBJECT_LENGTH - end_mod)),
end);
PRIMITIVE_RETURN (UNSPECIFIC);
}
DEFINE_PRIMITIVE ("WRITE-BITS!", Prim_write_bits_x, 3, 3,
"(POINTER OFFSET BIT-STRING)\n\
Write the contents of BIT-STRING in memory at the address (POINTER,OFFSET).")
{
READ_BITS_INITIALIZE ();
copy_bits ((MEMORY_LOC
(bit_string,
(BIT_STRING_INDEX_TO_WORD (bit_string, (end - 1))))),
((end_mod == 0) ? 0 : (OBJECT_LENGTH - end_mod)),
start,
offset,
end);
PRIMITIVE_RETURN (UNSPECIFIC);
}
/* Search Primitives */
#define SUBSTRING_FIND_INITIALIZE() \
SCHEME_OBJECT bit_string; \
long start, end; \
long word, bit, end_word, end_bit, mask; \
SCHEME_OBJECT *scan; \
PRIMITIVE_HEADER (3); \
CHECK_ARG (1, BIT_STRING_P); \
bit_string = (ARG_REF (1)); \
start = (arg_nonnegative_integer (2)); \
end = (arg_nonnegative_integer (3)); \
if (end > (BIT_STRING_LENGTH (bit_string))) \
error_bad_range_arg (3); \
if (start > end) \
error_bad_range_arg (2); \
if (start == end) \
PRIMITIVE_RETURN (SHARP_F)
#define SUBSTRING_FIND_NEXT_INITIALIZE() \
SUBSTRING_FIND_INITIALIZE (); \
word = (BIT_STRING_INDEX_TO_WORD (bit_string, start)); \
bit = (start % OBJECT_LENGTH); \
end_word = (BIT_STRING_INDEX_TO_WORD (bit_string, (end - 1))); \
end_bit = (((end - 1) % OBJECT_LENGTH) + 1); \
scan = (MEMORY_LOC (bit_string, word))
#define FIND_NEXT_SET_LOOP(init_bit) \
{ \
bit = (init_bit); \
mask = (1L << (init_bit)); \
while (true) \
{ \
if (((BIT_STRING_WORD (scan)) & mask) != 0) \
goto win; \
bit += 1; \
mask <<= 1; \
} \
}
DEFINE_PRIMITIVE ("BIT-SUBSTRING-FIND-NEXT-SET-BIT", Prim_bitstr_find_next_set_bit, 3, 3,
"(BIT-STRING START END)")
{
SUBSTRING_FIND_NEXT_INITIALIZE ();
if (word == end_word)
{
if ((((end_bit - bit) == OBJECT_LENGTH) &&
((BIT_STRING_WORD (scan)) != 0)) ||
(((BIT_STRING_WORD (scan)) & (ANY_MASK ((end_bit - bit), bit)))
!= 0))
{
FIND_NEXT_SET_LOOP (bit);
}
PRIMITIVE_RETURN (SHARP_F);
}
else if (((BIT_STRING_WORD (scan)) &
((bit == 0) ? (~ 0) : (ANY_MASK ((OBJECT_LENGTH - bit), bit))))
!= 0)
{
FIND_NEXT_SET_LOOP (bit);
}
INC_BIT_STRING_PTR (word);
while (word != end_word)
{
if ((* (INC_BIT_STRING_PTR (scan))) != 0)
{
FIND_NEXT_SET_LOOP (0);
}
INC_BIT_STRING_PTR (word);
}
if (((* (INC_BIT_STRING_PTR (scan))) &
((end_bit == OBJECT_LENGTH) ? (~ 0) : (LOW_MASK (end_bit))))
!= 0)
{
FIND_NEXT_SET_LOOP (0);
}
PRIMITIVE_RETURN (SHARP_F);
win:
PRIMITIVE_RETURN
(LONG_TO_UNSIGNED_FIXNUM
(BIT_STRING_INDEX_PAIR_TO_INDEX (bit_string, word, bit)));
}
extern void EXFUN (bit_string_set, (SCHEME_OBJECT, long, int));
void
DEFUN (bit_string_set, (bitstr, index, value),
SCHEME_OBJECT bitstr AND long index AND int value)
{
unsigned long mask;
SCHEME_OBJECT * ptr;
ptr = (MEMORY_LOC (bitstr, (BIT_STRING_INDEX_TO_WORD (bitstr, index))));
mask = (1L << (index % OBJECT_LENGTH));
if (value == 0)
(BIT_STRING_WORD (ptr)) &= (~mask);
else
(BIT_STRING_WORD (ptr)) |= mask;
return;
}