1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149
|
/* @(#)findbytes.c 1.3 06/09/13 Copyright 2000-2003 J. Schilling */
/*
* Find a byte with specific value in memory.
*
* Copyright (c) 2000-2003 J. Schilling
*
* Based on a strlen() idea from Torbjorn Granlund (tege@sics.se) and
* Dan Sahlin (dan@sics.se) and the memchr() suggestion
* from Dick Karpinski (dick@cca.ucsf.edu).
*/
/*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License, Version 1.0 only
* (the "License"). You may not use this file except in compliance
* with the License.
*
* See the file CDDL.Schily.txt in this distribution for details.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file CDDL.Schily.txt from this distribution.
*/
#include <schily/mconfig.h>
#include <schily/stdlib.h>
#include <schily/utypes.h>
#include <align.h>
#include <schily/standard.h>
#include <schily/string.h>
#include <schily/schily.h>
#ifdef PROTOTYPES
EXPORT char *
findbytes(const void *vp, int cnt, char val)
#else
EXPORT char *
findbytes(vp, cnt, val)
const void *vp;
register int cnt;
char val;
#endif
{
register Uchar uval = (Uchar)val;
register const Uchar *cp = (Uchar *)vp;
register const Ulong *lp;
register Ulong lval;
register Ulong lmask;
register Ulong magic_mask;
/*
* Test byte-wise until cp is properly aligned for a long pointer.
*/
while (--cnt >= 0 && !laligned(cp)) {
if (*cp++ == uval)
return ((char *)--cp);
}
cnt++;
/*
* The magic mask is a long word where all carry bits a clear.
* This are bits 8, 16, 24 ...
* In addition, the top bit is not set (e.g bit 31 or 63). The magic
* mask will look this way:
*
* bits: 01111110 11111110 ... 11111110 11111111
* bytes: AAAAAAAA BBBBBBBB ... CCCCCCCC DDDDDDDD
*
* If we add anything to this magic number, no carry bit will change if
* it is the first carry bit left to a 0 byte. Adding anything != 0
* to the magic number will just turn the carry bit left to the byte
* but does not propagate any further.
*/
#if SIZE_LONG == 4
magic_mask = 0x7EFEFEFFL;
#else
#if SIZE_LONG == 8
magic_mask = 0x7EFEFEFEFEFEFEFFL;
#else
/*
* #error will not work for all compilers (e.g. sunos4)
* The following line will abort compilation on all compilers
* if none of the above is defines. And that's what we want.
*/
error SIZE_LONG has unknown value
#endif
#endif
lmask = val & 0xFF;
lmask |= lmask << 8;
lmask |= lmask << 16;
#if SIZE_LONG > 4
lmask |= lmask << 32;
#endif
#if SIZE_LONG > 8
error SIZE_LONG has unknown value
#endif
for (lp = (const Ulong *)cp; cnt >= sizeof (long); cnt -= sizeof (long)) {
/*
* We are not looking for 0 bytes so we need to xor with the
* long mask of repeated bytes. If any of the bytes matches our
* wanted char, we will create a 0 byte in the current long.
* But how will we find if at least one byte in a long is zero?
*
* If we add 'magic_mask' and any of the holes in the magic
* mask do not change, we most likely found a 0 byte in the
* long word. It is only a most likely match because if bits
* 24..30 (ot bits 56..62) are 0 but bit 31 (or bit 63) is set
* we will believe that we found a match but there is none.
* This will happen if there is 0x80nnnnnn / 0x80nnnnnnnnnnnnnn
*/
lval = (*lp++ ^ lmask); /* create 0 byte on match */
lval = (lval + magic_mask) ^ ~lval; /* set bits unchanged by +*/
if ((lval & ~magic_mask) != 0) { /* a magic hole was set */
/*
* If any of the hole bits did not change by addition,
* we most likely had a match.
* If this was a correct match, find the matching byte.
*/
cp = (const Uchar *)(lp - 1);
if (cp[0] == uval)
return ((char *)cp);
if (cp[1] == uval)
return ((char *)&cp[1]);
if (cp[2] == uval)
return ((char *)&cp[2]);
if (cp[3] == uval)
return ((char *)&cp[3]);
#if SIZE_LONG > 4
if (cp[4] == uval)
return ((char *)&cp[4]);
if (cp[5] == uval)
return ((char *)&cp[5]);
if (cp[6] == uval)
return ((char *)&cp[6]);
if (cp[7] == uval)
return ((char *)&cp[7]);
#endif
#if SIZE_LONG > 8
error SIZE_LONG has unknown value
#endif
}
}
for (cp = (const Uchar *)lp; --cnt >= 0; ) {
if (*cp++ == uval)
return ((char *)--cp);
}
return ((char *)NULL);
}
|