223 lines
4.9 KiB
C
223 lines
4.9 KiB
C
/*
|
|
* $Id: buildindex.c,v 1.2 1999/02/14 09:50:42 dirk Exp $
|
|
*
|
|
* This file is part of WorkMan, the civilized CD player library
|
|
* (c) 1991-1997 by Steven Grimm (original author)
|
|
* (c) by Dirk Försterling (current 'author' = maintainer)
|
|
* The maintainer can be contacted by his e-mail address:
|
|
* milliByte@DeathsDoor.com
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the Free
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
*
|
|
* Build a WorkMan database index file from a flat text file. Requires
|
|
* 4.4BSD libdb library.
|
|
*/
|
|
|
|
static char buildindex_id[] = "$Id: buildindex.c,v 1.2 1999/02/14 09:50:42 dirk Exp $";
|
|
|
|
#include <stdio.h>
|
|
#include <db.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <netinet/in.h> /* for htonl() */
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
static char buildindex_id[]="$Id: buildindex.c,v 1.2 1999/02/14 09:50:42 dirk Exp $";
|
|
|
|
char *strrchr();
|
|
|
|
main(argc, argv)
|
|
int argc;
|
|
char **argv;
|
|
{
|
|
DB *db;
|
|
DBT key, data;
|
|
FILE *fp;
|
|
int lock = 1, i = 0, locked, frame;
|
|
char buf[1000], indname[MAXPATHLEN + 100], framebuf[8], *c;
|
|
unsigned long pos;
|
|
BTREEINFO bt;
|
|
struct stat st;
|
|
|
|
if (argc > 2 && !strcmp(argv[1], "-n"))
|
|
{
|
|
lock = 0;
|
|
i++;
|
|
}
|
|
|
|
if (argc < i + 2)
|
|
{
|
|
fprintf(stderr, "Usage: %s [-n] dbfile [dbfile ...]\n",
|
|
argv[0]);
|
|
exit(1);
|
|
}
|
|
|
|
data.data = &pos;
|
|
data.size = sizeof(pos);
|
|
key.data = framebuf;
|
|
key.size = 7; /* %07d */
|
|
|
|
while (++i < argc)
|
|
{
|
|
fprintf(stderr, "Building index for %s\n", argv[i]);
|
|
|
|
if ((fp = fopen(argv[i], "r")) == NULL)
|
|
{
|
|
perror(argv[i]);
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* Figure out the file's mode, uid, gid, so we can set the
|
|
* permissions on the index file to the same thing.
|
|
*/
|
|
if (fstat(fileno(fp), &st))
|
|
{
|
|
sprintf(indname, "%s: fstat", argv[i]);
|
|
perror(indname);
|
|
fclose(fp);
|
|
continue;
|
|
}
|
|
|
|
if (lock && lockit(fileno(fp), F_WRLCK))
|
|
{
|
|
sprintf(indname, "%s: Warning: Couldn't lock", argv[i]);
|
|
perror(indname);
|
|
locked = 0;
|
|
}
|
|
else
|
|
locked = 1;
|
|
|
|
/*
|
|
* Create a database file.
|
|
*/
|
|
bt.flags = R_DUP; /* allow duplicate keys */
|
|
bt.cachesize = 0;
|
|
bt.psize = 0;
|
|
bt.lorder = 4321;
|
|
bt.minkeypage = 0;
|
|
bt.compare = NULL; /* use lexical comparison */
|
|
bt.prefix = NULL; /* no prefix comparisons */
|
|
|
|
/* Index files have ".ind" extensions */
|
|
sprintf(indname, "%s.ind", argv[i]);
|
|
if ((db = dbopen(indname, O_CREAT | O_RDWR | O_TRUNC,
|
|
st.st_mode, DB_BTREE, &bt)) == NULL)
|
|
{
|
|
perror(indname);
|
|
if (locked)
|
|
lockit(fileno(fp), F_UNLCK);
|
|
fclose(fp);
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* Now loop through the text file, inserting a record into
|
|
* the index file for each "tracks" line.
|
|
*/
|
|
while (! feof(fp))
|
|
{
|
|
pos = ftell(fp);
|
|
buf[0] = '\0';
|
|
if (fgets(buf, sizeof(buf), fp) == NULL || ! buf[0])
|
|
{
|
|
/* End of file? */
|
|
if (feof(fp))
|
|
break;
|
|
|
|
/* Nope. A read error. Unlink the database. */
|
|
perror(argv[i]);
|
|
(void) unlink(indname);
|
|
break;
|
|
}
|
|
|
|
if (strncmp(buf, "tracks ", 7))
|
|
continue;
|
|
|
|
/*
|
|
* Found the start of a record. Figure out the start
|
|
* time of the last track and put an entry in the
|
|
* index file with that as the key.
|
|
*/
|
|
c = strrchr(buf, ' '); /* this will always succeed */
|
|
*c = '\0';
|
|
c = strrchr(buf, ' '); /* this should too, but... */
|
|
if (c == NULL)
|
|
{
|
|
fprintf(stderr,
|
|
"%s: Malformed tracks line at %lu\n",
|
|
argv[i], pos);
|
|
continue;
|
|
}
|
|
sscanf(c+1, "%d", &frame);
|
|
sprintf(framebuf, "%07d", frame);
|
|
pos = htonl(pos);
|
|
|
|
if ((db->put)(db, &key, &data, 0))
|
|
{
|
|
perror(indname);
|
|
unlink(indname);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Clean up.
|
|
*/
|
|
(void) (db->close)(db);
|
|
if (locked)
|
|
lockit(fileno(fp), F_UNLCK);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Lock a file. Time out after a little while if we can't get a lock;
|
|
* this usually means the locking system is broken.
|
|
*
|
|
* Unfortunately, if there are lots of people contending for a lock,
|
|
* this can result in the file not getting locked when it probably should.
|
|
*/
|
|
int
|
|
lockit(fd, type)
|
|
int fd;
|
|
int type;
|
|
{
|
|
struct flock fl;
|
|
int result, timer = 0;
|
|
|
|
fl.l_type = type;
|
|
fl.l_whence = 0;
|
|
fl.l_start = 0;
|
|
fl.l_len = 0;
|
|
|
|
while ((result = fcntl(fd, F_SETLK, &fl)) < 0)
|
|
{
|
|
if (errno != EACCES || errno != EAGAIN)
|
|
break;
|
|
if (timer++ == 30)
|
|
{
|
|
errno = ETIMEDOUT;
|
|
break;
|
|
}
|
|
|
|
sleep(1);
|
|
}
|
|
|
|
return (result);
|
|
}
|