import libterminfo, terminfo.db and stat command
This commit is contained in:
@@ -3,7 +3,7 @@
|
||||
.include <bsd.own.mk>
|
||||
|
||||
# NetBSD imports
|
||||
SUBDIR= indent m4 mkimage stat
|
||||
SUBDIR= indent m4 mkimage stat tic
|
||||
|
||||
# Non-NetBSD imports
|
||||
SUBDIR+= ministat
|
||||
|
||||
13
usr.bin/tic/Makefile
Normal file
13
usr.bin/tic/Makefile
Normal file
@@ -0,0 +1,13 @@
|
||||
# $NetBSD: Makefile,v 1.1 2010/02/03 15:16:32 roy Exp $
|
||||
|
||||
PROG= tic
|
||||
WARNS= 4
|
||||
|
||||
CPPFLAGS+= -I${.CURDIR}/../../lib/libterminfo
|
||||
|
||||
.ifndef HOSTPROG
|
||||
LDADD+= -lterminfo
|
||||
DPADD+= ${LIBTERMINFO}
|
||||
.endif
|
||||
|
||||
.include <bsd.prog.mk>
|
||||
0
usr.bin/tic/minix-port.patch
Normal file
0
usr.bin/tic/minix-port.patch
Normal file
105
usr.bin/tic/tic.1
Normal file
105
usr.bin/tic/tic.1
Normal file
@@ -0,0 +1,105 @@
|
||||
.\" $NetBSD: tic.1,v 1.8 2010/02/11 13:44:14 wiz Exp $
|
||||
.\"
|
||||
.\" Copyright (c) 2009, 2010 The NetBSD Foundation, Inc.
|
||||
.\" All rights reserved.
|
||||
.\"
|
||||
.\" This code is derived from software contributed to The NetBSD Foundation
|
||||
.\" by Roy Marples.
|
||||
.\"
|
||||
.\" 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.
|
||||
.\"
|
||||
.\" THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
|
||||
.\"
|
||||
.Dd February 11, 2010
|
||||
.Dt TIC 1
|
||||
.Os
|
||||
.Sh NAME
|
||||
.Nm tic
|
||||
.Nd terminfo compiler
|
||||
.Sh SYNOPSIS
|
||||
.Nm tic
|
||||
.Op Fl acSsx
|
||||
.Op Fl o Ar file
|
||||
.Ar source
|
||||
.Op Ar term1 term2 ...
|
||||
.Sh DESCRIPTION
|
||||
The
|
||||
.Nm
|
||||
utility compiles
|
||||
.Xr terminfo 5
|
||||
source into a database for use by other programs.
|
||||
The created database path name is the same as the source but with .db appended.
|
||||
.Pp
|
||||
The following options are available:
|
||||
.Bl -tag -width Fl
|
||||
.It Fl a
|
||||
Do not discard commented out capabilities.
|
||||
.It Fl c
|
||||
Only check for errors, don't write the final database.
|
||||
.It Fl o Ar file
|
||||
Write the database to
|
||||
.Ar file Ns .db
|
||||
instead of
|
||||
.Ar source Ns .db .
|
||||
.It Fl S
|
||||
For
|
||||
.Ar term1 , term2 , ...
|
||||
output a C structure containing name, compiled description, and compiled size.
|
||||
This can be used to embed terminal descriptons into a program.
|
||||
.It Fl s
|
||||
Display the number of terminal descritions written to the database.
|
||||
.It Fl x
|
||||
Include non standard capabilities defined in the
|
||||
.Ar source .
|
||||
.El
|
||||
.Ss Extensions To Terminfo
|
||||
When
|
||||
.Nm
|
||||
discovers a
|
||||
.Sy use Ns = Ns Va term
|
||||
capability, the terminal description for
|
||||
.Va term
|
||||
is merged in.
|
||||
Capabilities do not overwrite previously disovered ones and capabilities
|
||||
ending with @ are marked as absent so the terminal does not inherit the
|
||||
capability from
|
||||
.Sy use Ns d
|
||||
terminals.
|
||||
.Sh EXIT STATUS
|
||||
.Ex -std tic
|
||||
.Sh EXAMPLES
|
||||
To maintain your private terminfo definitions, if the system supplied
|
||||
ones do not support your terminal:
|
||||
.Bd -literal -offset indent
|
||||
.Ic tic ~/.terminfo
|
||||
.Ed
|
||||
.Sh SEE ALSO
|
||||
.Xr infocmp 1 ,
|
||||
.Xr tput 1 ,
|
||||
.Xr curses 3 ,
|
||||
.Xr terminfo 5
|
||||
.Sh STANDARDS
|
||||
The
|
||||
.Nm
|
||||
utility works with terminfo files that conform to the
|
||||
.St -xcurses4.2
|
||||
standard.
|
||||
.Sh AUTHORS
|
||||
.An Roy Marples Aq roy@NetBSD.org
|
||||
577
usr.bin/tic/tic.c
Normal file
577
usr.bin/tic/tic.c
Normal file
@@ -0,0 +1,577 @@
|
||||
/* $NetBSD: tic.c,v 1.10 2010/02/22 23:05:39 roy Exp $ */
|
||||
|
||||
/*
|
||||
* Copyright (c) 2009, 2010 The NetBSD Foundation, Inc.
|
||||
*
|
||||
* This code is derived from software contributed to The NetBSD Foundation
|
||||
* by Roy Marples.
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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 HAVE_NBTOOL_CONFIG_H
|
||||
#include "nbtool_config.h"
|
||||
#endif
|
||||
|
||||
#include <sys/cdefs.h>
|
||||
__RCSID("$NetBSD: tic.c,v 1.10 2010/02/22 23:05:39 roy Exp $");
|
||||
|
||||
#include <sys/types.h>
|
||||
|
||||
#if !HAVE_NBTOOL_CONFIG_H || HAVE_SYS_ENDIAN_H
|
||||
#include <sys/endian.h>
|
||||
#endif
|
||||
|
||||
#include <ctype.h>
|
||||
#include <err.h>
|
||||
#include <errno.h>
|
||||
#include <getopt.h>
|
||||
#include <limits.h>
|
||||
#include <fcntl.h>
|
||||
#include <ndbm.h>
|
||||
#include <stdarg.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <term_private.h>
|
||||
#include <term.h>
|
||||
|
||||
/* We store the full list of terminals we have instead of iterating
|
||||
through the database as the sequential iterator doesn't work
|
||||
the the data size stored changes N amount which ours will. */
|
||||
typedef struct term {
|
||||
struct term *next;
|
||||
char *name;
|
||||
char type;
|
||||
TIC *tic;
|
||||
} TERM;
|
||||
static TERM *terms;
|
||||
|
||||
static int error_exit;
|
||||
static int Sflag;
|
||||
static char *dbname;
|
||||
|
||||
static void
|
||||
do_unlink(void)
|
||||
{
|
||||
|
||||
if (dbname != NULL)
|
||||
unlink(dbname);
|
||||
}
|
||||
|
||||
static void __attribute__((__format__(__printf__, 1, 2)))
|
||||
dowarn(const char *fmt, ...)
|
||||
{
|
||||
va_list va;
|
||||
|
||||
error_exit = 1;
|
||||
va_start(va, fmt);
|
||||
vwarnx(fmt, va);
|
||||
va_end(va);
|
||||
}
|
||||
|
||||
static char *
|
||||
grow_tbuf(TBUF *tbuf, size_t len)
|
||||
{
|
||||
char *buf;
|
||||
|
||||
buf = _ti_grow_tbuf(tbuf, len);
|
||||
if (buf == NULL)
|
||||
err(1, "_ti_grow_tbuf");
|
||||
return buf;
|
||||
}
|
||||
|
||||
static int
|
||||
save_term(DBM *db, TERM *term)
|
||||
{
|
||||
uint8_t *buf;
|
||||
ssize_t len;
|
||||
datum key, value;
|
||||
|
||||
len = _ti_flatten(&buf, term->tic);
|
||||
if (len == -1)
|
||||
return -1;
|
||||
|
||||
key.dptr = term->name;
|
||||
key.dsize = strlen(term->name);
|
||||
value.dptr = buf;
|
||||
value.dsize = len;
|
||||
if (dbm_store(db, key, value, DBM_REPLACE) == -1)
|
||||
err(1, "dbm_store");
|
||||
free(buf);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static TERM *
|
||||
find_term(const char *name)
|
||||
{
|
||||
TERM *term;
|
||||
|
||||
for (term = terms; term != NULL; term = term->next)
|
||||
if (strcmp(term->name, name) == 0)
|
||||
return term;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static TERM *
|
||||
store_term(const char *name, char type)
|
||||
{
|
||||
TERM *term;
|
||||
|
||||
term = calloc(1, sizeof(*term));
|
||||
if (term == NULL)
|
||||
errx(1, "malloc");
|
||||
term->name = strdup(name);
|
||||
term->type = type;
|
||||
if (term->name == NULL)
|
||||
errx(1, "malloc");
|
||||
term->next = terms;
|
||||
terms = term;
|
||||
return term;
|
||||
}
|
||||
|
||||
static int
|
||||
process_entry(TBUF *buf, int flags)
|
||||
{
|
||||
char *p, *e, *alias;
|
||||
TERM *term;
|
||||
TIC *tic;
|
||||
|
||||
if (buf->bufpos == 0)
|
||||
return 0;
|
||||
/* Terminate the string */
|
||||
buf->buf[buf->bufpos - 1] = '\0';
|
||||
/* First rewind the buffer for new entries */
|
||||
buf->bufpos = 0;
|
||||
|
||||
if (isspace((unsigned char)*buf->buf))
|
||||
return 0;
|
||||
|
||||
tic = _ti_compile(buf->buf, flags);
|
||||
if (tic == NULL)
|
||||
return 0;
|
||||
|
||||
if (find_term(tic->name) != NULL) {
|
||||
dowarn("%s: duplicate entry", tic->name);
|
||||
_ti_freetic(tic);
|
||||
return 0;
|
||||
}
|
||||
term = store_term(tic->name, 't');
|
||||
term->tic = tic;
|
||||
|
||||
/* Create aliased terms */
|
||||
if (tic->alias != NULL) {
|
||||
alias = p = strdup(tic->alias);
|
||||
while (p != NULL && *p != '\0') {
|
||||
e = strchr(p, '|');
|
||||
if (e != NULL)
|
||||
*e++ = '\0';
|
||||
if (find_term(p) != NULL) {
|
||||
dowarn("%s: has alias for already assigned"
|
||||
" term %s", tic->name, p);
|
||||
} else {
|
||||
term = store_term(p, 'a');
|
||||
term->tic = calloc(sizeof(*term->tic), 1);
|
||||
if (term->tic == NULL)
|
||||
err(1, "malloc");
|
||||
term->tic->name = strdup(tic->name);
|
||||
if (term->tic->name == NULL)
|
||||
err(1, "malloc");
|
||||
}
|
||||
p = e;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
merge(TIC *rtic, TIC *utic, int flags)
|
||||
{
|
||||
char *cap, flag, *code, type, *str;
|
||||
short ind, num;
|
||||
size_t n;
|
||||
|
||||
cap = utic->flags.buf;
|
||||
for (n = utic->flags.entries; n > 0; n--) {
|
||||
ind = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
flag = *cap++;
|
||||
if (VALID_BOOLEAN(flag) &&
|
||||
_ti_find_cap(&rtic->flags, 'f', ind) == NULL)
|
||||
{
|
||||
_ti_grow_tbuf(&rtic->flags, sizeof(uint16_t) + 1);
|
||||
le16enc(rtic->flags.buf + rtic->flags.bufpos, ind);
|
||||
rtic->flags.bufpos += sizeof(uint16_t);
|
||||
rtic->flags.buf[rtic->flags.bufpos++] = flag;
|
||||
rtic->flags.entries++;
|
||||
}
|
||||
}
|
||||
|
||||
cap = utic->nums.buf;
|
||||
for (n = utic->nums.entries; n > 0; n--) {
|
||||
ind = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
num = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
if (VALID_NUMERIC(num) &&
|
||||
_ti_find_cap(&rtic->nums, 'n', ind) == NULL)
|
||||
{
|
||||
grow_tbuf(&rtic->nums, sizeof(uint16_t) * 2);
|
||||
le16enc(rtic->nums.buf + rtic->nums.bufpos, ind);
|
||||
rtic->nums.bufpos += sizeof(uint16_t);
|
||||
le16enc(rtic->nums.buf + rtic->nums.bufpos, num);
|
||||
rtic->nums.bufpos += sizeof(uint16_t);
|
||||
rtic->nums.entries++;
|
||||
}
|
||||
}
|
||||
|
||||
cap = utic->strs.buf;
|
||||
for (n = utic->strs.entries; n > 0; n--) {
|
||||
ind = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
num = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
if (num > 0 &&
|
||||
_ti_find_cap(&rtic->strs, 's', ind) == NULL)
|
||||
{
|
||||
grow_tbuf(&rtic->strs, (sizeof(uint16_t) * 2) + num);
|
||||
le16enc(rtic->strs.buf + rtic->strs.bufpos, ind);
|
||||
rtic->strs.bufpos += sizeof(uint16_t);
|
||||
le16enc(rtic->strs.buf + rtic->strs.bufpos, num);
|
||||
rtic->strs.bufpos += sizeof(uint16_t);
|
||||
memcpy(rtic->strs.buf + rtic->strs.bufpos,
|
||||
cap, num);
|
||||
rtic->strs.bufpos += num;
|
||||
rtic->strs.entries++;
|
||||
}
|
||||
cap += num;
|
||||
}
|
||||
|
||||
cap = utic->extras.buf;
|
||||
for (n = utic->extras.entries; n > 0; n--) {
|
||||
num = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
code = cap;
|
||||
cap += num;
|
||||
type = *cap++;
|
||||
flag = 0;
|
||||
str = NULL;
|
||||
switch (type) {
|
||||
case 'f':
|
||||
flag = *cap++;
|
||||
if (!VALID_BOOLEAN(flag))
|
||||
continue;
|
||||
break;
|
||||
case 'n':
|
||||
num = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
if (!VALID_NUMERIC(num))
|
||||
continue;
|
||||
break;
|
||||
case 's':
|
||||
num = le16dec(cap);
|
||||
cap += sizeof(uint16_t);
|
||||
str = cap;
|
||||
cap += num;
|
||||
if (num == 0)
|
||||
continue;
|
||||
break;
|
||||
}
|
||||
_ti_store_extra(rtic, 0, code, type, flag, num, str, num,
|
||||
flags);
|
||||
}
|
||||
}
|
||||
|
||||
static size_t
|
||||
merge_use(int flags)
|
||||
{
|
||||
size_t skipped, merged, memn;
|
||||
char *cap, *scap;
|
||||
uint16_t num;
|
||||
TIC *rtic, *utic;
|
||||
TERM *term, *uterm;;
|
||||
|
||||
skipped = merged = 0;
|
||||
for (term = terms; term != NULL; term = term->next) {
|
||||
if (term->type == 'a')
|
||||
continue;
|
||||
rtic = term->tic;
|
||||
while ((cap = _ti_find_extra(&rtic->extras, "use")) != NULL) {
|
||||
if (*cap++ != 's') {
|
||||
dowarn("%s: use is not string", rtic->name);
|
||||
break;
|
||||
}
|
||||
cap += sizeof(uint16_t);
|
||||
if (strcmp(rtic->name, cap) == 0) {
|
||||
dowarn("%s: uses itself", rtic->name);
|
||||
goto remove;
|
||||
}
|
||||
uterm = find_term(cap);
|
||||
if (uterm != NULL && uterm->type == 'a')
|
||||
uterm = find_term(uterm->tic->name);
|
||||
if (uterm == NULL) {
|
||||
dowarn("%s: no use record for %s",
|
||||
rtic->name, cap);
|
||||
goto remove;
|
||||
}
|
||||
utic = uterm->tic;
|
||||
if (strcmp(utic->name, rtic->name) == 0) {
|
||||
dowarn("%s: uses itself", rtic->name);
|
||||
goto remove;
|
||||
}
|
||||
if (_ti_find_extra(&utic->extras, "use") != NULL) {
|
||||
skipped++;
|
||||
break;
|
||||
}
|
||||
cap = _ti_find_extra(&rtic->extras, "use");
|
||||
merge(rtic, utic, flags);
|
||||
remove:
|
||||
/* The pointers may have changed, find the use again */
|
||||
cap = _ti_find_extra(&rtic->extras, "use");
|
||||
if (cap == NULL)
|
||||
dowarn("%s: use no longer exists - impossible",
|
||||
rtic->name);
|
||||
else {
|
||||
scap = cap - (4 + sizeof(uint16_t));
|
||||
cap++;
|
||||
num = le16dec(cap);
|
||||
cap += sizeof(uint16_t) + num;
|
||||
memn = rtic->extras.bufpos -
|
||||
(cap - rtic->extras.buf);
|
||||
memcpy(scap, cap, memn);
|
||||
rtic->extras.bufpos -= cap - scap;
|
||||
cap = scap;
|
||||
rtic->extras.entries--;
|
||||
merged++;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (merged == 0 && skipped != 0)
|
||||
dowarn("circular use detected");
|
||||
return merged;
|
||||
}
|
||||
|
||||
static int
|
||||
print_dump(int argc, char **argv)
|
||||
{
|
||||
TERM *term;
|
||||
uint8_t *buf;
|
||||
int i, n;
|
||||
size_t j, col;
|
||||
ssize_t len;
|
||||
|
||||
printf("struct compiled_term {\n");
|
||||
printf("\tconst char *name;\n");
|
||||
printf("\tconst char *cap;\n");
|
||||
printf("\tsize_t caplen;\n");
|
||||
printf("};\n\n");
|
||||
|
||||
printf("const struct compiled_term compiled_terms[] = {\n");
|
||||
|
||||
n = 0;
|
||||
for (i = 0; i < argc; i++) {
|
||||
term = find_term(argv[i]);
|
||||
if (term == NULL) {
|
||||
warnx("%s: no description for terminal", argv[i]);
|
||||
continue;
|
||||
}
|
||||
if (term->type == 'a') {
|
||||
warnx("%s: cannot dump alias", argv[i]);
|
||||
continue;
|
||||
}
|
||||
/* Don't compile the aliases in, save space */
|
||||
free(term->tic->alias);
|
||||
term->tic->alias = NULL;
|
||||
len = _ti_flatten(&buf, term->tic);
|
||||
if (len == 0 || len == -1)
|
||||
continue;
|
||||
|
||||
printf("\t{\n");
|
||||
printf("\t\t\"%s\",\n", argv[i]);
|
||||
n++;
|
||||
for (j = 0, col = 0; j < (size_t)len; j++) {
|
||||
if (col == 0) {
|
||||
printf("\t\t\"");
|
||||
col = 16;
|
||||
}
|
||||
|
||||
col += printf("\\%03o", (uint8_t)buf[j]);
|
||||
if (col > 75) {
|
||||
printf("\"%s\n",
|
||||
j + 1 == (size_t)len ? "," : "");
|
||||
col = 0;
|
||||
}
|
||||
}
|
||||
if (col != 0)
|
||||
printf("\",\n");
|
||||
printf("\t\t%zu\n", len);
|
||||
printf("\t}");
|
||||
if (i + 1 < argc)
|
||||
printf(",");
|
||||
printf("\n");
|
||||
free(buf);
|
||||
}
|
||||
printf("};\n");
|
||||
|
||||
return n;
|
||||
}
|
||||
|
||||
int
|
||||
main(int argc, char **argv)
|
||||
{
|
||||
int ch, cflag, sflag, flags;
|
||||
char *source, *p, *buf, *ofile;
|
||||
FILE *f;
|
||||
DBM *db;
|
||||
size_t len, buflen, nterm, nalias;
|
||||
TBUF tbuf;
|
||||
TERM *term;
|
||||
|
||||
cflag = sflag = 0;
|
||||
ofile = NULL;
|
||||
flags = TIC_ALIAS | TIC_DESCRIPTION | TIC_WARNING;
|
||||
while ((ch = getopt(argc, argv, "Saco:sx")) != -1)
|
||||
switch (ch) {
|
||||
case 'S':
|
||||
Sflag = 1;
|
||||
/* We still compile aliases so that use= works.
|
||||
* However, it's removed before we flatten to save space. */
|
||||
flags &= ~TIC_DESCRIPTION;
|
||||
break;
|
||||
case 'a':
|
||||
flags |= TIC_COMMENT;
|
||||
break;
|
||||
case 'c':
|
||||
cflag = 1;
|
||||
break;
|
||||
case 'o':
|
||||
ofile = optarg;
|
||||
break;
|
||||
case 's':
|
||||
sflag = 1;
|
||||
break;
|
||||
case 'x':
|
||||
flags |= TIC_EXTRA;
|
||||
break;
|
||||
case '?': /* FALLTHROUGH */
|
||||
default:
|
||||
fprintf(stderr, "usage: %s [-acSsx] [-o file] source\n",
|
||||
getprogname());
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
if (optind == argc)
|
||||
errx(1, "No source file given");
|
||||
source = argv[optind++];
|
||||
f = fopen(source, "r");
|
||||
if (f == NULL)
|
||||
err(1, "fopen: %s", source);
|
||||
if (!cflag && !Sflag) {
|
||||
if (ofile == NULL)
|
||||
ofile = source;
|
||||
len = strlen(ofile) + 9;
|
||||
dbname = malloc(len + 4); /* For adding .db after open */
|
||||
if (dbname == NULL)
|
||||
err(1, "malloc");
|
||||
snprintf(dbname, len, "%s.tmp", ofile);
|
||||
db = dbm_open(dbname, O_CREAT | O_RDWR | O_TRUNC, DEFFILEMODE);
|
||||
if (db == NULL)
|
||||
err(1, "dbopen: %s", source);
|
||||
p = dbname + strlen(dbname);
|
||||
*p++ = '.';
|
||||
*p++ = 'd';
|
||||
*p++ = 'b';
|
||||
*p++ = '\0';
|
||||
atexit(do_unlink);
|
||||
} else
|
||||
db = NULL; /* satisfy gcc warning */
|
||||
|
||||
tbuf.buflen = tbuf.bufpos = 0;
|
||||
while ((buf = fgetln(f, &buflen)) != NULL) {
|
||||
/* Skip comments */
|
||||
if (*buf == '#')
|
||||
continue;
|
||||
if (buf[buflen - 1] != '\n') {
|
||||
process_entry(&tbuf, flags);
|
||||
dowarn("last line is not a comment"
|
||||
" and does not end with a newline");
|
||||
continue;
|
||||
}
|
||||
/*
|
||||
If the first char is space not a space then we have a
|
||||
new entry, so process it.
|
||||
*/
|
||||
if (!isspace((unsigned char)*buf) && tbuf.bufpos != 0)
|
||||
process_entry(&tbuf, flags);
|
||||
|
||||
/* Grow the buffer if needed */
|
||||
grow_tbuf(&tbuf, buflen);
|
||||
/* Append the string */
|
||||
memcpy(tbuf.buf + tbuf.bufpos, buf, buflen);
|
||||
tbuf.bufpos += buflen;
|
||||
}
|
||||
/* Process the last entry if not done already */
|
||||
process_entry(&tbuf, flags);
|
||||
|
||||
/* Merge use entries until we have merged all we can */
|
||||
while (merge_use(flags) != 0)
|
||||
;
|
||||
|
||||
if (Sflag) {
|
||||
print_dump(argc - optind, argv + optind);
|
||||
return error_exit;
|
||||
}
|
||||
|
||||
if (cflag)
|
||||
return error_exit;
|
||||
|
||||
/* Save the terms */
|
||||
nterm = nalias = 0;
|
||||
for (term = terms; term != NULL; term = term->next) {
|
||||
save_term(db, term);
|
||||
if (term->type == 'a')
|
||||
nalias++;
|
||||
else
|
||||
nterm++;
|
||||
}
|
||||
|
||||
/* done! */
|
||||
dbm_close(db);
|
||||
|
||||
/* Rename the tmp db to the real one now */
|
||||
len = strlen(ofile) + 4;
|
||||
p = malloc(len);
|
||||
if (p == NULL)
|
||||
err(1, "malloc");
|
||||
snprintf(p, len, "%s.db", ofile);
|
||||
if (rename(dbname, p) == -1)
|
||||
err(1, "rename");
|
||||
free(dbname);
|
||||
dbname = NULL;
|
||||
|
||||
if (sflag != 0)
|
||||
fprintf(stderr, "%zu entries and %zu aliases written to %s\n",
|
||||
nterm, nalias, p);
|
||||
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
Reference in New Issue
Block a user