BSD 4_4_Lite1 release
[unix-history] / usr / src / lib / libc / db / recno / rec_open.c
index b2eaece..2a0f354 100644 (file)
@@ -1,15 +1,41 @@
 /*-
 /*-
- * Copyright (c) 1990 The Regents of the University of California.
- * All rights reserved.
+ * Copyright (c) 1990, 1993
+ *     The Regents of the University of California.  All rights reserved.
  *
  * This code is derived from software contributed to Berkeley by
  * Mike Olson.
  *
  *
  * This code is derived from software contributed to Berkeley by
  * Mike Olson.
  *
- * %sccs.include.redist.c%
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ *    must display the following acknowledgement:
+ *     This product includes software developed by the University of
+ *     California, Berkeley and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ *    may be used to endorse or promote products derived from this software
+ *    without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
  */
 
 #if defined(LIBC_SCCS) && !defined(lint)
  */
 
 #if defined(LIBC_SCCS) && !defined(lint)
-static char sccsid[] = "@(#)rec_open.c 5.16 (Berkeley) %G%";
+static char sccsid[] = "@(#)rec_open.c 8.6 (Berkeley) 2/22/94";
 #endif /* LIBC_SCCS and not lint */
 
 #include <sys/types.h>
 #endif /* LIBC_SCCS and not lint */
 
 #include <sys/types.h>
@@ -27,9 +53,9 @@ static char sccsid[] = "@(#)rec_open.c        5.16 (Berkeley) %G%";
 #include "recno.h"
 
 DB *
 #include "recno.h"
 
 DB *
-__rec_open(fname, flags, mode, openinfo)
+__rec_open(fname, flags, mode, openinfo, dflags)
        const char *fname;
        const char *fname;
-       int flags, mode;
+       int flags, mode, dflags;
        const RECNOINFO *openinfo;
 {
        BTREE *t;
        const RECNOINFO *openinfo;
 {
        BTREE *t;
@@ -52,13 +78,14 @@ __rec_open(fname, flags, mode, openinfo)
                btopeninfo.cachesize = openinfo->cachesize;
                btopeninfo.maxkeypage = 0;
                btopeninfo.minkeypage = 0;
                btopeninfo.cachesize = openinfo->cachesize;
                btopeninfo.maxkeypage = 0;
                btopeninfo.minkeypage = 0;
-               btopeninfo.psize = 0;
+               btopeninfo.psize = openinfo->psize;
                btopeninfo.compare = NULL;
                btopeninfo.prefix = NULL;
                btopeninfo.lorder = openinfo->lorder;
                btopeninfo.compare = NULL;
                btopeninfo.prefix = NULL;
                btopeninfo.lorder = openinfo->lorder;
-               dbp = __bt_open(NULL, O_RDWR, S_IRUSR | S_IWUSR, &btopeninfo);
+               dbp = __bt_open(openinfo->bfname,
+                   O_RDWR, S_IRUSR | S_IWUSR, &btopeninfo, dflags);
        } else
        } else
-               dbp = __bt_open(NULL, O_RDWR, S_IRUSR | S_IWUSR, NULL);
+               dbp = __bt_open(NULL, O_RDWR, S_IRUSR | S_IWUSR, NULL, dflags);
        if (dbp == NULL)
                goto err;
 
        if (dbp == NULL)
                goto err;
 
@@ -71,7 +98,7 @@ __rec_open(fname, flags, mode, openinfo)
        t = dbp->internal;
        if (openinfo) {
                if (openinfo->flags & R_FIXEDLEN) {
        t = dbp->internal;
        if (openinfo) {
                if (openinfo->flags & R_FIXEDLEN) {
-                       SET(t, BTF_FIXEDLEN);
+                       SET(t, R_FIXLEN);
                        t->bt_reclen = openinfo->reclen;
                        if (t->bt_reclen == 0)
                                goto einval;
                        t->bt_reclen = openinfo->reclen;
                        if (t->bt_reclen == 0)
                                goto einval;
@@ -80,35 +107,37 @@ __rec_open(fname, flags, mode, openinfo)
        } else
                t->bt_bval = '\n';
 
        } else
                t->bt_bval = '\n';
 
-       SET(t, BTF_RECNO);
+       SET(t, R_RECNO);
        if (fname == NULL)
        if (fname == NULL)
-               SET(t, BTF_EOF | BTF_RINMEM);
+               SET(t, R_EOF | R_INMEM);
        else
                t->bt_rfd = rfd;
        t->bt_rcursor = 0;
 
        else
                t->bt_rfd = rfd;
        t->bt_rcursor = 0;
 
-       /*
-        * In 4.4BSD stat(2) returns true for ISSOCK on pipes.  Until
-        * then, this is fairly close.  Pipes are read-only.
-        */
        if (fname != NULL) {
        if (fname != NULL) {
+               /*
+                * In 4.4BSD, stat(2) returns true for ISSOCK on pipes.
+                * Unfortunately, that's not portable, so we use lseek
+                * and check the errno values.
+                */
+               errno = 0;
                if (lseek(rfd, (off_t)0, SEEK_CUR) == -1 && errno == ESPIPE) {
                        switch (flags & O_ACCMODE) {
                        case O_RDONLY:
                if (lseek(rfd, (off_t)0, SEEK_CUR) == -1 && errno == ESPIPE) {
                        switch (flags & O_ACCMODE) {
                        case O_RDONLY:
-                               SET(t, BTF_RDONLY);
+                               SET(t, R_RDONLY);
                                break;
                        default:
                                goto einval;
                        }
 slow:                  if ((t->bt_rfp = fdopen(rfd, "r")) == NULL)
                                goto err;
                                break;
                        default:
                                goto einval;
                        }
 slow:                  if ((t->bt_rfp = fdopen(rfd, "r")) == NULL)
                                goto err;
-                       SET(t, BTF_CLOSEFP);
+                       SET(t, R_CLOSEFP);
                        t->bt_irec =
                        t->bt_irec =
-                           ISSET(t, BTF_FIXEDLEN) ? __rec_fpipe : __rec_vpipe;
+                           ISSET(t, R_FIXLEN) ? __rec_fpipe : __rec_vpipe;
                } else {
                        switch (flags & O_ACCMODE) {
                        case O_RDONLY:
                } else {
                        switch (flags & O_ACCMODE) {
                        case O_RDONLY:
-                               SET(t, BTF_RDONLY);
+                               SET(t, R_RDONLY);
                                break;
                        case O_RDWR:
                                break;
                                break;
                        case O_RDWR:
                                break;
@@ -118,23 +147,28 @@ slow:                     if ((t->bt_rfp = fdopen(rfd, "r")) == NULL)
 
                        if (fstat(rfd, &sb))
                                goto err;
 
                        if (fstat(rfd, &sb))
                                goto err;
-                       if (sb.st_size > (off_t)SIZE_T_MAX) {
-                               errno = EFBIG;
-                               goto err;
-                       }
+                       /*
+                        * Kluge -- we'd like to test to see if the file is too
+                        * big to mmap.  Since, we don't know what size or type
+                        * off_t's or size_t's are, what the largest unsigned
+                        * integral type is, or what random insanity the local
+                        * C compiler will perpetrate, doing the comparison in
+                        * a portable way is flatly impossible.  Hope that mmap
+                        * fails if the file is too large.
+                        */
                        if (sb.st_size == 0)
                        if (sb.st_size == 0)
-                               SET(t, BTF_EOF);
+                               SET(t, R_EOF);
                        else {
                                t->bt_msize = sb.st_size;
                        else {
                                t->bt_msize = sb.st_size;
-                               if ((t->bt_smap =
-                                   mmap(NULL, t->bt_msize, PROT_READ, 0, rfd,
+                               if ((t->bt_smap = mmap(NULL, t->bt_msize,
+                                   PROT_READ, MAP_PRIVATE, rfd,
                                    (off_t)0)) == (caddr_t)-1)
                                        goto slow;
                                t->bt_cmap = t->bt_smap;
                                t->bt_emap = t->bt_smap + sb.st_size;
                                    (off_t)0)) == (caddr_t)-1)
                                        goto slow;
                                t->bt_cmap = t->bt_smap;
                                t->bt_emap = t->bt_smap + sb.st_size;
-                               t->bt_irec = ISSET(t, BTF_FIXEDLEN) ?
+                               t->bt_irec = ISSET(t, R_FIXLEN) ?
                                    __rec_fmap : __rec_vmap;
                                    __rec_fmap : __rec_vmap;
-                               SET(t, BTF_MEMMAPPED);
+                               SET(t, R_MEMMAPPED);
                        }
                }
        }
                        }
                }
        }
@@ -142,6 +176,7 @@ slow:                       if ((t->bt_rfp = fdopen(rfd, "r")) == NULL)
        /* Use the recno routines. */
        dbp->close = __rec_close;
        dbp->del = __rec_delete;
        /* Use the recno routines. */
        dbp->close = __rec_close;
        dbp->del = __rec_delete;
+       dbp->fd = __rec_fd;
        dbp->get = __rec_get;
        dbp->put = __rec_put;
        dbp->seq = __rec_seq;
        dbp->get = __rec_get;
        dbp->put = __rec_put;
        dbp->seq = __rec_seq;
@@ -157,7 +192,7 @@ slow:                       if ((t->bt_rfp = fdopen(rfd, "r")) == NULL)
                mpool_put(t->bt_mp, h, 0);
 
        if (openinfo && openinfo->flags & R_SNAPSHOT &&
                mpool_put(t->bt_mp, h, 0);
 
        if (openinfo && openinfo->flags & R_SNAPSHOT &&
-           !ISSET(t, BTF_EOF | BTF_RINMEM) &&
+           !ISSET(t, R_EOF | R_INMEM) &&
            t->bt_irec(t, MAX_REC_NUMBER) == RET_ERROR)
                 goto err;
        return (dbp);
            t->bt_irec(t, MAX_REC_NUMBER) == RET_ERROR)
                 goto err;
        return (dbp);
@@ -171,3 +206,25 @@ err:       sverrno = errno;
        errno = sverrno;
        return (NULL);
 }
        errno = sverrno;
        return (NULL);
 }
+
+int
+__rec_fd(dbp)
+       const DB *dbp;
+{
+       BTREE *t;
+
+       t = dbp->internal;
+
+       /* Toss any page pinned across calls. */
+       if (t->bt_pinned != NULL) {
+               mpool_put(t->bt_mp, t->bt_pinned, 0);
+               t->bt_pinned = NULL;
+       }
+
+       /* In-memory database can't have a file descriptor. */
+       if (ISSET(t, R_INMEM)) {
+               errno = ENOENT;
+               return (-1);
+       }
+       return (t->bt_rfd);
+}