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