2000-05-31 23:49:23 +04:00
|
|
|
/* $NetBSD: ex_util.c,v 1.8 2000/05/31 19:49:26 jdc Exp $ */
|
1998-01-09 11:03:16 +03:00
|
|
|
|
1994-01-24 08:52:58 +03:00
|
|
|
/*-
|
1994-03-28 08:28:20 +04:00
|
|
|
* Copyright (c) 1993, 1994
|
1994-01-24 08:52:58 +03:00
|
|
|
* The Regents of the University of California. All rights reserved.
|
1996-05-20 07:47:00 +04:00
|
|
|
* Copyright (c) 1993, 1994, 1995, 1996
|
|
|
|
* Keith Bostic. All rights reserved.
|
1994-01-24 08:52:58 +03:00
|
|
|
*
|
1996-05-20 07:47:00 +04:00
|
|
|
* See the LICENSE file for redistribution information.
|
1994-01-24 08:52:58 +03:00
|
|
|
*/
|
|
|
|
|
1996-05-20 07:47:00 +04:00
|
|
|
#include "config.h"
|
|
|
|
|
1994-01-24 08:52:58 +03:00
|
|
|
#ifndef lint
|
1996-05-20 07:47:00 +04:00
|
|
|
static const char sccsid[] = "@(#)ex_util.c 10.21 (Berkeley) 5/8/96";
|
1994-01-24 08:52:58 +03:00
|
|
|
#endif /* not lint */
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
1994-03-28 08:28:20 +04:00
|
|
|
#include <sys/queue.h>
|
|
|
|
#include <sys/stat.h>
|
1994-01-24 08:52:58 +03:00
|
|
|
|
1994-03-28 08:28:20 +04:00
|
|
|
#include <bitstring.h>
|
1994-01-24 08:52:58 +03:00
|
|
|
#include <errno.h>
|
1994-03-28 08:28:20 +04:00
|
|
|
#include <limits.h>
|
|
|
|
#include <stdio.h>
|
1994-01-24 08:52:58 +03:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
1994-03-28 08:28:20 +04:00
|
|
|
#include <unistd.h>
|
|
|
|
|
1996-05-20 07:47:00 +04:00
|
|
|
#include "../common/common.h"
|
1994-01-24 08:52:58 +03:00
|
|
|
|
1996-05-20 07:47:00 +04:00
|
|
|
/*
|
|
|
|
* ex_cinit --
|
|
|
|
* Create an EX command structure.
|
|
|
|
*
|
|
|
|
* PUBLIC: void ex_cinit __P((EXCMD *,
|
|
|
|
* PUBLIC: int, int, recno_t, recno_t, int, ARGS **));
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
ex_cinit(cmdp, cmd_id, naddr, lno1, lno2, force, ap)
|
|
|
|
EXCMD *cmdp;
|
|
|
|
int cmd_id, force, naddr;
|
|
|
|
recno_t lno1, lno2;
|
|
|
|
ARGS **ap;
|
|
|
|
{
|
|
|
|
memset(cmdp, 0, sizeof(EXCMD));
|
|
|
|
cmdp->cmd = &cmds[cmd_id];
|
|
|
|
cmdp->addrcnt = naddr;
|
|
|
|
cmdp->addr1.lno = lno1;
|
|
|
|
cmdp->addr2.lno = lno2;
|
|
|
|
cmdp->addr1.cno = cmdp->addr2.cno = 1;
|
|
|
|
if (force)
|
|
|
|
cmdp->iflags |= E_C_FORCE;
|
|
|
|
cmdp->argc = 0;
|
|
|
|
if ((cmdp->argv = ap) != NULL)
|
|
|
|
cmdp->argv[0] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ex_cadd --
|
|
|
|
* Add an argument to an EX command structure.
|
|
|
|
*
|
|
|
|
* PUBLIC: void ex_cadd __P((EXCMD *, ARGS *, char *, size_t));
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
ex_cadd(cmdp, ap, arg, len)
|
|
|
|
EXCMD *cmdp;
|
|
|
|
ARGS *ap;
|
|
|
|
char *arg;
|
|
|
|
size_t len;
|
|
|
|
{
|
|
|
|
cmdp->argv[cmdp->argc] = ap;
|
|
|
|
ap->bp = arg;
|
|
|
|
ap->len = len;
|
|
|
|
cmdp->argv[++cmdp->argc] = NULL;
|
|
|
|
}
|
1994-01-24 08:52:58 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* ex_getline --
|
1996-05-20 07:47:00 +04:00
|
|
|
* Return a line from the file.
|
|
|
|
*
|
|
|
|
* PUBLIC: int ex_getline __P((SCR *, FILE *, size_t *));
|
1994-01-24 08:52:58 +03:00
|
|
|
*/
|
|
|
|
int
|
|
|
|
ex_getline(sp, fp, lenp)
|
|
|
|
SCR *sp;
|
|
|
|
FILE *fp;
|
|
|
|
size_t *lenp;
|
|
|
|
{
|
|
|
|
EX_PRIVATE *exp;
|
|
|
|
size_t off;
|
|
|
|
int ch;
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
exp = EXP(sp);
|
1994-08-17 20:35:35 +04:00
|
|
|
for (errno = 0, off = 0, p = exp->ibp;;) {
|
1994-01-24 08:52:58 +03:00
|
|
|
if (off >= exp->ibp_len) {
|
|
|
|
BINC_RET(sp, exp->ibp, exp->ibp_len, off + 1);
|
|
|
|
p = exp->ibp + off;
|
|
|
|
}
|
1994-08-17 20:35:35 +04:00
|
|
|
if ((ch = getc(fp)) == EOF && !feof(fp)) {
|
|
|
|
if (errno == EINTR) {
|
|
|
|
errno = 0;
|
|
|
|
clearerr(fp);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
}
|
1994-01-24 08:52:58 +03:00
|
|
|
if (ch == EOF || ch == '\n') {
|
|
|
|
if (ch == EOF && !off)
|
|
|
|
return (1);
|
|
|
|
*lenp = off;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
*p++ = ch;
|
1994-08-17 20:35:35 +04:00
|
|
|
++off;
|
1994-01-24 08:52:58 +03:00
|
|
|
}
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
1994-03-28 08:28:20 +04:00
|
|
|
|
|
|
|
/*
|
1996-05-20 07:47:00 +04:00
|
|
|
* ex_ncheck --
|
|
|
|
* Check for more files to edit.
|
|
|
|
*
|
|
|
|
* PUBLIC: int ex_ncheck __P((SCR *, int));
|
1994-03-28 08:28:20 +04:00
|
|
|
*/
|
|
|
|
int
|
1996-05-20 07:47:00 +04:00
|
|
|
ex_ncheck(sp, force)
|
1994-03-28 08:28:20 +04:00
|
|
|
SCR *sp;
|
1996-05-20 07:47:00 +04:00
|
|
|
int force;
|
1994-03-28 08:28:20 +04:00
|
|
|
{
|
|
|
|
/*
|
1996-05-20 07:47:00 +04:00
|
|
|
* !!!
|
|
|
|
* Historic practice: quit! or two quit's done in succession
|
|
|
|
* (where ZZ counts as a quit) didn't check for other files.
|
1994-03-28 08:28:20 +04:00
|
|
|
*/
|
1996-05-20 07:47:00 +04:00
|
|
|
if (!force && sp->ccnt != sp->q_ccnt + 1 &&
|
|
|
|
sp->cargv != NULL && sp->cargv[1] != NULL) {
|
|
|
|
sp->q_ccnt = sp->ccnt;
|
|
|
|
msgq(sp, M_ERR,
|
|
|
|
"167|More files to edit; use n[ext] to go to the next file, q[uit]! to quit");
|
|
|
|
return (1);
|
1994-03-28 08:28:20 +04:00
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
1996-05-20 07:47:00 +04:00
|
|
|
* ex_init --
|
|
|
|
* Init the screen for ex.
|
|
|
|
*
|
|
|
|
* PUBLIC: int ex_init __P((SCR *));
|
1994-03-28 08:28:20 +04:00
|
|
|
*/
|
1996-05-20 07:47:00 +04:00
|
|
|
int
|
|
|
|
ex_init(sp)
|
1994-03-28 08:28:20 +04:00
|
|
|
SCR *sp;
|
|
|
|
{
|
2000-05-31 23:49:23 +04:00
|
|
|
GS *gp;
|
|
|
|
|
|
|
|
gp = sp->gp;
|
|
|
|
|
|
|
|
if (gp->scr_screen(sp, SC_EX))
|
1996-05-20 07:47:00 +04:00
|
|
|
return (1);
|
2000-05-31 23:49:23 +04:00
|
|
|
(void)gp->scr_attr(sp, SA_ALTERNATE, 0);
|
1994-03-28 08:28:20 +04:00
|
|
|
|
1996-05-20 07:47:00 +04:00
|
|
|
sp->rows = O_VAL(sp, O_LINES);
|
|
|
|
sp->cols = O_VAL(sp, O_COLUMNS);
|
1994-03-28 08:28:20 +04:00
|
|
|
|
1996-05-20 07:47:00 +04:00
|
|
|
F_CLR(sp, SC_VI);
|
|
|
|
F_SET(sp, SC_EX | SC_SCR_EX);
|
|
|
|
return (0);
|
1994-03-28 08:28:20 +04:00
|
|
|
}
|
1994-08-17 20:35:35 +04:00
|
|
|
|
|
|
|
/*
|
1996-05-20 07:47:00 +04:00
|
|
|
* ex_emsg --
|
|
|
|
* Display a few common ex and vi error messages.
|
|
|
|
*
|
|
|
|
* PUBLIC: void ex_emsg __P((SCR *, char *, exm_t));
|
1994-08-17 20:35:35 +04:00
|
|
|
*/
|
1996-05-20 07:47:00 +04:00
|
|
|
void
|
|
|
|
ex_emsg(sp, p, which)
|
1994-08-17 20:35:35 +04:00
|
|
|
SCR *sp;
|
1996-05-20 07:47:00 +04:00
|
|
|
char *p;
|
|
|
|
exm_t which;
|
1994-08-17 20:35:35 +04:00
|
|
|
{
|
1996-05-20 07:47:00 +04:00
|
|
|
switch (which) {
|
|
|
|
case EXM_EMPTYBUF:
|
|
|
|
msgq(sp, M_ERR, "168|Buffer %s is empty", p);
|
|
|
|
break;
|
|
|
|
case EXM_FILECOUNT:
|
|
|
|
msgq_str(sp, M_ERR, p,
|
|
|
|
"144|%s: expanded into too many file names");
|
|
|
|
break;
|
|
|
|
case EXM_NOCANON:
|
1994-08-17 20:35:35 +04:00
|
|
|
msgq(sp, M_ERR,
|
1996-05-20 07:47:00 +04:00
|
|
|
"283|The %s command requires the ex terminal interface", p);
|
|
|
|
break;
|
|
|
|
case EXM_NOCANON_F:
|
|
|
|
msgq(sp, M_ERR,
|
|
|
|
"272|That form of %s requires the ex terminal interface",
|
|
|
|
p);
|
|
|
|
break;
|
|
|
|
case EXM_NOFILEYET:
|
|
|
|
if (p == NULL)
|
|
|
|
msgq(sp, M_ERR,
|
|
|
|
"274|Command failed, no file read in yet.");
|
|
|
|
else
|
|
|
|
msgq(sp, M_ERR,
|
|
|
|
"173|The %s command requires that a file have already been read in", p);
|
|
|
|
break;
|
|
|
|
case EXM_NOPREVBUF:
|
|
|
|
msgq(sp, M_ERR, "171|No previous buffer to execute");
|
|
|
|
break;
|
|
|
|
case EXM_NOPREVRE:
|
|
|
|
msgq(sp, M_ERR, "172|No previous regular expression");
|
|
|
|
break;
|
|
|
|
case EXM_NOSUSPEND:
|
|
|
|
msgq(sp, M_ERR, "230|This screen may not be suspended");
|
|
|
|
break;
|
|
|
|
case EXM_SECURE:
|
|
|
|
msgq(sp, M_ERR,
|
|
|
|
"290|The %s command is not supported when the secure edit option is set", p);
|
|
|
|
break;
|
|
|
|
case EXM_SECURE_F:
|
|
|
|
msgq(sp, M_ERR,
|
|
|
|
"284|That form of %s is not supported when the secure edit option is set", p);
|
|
|
|
break;
|
|
|
|
case EXM_USAGE:
|
|
|
|
msgq(sp, M_ERR, "174|Usage: %s", p);
|
|
|
|
break;
|
1994-08-17 20:35:35 +04:00
|
|
|
}
|
|
|
|
}
|