el.c 12 KB
Newer Older
1
/*	$NetBSD: el.c,v 1.68 2011/07/29 15:16:33 christos Exp $	*/
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

/*-
 * Copyright (c) 1992, 1993
 *	The Regents of the University of California.  All rights reserved.
 *
 * This code is derived from software contributed to Berkeley by
 * Christos Zoulas of Cornell University.
 *
 * 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.
unknown's avatar
unknown committed
18
 * 3. Neither the name of the University nor the names of its contributors
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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.
 */

35 36 37 38 39 40 41
#include "config.h"
#if !defined(lint) && !defined(SCCSID)
#if 0
static char sccsid[] = "@(#)el.c	8.2 (Berkeley) 1/3/94";
#else
#endif
#endif /* not lint && not SCCSID */
42 43 44 45 46 47 48 49 50

/*
 * el.c: EditLine interface functions
 */
#include <sys/types.h>
#include <sys/param.h>
#include <string.h>
#include <stdlib.h>
#include <stdarg.h>
51 52 53
#include <ctype.h>
#include <locale.h>
#include <langinfo.h>
54 55 56 57 58 59 60 61
#include "el.h"

/* el_init():
 *	Initialize editline and set default parameters.
 */
public EditLine *
el_init(const char *prog, FILE *fin, FILE *fout, FILE *ferr)
{
62
	EditLine *el = el_malloc(sizeof(*el));
63 64

	if (el == NULL)
65
		return NULL;
66 67 68

	memset(el, 0, sizeof(EditLine));

69
	el->el_infile = fin;
70 71
	el->el_outfile = fout;
	el->el_errfile = ferr;
72 73

	el->el_infd = fileno(fin);
74 75
	el->el_outfd = fileno(fout);
	el->el_errfd = fileno(ferr);
76

77 78
	el->el_prog = Strdup(ct_decode_string(prog, &el->el_scratch));
	if (el->el_prog == NULL) {
unknown's avatar
unknown committed
79 80 81
		el_free(el);
		return NULL;
	}
82 83 84 85 86

	/*
         * Initialize all the modules. Order is important!!!
         */
	el->el_flags = 0;
87 88 89 90 91
#ifdef WIDECHAR
	setlocale(LC_CTYPE, NULL);
        if (MB_CUR_MAX > 1)
		el->el_flags |= CHARSET_IS_MULTIBYTE;
#endif
92

93
	if (terminal_init(el) == -1) {
unknown's avatar
unknown committed
94
		el_free(el->el_prog);
unknown's avatar
unknown committed
95 96 97
		el_free(el);
		return NULL;
	}
98
	(void) keymacro_init(el);
99 100 101 102 103 104 105 106
	(void) map_init(el);
	if (tty_init(el) == -1)
		el->el_flags |= NO_TTY;
	(void) ch_init(el);
	(void) search_init(el);
	(void) hist_init(el);
	(void) prompt_init(el);
	(void) sig_init(el);
unknown's avatar
unknown committed
107
	(void) read_init(el);
108

109
	return el;
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
}


/* el_end():
 *	Clean up.
 */
public void
el_end(EditLine *el)
{

	if (el == NULL)
		return;

	el_reset(el);

125 126
	terminal_end(el);
	keymacro_end(el);
127 128 129 130 131 132 133 134
	map_end(el);
	tty_end(el);
	ch_end(el);
	search_end(el);
	hist_end(el);
	prompt_end(el);
	sig_end(el);

135 136 137 138 139 140 141 142
	el_free(el->el_prog);
#ifdef WIDECHAR
	el_free(el->el_scratch.cbuff);
	el_free(el->el_scratch.wbuff);
	el_free(el->el_lgcyconv.cbuff);
	el_free(el->el_lgcyconv.wbuff);
#endif
	el_free(el);
143 144 145 146 147 148 149 150 151 152 153
}


/* el_reset():
 *	Reset the tty and the parser
 */
public void
el_reset(EditLine *el)
{

	tty_cookedmode(el);
154
	ch_reset(el, 0);		/* XXX: Do we want that? */
155 156 157 158 159 160 161
}


/* el_set():
 *	set the editline parameters
 */
public int
162
FUN(el,set)(EditLine *el, int op, ...)
163
{
164
	va_list ap;
unknown's avatar
unknown committed
165
	int rv = 0;
166 167

	if (el == NULL)
168
		return -1;
169
	va_start(ap, op);
unknown's avatar
unknown committed
170

171 172
	switch (op) {
	case EL_PROMPT:
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
	case EL_RPROMPT: {
		el_pfunc_t p = va_arg(ap, el_pfunc_t);

		rv = prompt_set(el, p, 0, op, 1);
		break;
	}

	case EL_RESIZE: {
		el_zfunc_t p = va_arg(ap, el_zfunc_t);
		void *arg = va_arg(ap, void *);
		rv = ch_resizefun(el, p, arg);
		break;
	}

	case EL_PROMPT_ESC:
	case EL_RPROMPT_ESC: {
		el_pfunc_t p = va_arg(ap, el_pfunc_t);
		int c = va_arg(ap, int);

		rv = prompt_set(el, p, c, op, 1);
193
		break;
194
	}
195 196

	case EL_TERMINAL:
197
		rv = terminal_set(el, va_arg(ap, char *));
198 199 200
		break;

	case EL_EDITOR:
201
		rv = map_set_editor(el, va_arg(ap, Char *));
202 203 204
		break;

	case EL_SIGNAL:
205
		if (va_arg(ap, int))
206 207 208 209 210 211 212 213 214 215 216
			el->el_flags |= HANDLE_SIGNALS;
		else
			el->el_flags &= ~HANDLE_SIGNALS;
		break;

	case EL_BIND:
	case EL_TELLTC:
	case EL_SETTC:
	case EL_ECHOTC:
	case EL_SETTY:
	{
217
		const Char *argv[20];
218 219 220
		int i;

		for (i = 1; i < 20; i++)
221
			if ((argv[i] = va_arg(ap, Char *)) == NULL)
222 223 224 225
				break;

		switch (op) {
		case EL_BIND:
226
			argv[0] = STR("bind");
227 228 229 230
			rv = map_bind(el, i, argv);
			break;

		case EL_TELLTC:
231 232
			argv[0] = STR("telltc");
			rv = terminal_telltc(el, i, argv);
233 234 235
			break;

		case EL_SETTC:
236 237
			argv[0] = STR("settc");
			rv = terminal_settc(el, i, argv);
238 239 240
			break;

		case EL_ECHOTC:
241 242
			argv[0] = STR("echotc");
			rv = terminal_echotc(el, i, argv);
243 244 245
			break;

		case EL_SETTY:
246
			argv[0] = STR("setty");
247 248 249 250 251 252 253 254 255 256 257 258 259
			rv = tty_stty(el, i, argv);
			break;

		default:
			rv = -1;
			EL_ABORT((el->el_errfile, "Bad op %d\n", op));
			break;
		}
		break;
	}

	case EL_ADDFN:
	{
260 261
		Char *name = va_arg(ap, Char *);
		Char *help = va_arg(ap, Char *);
262
		el_func_t func = va_arg(ap, el_func_t);
263 264 265 266 267 268 269

		rv = map_addfunc(el, name, help, func);
		break;
	}

	case EL_HIST:
	{
270
		hist_fun_t func = va_arg(ap, hist_fun_t);
271
		void *ptr = va_arg(ap, void *);
272 273

		rv = hist_set(el, func, ptr);
274 275
		if (!(el->el_flags & CHARSET_IS_MULTIBYTE))
			el->el_flags &= ~NARROW_HISTORY;
276 277 278 279
		break;
	}

	case EL_EDITMODE:
280
		if (va_arg(ap, int))
281 282 283 284 285 286
			el->el_flags &= ~EDIT_DISABLED;
		else
			el->el_flags |= EDIT_DISABLED;
		rv = 0;
		break;

unknown's avatar
unknown committed
287 288
	case EL_GETCFN:
	{
289
		el_rfunc_t rc = va_arg(ap, el_rfunc_t);
unknown's avatar
unknown committed
290
		rv = el_read_setfn(el, rc);
291
		el->el_flags &= ~NARROW_READ;
unknown's avatar
unknown committed
292 293 294 295
		break;
	}

	case EL_CLIENTDATA:
296
		el->el_data = va_arg(ap, void *);
unknown's avatar
unknown committed
297 298
		break;

unknown's avatar
unknown committed
299
	case EL_UNBUFFERED:
300
		rv = va_arg(ap, int);
unknown's avatar
unknown committed
301 302 303 304 305 306 307 308 309 310 311
		if (rv && !(el->el_flags & UNBUFFERED)) {
			el->el_flags |= UNBUFFERED;
			read_prepare(el);
		} else if (!rv && (el->el_flags & UNBUFFERED)) {
			el->el_flags &= ~UNBUFFERED;
			read_finish(el);
		}
		rv = 0;
		break;

	case EL_PREP_TERM:
312
		rv = va_arg(ap, int);
unknown's avatar
unknown committed
313 314 315 316 317 318 319
		if (rv)
			(void) tty_rawmode(el);
		else
			(void) tty_cookedmode(el);
		rv = 0;
		break;

320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335
	case EL_SETFP:
	{
		FILE *fp;
		int what;

		what = va_arg(ap, int);
		fp = va_arg(ap, FILE *);

		rv = 0;
		switch (what) {
		case 0:
			el->el_infile = fp;
			el->el_infd = fileno(fp);
			break;
		case 1:
			el->el_outfile = fp;
336
			el->el_outfd = fileno(fp);
337 338 339
			break;
		case 2:
			el->el_errfile = fp;
340
			el->el_errfd = fileno(fp);
341 342 343 344 345 346 347 348 349 350 351
			break;
		default:
			rv = -1;
			break;
		}
		break;
	}

	case EL_REFRESH:
		re_clear_display(el);
		re_refresh(el);
352
		terminal__flush(el);
353 354
		break;

355 356
	default:
		rv = -1;
unknown's avatar
unknown committed
357
		break;
358 359
	}

360
	va_end(ap);
361
	return rv;
362 363 364 365 366 367 368
}


/* el_get():
 *	retrieve the editline parameters
 */
public int
369
FUN(el,get)(EditLine *el, int op, ...)
370
{
371
	va_list ap;
372 373
	int rv;

374 375 376 377 378
	if (el == NULL)
		return -1;

	va_start(ap, op);

379 380
	switch (op) {
	case EL_PROMPT:
381 382 383
	case EL_RPROMPT: {
		el_pfunc_t *p = va_arg(ap, el_pfunc_t *);
		rv = prompt_get(el, p, 0, op);
384
		break;
385 386 387 388 389 390 391 392 393
	}
	case EL_PROMPT_ESC:
	case EL_RPROMPT_ESC: {
		el_pfunc_t *p = va_arg(ap, el_pfunc_t *);
		Char *c = va_arg(ap, Char *);

		rv = prompt_get(el, p, c, op);
		break;
	}
394 395

	case EL_EDITOR:
396
		rv = map_get_editor(el, va_arg(ap, const Char **));
397 398 399
		break;

	case EL_SIGNAL:
400
		*va_arg(ap, int *) = (el->el_flags & HANDLE_SIGNALS);
401 402 403 404
		rv = 0;
		break;

	case EL_EDITMODE:
405
		*va_arg(ap, int *) = !(el->el_flags & EDIT_DISABLED);
406 407 408 409
		rv = 0;
		break;

	case EL_TERMINAL:
410
		terminal_get(el, va_arg(ap, const char **));
unknown's avatar
unknown committed
411
		rv = 0;
412 413
		break;

414
	case EL_GETTC:
415
	{
416 417
		static char name[] = "gettc";
		char *argv[20];
418 419
		int i;

420 421
 		for (i = 1; i < (int)(sizeof(argv) / sizeof(argv[0])); i++)
			if ((argv[i] = va_arg(ap, char *)) == NULL)
422 423 424
				break;

		switch (op) {
425 426
		case EL_GETTC:
			argv[0] = name;
427
			rv = terminal_gettc(el, i, argv);
428 429 430 431
			break;

		default:
			rv = -1;
432
			EL_ABORT((el->el_errfile, "Bad op %d\n", op));
433 434 435 436 437
			break;
		}
		break;
	}

unknown's avatar
unknown committed
438
	case EL_GETCFN:
439
		*va_arg(ap, el_rfunc_t *) = el_read_getfn(el);
unknown's avatar
unknown committed
440 441 442 443
		rv = 0;
		break;

	case EL_CLIENTDATA:
444
		*va_arg(ap, void **) = el->el_data;
unknown's avatar
unknown committed
445 446 447
		rv = 0;
		break;

unknown's avatar
unknown committed
448
	case EL_UNBUFFERED:
449
		*va_arg(ap, int *) = (!(el->el_flags & UNBUFFERED));
unknown's avatar
unknown committed
450 451 452
		rv = 0;
		break;

453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476
	case EL_GETFP:
	{
		int what;
		FILE **fpp;

		what = va_arg(ap, int);
		fpp = va_arg(ap, FILE **);
		rv = 0;
		switch (what) {
		case 0:
			*fpp = el->el_infile;
			break;
		case 1:
			*fpp = el->el_outfile;
			break;
		case 2:
			*fpp = el->el_errfile;
			break;
		default:
			rv = -1;
			break;
		}
		break;
	}
477 478
	default:
		rv = -1;
479
		break;
480
	}
481
	va_end(ap);
482

483
	return rv;
484 485 486 487 488 489
}


/* el_line():
 *	Return editing info
 */
490 491
public const TYPE(LineInfo) *
FUN(el,line)(EditLine *el)
492 493
{

494
	return (const TYPE(LineInfo) *)(void *)&el->el_line;
495 496
}

unknown's avatar
unknown committed
497

498 499 500 501 502 503 504 505
/* el_source():
 *	Source a file
 */
public int
el_source(EditLine *el, const char *fname)
{
	FILE *fp;
	size_t len;
unknown's avatar
unknown committed
506
	char *ptr;
507 508 509
	char *path = NULL;
	const Char *dptr;
	int error = 0;
510 511 512

	fp = NULL;
	if (fname == NULL) {
513 514 515 516 517 518 519
/* XXXMYSQL: Bug#49967 */
#if defined(HAVE_GETUID) && defined(HAVE_GETEUID) && \
    defined(HAVE_GETGID) && defined(HAVE_GETEGID)
#define HAVE_IDENTITY_FUNCS 1
#endif

#if (defined(HAVE_ISSETUGID) || defined(HAVE_IDENTITY_FUNCS))
unknown's avatar
unknown committed
520
		static const char elpath[] = "/.editrc";
521
		size_t plen = sizeof(elpath);
522
/* XXXMYSQL: Portability fix (for which platforms?) */
523
#ifdef HAVE_ISSETUGID
524
		if (issetugid())
525
			return -1;
526 527 528 529
#elif defined(HAVE_IDENTITY_FUNCS)
                if (getuid() != geteuid() || getgid() != getegid())
                  return (-1);
#endif
530
		if ((ptr = getenv("HOME")) == NULL)
531 532 533 534 535
			return -1;
		plen += strlen(ptr);
		if ((path = el_malloc(plen * sizeof(*path))) == NULL)
			return -1;
		(void)snprintf(path, plen, "%s%s", ptr, elpath);
536
		fname = path;
537 538
#else
		/*
539 540 541 542
		 * If issetugid() or the above mentioned get[e][u|g]id()
		 * functions are missing, always return an error, in order
		 * to keep from inadvertently opening up the user to a
		 * security hole.
543
		 */
544
		return -1;
545
#endif
546 547 548
	}
	if (fp == NULL)
		fp = fopen(fname, "r");
549 550 551 552
	if (fp == NULL) {
		el_free(path);
		return -1;
	}
553 554

	while ((ptr = fgetln(fp, &len)) != NULL) {
555 556 557 558 559 560
		if (*ptr == '\n')
			continue;	/* Empty line. */
		dptr = ct_decode_string(ptr, &el->el_scratch);
		if (!dptr)
			continue;
		if (len > 0 && dptr[len - 1] == '\n')
561
			--len;
562 563 564 565 566 567 568 569

		/* loop until first non-space char or EOL */
		while (*dptr != '\0' && Isspace(*dptr))
			dptr++;
		if (*dptr == '#')
			continue;   /* ignore, this is a comment line */
		if ((error = parse_line(el, dptr)) == -1)
			break;
570 571
	}

572
	el_free(path);
573
	(void) fclose(fp);
574
	return error;
575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591
}


/* el_resize():
 *	Called from program when terminal is resized
 */
public void
el_resize(EditLine *el)
{
	int lins, cols;
	sigset_t oset, nset;

	(void) sigemptyset(&nset);
	(void) sigaddset(&nset, SIGWINCH);
	(void) sigprocmask(SIG_BLOCK, &nset, &oset);

	/* get the correct window size */
592 593
	if (terminal_get_size(el, &lins, &cols))
		terminal_change_size(el, lins, cols);
594 595 596 597 598 599 600 601 602 603 604 605

	(void) sigprocmask(SIG_SETMASK, &oset, NULL);
}


/* el_beep():
 *	Called from the program to beep
 */
public void
el_beep(EditLine *el)
{

606
	terminal_beep(el);
607 608 609 610 611 612 613 614
}


/* el_editmode()
 *	Set the state of EDIT_DISABLED from the `edit' command.
 */
protected int
/*ARGSUSED*/
615
el_editmode(EditLine *el, int argc, const Char **argv)
616
{
617
	const Char *how;
618 619

	if (argv == NULL || argc != 2 || argv[1] == NULL)
620
		return -1;
621 622

	how = argv[1];
623
	if (Strcmp(how, STR("on")) == 0) {
624
		el->el_flags &= ~EDIT_DISABLED;
unknown's avatar
unknown committed
625
		tty_rawmode(el);
626
	} else if (Strcmp(how, STR("off")) == 0) {
unknown's avatar
unknown committed
627
		tty_cookedmode(el);
628
		el->el_flags |= EDIT_DISABLED;
unknown's avatar
unknown committed
629
	}
630
	else {
631 632 633
		(void) fprintf(el->el_errfile, "edit: Bad value `" FSTR "'.\n",
		    how);
		return -1;
634
	}
635
	return 0;
636
}