mirror of
git://git.code.sf.net/p/zsh/code
synced 2025-01-17 10:20:55 +01:00
320 lines
7.4 KiB
C
320 lines
7.4 KiB
C
/*
|
|
* mapfile.c - associative array interface to external files
|
|
*
|
|
* This file is part of zsh, the Z shell.
|
|
*
|
|
* Copyright (c) 1999 Sven Wischnowsky, Peter Stephenson
|
|
* All rights reserved.
|
|
*
|
|
* Permission is hereby granted, without written agreement and without
|
|
* license or royalty fees, to use, copy, modify, and distribute this
|
|
* software and to distribute modified versions of this software for any
|
|
* purpose, provided that the above copyright notice and the following
|
|
* two paragraphs appear in all copies of this software.
|
|
*
|
|
* In no event shall Sven Wischnowsky, Peter Stephenson or the Zsh Development
|
|
* Group be liable to any party for direct, indirect, special, incidental, or
|
|
* consequential damages arising out of the use of this software and its
|
|
* documentation, even if Peter Stephenson, Sven Wischnowsky and the Zsh
|
|
* Development Group have been advised of the possibility of such damage.
|
|
*
|
|
* Peter Stephenson, Sven Wischnowsky and the Zsh Development Group
|
|
* specifically disclaim any warranties, including, but not limited to, the
|
|
* implied warranties of merchantability and fitness for a particular purpose.
|
|
* The softwareprovided hereunder is on an "as is" basis, and Peter
|
|
* Stephenson, Sven Wischnowsky and the Zsh Development Group have no
|
|
* obligation to provide maintenance, support, updates, enhancements, or
|
|
* modifications.
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* To do: worry about when keys of associative arrays get unmeta'd.
|
|
*/
|
|
#include "mapfile.mdh"
|
|
#include "mapfile.pro"
|
|
|
|
/*
|
|
* Make sure we have all the bits I'm using for memory mapping, otherwise
|
|
* I don't know what I'm doing.
|
|
*/
|
|
#if defined(HAVE_SYS_MMAN_H) && defined(HAVE_FTRUNCATE)
|
|
#if defined(HAVE_MMAP) && defined(HAVE_MUNMAP) && defined(HAVE_MSYNC)
|
|
#define USE_MMAP 1
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#if !defined(MAP_VARIABLE)
|
|
#define MAP_VARIABLE 0
|
|
#endif
|
|
#if !defined(MAP_FILE)
|
|
#define MAP_FILE 0
|
|
#endif
|
|
#if !defined(MAP_NORESERVE)
|
|
#define MAP_NORESERVE 0
|
|
#endif
|
|
#define MMAP_ARGS (MAP_FILE | MAP_VARIABLE | MAP_SHARED | MAP_NORESERVE)
|
|
|
|
#endif /* HAVE_MMAP && HAVE_MUNMAP && HAVE_MSYNC */
|
|
#endif /* HAVE_SYS_MMAN_H && HAVE_FTRUNCATE */
|
|
|
|
static const struct gsu_hash mapfiles_gsu =
|
|
{ hashgetfn, setpmmapfiles, stdunsetfn };
|
|
|
|
/* Functions for the options special parameter. */
|
|
|
|
/**/
|
|
static void
|
|
setpmmapfile(Param pm, char *value)
|
|
{
|
|
int fd = -1, len;
|
|
char *name = ztrdup(pm->node.nam);
|
|
#ifdef USE_MMAP
|
|
caddr_t mmptr;
|
|
#else
|
|
FILE *fout;
|
|
#endif
|
|
|
|
/*
|
|
* First unmetafy the value, and the name since we don't
|
|
* where it's been.
|
|
*/
|
|
unmetafy(name, &len);
|
|
unmetafy(value, &len);
|
|
|
|
/* Open the file for writing */
|
|
#ifdef USE_MMAP
|
|
if (!(pm->node.flags & PM_READONLY) &&
|
|
(fd = open(name, O_RDWR|O_CREAT|O_NOCTTY, 0666)) >= 0 &&
|
|
(mmptr = (caddr_t)mmap((caddr_t)0, len, PROT_READ | PROT_WRITE,
|
|
MMAP_ARGS, fd, (off_t)0)) != (caddr_t)-1) {
|
|
/*
|
|
* First we need to make sure the file is long enough for
|
|
* when we msync. On AIX, at least, we just get zeroes otherwise.
|
|
*/
|
|
if (ftruncate(fd, len) < 0)
|
|
zwarn("ftruncate failed: %e", errno);
|
|
memcpy(mmptr, value, len);
|
|
#ifndef MS_SYNC
|
|
#define MS_SYNC 0
|
|
#endif
|
|
msync(mmptr, len, MS_SYNC);
|
|
/*
|
|
* Then we need to truncate again, since mmap() always maps complete
|
|
* pages. Honestly, I tried it without, and you need both.
|
|
*/
|
|
if (ftruncate(fd, len) < 0)
|
|
zwarn("ftruncate failed: %e", errno);
|
|
munmap(mmptr, len);
|
|
}
|
|
#else /* don't USE_MMAP */
|
|
/* can't be bothered to do anything too clever here */
|
|
if ((fout = fopen(name, "w"))) {
|
|
while (len--)
|
|
putc(*value++, fout);
|
|
fclose(fout);
|
|
}
|
|
#endif /* USE_MMAP */
|
|
if (fd >= 0)
|
|
close(fd);
|
|
free(name);
|
|
free(value);
|
|
}
|
|
|
|
/**/
|
|
static void
|
|
unsetpmmapfile(Param pm, UNUSED(int exp))
|
|
{
|
|
/* Unlink the file given by pm->nam */
|
|
char *fname = ztrdup(pm->node.nam);
|
|
int dummy;
|
|
unmetafy(fname, &dummy);
|
|
|
|
if (!(pm->node.flags & PM_READONLY))
|
|
unlink(fname);
|
|
|
|
free(fname);
|
|
}
|
|
|
|
/**/
|
|
static void
|
|
setpmmapfiles(Param pm, HashTable ht)
|
|
{
|
|
int i;
|
|
HashNode hn;
|
|
|
|
if (!ht)
|
|
return;
|
|
|
|
if (!(pm->node.flags & PM_READONLY))
|
|
for (i = 0; i < ht->hsize; i++)
|
|
for (hn = ht->nodes[i]; hn; hn = hn->next) {
|
|
struct value v;
|
|
|
|
v.isarr = v.flags = v.start = 0;
|
|
v.end = -1;
|
|
v.arr = NULL;
|
|
v.pm = (Param) hn;
|
|
|
|
setpmmapfile(v.pm, ztrdup(getstrvalue(&v)));
|
|
}
|
|
if (ht != pm->u.hash)
|
|
deleteparamtable(ht);
|
|
}
|
|
|
|
/**/
|
|
static char *
|
|
get_contents(char *fname)
|
|
{
|
|
int fd;
|
|
#ifdef USE_MMAP
|
|
caddr_t mmptr;
|
|
struct stat sbuf;
|
|
#endif
|
|
char *val;
|
|
unmetafy(fname = ztrdup(fname), &fd);
|
|
|
|
#ifdef USE_MMAP
|
|
if ((fd = open(fname, O_RDONLY | O_NOCTTY)) < 0 ||
|
|
fstat(fd, &sbuf) ||
|
|
(mmptr = (caddr_t)mmap((caddr_t)0, sbuf.st_size, PROT_READ,
|
|
MMAP_ARGS, fd, (off_t)0)) == (caddr_t)-1) {
|
|
if (fd >= 0)
|
|
close(fd);
|
|
free(fname);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Sadly, we need to copy the thing even if metafying doesn't
|
|
* change it. We just don't know when we might get a chance to
|
|
* munmap it, otherwise.
|
|
*/
|
|
val = metafy((char *)mmptr, sbuf.st_size, META_HEAPDUP);
|
|
|
|
munmap(mmptr, sbuf.st_size);
|
|
close(fd);
|
|
#else /* don't USE_MMAP */
|
|
val = NULL;
|
|
if ((fd = open(fname, O_RDONLY | O_NOCTTY)) >= 0) {
|
|
LinkList ll;
|
|
|
|
if ((ll = readoutput(fd, 1, 0)))
|
|
val = peekfirst(ll);
|
|
}
|
|
#endif /* USE_MMAP */
|
|
free(fname);
|
|
return val;
|
|
}
|
|
|
|
static const struct gsu_scalar mapfile_gsu =
|
|
{ strgetfn, setpmmapfile, unsetpmmapfile };
|
|
|
|
static struct paramdef partab[] = {
|
|
SPECIALPMDEF("mapfile", 0, &mapfiles_gsu, getpmmapfile, scanpmmapfile)
|
|
};
|
|
|
|
/**/
|
|
static HashNode
|
|
getpmmapfile(UNUSED(HashTable ht), const char *name)
|
|
{
|
|
char *contents;
|
|
Param pm = NULL;
|
|
|
|
pm = (Param) hcalloc(sizeof(struct param));
|
|
pm->node.nam = dupstring(name);
|
|
pm->node.flags = PM_SCALAR;
|
|
pm->gsu.s = &mapfile_gsu;
|
|
pm->node.flags |= (partab[0].pm->node.flags & PM_READONLY);
|
|
|
|
/* Set u.str to contents of file given by name */
|
|
if ((contents = get_contents(pm->node.nam)))
|
|
pm->u.str = contents;
|
|
else {
|
|
pm->u.str = "";
|
|
pm->node.flags |= PM_UNSET;
|
|
}
|
|
return &pm->node;
|
|
}
|
|
|
|
|
|
/**/
|
|
static void
|
|
scanpmmapfile(UNUSED(HashTable ht), ScanFunc func, int flags)
|
|
{
|
|
struct param pm;
|
|
DIR *dir;
|
|
|
|
if (!(dir = opendir(".")))
|
|
return;
|
|
|
|
memset((void *)&pm, 0, sizeof(struct param));
|
|
pm.node.flags = PM_SCALAR;
|
|
pm.gsu.s = &mapfile_gsu;
|
|
pm.node.flags |= (partab[0].pm->node.flags & PM_READONLY);
|
|
|
|
/* Here we scan the current directory, calling func() for each file */
|
|
while ((pm.node.nam = zreaddir(dir, 1))) {
|
|
/*
|
|
* Hmmm, it's rather wasteful always to read the contents.
|
|
* In fact, it's grotesequely wasteful, since that would mean
|
|
* we always read the entire contents of every single file
|
|
* in the directory into memory. Hence just leave it empty.
|
|
*/
|
|
pm.node.nam = dupstring(pm.node.nam);
|
|
pm.u.str = "";
|
|
func(&pm.node, flags);
|
|
}
|
|
closedir(dir);
|
|
}
|
|
|
|
static struct features module_features = {
|
|
NULL, 0,
|
|
NULL, 0,
|
|
NULL, 0,
|
|
partab, sizeof(partab)/sizeof(*partab),
|
|
0
|
|
};
|
|
|
|
/**/
|
|
int
|
|
setup_(UNUSED(Module m))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/**/
|
|
int
|
|
features_(Module m, char ***features)
|
|
{
|
|
*features = featuresarray(m, &module_features);
|
|
return 0;
|
|
}
|
|
|
|
/**/
|
|
int
|
|
enables_(Module m, int **enables)
|
|
{
|
|
return handlefeatures(m, &module_features, enables);
|
|
}
|
|
|
|
/**/
|
|
int
|
|
boot_(UNUSED(Module m))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/**/
|
|
int
|
|
cleanup_(Module m)
|
|
{
|
|
return setfeatureenables(m, &module_features, NULL);
|
|
}
|
|
|
|
/**/
|
|
int
|
|
finish_(UNUSED(Module m))
|
|
{
|
|
return 0;
|
|
}
|