termcap.c revision 331722
1/* A portion of this file is from ncurses: */
2/***************************************************************************
3*                            COPYRIGHT NOTICE                              *
4****************************************************************************
5*                ncurses is copyright (C) 1992-1995                        *
6*                          Zeyd M. Ben-Halim                               *
7*                          zmbenhal@netcom.com                             *
8*                          Eric S. Raymond                                 *
9*                          esr@snark.thyrsus.com                           *
10*                                                                          *
11*        Permission is hereby granted to reproduce and distribute ncurses  *
12*        by any means and for any fee, whether alone or as part of a       *
13*        larger distribution, in source or in binary form, PROVIDED        *
14*        this notice is included with any such distribution, and is not    *
15*        removed from any of its header files. Mention of ncurses in any   *
16*        applications linked with it is highly appreciated.                *
17*                                                                          *
18*        ncurses comes AS IS with no warranty, implied or expressed.       *
19*                                                                          *
20***************************************************************************/
21
22#include <curses.priv.h>
23
24#include <string.h>
25#include <term.h>
26#include <tic.h>
27#include <term_entry.h>
28
29/* The rest is from BSD */
30/*
31 * Copyright (c) 1980, 1993
32 *	The Regents of the University of California.  All rights reserved.
33 *
34 * Redistribution and use in source and binary forms, with or without
35 * modification, are permitted provided that the following conditions
36 * are met:
37 * 1. Redistributions of source code must retain the above copyright
38 *    notice, this list of conditions and the following disclaimer.
39 * 2. Redistributions in binary form must reproduce the above copyright
40 *    notice, this list of conditions and the following disclaimer in the
41 *    documentation and/or other materials provided with the distribution.
42 * 4. Neither the name of the University nor the names of its contributors
43 *    may be used to endorse or promote products derived from this software
44 *    without specific prior written permission.
45 *
46 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
47 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
48 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
49 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
50 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
51 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
52 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
53 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
54 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
55 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
56 * SUCH DAMAGE.
57 */
58
59#include <sys/cdefs.h>
60__FBSDID("$FreeBSD: stable/11/lib/ncurses/ncurses/termcap.c 331722 2018-03-29 02:50:57Z eadler $");
61
62#if 0
63#ifndef lint
64static const char sccsid[] = "@(#)termcap.c	8.1 (Berkeley) 6/4/93";
65#endif /* not lint */
66#endif
67
68#include <stdio.h>
69#include <ctype.h>
70#include <stdlib.h>
71#include <string.h>
72#include <unistd.h>
73#include <sys/param.h>
74#include "pathnames.h"
75
76#define	PBUFSIZ	MAXPATHLEN	/* max length of filename path */
77#define	PVECSIZ	32		/* max number of names in path */
78#define	TBUFSIZ 1024		/* max length of _nc_tgetent buffer */
79
80char _nc_termcap[TBUFSIZ + 1]; /* Last getcap, provided to tgetent() emul */
81
82/*
83 * termcap - routines for dealing with the terminal capability data base
84 *
85 * BUG:		Should use a "last" pointer in tbuf, so that searching
86 *		for capabilities alphabetically would not be a n**2/2
87 *		process when large numbers of capabilities are given.
88 * Note:	If we add a last pointer now we will screw up the
89 *		tc capability. We really should compile termcap.
90 *
91 * Essentially all the work here is scanning and decoding escapes
92 * in string capabilities.  We don't use stdio because the editor
93 * doesn't, and because living w/o it is not hard.
94 */
95
96/*
97 * Get an entry for terminal name in buffer _nc_termcap from the termcap
98 * file.
99 */
100int
101_nc_read_termcap_entry(const char *const name, TERMTYPE *const tp)
102{
103	ENTRY	*ep;
104	char *p;
105	char *cp;
106	char  *dummy;
107	char **fname;
108	char  *home;
109	int    i;
110	char   pathbuf[PBUFSIZ];	/* holds raw path of filenames */
111	char  *pathvec[PVECSIZ];	/* to point to names in pathbuf */
112	char **pvec;			/* holds usable tail of path vector */
113	char  *termpath;
114
115	_nc_termcap[0] = '\0';		/* in case */
116	dummy = NULL;
117	fname = pathvec;
118	pvec = pathvec;
119	p = pathbuf;
120	cp = getenv("TERMCAP");
121	/*
122	 * TERMCAP can have one of two things in it. It can be the
123	 * name of a file to use instead of /etc/termcap. In this
124	 * case it better start with a "/". Or it can be an entry to
125	 * use so we don't have to read the file. In this case it
126	 * has to already have the newlines crunched out.  If TERMCAP
127	 * does not hold a file name then a path of names is searched
128	 * instead.  The path is found in the TERMPATH variable, or
129	 * becomes "$HOME/.termcap /etc/termcap" if no TERMPATH exists.
130	 */
131	if (!cp || *cp != '/') {	/* no TERMCAP or it holds an entry */
132		if ( (termpath = getenv("TERMPATH")) )
133			strncpy(pathbuf, termpath, PBUFSIZ);
134		else {
135			if ( (home = getenv("HOME")) ) {/* set up default */
136				strncpy(pathbuf, home, PBUFSIZ - 1); /* $HOME first */
137				pathbuf[PBUFSIZ - 2] = '\0'; /* -2 because we add a slash */
138				p += strlen(pathbuf);	/* path, looking in */
139				*p++ = '/';
140			}	/* if no $HOME look in current directory */
141			strncpy(p, _PATH_DEF, PBUFSIZ - (p - pathbuf));
142		}
143	}
144	else				/* user-defined name in TERMCAP */
145		strncpy(pathbuf, cp, PBUFSIZ);	/* still can be tokenized */
146
147	/* For safety */
148	if (issetugid())
149		strcpy(pathbuf, _PATH_DEF_SEC);
150
151	pathbuf[PBUFSIZ - 1] = '\0';
152
153	*fname++ = pathbuf;	/* tokenize path into vector of names */
154	while (*++p)
155		if (*p == ' ' || *p == ':') {
156			*p = '\0';
157			while (*++p)
158				if (*p != ' ' && *p != ':')
159					break;
160			if (*p == '\0')
161				break;
162			*fname++ = p;
163			if (fname >= pathvec + PVECSIZ) {
164				fname--;
165				break;
166			}
167		}
168	*fname = (char *) 0;			/* mark end of vector */
169	if (cp && *cp && *cp != '/')
170		if (cgetset(cp) < 0)
171			return(-2);
172
173	i = cgetent(&dummy, pathvec, (char *)name);
174
175	if (i == 0) {
176		char *pd, *ps, *tok, *s, *tcs;
177		size_t len;
178
179		pd = _nc_termcap;
180		ps = dummy;
181		if ((tok = strchr(ps, ':')) == NULL) {
182			len = strlen(ps);
183			if (len >= TBUFSIZ)
184				i = -1;
185			else
186				strcpy(pd, ps);
187			goto done;
188		}
189		len = tok - ps + 1;
190		if (pd + len + 1 - _nc_termcap >= TBUFSIZ) {
191			i = -1;
192			goto done;
193		}
194		memcpy(pd, ps, len);
195		ps += len;
196		pd += len;
197		*pd = '\0';
198		tcs = pd - 1;
199		for (;;) {
200			while ((tok = strsep(&ps, ":")) != NULL &&
201			       *(tok - 2) != '\\' &&
202			       (*tok == '\0' || *tok == '\\' || !isgraph(UChar(*tok))))
203				;
204			if (tok == NULL)
205				break;
206			for (s = tcs; s != NULL && s[1] != '\0';
207			     s = strchr(s, ':')) {
208				s++;
209				if (s[0] == tok[0] && s[1] == tok[1])
210					goto skip_it;
211			}
212			len = strlen(tok);
213			if (pd + len + 1 - _nc_termcap >= TBUFSIZ) {
214				i = -1;
215				break;
216			}
217			memcpy(pd, tok, len);
218			pd += len;
219			*pd++ = ':';
220			*pd = '\0';
221		skip_it: ;
222		}
223	}
224done:
225	if (dummy)
226		free(dummy);
227
228
229/*
230 * From here on is ncurses-specific glue code
231 */
232
233	if (i < 0)
234		return(TGETENT_ERR);
235
236	_nc_set_source("TERMCAP");
237	_nc_read_entry_source((FILE *)NULL, _nc_termcap, FALSE, TRUE, NULLHOOK);
238
239	if (_nc_head == (ENTRY *)NULL)
240		return(TGETENT_ERR);
241
242	/* resolve all use references */
243	_nc_resolve_uses2(TRUE, FALSE);
244
245	for_entry_list(ep)
246		if (_nc_name_match(ep->tterm.term_names, name, "|:"))
247		{
248			/*
249			 * Make a local copy of the terminal capabilities, delinked
250			 * from the list.
251			 */
252			memcpy(tp, &ep->tterm, sizeof(TERMTYPE));
253			_nc_delink_entry(_nc_head, &(ep->tterm));
254			free(ep);
255			_nc_free_entries(_nc_head);
256			_nc_head = _nc_tail = NULL;	/* do not reuse! */
257
258			return TGETENT_YES;	/* OK */
259		}
260
261	_nc_free_entries(_nc_head);
262	_nc_head = _nc_tail = NULL;	/* do not reuse! */
263	return(TGETENT_NO);	/* not found */
264}
265