OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [tags/] [gnu-src/] [newlib-1.18.0/] [newlib-1.18.0-or32-1.0rc1/] [newlib/] [libc/] [posix/] [telldir.c] - Diff between revs 207 and 345

Go to most recent revision | Only display areas with differences | Details | Blame | View Log

Rev 207 Rev 345
#ifndef HAVE_OPENDIR
#ifndef HAVE_OPENDIR
 
 
/*
/*
 * Copyright (c) 1983 Regents of the University of California.
 * Copyright (c) 1983 Regents of the University of California.
 * All rights reserved.
 * All rights reserved.
 *
 *
 * Redistribution and use in source and binary forms, with or without
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * modification, are permitted provided that the following conditions
 * are met:
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *    documentation and/or other materials provided with the distribution.
 * 3. All advertising materials mentioning features or use of this software
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgement:
 *    must display the following acknowledgement:
 *      This product includes software developed by the University of
 *      This product includes software developed by the University of
 *      California, Berkeley and its contributors.
 *      California, Berkeley and its contributors.
 * 4. Neither the name of the University nor the names of 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
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *    without specific prior written permission.
 *
 *
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * 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
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * SUCH DAMAGE.
 */
 */
 
 
#if defined(LIBC_SCCS) && !defined(lint)
#if defined(LIBC_SCCS) && !defined(lint)
static char sccsid[] = "@(#)telldir.c   5.9 (Berkeley) 2/23/91";
static char sccsid[] = "@(#)telldir.c   5.9 (Berkeley) 2/23/91";
#endif /* LIBC_SCCS and not lint */
#endif /* LIBC_SCCS and not lint */
 
 
#include <sys/param.h>
#include <sys/param.h>
#include <dirent.h>
#include <dirent.h>
#include <stdlib.h>
#include <stdlib.h>
#include <unistd.h>
#include <unistd.h>
#include <sys/lock.h>
#include <sys/lock.h>
 
 
/*
/*
 * The option SINGLEUSE may be defined to say that a telldir
 * The option SINGLEUSE may be defined to say that a telldir
 * cookie may be used only once before it is freed. This option
 * cookie may be used only once before it is freed. This option
 * is used to avoid having memory usage grow without bound.
 * is used to avoid having memory usage grow without bound.
 */
 */
#define SINGLEUSE
#define SINGLEUSE
 
 
/*
/*
 * One of these structures is malloced to describe the current directory
 * One of these structures is malloced to describe the current directory
 * position each time telldir is called. It records the current magic
 * position each time telldir is called. It records the current magic
 * cookie returned by getdirentries and the offset within the buffer
 * cookie returned by getdirentries and the offset within the buffer
 * associated with that return value.
 * associated with that return value.
 */
 */
struct ddloc {
struct ddloc {
        struct  ddloc *loc_next;/* next structure in list */
        struct  ddloc *loc_next;/* next structure in list */
        long    loc_index;      /* key associated with structure */
        long    loc_index;      /* key associated with structure */
        long    loc_seek;       /* magic cookie returned by getdirentries */
        long    loc_seek;       /* magic cookie returned by getdirentries */
        long    loc_loc;        /* offset of entry in buffer */
        long    loc_loc;        /* offset of entry in buffer */
        DIR    *loc_dirp;       /* DIR pointer */
        DIR    *loc_dirp;       /* DIR pointer */
};
};
 
 
#define NDIRHASH        32      /* Num of hash lists, must be a power of 2 */
#define NDIRHASH        32      /* Num of hash lists, must be a power of 2 */
#define LOCHASH(i)      ((i)&(NDIRHASH-1))
#define LOCHASH(i)      ((i)&(NDIRHASH-1))
 
 
static long     dd_loccnt;      /* Index of entry for sequential readdir's */
static long     dd_loccnt;      /* Index of entry for sequential readdir's */
static struct   ddloc *dd_hash[NDIRHASH];   /* Hash list heads for ddlocs */
static struct   ddloc *dd_hash[NDIRHASH];   /* Hash list heads for ddlocs */
__LOCK_INIT(static, dd_hash_lock);
__LOCK_INIT(static, dd_hash_lock);
 
 
/*
/*
 * return a pointer into a directory
 * return a pointer into a directory
 */
 */
 
 
#if !defined(_ELIX_LEVEL) || (_ELIX_LEVEL >= 2)
#if !defined(_ELIX_LEVEL) || (_ELIX_LEVEL >= 2)
 
 
long
long
_DEFUN(telldir, (dirp),
_DEFUN(telldir, (dirp),
       DIR *dirp)
       DIR *dirp)
{
{
        register int index;
        register int index;
        register struct ddloc *lp;
        register struct ddloc *lp;
 
 
        if ((lp = (struct ddloc *)malloc(sizeof(struct ddloc))) == NULL)
        if ((lp = (struct ddloc *)malloc(sizeof(struct ddloc))) == NULL)
                return (-1);
                return (-1);
 
 
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
        __lock_acquire_recursive(dirp->dd_lock);
        __lock_acquire_recursive(dirp->dd_lock);
        __lock_acquire(dd_hash_lock);
        __lock_acquire(dd_hash_lock);
#endif
#endif
        index = dd_loccnt++;
        index = dd_loccnt++;
        lp->loc_index = index;
        lp->loc_index = index;
        lp->loc_seek = dirp->dd_seek;
        lp->loc_seek = dirp->dd_seek;
        lp->loc_loc = dirp->dd_loc;
        lp->loc_loc = dirp->dd_loc;
        lp->loc_dirp = dirp;
        lp->loc_dirp = dirp;
        lp->loc_next = dd_hash[LOCHASH(index)];
        lp->loc_next = dd_hash[LOCHASH(index)];
        dd_hash[LOCHASH(index)] = lp;
        dd_hash[LOCHASH(index)] = lp;
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
        __lock_release(dd_hash_lock);
        __lock_release(dd_hash_lock);
        __lock_release_recursive(dirp->dd_lock);
        __lock_release_recursive(dirp->dd_lock);
#endif
#endif
        return (index);
        return (index);
}
}
 
 
#endif /* !_ELIX_LEVEL || _ELIX_LEVEL >= 2 */
#endif /* !_ELIX_LEVEL || _ELIX_LEVEL >= 2 */
 
 
/*
/*
 * seek to an entry in a directory.
 * seek to an entry in a directory.
 * Only values returned by "telldir" should be passed to seekdir.
 * Only values returned by "telldir" should be passed to seekdir.
 */
 */
void
void
_DEFUN(_seekdir, (dirp, loc),
_DEFUN(_seekdir, (dirp, loc),
        register DIR *dirp _AND
        register DIR *dirp _AND
        long loc)
        long loc)
{
{
        register struct ddloc *lp;
        register struct ddloc *lp;
        register struct ddloc **prevlp;
        register struct ddloc **prevlp;
        struct dirent *dp;
        struct dirent *dp;
        extern long lseek();
        extern long lseek();
 
 
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
        __lock_acquire(dd_hash_lock);
        __lock_acquire(dd_hash_lock);
#endif
#endif
        prevlp = &dd_hash[LOCHASH(loc)];
        prevlp = &dd_hash[LOCHASH(loc)];
        lp = *prevlp;
        lp = *prevlp;
        while (lp != NULL) {
        while (lp != NULL) {
                if (lp->loc_index == loc)
                if (lp->loc_index == loc)
                        break;
                        break;
                prevlp = &lp->loc_next;
                prevlp = &lp->loc_next;
                lp = lp->loc_next;
                lp = lp->loc_next;
        }
        }
        if (lp == NULL) {
        if (lp == NULL) {
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
                __lock_release(dd_hash_lock);
                __lock_release(dd_hash_lock);
#endif
#endif
                return;
                return;
        }
        }
        if (lp->loc_loc == dirp->dd_loc && lp->loc_seek == dirp->dd_seek)
        if (lp->loc_loc == dirp->dd_loc && lp->loc_seek == dirp->dd_seek)
                goto found;
                goto found;
        (void) lseek(dirp->dd_fd, lp->loc_seek, 0);
        (void) lseek(dirp->dd_fd, lp->loc_seek, 0);
        dirp->dd_seek = lp->loc_seek;
        dirp->dd_seek = lp->loc_seek;
        dirp->dd_loc = 0;
        dirp->dd_loc = 0;
        while (dirp->dd_loc < lp->loc_loc) {
        while (dirp->dd_loc < lp->loc_loc) {
                dp = readdir(dirp);
                dp = readdir(dirp);
                if (dp == NULL)
                if (dp == NULL)
                        break;
                        break;
        }
        }
found:
found:
#ifdef SINGLEUSE
#ifdef SINGLEUSE
        *prevlp = lp->loc_next;
        *prevlp = lp->loc_next;
        free((caddr_t)lp);
        free((caddr_t)lp);
#endif
#endif
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
        __lock_release(dd_hash_lock);
        __lock_release(dd_hash_lock);
#endif
#endif
}
}
 
 
/* clean out any hash entries from a closed directory */
/* clean out any hash entries from a closed directory */
void
void
_DEFUN(_cleanupdir, (dirp),
_DEFUN(_cleanupdir, (dirp),
        register DIR *dirp)
        register DIR *dirp)
{
{
        int i;
        int i;
 
 
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
        __lock_acquire(dd_hash_lock);
        __lock_acquire(dd_hash_lock);
#endif
#endif
        for (i = 0; i < NDIRHASH; ++i) {
        for (i = 0; i < NDIRHASH; ++i) {
                register struct ddloc *lp;
                register struct ddloc *lp;
                register struct ddloc *prevlp;
                register struct ddloc *prevlp;
                lp = dd_hash[i];
                lp = dd_hash[i];
                while (lp != NULL && lp->loc_dirp == dirp) {
                while (lp != NULL && lp->loc_dirp == dirp) {
                        dd_hash[i] = lp->loc_next;
                        dd_hash[i] = lp->loc_next;
                        prevlp = lp;
                        prevlp = lp;
                        free((caddr_t)lp);
                        free((caddr_t)lp);
                        lp = prevlp->loc_next;
                        lp = prevlp->loc_next;
                }
                }
                prevlp = lp;
                prevlp = lp;
                while (lp != NULL) {
                while (lp != NULL) {
                        lp = lp->loc_next;
                        lp = lp->loc_next;
                        if (lp != NULL && lp->loc_dirp == dirp) {
                        if (lp != NULL && lp->loc_dirp == dirp) {
                                prevlp->loc_next = lp->loc_next;
                                prevlp->loc_next = lp->loc_next;
                                free((caddr_t)lp);
                                free((caddr_t)lp);
                                lp = prevlp;
                                lp = prevlp;
                        }
                        }
                        else
                        else
                                prevlp = lp;
                                prevlp = lp;
                }
                }
        }
        }
#ifdef HAVE_DD_LOCK
#ifdef HAVE_DD_LOCK
        __lock_release(dd_hash_lock);
        __lock_release(dd_hash_lock);
#endif
#endif
 
 
}
}
#endif /* ! HAVE_OPENDIR */
#endif /* ! HAVE_OPENDIR */
 
 

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.