usr/src/lib/libc/port/gen/ndbm.c
author eschrock
Fri, 19 Aug 2005 11:56:37 -0700
changeset 380 4ae8f505c115
parent 0 68f95e015346
child 5891 0d5c6468bb04
permissions -rw-r--r--
6307489 getmntany() should not stat meaningless special devices
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     5
 * Common Development and Distribution License, Version 1.0 only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     6
 * (the "License").  You may not use this file except in compliance
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 * with the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
/*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
/*	  All Rights Reserved  	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
 * University Copyright- Copyright (c) 1982, 1986, 1988
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
 * The Regents of the University of California
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
 * All Rights Reserved
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
 * University Acknowledgment- Portions of this document are derived from
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
 * software developed by the University of California, Berkeley, and its
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
 * contributors.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
#include "synonyms.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
#include <sys/stat.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
#include <fcntl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
#include <sys/file.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
#include <stdio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
#include <stdlib.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
#include <errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
#include <ndbm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
#include <unistd.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
#include <string.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
/* add support for batched writing for NIS */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
#define	_DBM_DEFWRITE 0x4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
#define	_DBM_DIRTY 0x8
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
#define	_DBM_DIRDIRTY 0x10
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
#define	dbm_dirty(db) ((db)->dbm_flags & _DBM_DIRTY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
#define	dbm_dirdirty(db) ((db)->dbm_flags & _DBM_DIRDIRTY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
#define	dbm_defwrite(db) ((db)->dbm_flags & _DBM_DEFWRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
#define	dbm_setdirty(db) (db)->dbm_flags |= _DBM_DIRTY
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
#define	dbm_clrdirty(db) (db)->dbm_flags &= ~_DBM_DIRTY
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
#define	dbm_setdirdirty(db) (db)->dbm_flags |= _DBM_DIRDIRTY
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
#define	dbm_clrdirdirty(db) (db)->dbm_flags &= ~_DBM_DIRDIRTY
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
 * forward declarations
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
static datum makdatum(char *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
static unsigned long dcalchash(datum);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
static void dbm_access(DBM *, unsigned long);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
static int finddatum(char *, datum);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
static int delitem(char *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
static int additem(char *, datum, datum);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
static int cmpdatum(datum, datum);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
static int setbit(DBM *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
static int getbit(DBM *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
static int dbm_flushdir(DBM *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
static int dbm_flushpag(DBM *db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
/* the following three are required by mapfile-vers */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
datum dbm_do_nextkey(DBM *, datum);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
int dbm_close_status(DBM *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
/* used to make a dbm file all at once instead of incrementally */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
dbm_setdefwrite(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
	db->dbm_flags |= _DBM_DEFWRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
#undef	dbm_error
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
dbm_error(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
	return (db->dbm_flags & _DBM_IOERR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
#undef	dbm_clearerr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
dbm_clearerr(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
	return (db->dbm_flags &= ~_DBM_IOERR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
dbm_flush(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
	int ok = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
	if (dbm_flushpag(db) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
		ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
	if (dbm_flushdir(db) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
		ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
	return (ok);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
dbm_flushpag(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
	int ok = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
	if (dbm_dirty(db)) { /* must page out the page */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
		where = (((off64_t)db->dbm_pagbno) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
		if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
		    (write(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) != PBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
			db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
			ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
		dbm_clrdirty(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
	return (ok);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   137
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
dbm_flushdir(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   141
	int ok = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   142
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   143
	if (dbm_dirdirty(db)) { /* must page out the dir */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   144
		where = (((off64_t)db->dbm_dirbno) * DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   145
		if ((lseek64(db->dbm_dirf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   146
		    (write(db->dbm_dirf, db->dbm_dirbuf, DBLKSIZ) != DBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   147
			ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   148
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   149
		dbm_clrdirdirty(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   150
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   151
	return (ok);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   152
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   153
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   154
#define	BYTESIZ 8
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   155
#undef setbit
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   156
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   157
DBM *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   158
dbm_open(const char *file, int flags, mode_t mode)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   159
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   160
	struct stat64 statb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   161
	DBM *db;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   162
	int	serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   163
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   164
	if ((db = (DBM *)malloc(sizeof (*db))) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   165
		errno = ENOMEM;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   166
		return ((DBM *)0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   167
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   168
	db->dbm_flags = (flags & 03) == O_RDONLY ? _DBM_RDONLY : 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   169
	if ((flags & 03) == O_WRONLY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   170
		flags = (flags & ~03) | O_RDWR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   171
	if (strlcpy(db->dbm_pagbuf, file, sizeof (db->dbm_pagbuf)) >=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   172
	    sizeof (db->dbm_pagbuf) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   173
	    strlcat(db->dbm_pagbuf, ".pag", sizeof (db->dbm_pagbuf)) >=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   174
	    sizeof (db->dbm_pagbuf)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   175
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   176
		 * file.pag does not fit into dbm_pagbuf.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   177
		 * fail with ENAMETOOLONG.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   178
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   179
		serrno = ENAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   180
		goto bad;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   181
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   182
	db->dbm_pagf = open64(db->dbm_pagbuf, flags, mode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   183
	if (db->dbm_pagf < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   184
		serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   185
		goto bad;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   186
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   187
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   188
	 * We know this won't overflow so it is safe to ignore the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   189
	 * return value; we use strl* to prevent false hits in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   190
	 * code sweeps.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   191
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   192
	(void) strlcpy(db->dbm_pagbuf, file, sizeof (db->dbm_pagbuf));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   193
	(void) strlcat(db->dbm_pagbuf, ".dir", sizeof (db->dbm_pagbuf));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   194
	db->dbm_dirf = open64(db->dbm_pagbuf, flags, mode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   195
	if (db->dbm_dirf < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   196
		serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   197
		goto bad1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   198
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   199
	(void) fstat64(db->dbm_dirf, &statb);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   200
	db->dbm_maxbno = statb.st_size * BYTESIZ-1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   201
	db->dbm_pagbno = db->dbm_dirbno = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   202
	return (db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   203
bad1:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   204
	(void) close(db->dbm_pagf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   205
bad:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   206
	free((char *)db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   207
	errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   208
	return ((DBM *)0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   209
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   210
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   211
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   212
dbm_close(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   213
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   214
(void) dbm_close_status(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   215
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   216
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   217
/* close with return code */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   218
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   219
dbm_close_status(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   220
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   221
	int ok;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   222
	ok = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   223
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   224
	if (dbm_flush(db) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   225
		ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   226
	if (close(db->dbm_dirf) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   227
		ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   228
	if (close(db->dbm_pagf) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   229
		ok = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   230
	free((char *)db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   231
	return (ok);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   232
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   233
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   234
long
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   235
dbm_forder(DBM *db, datum key)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   236
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   237
	unsigned long hash;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   238
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   239
	hash = dcalchash(key);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   240
	for (db->dbm_hmask = 0; ; db->dbm_hmask = (db->dbm_hmask<<1)+1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   241
		db->dbm_blkno = hash & db->dbm_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   242
		db->dbm_bitno = db->dbm_blkno + db->dbm_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   243
		if (getbit(db) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   244
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   245
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   246
	return (db->dbm_blkno);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   247
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   248
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   249
datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   250
dbm_fetch(DBM *db, datum key)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   251
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   252
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   253
	datum item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   254
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   255
	if (dbm_error(db))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   256
		goto err;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   257
	dbm_access(db, dcalchash(key));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   258
	if ((i = finddatum(db->dbm_pagbuf, key)) >= 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   259
		item = makdatum(db->dbm_pagbuf, i+1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   260
		if (item.dptr != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   261
			return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   262
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   263
err:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   264
	item.dptr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   265
	item.dsize = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   266
	return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   267
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   268
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   269
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   270
dbm_delete(DBM *db, datum key)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   271
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   272
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   273
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   274
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   275
	if (dbm_error(db))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   276
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   277
	if (dbm_rdonly(db)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   278
		errno = EPERM;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   279
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   280
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   281
	dbm_access(db, dcalchash(key));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   282
	if ((i = finddatum(db->dbm_pagbuf, key)) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   283
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   284
	if (!delitem(db->dbm_pagbuf, i))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   285
		goto err;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   286
	db->dbm_pagbno = db->dbm_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   287
	if (dbm_defwrite(db)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   288
		dbm_setdirty(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   289
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   290
		where = (((off64_t)db->dbm_blkno) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   291
		if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   292
		    (write(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) != PBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   293
			err:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   294
				db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   295
				return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   296
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   297
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   298
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   299
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   300
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   301
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   302
dbm_store(DBM *db, datum key, datum dat, int replace)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   303
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   304
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   305
	datum item, item1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   306
	char ovfbuf[PBLKSIZ];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   307
	unsigned long hashdiff, key_hash, item_hash;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   308
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   309
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   310
	if (dbm_error(db))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   311
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   312
	if (dbm_rdonly(db)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   313
		errno = EPERM;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   314
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   315
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   316
loop:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   317
	key_hash = dcalchash(key);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   318
	dbm_access(db, key_hash);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   319
	if ((i = finddatum(db->dbm_pagbuf, key)) >= 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   320
		if (!replace)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   321
			return (1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   322
		if (!delitem(db->dbm_pagbuf, i)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   323
			db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   324
			return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   325
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   326
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   327
	if (!additem(db->dbm_pagbuf, key, dat))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   328
		goto split;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   329
	db->dbm_pagbno = db->dbm_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   330
	if (dbm_defwrite(db)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   331
		dbm_setdirty(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   332
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   333
		where = (((off64_t)db->dbm_blkno) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   334
		if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   335
		    (write(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) != PBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   336
			db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   337
			return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   338
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   339
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   340
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   341
split:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   342
	hashdiff = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   343
	if (key.dsize + dat.dsize + 3 * (int)sizeof (short) >= PBLKSIZ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   344
		db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   345
		errno = ENOSPC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   346
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   347
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   348
	(void) memset(ovfbuf, 0, PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   349
	for (i = 0; ; ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   350
		item = makdatum(db->dbm_pagbuf, i);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   351
		if (item.dptr == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   352
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   353
		item_hash = dcalchash(item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   354
		if (item_hash != key_hash) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   355
			hashdiff++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   356
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   357
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   358
		if (item_hash & (db->dbm_hmask+1)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   359
			item1 = makdatum(db->dbm_pagbuf, i+1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   360
			if (item1.dptr == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   361
				/* (void) fprintf(stderr, "ndbm: */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   362
				/* split not paired\n"); */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   363
				db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   364
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   365
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   366
			if (!additem(ovfbuf, item, item1) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   367
			    !delitem(db->dbm_pagbuf, i)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   368
				db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   369
				return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   370
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   371
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   372
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   373
		i += 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   374
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   375
	db->dbm_pagbno = db->dbm_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   376
	where = (((off64_t)db->dbm_blkno) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   377
	if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   378
	    (write(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) != PBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   379
		db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   380
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   381
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   382
	dbm_clrdirty(db); /* clear dirty */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   383
	where = (((off64_t)db->dbm_blkno + db->dbm_hmask + 1) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   384
	if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   385
	    (write(db->dbm_pagf, ovfbuf, PBLKSIZ) != PBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   386
		db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   387
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   388
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   389
	if (setbit(db) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   390
		db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   391
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   392
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   393
	if (hashdiff == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   394
		db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   395
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   396
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   397
	goto loop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   398
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   399
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   400
static unsigned long
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   401
dbm_hashinc(DBM *db, unsigned long hash)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   402
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   403
	long bit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   404
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   405
	hash &= db->dbm_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   406
	bit = db->dbm_hmask+1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   407
	for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   408
		bit >>= 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   409
		if (bit == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   410
			return (0L);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   411
		if ((hash&bit) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   412
			return (hash|bit);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   413
		hash &= ~bit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   414
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   415
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   416
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   417
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   418
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   419
static datum nullkey = {NULL, 0};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   420
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   421
datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   422
dbm_firsthash(DBM *db, unsigned long hash)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   423
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   424
	int i, j;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   425
	datum item, bitem;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   426
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   427
loop:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   428
	dbm_access(db, hash);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   429
	j = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   430
	bitem = makdatum(db->dbm_pagbuf, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   431
	for (i = 2; ; i += 2) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   432
		item = makdatum(db->dbm_pagbuf, i);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   433
		if (item.dptr == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   434
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   435
		if (cmpdatum(bitem, item) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   436
			j = i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   437
			bitem = item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   438
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   439
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   440
	if (bitem.dptr != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   441
		db->dbm_keyptr = j + 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   442
		db->dbm_blkptr = db->dbm_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   443
		return (bitem);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   444
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   445
	hash = dbm_hashinc(db, hash);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   446
	if (hash == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   447
		return (item); /* null item */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   448
	goto loop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   449
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   450
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   451
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   452
datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   453
dbm_firstkey(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   454
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   455
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   456
	db->dbm_blkptr = 0L;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   457
	db->dbm_keyptr = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   458
	return (dbm_firsthash(db, 0L));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   459
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   460
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   461
datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   462
dbm_nextkey(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   463
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   464
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   465
	return (dbm_do_nextkey(db, nullkey));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   466
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   467
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   468
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   469
 * this is used if keyptr-2, blocknum doesn't point to the previous
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   470
 * specific key allowing the fast hash order search --
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   471
 * its use indicates user tampering with our state variables,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   472
 * which some evil users might do to search from some specific place.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   473
 * It finds the first key at or after blkptr, keyptr in block seq order
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   474
 * this requires looking at all sorts of emtpy blocks in many cases
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   475
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   476
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   477
static
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   478
datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   479
dbm_slow_nextkey(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   480
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   481
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   482
	struct stat64 statb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   483
	datum item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   484
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   485
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   486
	if (dbm_error(db) || fstat64(db->dbm_pagf, &statb) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   487
		goto err;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   488
	statb.st_size /= PBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   489
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   490
	for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   491
		if (db->dbm_blkptr != db->dbm_pagbno) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   492
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   493
			if (dbm_dirty(db)) (void) dbm_flushpag(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   494
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   495
			db->dbm_pagbno = db->dbm_blkptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   496
			where = (((off64_t)db->dbm_blkptr) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   497
			if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   498
			    (read(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) !=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   499
				PBLKSIZ))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   500
				(void) memset(db->dbm_pagbuf, 0, PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   501
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   502
			else if (chkblk(db->dbm_pagbuf) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   503
				db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   504
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   505
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   506
		/* Am I an empty block? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   507
		if (((short *)(uintptr_t)db->dbm_pagbuf)[0] != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   508
			item = makdatum(db->dbm_pagbuf, db->dbm_keyptr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   509
			if (item.dptr != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   510
				db->dbm_keyptr += 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   511
				return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   512
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   513
			db->dbm_keyptr = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   514
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   515
		/* go to next sequential block */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   516
		if (++db->dbm_blkptr >= statb.st_size)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   517
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   518
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   519
err:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   520
	item.dptr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   521
	item.dsize = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   522
	return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   523
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   524
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   525
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   526
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   527
datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   528
dbm_do_nextkey(DBM *db, datum inkey)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   529
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   530
	datum item, bitem;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   531
	unsigned long hash;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   532
	datum key;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   533
	int f;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   534
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   535
	int j;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   536
	short *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   537
	long n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   538
	char *p1, *p2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   539
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   540
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   541
	if (dbm_error(db)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   542
		item.dptr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   543
		item.dsize = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   544
		return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   545
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   546
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   547
	/* user has supplied lastkey */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   548
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   549
	if (inkey.dptr != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   550
		dbm_access(db, (hash = dcalchash(inkey)));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   551
		if ((i = finddatum(db->dbm_pagbuf, inkey)) >= 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   552
			db->dbm_keyptr = i + 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   553
			db->dbm_blkptr = db->dbm_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   554
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   555
		key = inkey;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   556
	} else  {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   557
		/* is this a manual firstkey request? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   558
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   559
		if (db->dbm_blkptr == 0L &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   560
			db->dbm_keyptr == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   561
			return (dbm_firsthash(db, 0L));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   562
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   563
		/* no -- get lastkey this is like dbm_access by blkptr */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   564
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   565
		if (db->dbm_blkptr != db->dbm_pagbno) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   566
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   567
			if (dbm_dirty(db)) (void) dbm_flushpag(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   568
			db->dbm_pagbno = db->dbm_blkptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   569
			where = (((off64_t)db->dbm_blkptr) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   570
			if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   571
			    (read(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) !=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   572
				PBLKSIZ))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   573
				(void) memset(db->dbm_pagbuf, 0, PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   574
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   575
			else if (chkblk(db->dbm_pagbuf) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   576
			db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   577
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   578
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   579
		/* now just make up last key datum */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   580
		if (db->dbm_keyptr >= 2)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   581
			key = makdatum(db->dbm_pagbuf, (db->dbm_keyptr-2));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   582
		else key = nullkey;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   583
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   584
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   585
		 * the keyptr pagbuf have failed us, the user must
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   586
		 * be an extra clever moron who depends on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   587
		 * these variables and their former meaning.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   588
		 * If we set the variables this would have got
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   589
		 * us the key for sure! So give him the old algorithm.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   590
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   591
		if (key.dptr == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   592
			return (dbm_slow_nextkey(db));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   593
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   594
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   595
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   596
	 * at this point the last key is paged in and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   597
	 * we can proceed as in old dbm -- like Ken did his.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   598
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   599
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   600
	f = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   601
	j = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   602
	sp = (short *)(uintptr_t)db->dbm_pagbuf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   603
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   604
	for (i = 0; ; i += 2) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   605
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   606
		/* begin put makdatum inline */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   607
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   608
		if ((unsigned)i >= sp[0]) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   609
			item.dptr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   610
			item.dsize = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   611
			break; /* from below */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   612
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   613
			if (i > 0) item.dsize = sp[i] - sp[i + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   614
			else item.dsize = PBLKSIZ - sp[i + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   615
			item.dptr = db->dbm_pagbuf+sp[i + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   616
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   617
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   618
		/* item = makdatum(db->dbm_pagbuf, i); */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   619
		/* end put makdatum inline */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   620
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   621
		if (item.dptr == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   622
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   623
/* inline cmpdatum */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   624
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   625
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   626
		n = key.dsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   627
		if (n != item.dsize) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   628
			if ((n - item.dsize) <= 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   629
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   630
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   631
			if (n == 0) continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   632
			p1 = key.dptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   633
			p2 = item.dptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   634
			do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   635
				if (*p1++ != *p2++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   636
					if ((*--p1 - *--p2) > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   637
						goto keep_going;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   638
					else continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   639
			} while (--n);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   640
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   641
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   642
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   643
keep_going:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   644
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   645
/* end inline cmpdatum */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   646
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   647
		 * if (cmpdatum(key, item) <= 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   648
		 *	continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   649
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   650
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   651
		if (f) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   652
			bitem = item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   653
			j = i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   654
			f = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   655
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   656
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   657
			/* if (cmpdatum(bitem, item) < 0) */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   658
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   659
			n = bitem.dsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   660
			if (n != item.dsize) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   661
				if ((n - item.dsize) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   662
					bitem = item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   663
					j = i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   664
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   665
			} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   666
				if (n != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   667
					p1 = bitem.dptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   668
					p2 = item.dptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   669
					do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   670
					if (*p1++ != *p2++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   671
						if ((*--p1 - *--p2) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   672
							bitem = item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   673
							j = i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   674
						}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   675
						break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   676
					}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   677
					} while (--n);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   678
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   679
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   680
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   681
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   682
	if (f == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   683
		db->dbm_keyptr = j + 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   684
		db->dbm_blkptr = db->dbm_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   685
		return (bitem);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   686
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   687
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   688
	/* really need hash at this point */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   689
	/* if he gave us a key we have already calculated the hash */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   690
	/* if not get it */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   691
	if (inkey.dptr == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   692
		hash = dcalchash(key);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   693
	hash = dbm_hashinc(db, hash);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   694
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   695
	if (hash == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   696
		return (item); /* null */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   697
	/* get first item on next page in hash table order */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   698
	return (dbm_firsthash(db, hash));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   699
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   700
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   701
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   702
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   703
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   704
dbm_access(DBM *db, unsigned long hash)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   705
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   706
	long b, i, n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   707
	long bn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   708
	long my_bitno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   709
	long my_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   710
	long my_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   711
	int j = (sizeof (my_hmask)) * 8;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   712
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   713
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   714
	for (my_hmask = 0; j-- > 0; my_hmask = (my_hmask<<1) + 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   715
		my_blkno = hash & my_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   716
		my_bitno = my_blkno + my_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   717
		/* getbit inline */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   718
		if (my_bitno > db->dbm_maxbno) break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   719
		n = my_bitno % BYTESIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   720
		bn = my_bitno / BYTESIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   721
		i = bn % DBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   722
		b = bn / DBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   723
		if (b != db->dbm_dirbno) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   724
			if (dbm_dirdirty(db))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   725
				(void) dbm_flushdir(db); /* must flush */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   726
			db->dbm_dirbno = b;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   727
			where = (((off64_t)b) * DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   728
			if ((lseek64(db->dbm_dirf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   729
			    (read(db->dbm_dirf, db->dbm_dirbuf, DBLKSIZ) !=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   730
				DBLKSIZ))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   731
				(void) memset(db->dbm_dirbuf, 0, DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   732
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   733
		if ((db->dbm_dirbuf[i] & (1<<n)) == 0) break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   734
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   735
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   736
		 * if (getbit(db) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   737
		 *	break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   738
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   739
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   740
	/* copy */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   741
	db->dbm_blkno = my_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   742
	db->dbm_bitno = my_bitno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   743
	db->dbm_hmask = my_hmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   744
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   745
	if (my_blkno != db->dbm_pagbno) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   746
		if (dbm_dirty(db)) { /* must page out the page */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   747
			where = (((off64_t)db->dbm_pagbno) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   748
			if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   749
			    (write(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) !=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   750
				PBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   751
				db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   752
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   753
		dbm_clrdirty(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   754
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   755
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   756
		db->dbm_pagbno = my_blkno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   757
		where = (((off64_t)my_blkno) * PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   758
		if ((lseek64(db->dbm_pagf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   759
		    (read(db->dbm_pagf, db->dbm_pagbuf, PBLKSIZ) != PBLKSIZ))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   760
			(void) memset(db->dbm_pagbuf, 0, PBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   761
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   762
		else if (chkblk(db->dbm_pagbuf) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   763
			db->dbm_flags |= _DBM_IOERR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   764
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   765
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   766
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   767
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   768
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   769
getbit(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   770
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   771
	long bn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   772
	long b, i, n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   773
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   774
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   775
	if (db->dbm_bitno > db->dbm_maxbno)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   776
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   777
	n = db->dbm_bitno % BYTESIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   778
	bn = db->dbm_bitno / BYTESIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   779
	i = bn % DBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   780
	b = bn / DBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   781
	if (b != db->dbm_dirbno) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   782
		if (dbm_dirdirty(db))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   783
			(void) dbm_flushdir(db); /* must flush */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   784
		db->dbm_dirbno = b;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   785
		where = (((off64_t)b) * DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   786
		if ((lseek64(db->dbm_dirf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   787
		    (read(db->dbm_dirf, db->dbm_dirbuf, DBLKSIZ) != DBLKSIZ))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   788
			(void) memset(db->dbm_dirbuf, 0, DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   789
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   790
	return (db->dbm_dirbuf[i] & (1<<n));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   791
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   792
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   793
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   794
setbit(DBM *db)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   795
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   796
	long bn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   797
	long i, n, b;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   798
	off64_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   799
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   800
	if (db->dbm_bitno > db->dbm_maxbno)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   801
		db->dbm_maxbno = db->dbm_bitno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   802
	n = db->dbm_bitno % BYTESIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   803
	bn = db->dbm_bitno / BYTESIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   804
	i = bn % DBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   805
	b = bn / DBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   806
	if (b != db->dbm_dirbno) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   807
		if (dbm_dirdirty(db))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   808
			(void) dbm_flushdir(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   809
		db->dbm_dirbno = b;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   810
		where = (((off64_t)b) * DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   811
		if ((lseek64(db->dbm_dirf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   812
		    (read(db->dbm_dirf, db->dbm_dirbuf, DBLKSIZ) != DBLKSIZ))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   813
			(void) memset(db->dbm_dirbuf, 0, DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   814
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   815
	db->dbm_dirbuf[i] |= 1<<n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   816
	db->dbm_dirbno = b;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   817
	if (dbm_defwrite(db)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   818
		dbm_setdirdirty(db);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   819
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   820
		where = (((off64_t)b) * DBLKSIZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   821
		if ((lseek64(db->dbm_dirf, where, L_SET) != where) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   822
		    (write(db->dbm_dirf, db->dbm_dirbuf, DBLKSIZ) != DBLKSIZ)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   823
			return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   824
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   825
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   826
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   827
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   828
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   829
static datum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   830
makdatum(char buf[PBLKSIZ], int n)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   831
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   832
	short *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   833
	int t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   834
	datum item;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   835
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   836
	sp = (short *)(uintptr_t)buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   837
	if ((unsigned)n >= sp[0]) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   838
		item.dptr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   839
		item.dsize = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   840
		return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   841
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   842
	t = PBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   843
	if (n > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   844
		t = sp[n];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   845
	item.dptr = buf + sp[n + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   846
	item.dsize = t - sp[n + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   847
	return (item);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   848
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   849
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   850
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   851
cmpdatum(datum d1, datum d2)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   852
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   853
	long n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   854
	char *p1, *p2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   855
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   856
	n = d1.dsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   857
	if (n != d2.dsize)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   858
		return ((int)(n - d2.dsize));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   859
	if (n == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   860
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   861
	p1 = d1.dptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   862
	p2 = d2.dptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   863
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   864
		if (*p1 != *p2)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   865
			return (*p1 - *p2);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   866
		else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   867
			p1++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   868
			p2++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   869
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   870
	} while (--n);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   871
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   872
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   873
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   874
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   875
finddatum(char buf[PBLKSIZ], datum item)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   876
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   877
	short *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   878
	int i, n, j;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   879
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   880
	sp = (short *)(uintptr_t)buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   881
	n = PBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   882
	for (i = 0, j = sp[0]; i < j; i += 2, n = sp[i]) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   883
		n -= sp[i + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   884
		if (n != item.dsize)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   885
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   886
		if (n == 0 || memcmp(&buf[sp[i+1]], item.dptr, n) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   887
			return (i);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   888
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   889
	return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   890
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   891
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   892
static const int hitab[16]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   893
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   894
 * ken's
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   895
 * {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   896
 *	055, 043, 036, 054, 063, 014, 004, 005,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   897
 *	010, 064, 077, 000, 035, 027, 025, 071,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   898
 * };
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   899
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   900
	= {    61, 57, 53, 49, 45, 41, 37, 33,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   901
		29, 25, 21, 17, 13,  9,  5,  1,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   902
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   903
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   904
/* could consider to make it 32-bit int table to minimize memory usage */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   905
static const long hltab[64]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   906
	= {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   907
	06100151277L, 06106161736L, 06452611562L, 05001724107L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   908
	02614772546L, 04120731531L, 04665262210L, 07347467531L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   909
	06735253126L, 06042345173L, 03072226605L, 01464164730L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   910
	03247435524L, 07652510057L, 01546775256L, 05714532133L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   911
	06173260402L, 07517101630L, 02431460343L, 01743245566L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   912
	00261675137L, 02433103631L, 03421772437L, 04447707466L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   913
	04435620103L, 03757017115L, 03641531772L, 06767633246L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   914
	02673230344L, 00260612216L, 04133454451L, 00615531516L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   915
	06137717526L, 02574116560L, 02304023373L, 07061702261L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   916
	05153031405L, 05322056705L, 07401116734L, 06552375715L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   917
	06165233473L, 05311063631L, 01212221723L, 01052267235L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   918
	06000615237L, 01075222665L, 06330216006L, 04402355630L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   919
	01451177262L, 02000133436L, 06025467062L, 07121076461L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   920
	03123433522L, 01010635225L, 01716177066L, 05161746527L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   921
	01736635071L, 06243505026L, 03637211610L, 01756474365L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   922
	04723077174L, 03642763134L, 05750130273L, 03655541561L,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   923
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   924
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   925
static unsigned long
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   926
dcalchash(datum item)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   927
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   928
	long s;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   929
	int c, j;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   930
	char *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   931
	unsigned long hashl;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   932
	long hashi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   933
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   934
	hashl = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   935
	hashi = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   936
	for (cp = item.dptr, s = item.dsize; --s >= 0; ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   937
		c = *cp++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   938
		for (j = 0; j < BYTESIZ; j += 4) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   939
			hashi += hitab[c&017];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   940
			hashl += hltab[hashi&63];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   941
			c >>= 4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   942
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   943
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   944
	return (hashl);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   945
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   946
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   947
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   948
 * Delete pairs of items (n & n+1).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   949
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   950
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   951
delitem(char buf[PBLKSIZ], int n)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   952
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   953
	short *sp, *sp1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   954
	int i1, i2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   955
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   956
	sp = (short *)(uintptr_t)buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   957
	i2 = sp[0];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   958
	if ((unsigned)n >= i2 || (n & 1))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   959
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   960
	if (n == i2-2) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   961
		sp[0] -= 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   962
		return (1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   963
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   964
	i1 = PBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   965
	if (n > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   966
		i1 = sp[n];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   967
	i1 -= sp[n+2];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   968
	if (i1 > 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   969
		i2 = sp[i2];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   970
		(void) memmove(&buf[i2 + i1], &buf[i2], sp[n+2] - i2);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   971
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   972
	sp[0] -= 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   973
	for (sp1 = sp + sp[0], sp += n+1; sp <= sp1; sp++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   974
		sp[0] = sp[2] + i1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   975
	return (1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   976
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   977
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   978
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   979
 * Add pairs of items (item & item1).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   980
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   981
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   982
additem(char buf[PBLKSIZ], datum item, datum item1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   983
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   984
	short *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   985
	int i1, i2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   986
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   987
	sp = (short *)(uintptr_t)buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   988
	i1 = PBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   989
	i2 = sp[0];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   990
	if (i2 > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   991
		i1 = sp[i2];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   992
	i1 -= item.dsize + item1.dsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   993
	if (i1 <= (i2+3) * (int)sizeof (short))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   994
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   995
	sp[0] += 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   996
	sp[++i2] = (short)(i1 + item1.dsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   997
	(void) memmove(&buf[i1 + item1.dsize], item.dptr, item.dsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   998
	sp[++i2] = i1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   999
	(void) memmove(&buf[i1], item1.dptr, item1.dsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1000
	return (1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1001
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1002
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1003
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1004
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1005
chkblk(char buf[PBLKSIZ])
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1006
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1007
	short *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1008
	int t, i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1009
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1010
	sp = (short *)buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1011
	t = PBLKSIZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1012
	for (i = 0; i < sp[0]; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1013
		if (sp[i + 1] > t)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1014
			return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1015
		t = sp[i + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1016
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1017
	if (t < (sp[0] + 1) * sizeof (short))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1018
		return (-1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1019
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1020
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1021
#endif