summaryrefslogtreecommitdiff
blob: cc6c6ff1b956487446e7af4332ea43f7cd99090b (plain)
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
/* Copyright (C) 2001-2019 Artifex Software, Inc.
   All Rights Reserved.

   This software is provided AS-IS with no warranty, either express or
   implied.

   This software is distributed under license and may not be copied,
   modified or distributed except as expressly authorized under the terms
   of the license contained in the file LICENSE in this distribution.

   Refer to licensing information at http://www.artifex.com or contact
   Artifex Software, Inc.,  1305 Grant Avenue - Suite 200, Novato,
   CA 94945, U.S.A., +1(415)492-9861, for further information.
*/


/* A template for packed dictionary search method */

#if defined(found) && defined(deleted) && defined(missing)

/*
 * Define template for searching a packed dictionary.
 *
 * Free variables:
 *      ref_packed kpack - holds the packed key.
 *      uint hash - holds the hash of the name.
 *      dict *pdict - points to the dictionary.
 *      uint size - holds npairs(pdict).
 *
 * Template parameters are :
 *	found   - the found key action.
 *	deleted - the deleted key action.
 *	missing - the missed key action.
 *
 * Note that the template is *not* enclosed in {}, so that we can access
 * the values of kbot and kp after leaving the template.
 */

    const ref_packed *kbot = pdict->keys.value.packed;
    const int start = dict_hash_mod(hash, size) + 1;
    register const ref_packed *kp = kbot + start;
    int wrap = 0;

    again:
    for (; ; kp-- ) {
        if_debug2('D', "[D]probe 0x%lx: 0x%x\n", (ulong)kp, *kp);
        if ( *kp == kpack ) {
            found;
        } else if ( !r_packed_is_name(kp) ) {
            /* Empty, deleted, or wraparound. Figure out which. */
            if ( *kp == packed_key_empty )
                missing;
            if ( kp == kbot ) {
                if (wrap)
                    break;
                else {
                    wrap++;
                    kp += size; /* wrap */
                    goto again; /* skip "kp--". */
                }
            } else {
                deleted;
            }
        }
   }
#else
int dummy;
#endif