root/usr/src/lib/libnsl/yp/dbm.c
/*
 * CDDL HEADER START
 *
 * 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.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */

/*
 * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

/*      Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */
/*        All Rights Reserved   */

/*
 * Portions of this source code were derived from Berkeley
 * under license from the Regents of the University of
 * California.
 */

#include "mt.h"
#include <rpcsvc/dbm.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <stdio.h>
#include <errno.h>

void dbm_access(long);
void delitem(char *, int);
void chkblk(char *);
int  additem(char *, datum);
int  getbit(void);
int  setbit(void);
int  cmpdatum(datum, datum);

int
dbminit(char *file)
{
        struct stat statb;

        dbrdonly = 0;
        if (strlcpy(pagbuf, file, sizeof (pagbuf)) >= sizeof (pagbuf) ||
            strlcat(pagbuf, ".pag", sizeof (pagbuf)) >= sizeof (pagbuf)) {
                /*
                 * file.pag does not fit into pagbuf.
                 * fails with ENAMETOOLONG.
                 */
                errno = ENAMETOOLONG;
                return (-1);
        }
        pagf = open(pagbuf, 2);
        if (pagf < 0) {
                pagf = open(pagbuf, 0);
                dbrdonly = 1;
        }
        /*
         * We know this won't overflow so it is safe to ignore the
         * return value; we use strl* to prevent false hits in
         * code sweeps.
         */
        (void) strlcpy(pagbuf, file, sizeof (pagbuf));
        (void) strlcat(pagbuf, ".dir", sizeof (pagbuf));
        dirf = open(pagbuf, 2);
        if (dirf < 0) {
                dirf = open(pagbuf, 0);
                dbrdonly = 1;
        }
        if (pagf < 0 || dirf < 0)
                return (-1);
        (void) fstat(dirf, &statb);
        maxbno = statb.st_size*BYTESIZ-1;
        return (0);
}

static long oldb1 = -1;
static long oldb2 = -1;

/* Avoid using cached data for subsequent accesses. */
int
dbmflush(void)
{
        oldb1 = -1;
        oldb2 = -1;
        return (0);
}

/* Clean up after ourself. */
int
dbmclose(void)
{
        (void) close(pagf);
        (void) close(dirf);
        bitno = 0;
        maxbno = 0;
        blkno = 0;
        hmask = 0;
        oldb1 = -1;
        oldb2 = -1;
        return (0);
}

long
forder(datum key)
{
        long hash;

        hash = calchash(key);
        for (hmask = 0; ; hmask = (hmask<<1) + 1) {
                blkno = hash & hmask;
                bitno = blkno + hmask;
                if (getbit() == 0)
                        break;
        }
        return (blkno);
}

datum
fetch(datum key)
{
        int i;
        datum item;

        dbm_access(calchash(key));
        for (i = 0; ; i += 2) {
                item = makdatum(pagbuf, i);
                if (item.dptr == NULL) {
                        return (item);
                }
                if (cmpdatum(key, item) == 0) {
                        item = makdatum(pagbuf, i+1);
                        if (item.dptr == NULL)
                                (void) printf("items not in pairs\n");
                        return (item);
                }
        }
}

int
delete(datum key)
{
        int i;
        datum item;

        if (dbrdonly)
                return (-1);
        dbm_access(calchash(key));
        for (i = 0; ; i += 2) {
                item = makdatum(pagbuf, i);
                if (item.dptr == NULL)
                        return (-1);
                if (cmpdatum(key, item) == 0) {
                        delitem(pagbuf, i);
                        delitem(pagbuf, i);
                        break;
                }
        }
        (void) lseek(pagf, blkno*PBLKSIZ, 0);
        (void) write(pagf, pagbuf, PBLKSIZ);
        return (0);
}

int
store(datum key, datum dat)
{
        int i;
        datum item;
        char ovfbuf[PBLKSIZ];

        if (dbrdonly)
                return (-1);
loop:
        dbm_access(calchash(key));
        for (i = 0; ; i += 2) {
                item = makdatum(pagbuf, i);
                if (item.dptr == NULL)
                        break;
                if (cmpdatum(key, item) == 0) {
                        delitem(pagbuf, i);
                        delitem(pagbuf, i);
                        break;
                }
        }
        i = additem(pagbuf, key);
        if (i < 0)
                goto split;
        if (additem(pagbuf, dat) < 0) {
                delitem(pagbuf, i);
                goto split;
        }
        (void) lseek(pagf, blkno*PBLKSIZ, 0);
        (void) write(pagf, pagbuf, PBLKSIZ);
        return (0);

split:
        if (key.dsize + dat.dsize + 3 * sizeof (short) >= PBLKSIZ) {
                (void) printf("entry too big\n");
                return (-1);
        }
        (void) memset(&ovfbuf, 0, PBLKSIZ);
        for (i = 0; ; ) {
                item = makdatum(pagbuf, i);
                if (item.dptr == NULL)
                        break;
                if (calchash(item) & (hmask+1)) {
                        (void) additem(ovfbuf, item);
                        delitem(pagbuf, i);
                        item = makdatum(pagbuf, i);
                        if (item.dptr == NULL) {
                                (void) printf("split not paired\n");
                                break;
                        }
                        (void) additem(ovfbuf, item);
                        delitem(pagbuf, i);
                        continue;
                }
                i += 2;
        }
        (void) lseek(pagf, blkno*PBLKSIZ, 0);
        if (write(pagf, pagbuf, PBLKSIZ) < 0) {
                return (-1);
        }
        (void) lseek(pagf, (blkno+hmask+1)*PBLKSIZ, 0);
        if (write(pagf, ovfbuf, PBLKSIZ) < 0) {
                return (-1);
        }
        if (setbit() < 0) {
                return (-1);
        }
        goto loop;
}

datum
firstkey(void)
{
        return (firsthash(0L));
}

datum
nextkey(datum key)
{
        int i;
        datum item, bitem;
        long hash;
        int f;

#ifdef lint
        bitem.dptr = NULL;
        bitem.dsize = 0;
#endif /* lint */
        hash = calchash(key);
        dbm_access(hash);
        f = 1;
        for (i = 0; ; i += 2) {
                item = makdatum(pagbuf, i);
                if (item.dptr == NULL)
                        break;
                if (cmpdatum(key, item) <= 0)
                        continue;
                if (f || cmpdatum(bitem, item) < 0) {
                        bitem = item;
                        f = 0;
                }
        }
        if (f == 0)
                return (bitem);
        hash = hashinc(hash);
        if (hash == 0)
                return (item);
        return (firsthash(hash));
}

datum
firsthash(long hash)
{
        int i;
        datum item, bitem;

loop:
        dbm_access(hash);
        bitem = makdatum(pagbuf, 0);
        for (i = 2; ; i += 2) {
                item = makdatum(pagbuf, i);
                if (item.dptr == NULL)
                        break;
                if (cmpdatum(bitem, item) < 0)
                        bitem = item;
        }
        if (bitem.dptr != NULL)
                return (bitem);
        hash = hashinc(hash);
        if (hash == 0)
                return (item);
        goto loop;
}

void
dbm_access(long hash)
{
        ssize_t readsize;

        for (hmask = 0; ; hmask = (hmask<<1) + 1) {
                blkno = hash & hmask;
                bitno = blkno + hmask;
                if (getbit() == 0)
                        break;
        }
        if (blkno != oldb1) {
                (void) lseek(pagf, blkno*PBLKSIZ, 0);
                readsize = read(pagf, pagbuf, PBLKSIZ);
                if (readsize != PBLKSIZ) {
                        if (readsize < 0)
                                readsize = 0;
                        (void) memset((&pagbuf+readsize), 0, PBLKSIZ-readsize);
                }
                chkblk(pagbuf);
                oldb1 = blkno;
        }
}

int
getbit(void)
{
        long bn;
        ssize_t readsize;
        long b, i, n;

        if (bitno > maxbno)
                return (0);
        n = bitno % BYTESIZ;
        bn = bitno / BYTESIZ;
        i = bn % DBLKSIZ;
        b = bn / DBLKSIZ;
        if (b != oldb2) {
                (void) lseek(dirf, (long)b*DBLKSIZ, 0);
                readsize = read(dirf, dirbuf, DBLKSIZ);
                if (readsize != DBLKSIZ) {
                        if (readsize < 0)
                                readsize = 0;
                        (void) memset(&dirbuf+readsize, 0, DBLKSIZ-readsize);
                }
                oldb2 = b;
        }
        if (dirbuf[i] & (1<<n))
                return (1);
        return (0);
}

int
setbit(void)
{
        long bn;
        long i, n, b;

        if (dbrdonly)
                return (-1);
        if (bitno > maxbno) {
                maxbno = bitno;
                (void) getbit();
        }
        n = bitno % BYTESIZ;
        bn = bitno / BYTESIZ;
        i = bn % DBLKSIZ;
        b = bn / DBLKSIZ;
        dirbuf[i] |= 1<<n;
        (void) lseek(dirf, (long)b*DBLKSIZ, 0);
        if (write(dirf, dirbuf, DBLKSIZ) < 0)
                return (-1);
        return (0);
}

datum
makdatum(char *buf, int n)
{
        short *sp;
        int t;
        datum item;

        /* LINTED pointer cast */
        sp = (short *)buf;
        if (n < 0 || n >= sp[0])
                goto null;
        t = PBLKSIZ;
        if (n > 0)
                t = sp[n+1-1];
        item.dptr = buf+sp[n+1];
        item.dsize = t - sp[n+1];
        return (item);

null:
        item.dptr = NULL;
        item.dsize = 0;
        return (item);
}

int
cmpdatum(datum d1, datum d2)
{
        int n;
        char *p1, *p2;

        n = d1.dsize;
        if (n != d2.dsize)
                return (n - d2.dsize);
        if (n == 0)
                return (0);
        p1 = d1.dptr;
        p2 = d2.dptr;
        do {
                if (*p1++ != *p2++)
                        return (*--p1 - *--p2);
        } while (--n);
        return (0);
}

int     hitab[16]
/*
 * ken's
 * {
 *      055, 043, 036, 054, 063, 014, 004, 005,
 *      010, 064, 077, 000, 035, 027, 025, 071,
 * };
 */
        = {     61, 57, 53, 49, 45, 41, 37, 33,
        29, 25, 21, 17, 13,  9,  5,  1,
};
long    hltab[64]
        = {
        06100151277L, 06106161736L, 06452611562L, 05001724107L,
        02614772546L, 04120731531L, 04665262210L, 07347467531L,
        06735253126L, 06042345173L, 03072226605L, 01464164730L,
        03247435524L, 07652510057L, 01546775256L, 05714532133L,
        06173260402L, 07517101630L, 02431460343L, 01743245566L,
        00261675137L, 02433103631L, 03421772437L, 04447707466L,
        04435620103L, 03757017115L, 03641531772L, 06767633246L,
        02673230344L, 00260612216L, 04133454451L, 00615531516L,
        06137717526L, 02574116560L, 02304023373L, 07061702261L,
        05153031405L, 05322056705L, 07401116734L, 06552375715L,
        06165233473L, 05311063631L, 01212221723L, 01052267235L,
        06000615237L, 01075222665L, 06330216006L, 04402355630L,
        01451177262L, 02000133436L, 06025467062L, 07121076461L,
        03123433522L, 01010635225L, 01716177066L, 05161746527L,
        01736635071L, 06243505026L, 03637211610L, 01756474365L,
        04723077174L, 03642763134L, 05750130273L, 03655541561L,
};

long
hashinc(long hash)
{
        long bit;

        hash &= hmask;
        bit = hmask+1;
        for (; ; ) {
                bit >>= 1;
                if (bit == 0)
                        return (0L);
                if ((hash&bit) == 0)
                        return (hash|bit);
                hash &= ~bit;
        }
}

long
calchash(datum item)
{
        int i, j, f;
        long hashl;
        int hashi;

        hashl = 0;
        hashi = 0;
        for (i = 0; i < item.dsize; i++) {
                f = item.dptr[i];
                for (j = 0; j < BYTESIZ; j += 4) {
                        hashi += hitab[f&017];
                        hashl += hltab[hashi&63];
                        f >>= 4;
                }
        }
        return (hashl);
}

void
delitem(char *buf, int n)
{
        short *sp;
        int i1, i2, i3;

        /* LINTED pointer cast */
        sp = (short *)buf;
        if (n < 0 || n >= sp[0])
                goto bad;
        i1 = sp[n+1];
        i2 = PBLKSIZ;
        if (n > 0)
                i2 = sp[n+1-1];
        i3 = sp[sp[0]+1-1];
        if (i2 > i1)
        while (i1 > i3) {
                i1--;
                i2--;
                buf[i2] = buf[i1];
                buf[i1] = 0;
        }
        i2 -= i1;
        for (i1 = n + 1; i1 < sp[0]; i1++)
                sp[i1+1-1] = sp[i1+1] + i2;
        sp[0]--;
        sp[sp[0]+1] = 0;
        return;

bad:
        (void) printf("bad delitem\n");
        abort();
}

int
additem(char *buf, datum item)
{
        short *sp;
        int i1, i2;

        sp = (short *)buf;
        i1 = PBLKSIZ;
        if (sp[0] > 0)
                i1 = sp[sp[0]+1-1];
        i1 -= item.dsize;
        i2 = (sp[0]+2) * (int)sizeof (short);
        if (i1 <= i2)
                return (-1);
        sp[sp[0]+1] = (short)i1;
        for (i2 = 0; i2 < item.dsize; i2++) {
                buf[i1] = item.dptr[i2];
                i1++;
        }
        sp[0]++;
        return (sp[0]-1);
}

void
chkblk(char *buf)
{
        short *sp;
        int t, i;

        /* LINTED pointer cast */
        sp = (short *)buf;
        t = PBLKSIZ;
        for (i = 0; i < sp[0]; i++) {
                if (sp[i+1] > t)
                        goto bad;
                t = sp[i+1];
        }
        if (t < (sp[0]+1) * sizeof (short))
                goto bad;
        return;

bad:
        (void) printf("bad block\n");
        abort();
        (void) memset(&buf, 0, PBLKSIZ);
}