xref: /plan9/sys/src/ape/cmd/pdksh/history.c (revision 7dd7cddf99dd7472612f1413b4da293630e6b1bc)
1*7dd7cddfSDavid du Colombier /*
2*7dd7cddfSDavid du Colombier  * command history
3*7dd7cddfSDavid du Colombier  *
4*7dd7cddfSDavid du Colombier  * only implements in-memory history.
5*7dd7cddfSDavid du Colombier  */
6*7dd7cddfSDavid du Colombier 
7*7dd7cddfSDavid du Colombier /*
8*7dd7cddfSDavid du Colombier  *	This file contains
9*7dd7cddfSDavid du Colombier  *	a)	the original in-memory history  mechanism
10*7dd7cddfSDavid du Colombier  *	b)	a simple file saving history mechanism done by  sjg@zen
11*7dd7cddfSDavid du Colombier  *		define EASY_HISTORY to get this
12*7dd7cddfSDavid du Colombier  *	c)	a more complicated mechanism done by  pc@hillside.co.uk
13*7dd7cddfSDavid du Colombier  *		that more closely follows the real ksh way of doing
14*7dd7cddfSDavid du Colombier  *		things. You need to have the mmap system call for this
15*7dd7cddfSDavid du Colombier  *		to work on your system
16*7dd7cddfSDavid du Colombier  */
17*7dd7cddfSDavid du Colombier 
18*7dd7cddfSDavid du Colombier #include "sh.h"
19*7dd7cddfSDavid du Colombier #include "ksh_stat.h"
20*7dd7cddfSDavid du Colombier 
21*7dd7cddfSDavid du Colombier #ifdef HISTORY
22*7dd7cddfSDavid du Colombier # ifdef EASY_HISTORY
23*7dd7cddfSDavid du Colombier 
24*7dd7cddfSDavid du Colombier #  ifndef HISTFILE
25*7dd7cddfSDavid du Colombier #   ifdef OS2
26*7dd7cddfSDavid du Colombier #    define HISTFILE "history.ksh"
27*7dd7cddfSDavid du Colombier #   else /* OS2 */
28*7dd7cddfSDavid du Colombier #    define HISTFILE ".pdksh_history"
29*7dd7cddfSDavid du Colombier #   endif /* OS2 */
30*7dd7cddfSDavid du Colombier #  endif
31*7dd7cddfSDavid du Colombier 
32*7dd7cddfSDavid du Colombier # else
33*7dd7cddfSDavid du Colombier /*	Defines and includes for the complicated case */
34*7dd7cddfSDavid du Colombier 
35*7dd7cddfSDavid du Colombier #  include <sys/file.h>
36*7dd7cddfSDavid du Colombier #  include <sys/mman.h>
37*7dd7cddfSDavid du Colombier 
38*7dd7cddfSDavid du Colombier /*
39*7dd7cddfSDavid du Colombier  *	variables for handling the data file
40*7dd7cddfSDavid du Colombier  */
41*7dd7cddfSDavid du Colombier static int	histfd;
42*7dd7cddfSDavid du Colombier static int	hsize;
43*7dd7cddfSDavid du Colombier 
44*7dd7cddfSDavid du Colombier static int hist_count_lines ARGS((unsigned char *, int));
45*7dd7cddfSDavid du Colombier static int hist_shrink ARGS((unsigned char *, int));
46*7dd7cddfSDavid du Colombier static unsigned char *hist_skip_back ARGS((unsigned char *,int *,int));
47*7dd7cddfSDavid du Colombier static void histload ARGS((Source *, unsigned char *, int));
48*7dd7cddfSDavid du Colombier static void histinsert ARGS((Source *, int, unsigned char *));
49*7dd7cddfSDavid du Colombier static void writehistfile ARGS((int, char *));
50*7dd7cddfSDavid du Colombier static int sprinkle ARGS((int));
51*7dd7cddfSDavid du Colombier 
52*7dd7cddfSDavid du Colombier #  ifdef MAP_FILE
53*7dd7cddfSDavid du Colombier #   define MAP_FLAGS	(MAP_FILE|MAP_PRIVATE)
54*7dd7cddfSDavid du Colombier #  else
55*7dd7cddfSDavid du Colombier #   define MAP_FLAGS	MAP_PRIVATE
56*7dd7cddfSDavid du Colombier #  endif
57*7dd7cddfSDavid du Colombier 
58*7dd7cddfSDavid du Colombier # endif	/* of EASY_HISTORY */
59*7dd7cddfSDavid du Colombier 
60*7dd7cddfSDavid du Colombier static int	hist_execute ARGS((char *cmd));
61*7dd7cddfSDavid du Colombier static int	hist_replace ARGS((char **hp, const char *pat, const char *rep,
62*7dd7cddfSDavid du Colombier 				   int global));
63*7dd7cddfSDavid du Colombier static char   **hist_get ARGS((const char *str, int approx, int allow_cur));
64*7dd7cddfSDavid du Colombier static char   **hist_get_newest ARGS((int allow_cur));
65*7dd7cddfSDavid du Colombier static char   **hist_get_oldest ARGS(());
66*7dd7cddfSDavid du Colombier static void	histbackup ARGS((void));
67*7dd7cddfSDavid du Colombier 
68*7dd7cddfSDavid du Colombier static char   **current;	/* current postition in history[] */
69*7dd7cddfSDavid du Colombier static int	curpos;		/* current index in history[] */
70*7dd7cddfSDavid du Colombier static char    *hname;		/* current name of history file */
71*7dd7cddfSDavid du Colombier static int	hstarted;	/* set after hist_init() called */
72*7dd7cddfSDavid du Colombier static Source	*hist_source;
73*7dd7cddfSDavid du Colombier 
74*7dd7cddfSDavid du Colombier 
75*7dd7cddfSDavid du Colombier int
c_fc(wp)76*7dd7cddfSDavid du Colombier c_fc(wp)
77*7dd7cddfSDavid du Colombier 	char **wp;
78*7dd7cddfSDavid du Colombier {
79*7dd7cddfSDavid du Colombier 	struct shf *shf;
80*7dd7cddfSDavid du Colombier 	struct temp UNINITIALIZED(*tf);
81*7dd7cddfSDavid du Colombier 	char *p, *editor = (char *) 0;
82*7dd7cddfSDavid du Colombier 	int gflag = 0, lflag = 0, nflag = 0, sflag = 0, rflag = 0;
83*7dd7cddfSDavid du Colombier 	int optc;
84*7dd7cddfSDavid du Colombier 	char *first = (char *) 0, *last = (char *) 0;
85*7dd7cddfSDavid du Colombier 	char **hfirst, **hlast, **hp;
86*7dd7cddfSDavid du Colombier 
87*7dd7cddfSDavid du Colombier 	while ((optc = ksh_getopt(wp, &builtin_opt, "e:glnrs0,1,2,3,4,5,6,7,8,9,")) != EOF)
88*7dd7cddfSDavid du Colombier 		switch (optc) {
89*7dd7cddfSDavid du Colombier 		  case 'e':
90*7dd7cddfSDavid du Colombier 			p = builtin_opt.optarg;
91*7dd7cddfSDavid du Colombier 			if (strcmp(p, "-") == 0)
92*7dd7cddfSDavid du Colombier 				sflag++;
93*7dd7cddfSDavid du Colombier 			else {
94*7dd7cddfSDavid du Colombier 				editor = str_nsave(p, strlen(p) + 4, ATEMP);
95*7dd7cddfSDavid du Colombier 				strcat(editor, " $_");
96*7dd7cddfSDavid du Colombier 			}
97*7dd7cddfSDavid du Colombier 			break;
98*7dd7cddfSDavid du Colombier 		  case 'g': /* non-at&t ksh */
99*7dd7cddfSDavid du Colombier 			gflag++;
100*7dd7cddfSDavid du Colombier 			break;
101*7dd7cddfSDavid du Colombier 		  case 'l':
102*7dd7cddfSDavid du Colombier 			lflag++;
103*7dd7cddfSDavid du Colombier 			break;
104*7dd7cddfSDavid du Colombier 		  case 'n':
105*7dd7cddfSDavid du Colombier 			nflag++;
106*7dd7cddfSDavid du Colombier 			break;
107*7dd7cddfSDavid du Colombier 		  case 'r':
108*7dd7cddfSDavid du Colombier 			rflag++;
109*7dd7cddfSDavid du Colombier 			break;
110*7dd7cddfSDavid du Colombier 		  case 's':	/* posix version of -e - */
111*7dd7cddfSDavid du Colombier 			sflag++;
112*7dd7cddfSDavid du Colombier 			break;
113*7dd7cddfSDavid du Colombier 		  /* kludge city - accept -num as -- -num (kind of) */
114*7dd7cddfSDavid du Colombier 		  case '0': case '1': case '2': case '3': case '4':
115*7dd7cddfSDavid du Colombier 		  case '5': case '6': case '7': case '8': case '9':
116*7dd7cddfSDavid du Colombier 			p = shf_smprintf("-%c%s",
117*7dd7cddfSDavid du Colombier 					optc, builtin_opt.optarg);
118*7dd7cddfSDavid du Colombier 			if (!first)
119*7dd7cddfSDavid du Colombier 				first = p;
120*7dd7cddfSDavid du Colombier 			else if (!last)
121*7dd7cddfSDavid du Colombier 				last = p;
122*7dd7cddfSDavid du Colombier 			else {
123*7dd7cddfSDavid du Colombier 				bi_errorf("too many arguments");
124*7dd7cddfSDavid du Colombier 				return 1;
125*7dd7cddfSDavid du Colombier 			}
126*7dd7cddfSDavid du Colombier 			break;
127*7dd7cddfSDavid du Colombier 		  case '?':
128*7dd7cddfSDavid du Colombier 			return 1;
129*7dd7cddfSDavid du Colombier 		}
130*7dd7cddfSDavid du Colombier 	wp += builtin_opt.optind;
131*7dd7cddfSDavid du Colombier 
132*7dd7cddfSDavid du Colombier 	/* Substitute and execute command */
133*7dd7cddfSDavid du Colombier 	if (sflag) {
134*7dd7cddfSDavid du Colombier 		char *pat = (char *) 0, *rep = (char *) 0;
135*7dd7cddfSDavid du Colombier 
136*7dd7cddfSDavid du Colombier 		if (editor || lflag || nflag || rflag) {
137*7dd7cddfSDavid du Colombier 			bi_errorf("can't use -e, -l, -n, -r with -s (-e -)");
138*7dd7cddfSDavid du Colombier 			return 1;
139*7dd7cddfSDavid du Colombier 		}
140*7dd7cddfSDavid du Colombier 
141*7dd7cddfSDavid du Colombier 		/* Check for pattern replacement argument */
142*7dd7cddfSDavid du Colombier 		if (*wp && **wp && (p = strchr(*wp + 1, '='))) {
143*7dd7cddfSDavid du Colombier 			pat = str_save(*wp, ATEMP);
144*7dd7cddfSDavid du Colombier 			p = pat + (p - *wp);
145*7dd7cddfSDavid du Colombier 			*p++ = '\0';
146*7dd7cddfSDavid du Colombier 			rep = p;
147*7dd7cddfSDavid du Colombier 			wp++;
148*7dd7cddfSDavid du Colombier 		}
149*7dd7cddfSDavid du Colombier 		/* Check for search prefix */
150*7dd7cddfSDavid du Colombier 		if (!first && (first = *wp))
151*7dd7cddfSDavid du Colombier 			wp++;
152*7dd7cddfSDavid du Colombier 		if (last || *wp) {
153*7dd7cddfSDavid du Colombier 			bi_errorf("too many arguments");
154*7dd7cddfSDavid du Colombier 			return 1;
155*7dd7cddfSDavid du Colombier 		}
156*7dd7cddfSDavid du Colombier 
157*7dd7cddfSDavid du Colombier 		hp = first ? hist_get(first, FALSE, FALSE)
158*7dd7cddfSDavid du Colombier 			   : hist_get_newest(FALSE);
159*7dd7cddfSDavid du Colombier 		if (!hp)
160*7dd7cddfSDavid du Colombier 			return 1;
161*7dd7cddfSDavid du Colombier 		return hist_replace(hp, pat, rep, gflag);
162*7dd7cddfSDavid du Colombier 	}
163*7dd7cddfSDavid du Colombier 
164*7dd7cddfSDavid du Colombier 	if (editor && (lflag || nflag)) {
165*7dd7cddfSDavid du Colombier 		bi_errorf("can't use -l, -n with -e");
166*7dd7cddfSDavid du Colombier 		return 1;
167*7dd7cddfSDavid du Colombier 	}
168*7dd7cddfSDavid du Colombier 
169*7dd7cddfSDavid du Colombier 	if (!first && (first = *wp))
170*7dd7cddfSDavid du Colombier 		wp++;
171*7dd7cddfSDavid du Colombier 	if (!last && (last = *wp))
172*7dd7cddfSDavid du Colombier 		wp++;
173*7dd7cddfSDavid du Colombier 	if (*wp) {
174*7dd7cddfSDavid du Colombier 		bi_errorf("too many arguments");
175*7dd7cddfSDavid du Colombier 		return 1;
176*7dd7cddfSDavid du Colombier 	}
177*7dd7cddfSDavid du Colombier 	if (!first) {
178*7dd7cddfSDavid du Colombier 		hfirst = lflag ? hist_get("-16", TRUE, TRUE)
179*7dd7cddfSDavid du Colombier 			       : hist_get_newest(FALSE);
180*7dd7cddfSDavid du Colombier 		if (!hfirst)
181*7dd7cddfSDavid du Colombier 			return 1;
182*7dd7cddfSDavid du Colombier 		/* can't fail if hfirst didn't fail */
183*7dd7cddfSDavid du Colombier 		hlast = hist_get_newest(FALSE);
184*7dd7cddfSDavid du Colombier 	} else {
185*7dd7cddfSDavid du Colombier 		/* POSIX says not an error if first/last out of bounds
186*7dd7cddfSDavid du Colombier 		 * when range is specified; at&t ksh and pdksh allow out of
187*7dd7cddfSDavid du Colombier 		 * bounds for -l as well.
188*7dd7cddfSDavid du Colombier 		 */
189*7dd7cddfSDavid du Colombier 		hfirst = hist_get(first, (lflag || last) ? TRUE : FALSE,
190*7dd7cddfSDavid du Colombier 				lflag ? TRUE : FALSE);
191*7dd7cddfSDavid du Colombier 		if (!hfirst)
192*7dd7cddfSDavid du Colombier 			return 1;
193*7dd7cddfSDavid du Colombier 		hlast = last ? hist_get(last, TRUE, lflag ? TRUE : FALSE)
194*7dd7cddfSDavid du Colombier 			    : (lflag ? hist_get_newest(FALSE) : hfirst);
195*7dd7cddfSDavid du Colombier 		if (!hlast)
196*7dd7cddfSDavid du Colombier 			return 1;
197*7dd7cddfSDavid du Colombier 	}
198*7dd7cddfSDavid du Colombier 	if (hfirst > hlast) {
199*7dd7cddfSDavid du Colombier 		char **temp;
200*7dd7cddfSDavid du Colombier 
201*7dd7cddfSDavid du Colombier 		temp = hfirst; hfirst = hlast; hlast = temp;
202*7dd7cddfSDavid du Colombier 		rflag = !rflag; /* POSIX */
203*7dd7cddfSDavid du Colombier 	}
204*7dd7cddfSDavid du Colombier 
205*7dd7cddfSDavid du Colombier 	/* List history */
206*7dd7cddfSDavid du Colombier 	if (lflag) {
207*7dd7cddfSDavid du Colombier 		char *s, *t;
208*7dd7cddfSDavid du Colombier 		const char *nfmt = nflag ? "\t" : "%d\t";
209*7dd7cddfSDavid du Colombier 
210*7dd7cddfSDavid du Colombier 		for (hp = rflag ? hlast : hfirst;
211*7dd7cddfSDavid du Colombier 		     hp >= hfirst && hp <= hlast; hp += rflag ? -1 : 1)
212*7dd7cddfSDavid du Colombier 		{
213*7dd7cddfSDavid du Colombier 			shf_fprintf(shl_stdout, nfmt,
214*7dd7cddfSDavid du Colombier 				hist_source->line - (int) (histptr - hp));
215*7dd7cddfSDavid du Colombier 			/* print multi-line commands correctly */
216*7dd7cddfSDavid du Colombier 			for (s = *hp; (t = strchr(s, '\n')); s = t)
217*7dd7cddfSDavid du Colombier 				shf_fprintf(shl_stdout, "%.*s\t", ++t - s, s);
218*7dd7cddfSDavid du Colombier 			shf_fprintf(shl_stdout, "%s\n", s);
219*7dd7cddfSDavid du Colombier 		}
220*7dd7cddfSDavid du Colombier 		shf_flush(shl_stdout);
221*7dd7cddfSDavid du Colombier 		return 0;
222*7dd7cddfSDavid du Colombier 	}
223*7dd7cddfSDavid du Colombier 
224*7dd7cddfSDavid du Colombier 	/* Run editor on selected lines, then run resulting commands */
225*7dd7cddfSDavid du Colombier 
226*7dd7cddfSDavid du Colombier 	tf = maketemp(ATEMP, TT_HIST_EDIT, &e->temps);
227*7dd7cddfSDavid du Colombier 	if (!(shf = tf->shf)) {
228*7dd7cddfSDavid du Colombier 		bi_errorf("cannot create temp file %s - %s",
229*7dd7cddfSDavid du Colombier 			tf->name, strerror(errno));
230*7dd7cddfSDavid du Colombier 		return 1;
231*7dd7cddfSDavid du Colombier 	}
232*7dd7cddfSDavid du Colombier 	for (hp = rflag ? hlast : hfirst;
233*7dd7cddfSDavid du Colombier 	     hp >= hfirst && hp <= hlast; hp += rflag ? -1 : 1)
234*7dd7cddfSDavid du Colombier 		shf_fprintf(shf, "%s\n", *hp);
235*7dd7cddfSDavid du Colombier 	if (shf_close(shf) == EOF) {
236*7dd7cddfSDavid du Colombier 		bi_errorf("error writing temporary file - %s", strerror(errno));
237*7dd7cddfSDavid du Colombier 		return 1;
238*7dd7cddfSDavid du Colombier 	}
239*7dd7cddfSDavid du Colombier 
240*7dd7cddfSDavid du Colombier 	/* Ignore setstr errors here (arbitrary) */
241*7dd7cddfSDavid du Colombier 	setstr(local("_", FALSE), tf->name, KSH_RETURN_ERROR);
242*7dd7cddfSDavid du Colombier 
243*7dd7cddfSDavid du Colombier 	/* XXX: source should not get trashed by this.. */
244*7dd7cddfSDavid du Colombier 	{
245*7dd7cddfSDavid du Colombier 		Source *sold = source;
246*7dd7cddfSDavid du Colombier 		int ret;
247*7dd7cddfSDavid du Colombier 
248*7dd7cddfSDavid du Colombier 		ret = command(editor ? editor : "${FCEDIT:-/bin/ed} $_");
249*7dd7cddfSDavid du Colombier 		source = sold;
250*7dd7cddfSDavid du Colombier 		if (ret)
251*7dd7cddfSDavid du Colombier 			return ret;
252*7dd7cddfSDavid du Colombier 	}
253*7dd7cddfSDavid du Colombier 
254*7dd7cddfSDavid du Colombier 	{
255*7dd7cddfSDavid du Colombier 		struct stat statb;
256*7dd7cddfSDavid du Colombier 		XString xs;
257*7dd7cddfSDavid du Colombier 		char *xp;
258*7dd7cddfSDavid du Colombier 		int n;
259*7dd7cddfSDavid du Colombier 
260*7dd7cddfSDavid du Colombier 		if (!(shf = shf_open(tf->name, O_RDONLY, 0, 0))) {
261*7dd7cddfSDavid du Colombier 			bi_errorf("cannot open temp file %s", tf->name);
262*7dd7cddfSDavid du Colombier 			return 1;
263*7dd7cddfSDavid du Colombier 		}
264*7dd7cddfSDavid du Colombier 
265*7dd7cddfSDavid du Colombier 		n = fstat(shf_fileno(shf), &statb) < 0 ? 128
266*7dd7cddfSDavid du Colombier 			: statb.st_size + 1;
267*7dd7cddfSDavid du Colombier 		Xinit(xs, xp, n, hist_source->areap);
268*7dd7cddfSDavid du Colombier 		while ((n = shf_read(xp, Xnleft(xs, xp), shf)) > 0) {
269*7dd7cddfSDavid du Colombier 			xp += n;
270*7dd7cddfSDavid du Colombier 			if (Xnleft(xs, xp) <= 0)
271*7dd7cddfSDavid du Colombier 				XcheckN(xs, xp, Xlength(xs, xp));
272*7dd7cddfSDavid du Colombier 		}
273*7dd7cddfSDavid du Colombier 		if (n < 0) {
274*7dd7cddfSDavid du Colombier 			bi_errorf("error reading temp file %s - %s",
275*7dd7cddfSDavid du Colombier 				tf->name, strerror(shf_errno(shf)));
276*7dd7cddfSDavid du Colombier 			shf_close(shf);
277*7dd7cddfSDavid du Colombier 			return 1;
278*7dd7cddfSDavid du Colombier 		}
279*7dd7cddfSDavid du Colombier 		shf_close(shf);
280*7dd7cddfSDavid du Colombier 		*xp = '\0';
281*7dd7cddfSDavid du Colombier 		strip_nuls(Xstring(xs, xp), Xlength(xs, xp));
282*7dd7cddfSDavid du Colombier 		return hist_execute(Xstring(xs, xp));
283*7dd7cddfSDavid du Colombier 	}
284*7dd7cddfSDavid du Colombier }
285*7dd7cddfSDavid du Colombier 
286*7dd7cddfSDavid du Colombier /* Save cmd in history, execute cmd (cmd gets trashed) */
287*7dd7cddfSDavid du Colombier static int
hist_execute(cmd)288*7dd7cddfSDavid du Colombier hist_execute(cmd)
289*7dd7cddfSDavid du Colombier 	char *cmd;
290*7dd7cddfSDavid du Colombier {
291*7dd7cddfSDavid du Colombier 	Source *sold;
292*7dd7cddfSDavid du Colombier 	int ret;
293*7dd7cddfSDavid du Colombier 	char *p, *q;
294*7dd7cddfSDavid du Colombier 
295*7dd7cddfSDavid du Colombier 	histbackup();
296*7dd7cddfSDavid du Colombier 
297*7dd7cddfSDavid du Colombier 	for (p = cmd; p; p = q) {
298*7dd7cddfSDavid du Colombier 		if ((q = strchr(p, '\n'))) {
299*7dd7cddfSDavid du Colombier 			*q++ = '\0'; /* kill the newline */
300*7dd7cddfSDavid du Colombier 			if (!*q) /* ignore trailing newline */
301*7dd7cddfSDavid du Colombier 				q = (char *) 0;
302*7dd7cddfSDavid du Colombier 		}
303*7dd7cddfSDavid du Colombier #ifdef EASY_HISTORY
304*7dd7cddfSDavid du Colombier 		if (p != cmd)
305*7dd7cddfSDavid du Colombier 			histappend(p, TRUE);
306*7dd7cddfSDavid du Colombier 		else
307*7dd7cddfSDavid du Colombier #endif /* EASY_HISTORY */
308*7dd7cddfSDavid du Colombier 			histsave(++(hist_source->line), p, 1);
309*7dd7cddfSDavid du Colombier 
310*7dd7cddfSDavid du Colombier 		shellf("%s\n", p); /* POSIX doesn't say this is done... */
311*7dd7cddfSDavid du Colombier 		if ((p = q)) /* restore \n (trailing \n not restored) */
312*7dd7cddfSDavid du Colombier 			q[-1] = '\n';
313*7dd7cddfSDavid du Colombier 	}
314*7dd7cddfSDavid du Colombier 
315*7dd7cddfSDavid du Colombier 	/* Commands are executed here instead of pushing them onto the
316*7dd7cddfSDavid du Colombier 	 * input 'cause posix says the redirection and variable assignments
317*7dd7cddfSDavid du Colombier 	 * in
318*7dd7cddfSDavid du Colombier 	 *	X=y fc -e - 42 2> /dev/null
319*7dd7cddfSDavid du Colombier 	 * are to effect the repeated commands environment.
320*7dd7cddfSDavid du Colombier 	 */
321*7dd7cddfSDavid du Colombier 	/* XXX: source should not get trashed by this.. */
322*7dd7cddfSDavid du Colombier 	sold = source;
323*7dd7cddfSDavid du Colombier 	ret = command(cmd);
324*7dd7cddfSDavid du Colombier 	source = sold;
325*7dd7cddfSDavid du Colombier 	return ret;
326*7dd7cddfSDavid du Colombier }
327*7dd7cddfSDavid du Colombier 
328*7dd7cddfSDavid du Colombier static int
hist_replace(hp,pat,rep,global)329*7dd7cddfSDavid du Colombier hist_replace(hp, pat, rep, global)
330*7dd7cddfSDavid du Colombier 	char **hp;
331*7dd7cddfSDavid du Colombier 	const char *pat;
332*7dd7cddfSDavid du Colombier 	const char *rep;
333*7dd7cddfSDavid du Colombier 	int global;
334*7dd7cddfSDavid du Colombier {
335*7dd7cddfSDavid du Colombier 	char *line;
336*7dd7cddfSDavid du Colombier 
337*7dd7cddfSDavid du Colombier 	if (!pat)
338*7dd7cddfSDavid du Colombier 		line = str_save(*hp, ATEMP);
339*7dd7cddfSDavid du Colombier 	else {
340*7dd7cddfSDavid du Colombier 		char *s, *s1;
341*7dd7cddfSDavid du Colombier 		int pat_len = strlen(pat);
342*7dd7cddfSDavid du Colombier 		int rep_len = strlen(rep);
343*7dd7cddfSDavid du Colombier 		int len;
344*7dd7cddfSDavid du Colombier 		XString xs;
345*7dd7cddfSDavid du Colombier 		char *xp;
346*7dd7cddfSDavid du Colombier 		int any_subst = 0;
347*7dd7cddfSDavid du Colombier 
348*7dd7cddfSDavid du Colombier 		Xinit(xs, xp, 128, ATEMP);
349*7dd7cddfSDavid du Colombier 		for (s = *hp; (s1 = strstr(s, pat))
350*7dd7cddfSDavid du Colombier 			      && (!any_subst || global) ; s = s1 + pat_len)
351*7dd7cddfSDavid du Colombier 		{
352*7dd7cddfSDavid du Colombier 			any_subst = 1;
353*7dd7cddfSDavid du Colombier 			len = s1 - s;
354*7dd7cddfSDavid du Colombier 			XcheckN(xs, xp, len + rep_len);
355*7dd7cddfSDavid du Colombier 			memcpy(xp, s, len);		/* first part */
356*7dd7cddfSDavid du Colombier 			xp += len;
357*7dd7cddfSDavid du Colombier 			memcpy(xp, rep, rep_len);	/* replacement */
358*7dd7cddfSDavid du Colombier 			xp += rep_len;
359*7dd7cddfSDavid du Colombier 		}
360*7dd7cddfSDavid du Colombier 		if (!any_subst) {
361*7dd7cddfSDavid du Colombier 			bi_errorf("substitution failed");
362*7dd7cddfSDavid du Colombier 			return 1;
363*7dd7cddfSDavid du Colombier 		}
364*7dd7cddfSDavid du Colombier 		len = strlen(s) + 1;
365*7dd7cddfSDavid du Colombier 		XcheckN(xs, xp, len);
366*7dd7cddfSDavid du Colombier 		memcpy(xp, s, len);
367*7dd7cddfSDavid du Colombier 		xp += len;
368*7dd7cddfSDavid du Colombier 		line = Xclose(xs, xp);
369*7dd7cddfSDavid du Colombier 	}
370*7dd7cddfSDavid du Colombier 	return hist_execute(line);
371*7dd7cddfSDavid du Colombier }
372*7dd7cddfSDavid du Colombier 
373*7dd7cddfSDavid du Colombier /*
374*7dd7cddfSDavid du Colombier  * get pointer to history given pattern
375*7dd7cddfSDavid du Colombier  * pattern is a number or string
376*7dd7cddfSDavid du Colombier  */
377*7dd7cddfSDavid du Colombier static char **
hist_get(str,approx,allow_cur)378*7dd7cddfSDavid du Colombier hist_get(str, approx, allow_cur)
379*7dd7cddfSDavid du Colombier 	const char *str;
380*7dd7cddfSDavid du Colombier 	int approx;
381*7dd7cddfSDavid du Colombier 	int allow_cur;
382*7dd7cddfSDavid du Colombier {
383*7dd7cddfSDavid du Colombier 	char **hp = (char **) 0;
384*7dd7cddfSDavid du Colombier 	int n;
385*7dd7cddfSDavid du Colombier 
386*7dd7cddfSDavid du Colombier 	if (getn(str, &n)) {
387*7dd7cddfSDavid du Colombier 		hp = histptr + (n < 0 ? n : (n - hist_source->line));
388*7dd7cddfSDavid du Colombier 		if (hp < history) {
389*7dd7cddfSDavid du Colombier 			if (approx)
390*7dd7cddfSDavid du Colombier 				hp = hist_get_oldest();
391*7dd7cddfSDavid du Colombier 			else {
392*7dd7cddfSDavid du Colombier 				bi_errorf("%s: not in history", str);
393*7dd7cddfSDavid du Colombier 				hp = (char **) 0;
394*7dd7cddfSDavid du Colombier 			}
395*7dd7cddfSDavid du Colombier 		} else if (hp > histptr) {
396*7dd7cddfSDavid du Colombier 			if (approx)
397*7dd7cddfSDavid du Colombier 				hp = hist_get_newest(allow_cur);
398*7dd7cddfSDavid du Colombier 			else {
399*7dd7cddfSDavid du Colombier 				bi_errorf("%s: not in history", str);
400*7dd7cddfSDavid du Colombier 				hp = (char **) 0;
401*7dd7cddfSDavid du Colombier 			}
402*7dd7cddfSDavid du Colombier 		} else if (!allow_cur && hp == histptr) {
403*7dd7cddfSDavid du Colombier 			bi_errorf("%s: invalid range", str);
404*7dd7cddfSDavid du Colombier 			hp = (char **) 0;
405*7dd7cddfSDavid du Colombier 		}
406*7dd7cddfSDavid du Colombier 	} else {
407*7dd7cddfSDavid du Colombier 		int anchored = *str == '?' ? (++str, 0) : 1;
408*7dd7cddfSDavid du Colombier 
409*7dd7cddfSDavid du Colombier 		/* the -1 is to avoid the current fc command */
410*7dd7cddfSDavid du Colombier 		n = findhist(histptr - history - 1, 0, str, anchored);
411*7dd7cddfSDavid du Colombier 		if (n < 0) {
412*7dd7cddfSDavid du Colombier 			bi_errorf("%s: not in history", str);
413*7dd7cddfSDavid du Colombier 			hp = (char **) 0;
414*7dd7cddfSDavid du Colombier 		} else
415*7dd7cddfSDavid du Colombier 			hp = &history[n];
416*7dd7cddfSDavid du Colombier 	}
417*7dd7cddfSDavid du Colombier 	return hp;
418*7dd7cddfSDavid du Colombier }
419*7dd7cddfSDavid du Colombier 
420*7dd7cddfSDavid du Colombier /* Return a pointer to the newest command in the history */
421*7dd7cddfSDavid du Colombier static char **
hist_get_newest(allow_cur)422*7dd7cddfSDavid du Colombier hist_get_newest(allow_cur)
423*7dd7cddfSDavid du Colombier 	int allow_cur;
424*7dd7cddfSDavid du Colombier {
425*7dd7cddfSDavid du Colombier 	if (histptr < history || (!allow_cur && histptr == history)) {
426*7dd7cddfSDavid du Colombier 		bi_errorf("no history (yet)");
427*7dd7cddfSDavid du Colombier 		return (char **) 0;
428*7dd7cddfSDavid du Colombier 	}
429*7dd7cddfSDavid du Colombier 	if (allow_cur)
430*7dd7cddfSDavid du Colombier 		return histptr;
431*7dd7cddfSDavid du Colombier 	return histptr - 1;
432*7dd7cddfSDavid du Colombier }
433*7dd7cddfSDavid du Colombier 
434*7dd7cddfSDavid du Colombier /* Return a pointer to the newest command in the history */
435*7dd7cddfSDavid du Colombier static char **
hist_get_oldest()436*7dd7cddfSDavid du Colombier hist_get_oldest()
437*7dd7cddfSDavid du Colombier {
438*7dd7cddfSDavid du Colombier 	if (histptr <= history) {
439*7dd7cddfSDavid du Colombier 		bi_errorf("no history (yet)");
440*7dd7cddfSDavid du Colombier 		return (char **) 0;
441*7dd7cddfSDavid du Colombier 	}
442*7dd7cddfSDavid du Colombier 	return history;
443*7dd7cddfSDavid du Colombier }
444*7dd7cddfSDavid du Colombier 
445*7dd7cddfSDavid du Colombier /******************************/
446*7dd7cddfSDavid du Colombier /* Back up over last histsave */
447*7dd7cddfSDavid du Colombier /******************************/
448*7dd7cddfSDavid du Colombier static void
histbackup()449*7dd7cddfSDavid du Colombier histbackup()
450*7dd7cddfSDavid du Colombier {
451*7dd7cddfSDavid du Colombier 	static int last_line = -1;
452*7dd7cddfSDavid du Colombier 
453*7dd7cddfSDavid du Colombier 	if (histptr >= history && last_line != hist_source->line) {
454*7dd7cddfSDavid du Colombier 		hist_source->line--;
455*7dd7cddfSDavid du Colombier 		afree((void*)*histptr, APERM);
456*7dd7cddfSDavid du Colombier 		histptr--;
457*7dd7cddfSDavid du Colombier 		last_line = hist_source->line;
458*7dd7cddfSDavid du Colombier 	}
459*7dd7cddfSDavid du Colombier }
460*7dd7cddfSDavid du Colombier 
461*7dd7cddfSDavid du Colombier /*
462*7dd7cddfSDavid du Colombier  * Return the current position.
463*7dd7cddfSDavid du Colombier  */
464*7dd7cddfSDavid du Colombier char **
histpos()465*7dd7cddfSDavid du Colombier histpos()
466*7dd7cddfSDavid du Colombier {
467*7dd7cddfSDavid du Colombier 	return current;
468*7dd7cddfSDavid du Colombier }
469*7dd7cddfSDavid du Colombier 
470*7dd7cddfSDavid du Colombier int
histN()471*7dd7cddfSDavid du Colombier histN()
472*7dd7cddfSDavid du Colombier {
473*7dd7cddfSDavid du Colombier 	return curpos;
474*7dd7cddfSDavid du Colombier }
475*7dd7cddfSDavid du Colombier 
476*7dd7cddfSDavid du Colombier int
histnum(n)477*7dd7cddfSDavid du Colombier histnum(n)
478*7dd7cddfSDavid du Colombier 	int	n;
479*7dd7cddfSDavid du Colombier {
480*7dd7cddfSDavid du Colombier 	int	last = histptr - history;
481*7dd7cddfSDavid du Colombier 
482*7dd7cddfSDavid du Colombier 	if (n < 0 || n >= last) {
483*7dd7cddfSDavid du Colombier 		current = histptr;
484*7dd7cddfSDavid du Colombier 		curpos = last;
485*7dd7cddfSDavid du Colombier 		return last;
486*7dd7cddfSDavid du Colombier 	} else {
487*7dd7cddfSDavid du Colombier 		current = &history[n];
488*7dd7cddfSDavid du Colombier 		curpos = n;
489*7dd7cddfSDavid du Colombier 		return n;
490*7dd7cddfSDavid du Colombier 	}
491*7dd7cddfSDavid du Colombier }
492*7dd7cddfSDavid du Colombier 
493*7dd7cddfSDavid du Colombier /*
494*7dd7cddfSDavid du Colombier  * This will become unecessary if hist_get is modified to allow
495*7dd7cddfSDavid du Colombier  * searching from positions other than the end, and in either
496*7dd7cddfSDavid du Colombier  * direction.
497*7dd7cddfSDavid du Colombier  */
498*7dd7cddfSDavid du Colombier int
findhist(start,fwd,str,anchored)499*7dd7cddfSDavid du Colombier findhist(start, fwd, str, anchored)
500*7dd7cddfSDavid du Colombier 	int	start;
501*7dd7cddfSDavid du Colombier 	int	fwd;
502*7dd7cddfSDavid du Colombier 	const char  *str;
503*7dd7cddfSDavid du Colombier 	int	anchored;
504*7dd7cddfSDavid du Colombier {
505*7dd7cddfSDavid du Colombier 	char	**hp;
506*7dd7cddfSDavid du Colombier 	int	maxhist = histptr - history;
507*7dd7cddfSDavid du Colombier 	int	incr = fwd ? 1 : -1;
508*7dd7cddfSDavid du Colombier 	int	len = strlen(str);
509*7dd7cddfSDavid du Colombier 
510*7dd7cddfSDavid du Colombier 	if (start < 0 || start >= maxhist)
511*7dd7cddfSDavid du Colombier 		start = maxhist;
512*7dd7cddfSDavid du Colombier 
513*7dd7cddfSDavid du Colombier 	hp = &history[start];
514*7dd7cddfSDavid du Colombier 	for (; hp >= history && hp <= histptr; hp += incr)
515*7dd7cddfSDavid du Colombier 		if ((anchored && strncmp(*hp, str, len) == 0)
516*7dd7cddfSDavid du Colombier 		    || (!anchored && strstr(*hp, str)))
517*7dd7cddfSDavid du Colombier 			return hp - history;
518*7dd7cddfSDavid du Colombier 
519*7dd7cddfSDavid du Colombier 	return -1;
520*7dd7cddfSDavid du Colombier }
521*7dd7cddfSDavid du Colombier 
522*7dd7cddfSDavid du Colombier /*
523*7dd7cddfSDavid du Colombier  *	set history
524*7dd7cddfSDavid du Colombier  *	this means reallocating the dataspace
525*7dd7cddfSDavid du Colombier  */
526*7dd7cddfSDavid du Colombier void
sethistsize(n)527*7dd7cddfSDavid du Colombier sethistsize(n)
528*7dd7cddfSDavid du Colombier 	int n;
529*7dd7cddfSDavid du Colombier {
530*7dd7cddfSDavid du Colombier 	if (n > 0 && n != histsize) {
531*7dd7cddfSDavid du Colombier 		int cursize = histptr - history;
532*7dd7cddfSDavid du Colombier 
533*7dd7cddfSDavid du Colombier 		/* save most recent history */
534*7dd7cddfSDavid du Colombier 		if (n < cursize) {
535*7dd7cddfSDavid du Colombier 			memmove(history, histptr - n, n * sizeof(char *));
536*7dd7cddfSDavid du Colombier 			cursize = n;
537*7dd7cddfSDavid du Colombier 		}
538*7dd7cddfSDavid du Colombier 
539*7dd7cddfSDavid du Colombier 		history = (char **)aresize(history, n*sizeof(char *), APERM);
540*7dd7cddfSDavid du Colombier 
541*7dd7cddfSDavid du Colombier 		histsize = n;
542*7dd7cddfSDavid du Colombier 		histptr = history + cursize;
543*7dd7cddfSDavid du Colombier 	}
544*7dd7cddfSDavid du Colombier }
545*7dd7cddfSDavid du Colombier 
546*7dd7cddfSDavid du Colombier /*
547*7dd7cddfSDavid du Colombier  *	set history file
548*7dd7cddfSDavid du Colombier  *	This can mean reloading/resetting/starting history file
549*7dd7cddfSDavid du Colombier  *	maintenance
550*7dd7cddfSDavid du Colombier  */
551*7dd7cddfSDavid du Colombier void
sethistfile(name)552*7dd7cddfSDavid du Colombier sethistfile(name)
553*7dd7cddfSDavid du Colombier 	const char *name;
554*7dd7cddfSDavid du Colombier {
555*7dd7cddfSDavid du Colombier 	/* if not started then nothing to do */
556*7dd7cddfSDavid du Colombier 	if (hstarted == 0)
557*7dd7cddfSDavid du Colombier 		return;
558*7dd7cddfSDavid du Colombier 
559*7dd7cddfSDavid du Colombier 	/* if the name is the same as the name we have */
560*7dd7cddfSDavid du Colombier 	if (hname && strcmp(hname, name) == 0)
561*7dd7cddfSDavid du Colombier 		return;
562*7dd7cddfSDavid du Colombier 
563*7dd7cddfSDavid du Colombier 	/*
564*7dd7cddfSDavid du Colombier 	 * its a new name - possibly
565*7dd7cddfSDavid du Colombier 	 */
566*7dd7cddfSDavid du Colombier # ifdef EASY_HISTORY
567*7dd7cddfSDavid du Colombier 	if (hname) {
568*7dd7cddfSDavid du Colombier 		afree(hname, APERM);
569*7dd7cddfSDavid du Colombier 		hname = NULL;
570*7dd7cddfSDavid du Colombier 	}
571*7dd7cddfSDavid du Colombier # else
572*7dd7cddfSDavid du Colombier 	if (histfd) {
573*7dd7cddfSDavid du Colombier 		/* yes the file is open */
574*7dd7cddfSDavid du Colombier 		(void) close(histfd);
575*7dd7cddfSDavid du Colombier 		histfd = 0;
576*7dd7cddfSDavid du Colombier 		hsize = 0;
577*7dd7cddfSDavid du Colombier 		afree(hname, APERM);
578*7dd7cddfSDavid du Colombier 		hname = NULL;
579*7dd7cddfSDavid du Colombier 		/* let's reset the history */
580*7dd7cddfSDavid du Colombier 		histptr = history - 1;
581*7dd7cddfSDavid du Colombier 		hist_source->line = 0;
582*7dd7cddfSDavid du Colombier 	}
583*7dd7cddfSDavid du Colombier # endif
584*7dd7cddfSDavid du Colombier 
585*7dd7cddfSDavid du Colombier 	hist_init(hist_source);
586*7dd7cddfSDavid du Colombier }
587*7dd7cddfSDavid du Colombier 
588*7dd7cddfSDavid du Colombier /*
589*7dd7cddfSDavid du Colombier  *	initialise the history vector
590*7dd7cddfSDavid du Colombier  */
591*7dd7cddfSDavid du Colombier void
init_histvec()592*7dd7cddfSDavid du Colombier init_histvec()
593*7dd7cddfSDavid du Colombier {
594*7dd7cddfSDavid du Colombier 	if (history == (char **)NULL) {
595*7dd7cddfSDavid du Colombier 		histsize = HISTORYSIZE;
596*7dd7cddfSDavid du Colombier 		history = (char **)alloc(histsize*sizeof (char *), APERM);
597*7dd7cddfSDavid du Colombier 		histptr = history - 1;
598*7dd7cddfSDavid du Colombier 	}
599*7dd7cddfSDavid du Colombier }
600*7dd7cddfSDavid du Colombier 
601*7dd7cddfSDavid du Colombier # ifdef EASY_HISTORY
602*7dd7cddfSDavid du Colombier /*
603*7dd7cddfSDavid du Colombier  * save command in history
604*7dd7cddfSDavid du Colombier  */
605*7dd7cddfSDavid du Colombier void
histsave(lno,cmd,dowrite)606*7dd7cddfSDavid du Colombier histsave(lno, cmd, dowrite)
607*7dd7cddfSDavid du Colombier 	int lno;	/* ignored (compatibility with COMPLEX_HISTORY) */
608*7dd7cddfSDavid du Colombier 	const char *cmd;
609*7dd7cddfSDavid du Colombier 	int dowrite;	/* ignored (compatibility with COMPLEX_HISTORY) */
610*7dd7cddfSDavid du Colombier {
611*7dd7cddfSDavid du Colombier 	register char **hp = histptr;
612*7dd7cddfSDavid du Colombier 	char *cp;
613*7dd7cddfSDavid du Colombier 
614*7dd7cddfSDavid du Colombier 	if (++hp >= history + histsize) { /* remove oldest command */
615*7dd7cddfSDavid du Colombier 		afree((void*)history[0], APERM);
616*7dd7cddfSDavid du Colombier 		memmove(history, history + 1,
617*7dd7cddfSDavid du Colombier 			sizeof(history[0]) * (histsize - 1));
618*7dd7cddfSDavid du Colombier 		hp = &history[histsize - 1];
619*7dd7cddfSDavid du Colombier 	}
620*7dd7cddfSDavid du Colombier 	*hp = str_save(cmd, APERM);
621*7dd7cddfSDavid du Colombier 	/* trash trailing newline but allow imbedded newlines */
622*7dd7cddfSDavid du Colombier 	cp = *hp + strlen(*hp);
623*7dd7cddfSDavid du Colombier 	if (cp > *hp && cp[-1] == '\n')
624*7dd7cddfSDavid du Colombier 		cp[-1] = '\0';
625*7dd7cddfSDavid du Colombier 	histptr = hp;
626*7dd7cddfSDavid du Colombier }
627*7dd7cddfSDavid du Colombier 
628*7dd7cddfSDavid du Colombier /*
629*7dd7cddfSDavid du Colombier  * Append an entry to the last saved command. Used for multiline
630*7dd7cddfSDavid du Colombier  * commands
631*7dd7cddfSDavid du Colombier  */
632*7dd7cddfSDavid du Colombier void
histappend(cmd,nl_separate)633*7dd7cddfSDavid du Colombier histappend(cmd, nl_separate)
634*7dd7cddfSDavid du Colombier 	const char *cmd;
635*7dd7cddfSDavid du Colombier 	int	nl_separate;
636*7dd7cddfSDavid du Colombier {
637*7dd7cddfSDavid du Colombier 	int	hlen, clen;
638*7dd7cddfSDavid du Colombier 	char	*p;
639*7dd7cddfSDavid du Colombier 
640*7dd7cddfSDavid du Colombier 	hlen = strlen(*histptr);
641*7dd7cddfSDavid du Colombier 	clen = strlen(cmd);
642*7dd7cddfSDavid du Colombier 	if (clen > 0 && cmd[clen-1] == '\n')
643*7dd7cddfSDavid du Colombier 		clen--;
644*7dd7cddfSDavid du Colombier 	p = *histptr = (char *) aresize(*histptr, hlen + clen + 2, APERM);
645*7dd7cddfSDavid du Colombier 	p += hlen;
646*7dd7cddfSDavid du Colombier 	if (nl_separate)
647*7dd7cddfSDavid du Colombier 		*p++ = '\n';
648*7dd7cddfSDavid du Colombier 	memcpy(p, cmd, clen);
649*7dd7cddfSDavid du Colombier 	p[clen] = '\0';
650*7dd7cddfSDavid du Colombier }
651*7dd7cddfSDavid du Colombier 
652*7dd7cddfSDavid du Colombier /*
653*7dd7cddfSDavid du Colombier  * 92-04-25 <sjg@zen>
654*7dd7cddfSDavid du Colombier  * A simple history file implementation.
655*7dd7cddfSDavid du Colombier  * At present we only save the history when we exit.
656*7dd7cddfSDavid du Colombier  * This can cause problems when there are multiple shells are
657*7dd7cddfSDavid du Colombier  * running under the same user-id.  The last shell to exit gets
658*7dd7cddfSDavid du Colombier  * to save its history.
659*7dd7cddfSDavid du Colombier  */
660*7dd7cddfSDavid du Colombier void
hist_init(s)661*7dd7cddfSDavid du Colombier hist_init(s)
662*7dd7cddfSDavid du Colombier 	Source *s;
663*7dd7cddfSDavid du Colombier {
664*7dd7cddfSDavid du Colombier 	char *f;
665*7dd7cddfSDavid du Colombier 	FILE *fh;
666*7dd7cddfSDavid du Colombier 
667*7dd7cddfSDavid du Colombier 	if (Flag(FTALKING) == 0)
668*7dd7cddfSDavid du Colombier 		return;
669*7dd7cddfSDavid du Colombier 
670*7dd7cddfSDavid du Colombier 	hstarted = 1;
671*7dd7cddfSDavid du Colombier 
672*7dd7cddfSDavid du Colombier 	hist_source = s;
673*7dd7cddfSDavid du Colombier 
674*7dd7cddfSDavid du Colombier 	if ((f = str_val(global("HISTFILE"))) == NULL || *f == '\0') {
675*7dd7cddfSDavid du Colombier # if 1 /* Don't use history file unless the user asks for it */
676*7dd7cddfSDavid du Colombier 		hname = NULL;
677*7dd7cddfSDavid du Colombier 		return;
678*7dd7cddfSDavid du Colombier # else
679*7dd7cddfSDavid du Colombier 		char *home = str_val(global("HOME"));
680*7dd7cddfSDavid du Colombier 		int len;
681*7dd7cddfSDavid du Colombier 
682*7dd7cddfSDavid du Colombier 		if (home == NULL)
683*7dd7cddfSDavid du Colombier 			home = null;
684*7dd7cddfSDavid du Colombier 		f = HISTFILE;
685*7dd7cddfSDavid du Colombier 		hname = alloc(len = strlen(home) + strlen(f) + 2, APERM);
686*7dd7cddfSDavid du Colombier 		shf_snprintf(hname, len, "%s/%s", home, f);
687*7dd7cddfSDavid du Colombier # endif
688*7dd7cddfSDavid du Colombier 	} else
689*7dd7cddfSDavid du Colombier 		hname = str_save(f, APERM);
690*7dd7cddfSDavid du Colombier 
691*7dd7cddfSDavid du Colombier 	if ((fh = fopen(hname, "r"))) {
692*7dd7cddfSDavid du Colombier 		int pos = 0, nread = 0;
693*7dd7cddfSDavid du Colombier 		int contin = 0;		/* continuation of previous command */
694*7dd7cddfSDavid du Colombier 		char *end;
695*7dd7cddfSDavid du Colombier 		char hline[LINE + 1];
696*7dd7cddfSDavid du Colombier 
697*7dd7cddfSDavid du Colombier 		while (1) {
698*7dd7cddfSDavid du Colombier 			if (pos >= nread) {
699*7dd7cddfSDavid du Colombier 				pos = 0;
700*7dd7cddfSDavid du Colombier 				nread = fread(hline, 1, LINE, fh);
701*7dd7cddfSDavid du Colombier 				if (nread <= 0)
702*7dd7cddfSDavid du Colombier 					break;
703*7dd7cddfSDavid du Colombier 				hline[nread] = '\0';
704*7dd7cddfSDavid du Colombier 			}
705*7dd7cddfSDavid du Colombier 			end = strchr(hline + pos, 0); /* will always succeed */
706*7dd7cddfSDavid du Colombier 			if (contin)
707*7dd7cddfSDavid du Colombier 				histappend(hline + pos, 0);
708*7dd7cddfSDavid du Colombier 			else {
709*7dd7cddfSDavid du Colombier 				hist_source->line++;
710*7dd7cddfSDavid du Colombier 				histsave(0, hline + pos, 0);
711*7dd7cddfSDavid du Colombier 			}
712*7dd7cddfSDavid du Colombier 			pos = end - hline + 1;
713*7dd7cddfSDavid du Colombier 			contin = end == &hline[nread];
714*7dd7cddfSDavid du Colombier 		}
715*7dd7cddfSDavid du Colombier 		fclose(fh);
716*7dd7cddfSDavid du Colombier 	}
717*7dd7cddfSDavid du Colombier }
718*7dd7cddfSDavid du Colombier 
719*7dd7cddfSDavid du Colombier /*
720*7dd7cddfSDavid du Colombier  * save our history.
721*7dd7cddfSDavid du Colombier  * We check that we do not have more than we are allowed.
722*7dd7cddfSDavid du Colombier  * If the history file is read-only we do nothing.
723*7dd7cddfSDavid du Colombier  * Handy for having all shells start with a useful history set.
724*7dd7cddfSDavid du Colombier  */
725*7dd7cddfSDavid du Colombier 
726*7dd7cddfSDavid du Colombier void
hist_finish()727*7dd7cddfSDavid du Colombier hist_finish()
728*7dd7cddfSDavid du Colombier {
729*7dd7cddfSDavid du Colombier   static int once;
730*7dd7cddfSDavid du Colombier   FILE *fh;
731*7dd7cddfSDavid du Colombier   register int i;
732*7dd7cddfSDavid du Colombier   register char **hp;
733*7dd7cddfSDavid du Colombier 
734*7dd7cddfSDavid du Colombier   if (once++)
735*7dd7cddfSDavid du Colombier     return;
736*7dd7cddfSDavid du Colombier   /* check how many we have */
737*7dd7cddfSDavid du Colombier   i = histptr - history;
738*7dd7cddfSDavid du Colombier   if (i >= histsize)
739*7dd7cddfSDavid du Colombier     hp = &histptr[-histsize];
740*7dd7cddfSDavid du Colombier   else
741*7dd7cddfSDavid du Colombier     hp = history;
742*7dd7cddfSDavid du Colombier   if (hname && (fh = fopen(hname, "w")))
743*7dd7cddfSDavid du Colombier   {
744*7dd7cddfSDavid du Colombier     for (i = 0; hp + i <= histptr && hp[i]; i++)
745*7dd7cddfSDavid du Colombier       fprintf(fh, "%s%c", hp[i], '\0');
746*7dd7cddfSDavid du Colombier     fclose(fh);
747*7dd7cddfSDavid du Colombier   }
748*7dd7cddfSDavid du Colombier }
749*7dd7cddfSDavid du Colombier 
750*7dd7cddfSDavid du Colombier # else /* EASY_HISTORY */
751*7dd7cddfSDavid du Colombier 
752*7dd7cddfSDavid du Colombier /*
753*7dd7cddfSDavid du Colombier  *	Routines added by Peter Collinson BSDI(Europe)/Hillside Systems to
754*7dd7cddfSDavid du Colombier  *	a) permit HISTSIZE to control number of lines of history stored
755*7dd7cddfSDavid du Colombier  *	b) maintain a physical history file
756*7dd7cddfSDavid du Colombier  *
757*7dd7cddfSDavid du Colombier  *	It turns out that there is a lot of ghastly hackery here
758*7dd7cddfSDavid du Colombier  */
759*7dd7cddfSDavid du Colombier 
760*7dd7cddfSDavid du Colombier 
761*7dd7cddfSDavid du Colombier /*
762*7dd7cddfSDavid du Colombier  * save command in history
763*7dd7cddfSDavid du Colombier  */
764*7dd7cddfSDavid du Colombier void
histsave(lno,cmd,dowrite)765*7dd7cddfSDavid du Colombier histsave(lno, cmd, dowrite)
766*7dd7cddfSDavid du Colombier 	int lno;
767*7dd7cddfSDavid du Colombier 	const char *cmd;
768*7dd7cddfSDavid du Colombier 	int dowrite;
769*7dd7cddfSDavid du Colombier {
770*7dd7cddfSDavid du Colombier 	register char **hp;
771*7dd7cddfSDavid du Colombier 	char *c, *cp;
772*7dd7cddfSDavid du Colombier 
773*7dd7cddfSDavid du Colombier 	c = str_save(cmd, APERM);
774*7dd7cddfSDavid du Colombier 	if ((cp = strchr(c, '\n')) != NULL)
775*7dd7cddfSDavid du Colombier 		*cp = '\0';
776*7dd7cddfSDavid du Colombier 
777*7dd7cddfSDavid du Colombier 	if (histfd && dowrite)
778*7dd7cddfSDavid du Colombier 		writehistfile(lno, c);
779*7dd7cddfSDavid du Colombier 
780*7dd7cddfSDavid du Colombier 	hp = histptr;
781*7dd7cddfSDavid du Colombier 
782*7dd7cddfSDavid du Colombier 	if (++hp >= history + histsize) { /* remove oldest command */
783*7dd7cddfSDavid du Colombier 		afree((void*)*history, APERM);
784*7dd7cddfSDavid du Colombier 		for (hp = history; hp < history + histsize - 1; hp++)
785*7dd7cddfSDavid du Colombier 			hp[0] = hp[1];
786*7dd7cddfSDavid du Colombier 	}
787*7dd7cddfSDavid du Colombier 	*hp = c;
788*7dd7cddfSDavid du Colombier 	histptr = hp;
789*7dd7cddfSDavid du Colombier }
790*7dd7cddfSDavid du Colombier 
791*7dd7cddfSDavid du Colombier /*
792*7dd7cddfSDavid du Colombier  *	Write history data to a file nominated by HISTFILE
793*7dd7cddfSDavid du Colombier  *	if HISTFILE is unset then history still happens, but
794*7dd7cddfSDavid du Colombier  *	the data is not written to a file
795*7dd7cddfSDavid du Colombier  *	All copies of ksh looking at the file will maintain the
796*7dd7cddfSDavid du Colombier  *	same history. This is ksh behaviour.
797*7dd7cddfSDavid du Colombier  *
798*7dd7cddfSDavid du Colombier  *	This stuff uses mmap()
799*7dd7cddfSDavid du Colombier  *	if your system ain't got it - then you'll have to undef HISTORYFILE
800*7dd7cddfSDavid du Colombier  */
801*7dd7cddfSDavid du Colombier 
802*7dd7cddfSDavid du Colombier /*
803*7dd7cddfSDavid du Colombier  *	Open a history file
804*7dd7cddfSDavid du Colombier  *	Format is:
805*7dd7cddfSDavid du Colombier  *	Bytes 1, 2: HMAGIC - just to check that we are dealing with
806*7dd7cddfSDavid du Colombier  *		    the correct object
807*7dd7cddfSDavid du Colombier  *	Then follows a number of stored commands
808*7dd7cddfSDavid du Colombier  *	Each command is
809*7dd7cddfSDavid du Colombier  *	<command byte><command number(4 bytes)><bytes><null>
810*7dd7cddfSDavid du Colombier  */
811*7dd7cddfSDavid du Colombier # define HMAGIC1		0xab
812*7dd7cddfSDavid du Colombier # define HMAGIC2		0xcd
813*7dd7cddfSDavid du Colombier # define COMMAND		0xff
814*7dd7cddfSDavid du Colombier 
815*7dd7cddfSDavid du Colombier void
hist_init(s)816*7dd7cddfSDavid du Colombier hist_init(s)
817*7dd7cddfSDavid du Colombier 	Source *s;
818*7dd7cddfSDavid du Colombier {
819*7dd7cddfSDavid du Colombier 	unsigned char	*base;
820*7dd7cddfSDavid du Colombier 	int	lines;
821*7dd7cddfSDavid du Colombier 	int	fd;
822*7dd7cddfSDavid du Colombier 
823*7dd7cddfSDavid du Colombier 	if (Flag(FTALKING) == 0)
824*7dd7cddfSDavid du Colombier 		return;
825*7dd7cddfSDavid du Colombier 
826*7dd7cddfSDavid du Colombier 	hstarted = 1;
827*7dd7cddfSDavid du Colombier 
828*7dd7cddfSDavid du Colombier 	hist_source = s;
829*7dd7cddfSDavid du Colombier 
830*7dd7cddfSDavid du Colombier 	hname = str_val(global("HISTFILE"));
831*7dd7cddfSDavid du Colombier 	if (hname == NULL)
832*7dd7cddfSDavid du Colombier 		return;
833*7dd7cddfSDavid du Colombier 	hname = str_save(hname, APERM);
834*7dd7cddfSDavid du Colombier 
835*7dd7cddfSDavid du Colombier   retry:
836*7dd7cddfSDavid du Colombier 	/* we have a file and are interactive */
837*7dd7cddfSDavid du Colombier 	if ((fd = open(hname, O_RDWR|O_CREAT|O_APPEND, 0600)) < 0)
838*7dd7cddfSDavid du Colombier 		return;
839*7dd7cddfSDavid du Colombier 
840*7dd7cddfSDavid du Colombier 	histfd = savefd(fd, 0);
841*7dd7cddfSDavid du Colombier 
842*7dd7cddfSDavid du Colombier 	(void) flock(histfd, LOCK_EX);
843*7dd7cddfSDavid du Colombier 
844*7dd7cddfSDavid du Colombier 	hsize = lseek(histfd, 0L, SEEK_END);
845*7dd7cddfSDavid du Colombier 
846*7dd7cddfSDavid du Colombier 	if (hsize == 0) {
847*7dd7cddfSDavid du Colombier 		/* add magic */
848*7dd7cddfSDavid du Colombier 		if (sprinkle(histfd)) {
849*7dd7cddfSDavid du Colombier 			hist_finish();
850*7dd7cddfSDavid du Colombier 			return;
851*7dd7cddfSDavid du Colombier 		}
852*7dd7cddfSDavid du Colombier 	}
853*7dd7cddfSDavid du Colombier 	else if (hsize > 0) {
854*7dd7cddfSDavid du Colombier 		/*
855*7dd7cddfSDavid du Colombier 		 * we have some data
856*7dd7cddfSDavid du Colombier 		 */
857*7dd7cddfSDavid du Colombier 		base = (unsigned char *)mmap(0, hsize, PROT_READ, MAP_FLAGS, histfd, 0);
858*7dd7cddfSDavid du Colombier 		/*
859*7dd7cddfSDavid du Colombier 		 * check on its validity
860*7dd7cddfSDavid du Colombier 		 */
861*7dd7cddfSDavid du Colombier 		if ((int)base == -1 || *base != HMAGIC1 || base[1] != HMAGIC2) {
862*7dd7cddfSDavid du Colombier 			if ((int)base !=  -1)
863*7dd7cddfSDavid du Colombier 				munmap((caddr_t)base, hsize);
864*7dd7cddfSDavid du Colombier 			hist_finish();
865*7dd7cddfSDavid du Colombier 			unlink(hname);
866*7dd7cddfSDavid du Colombier 			goto retry;
867*7dd7cddfSDavid du Colombier 		}
868*7dd7cddfSDavid du Colombier 		if (hsize > 2) {
869*7dd7cddfSDavid du Colombier 			lines = hist_count_lines(base+2, hsize-2);
870*7dd7cddfSDavid du Colombier 			if (lines > histsize) {
871*7dd7cddfSDavid du Colombier 				/* we need to make the file smaller */
872*7dd7cddfSDavid du Colombier 				if (hist_shrink(base, hsize))
873*7dd7cddfSDavid du Colombier 					unlink(hname);
874*7dd7cddfSDavid du Colombier 				munmap((caddr_t)base, hsize);
875*7dd7cddfSDavid du Colombier 				hist_finish();
876*7dd7cddfSDavid du Colombier 				goto retry;
877*7dd7cddfSDavid du Colombier 			}
878*7dd7cddfSDavid du Colombier 		}
879*7dd7cddfSDavid du Colombier 		histload(hist_source, base+2, hsize-2);
880*7dd7cddfSDavid du Colombier 		munmap((caddr_t)base, hsize);
881*7dd7cddfSDavid du Colombier 	}
882*7dd7cddfSDavid du Colombier 	(void) flock(histfd, LOCK_UN);
883*7dd7cddfSDavid du Colombier 	hsize = lseek(histfd, 0L, SEEK_END);
884*7dd7cddfSDavid du Colombier }
885*7dd7cddfSDavid du Colombier 
886*7dd7cddfSDavid du Colombier typedef enum state {
887*7dd7cddfSDavid du Colombier 	shdr,		/* expecting a header */
888*7dd7cddfSDavid du Colombier 	sline,		/* looking for a null byte to end the line */
889*7dd7cddfSDavid du Colombier 	sn1,		/* bytes 1 to 4 of a line no */
890*7dd7cddfSDavid du Colombier 	sn2, sn3, sn4,
891*7dd7cddfSDavid du Colombier } State;
892*7dd7cddfSDavid du Colombier 
893*7dd7cddfSDavid du Colombier static int
hist_count_lines(base,bytes)894*7dd7cddfSDavid du Colombier hist_count_lines(base, bytes)
895*7dd7cddfSDavid du Colombier 	register unsigned char *base;
896*7dd7cddfSDavid du Colombier 	register int bytes;
897*7dd7cddfSDavid du Colombier {
898*7dd7cddfSDavid du Colombier 	State state = shdr;
899*7dd7cddfSDavid du Colombier 	register lines = 0;
900*7dd7cddfSDavid du Colombier 
901*7dd7cddfSDavid du Colombier 	while (bytes--) {
902*7dd7cddfSDavid du Colombier 		switch (state)
903*7dd7cddfSDavid du Colombier 		{
904*7dd7cddfSDavid du Colombier 		case shdr:
905*7dd7cddfSDavid du Colombier 			if (*base == COMMAND)
906*7dd7cddfSDavid du Colombier 				state = sn1;
907*7dd7cddfSDavid du Colombier 			break;
908*7dd7cddfSDavid du Colombier 		case sn1:
909*7dd7cddfSDavid du Colombier 			state = sn2; break;
910*7dd7cddfSDavid du Colombier 		case sn2:
911*7dd7cddfSDavid du Colombier 			state = sn3; break;
912*7dd7cddfSDavid du Colombier 		case sn3:
913*7dd7cddfSDavid du Colombier 			state = sn4; break;
914*7dd7cddfSDavid du Colombier 		case sn4:
915*7dd7cddfSDavid du Colombier 			state = sline; break;
916*7dd7cddfSDavid du Colombier 		case sline:
917*7dd7cddfSDavid du Colombier 			if (*base == '\0')
918*7dd7cddfSDavid du Colombier 				lines++, state = shdr;
919*7dd7cddfSDavid du Colombier 		}
920*7dd7cddfSDavid du Colombier 		base++;
921*7dd7cddfSDavid du Colombier 	}
922*7dd7cddfSDavid du Colombier 	return lines;
923*7dd7cddfSDavid du Colombier }
924*7dd7cddfSDavid du Colombier 
925*7dd7cddfSDavid du Colombier /*
926*7dd7cddfSDavid du Colombier  *	Shrink the history file to histsize lines
927*7dd7cddfSDavid du Colombier  */
928*7dd7cddfSDavid du Colombier static int
hist_shrink(oldbase,oldbytes)929*7dd7cddfSDavid du Colombier hist_shrink(oldbase, oldbytes)
930*7dd7cddfSDavid du Colombier 	unsigned char *oldbase;
931*7dd7cddfSDavid du Colombier 	int oldbytes;
932*7dd7cddfSDavid du Colombier {
933*7dd7cddfSDavid du Colombier 	int fd;
934*7dd7cddfSDavid du Colombier 	char	nfile[1024];
935*7dd7cddfSDavid du Colombier 	struct	stat statb;
936*7dd7cddfSDavid du Colombier 	unsigned char *nbase = oldbase;
937*7dd7cddfSDavid du Colombier 	int nbytes = oldbytes;
938*7dd7cddfSDavid du Colombier 
939*7dd7cddfSDavid du Colombier 	nbase = hist_skip_back(nbase, &nbytes, histsize);
940*7dd7cddfSDavid du Colombier 	if (nbase == NULL)
941*7dd7cddfSDavid du Colombier 		return 1;
942*7dd7cddfSDavid du Colombier 	if (nbase == oldbase)
943*7dd7cddfSDavid du Colombier 		return 0;
944*7dd7cddfSDavid du Colombier 
945*7dd7cddfSDavid du Colombier 	/*
946*7dd7cddfSDavid du Colombier 	 *	create temp file
947*7dd7cddfSDavid du Colombier 	 */
948*7dd7cddfSDavid du Colombier 	(void) shf_snprintf(nfile, sizeof(nfile), "%s.%d", hname, procpid);
949*7dd7cddfSDavid du Colombier 	if ((fd = creat(nfile, 0600)) < 0)
950*7dd7cddfSDavid du Colombier 		return 1;
951*7dd7cddfSDavid du Colombier 
952*7dd7cddfSDavid du Colombier 	if (sprinkle(fd)) {
953*7dd7cddfSDavid du Colombier 		close(fd);
954*7dd7cddfSDavid du Colombier 		unlink(nfile);
955*7dd7cddfSDavid du Colombier 		return 1;
956*7dd7cddfSDavid du Colombier 	}
957*7dd7cddfSDavid du Colombier 	if (write(fd, nbase, nbytes) != nbytes) {
958*7dd7cddfSDavid du Colombier 		close(fd);
959*7dd7cddfSDavid du Colombier 		unlink(nfile);
960*7dd7cddfSDavid du Colombier 		return 1;
961*7dd7cddfSDavid du Colombier 	}
962*7dd7cddfSDavid du Colombier 	/*
963*7dd7cddfSDavid du Colombier 	 *	worry about who owns this file
964*7dd7cddfSDavid du Colombier 	 */
965*7dd7cddfSDavid du Colombier 	if (fstat(histfd, &statb) >= 0)
966*7dd7cddfSDavid du Colombier 		fchown(fd, statb.st_uid, statb.st_gid);
967*7dd7cddfSDavid du Colombier 	close(fd);
968*7dd7cddfSDavid du Colombier 
969*7dd7cddfSDavid du Colombier 	/*
970*7dd7cddfSDavid du Colombier 	 *	rename
971*7dd7cddfSDavid du Colombier 	 */
972*7dd7cddfSDavid du Colombier 	if (rename(nfile, hname) < 0)
973*7dd7cddfSDavid du Colombier 		return 1;
974*7dd7cddfSDavid du Colombier 	return 0;
975*7dd7cddfSDavid du Colombier }
976*7dd7cddfSDavid du Colombier 
977*7dd7cddfSDavid du Colombier 
978*7dd7cddfSDavid du Colombier /*
979*7dd7cddfSDavid du Colombier  *	find a pointer to the data `no' back from the end of the file
980*7dd7cddfSDavid du Colombier  *	return the pointer and the number of bytes left
981*7dd7cddfSDavid du Colombier  */
982*7dd7cddfSDavid du Colombier static unsigned char *
hist_skip_back(base,bytes,no)983*7dd7cddfSDavid du Colombier hist_skip_back(base, bytes, no)
984*7dd7cddfSDavid du Colombier 	unsigned char *base;
985*7dd7cddfSDavid du Colombier 	int *bytes;
986*7dd7cddfSDavid du Colombier 	int no;
987*7dd7cddfSDavid du Colombier {
988*7dd7cddfSDavid du Colombier 	register int lines = 0;
989*7dd7cddfSDavid du Colombier 	register unsigned char *ep;
990*7dd7cddfSDavid du Colombier 
991*7dd7cddfSDavid du Colombier 	for (ep = base + *bytes; --ep > base; ) {
992*7dd7cddfSDavid du Colombier 		/* this doesn't really work: the 4 byte line number that is
993*7dd7cddfSDavid du Colombier 		 * encoded after the COMMAND byte can itself contain the
994*7dd7cddfSDavid du Colombier 		 * COMMAND byte....
995*7dd7cddfSDavid du Colombier 		 */
996*7dd7cddfSDavid du Colombier 		for (; ep > base && *ep != COMMAND; ep--)
997*7dd7cddfSDavid du Colombier 			;
998*7dd7cddfSDavid du Colombier 		if (ep == base)
999*7dd7cddfSDavid du Colombier 			break;
1000*7dd7cddfSDavid du Colombier 		if (++lines == no) {
1001*7dd7cddfSDavid du Colombier 			*bytes = *bytes - ((char *)ep - (char *)base);
1002*7dd7cddfSDavid du Colombier 			return ep;
1003*7dd7cddfSDavid du Colombier 		}
1004*7dd7cddfSDavid du Colombier 	}
1005*7dd7cddfSDavid du Colombier 	return NULL;
1006*7dd7cddfSDavid du Colombier }
1007*7dd7cddfSDavid du Colombier 
1008*7dd7cddfSDavid du Colombier /*
1009*7dd7cddfSDavid du Colombier  *	load the history structure from the stored data
1010*7dd7cddfSDavid du Colombier  */
1011*7dd7cddfSDavid du Colombier static void
histload(s,base,bytes)1012*7dd7cddfSDavid du Colombier histload(s, base, bytes)
1013*7dd7cddfSDavid du Colombier 	Source *s;
1014*7dd7cddfSDavid du Colombier 	register unsigned char *base;
1015*7dd7cddfSDavid du Colombier 	register int bytes;
1016*7dd7cddfSDavid du Colombier {
1017*7dd7cddfSDavid du Colombier 	State state;
1018*7dd7cddfSDavid du Colombier 	int	lno;
1019*7dd7cddfSDavid du Colombier 	unsigned char	*line;
1020*7dd7cddfSDavid du Colombier 
1021*7dd7cddfSDavid du Colombier 	for (state = shdr; bytes-- > 0; base++) {
1022*7dd7cddfSDavid du Colombier 		switch (state) {
1023*7dd7cddfSDavid du Colombier 		case shdr:
1024*7dd7cddfSDavid du Colombier 			if (*base == COMMAND)
1025*7dd7cddfSDavid du Colombier 				state = sn1;
1026*7dd7cddfSDavid du Colombier 			break;
1027*7dd7cddfSDavid du Colombier 		case sn1:
1028*7dd7cddfSDavid du Colombier 			lno = (((*base)&0xff)<<24);
1029*7dd7cddfSDavid du Colombier 			state = sn2;
1030*7dd7cddfSDavid du Colombier 			break;
1031*7dd7cddfSDavid du Colombier 		case sn2:
1032*7dd7cddfSDavid du Colombier 			lno |= (((*base)&0xff)<<16);
1033*7dd7cddfSDavid du Colombier 			state = sn3;
1034*7dd7cddfSDavid du Colombier 			break;
1035*7dd7cddfSDavid du Colombier 		case sn3:
1036*7dd7cddfSDavid du Colombier 			lno |= (((*base)&0xff)<<8);
1037*7dd7cddfSDavid du Colombier 			state = sn4;
1038*7dd7cddfSDavid du Colombier 			break;
1039*7dd7cddfSDavid du Colombier 		case sn4:
1040*7dd7cddfSDavid du Colombier 			lno |= (*base)&0xff;
1041*7dd7cddfSDavid du Colombier 			line = base+1;
1042*7dd7cddfSDavid du Colombier 			state = sline;
1043*7dd7cddfSDavid du Colombier 			break;
1044*7dd7cddfSDavid du Colombier 		case sline:
1045*7dd7cddfSDavid du Colombier 			if (*base == '\0') {
1046*7dd7cddfSDavid du Colombier 				/* worry about line numbers */
1047*7dd7cddfSDavid du Colombier 				if (histptr >= history && lno-1 != s->line) {
1048*7dd7cddfSDavid du Colombier 					/* a replacement ? */
1049*7dd7cddfSDavid du Colombier 					histinsert(s, lno, line);
1050*7dd7cddfSDavid du Colombier 				}
1051*7dd7cddfSDavid du Colombier 				else {
1052*7dd7cddfSDavid du Colombier 					s->line = lno;
1053*7dd7cddfSDavid du Colombier 					histsave(lno, (char *)line, 0);
1054*7dd7cddfSDavid du Colombier 				}
1055*7dd7cddfSDavid du Colombier 				state = shdr;
1056*7dd7cddfSDavid du Colombier 			}
1057*7dd7cddfSDavid du Colombier 		}
1058*7dd7cddfSDavid du Colombier 	}
1059*7dd7cddfSDavid du Colombier }
1060*7dd7cddfSDavid du Colombier 
1061*7dd7cddfSDavid du Colombier /*
1062*7dd7cddfSDavid du Colombier  *	Insert a line into the history at a specified number
1063*7dd7cddfSDavid du Colombier  */
1064*7dd7cddfSDavid du Colombier static void
histinsert(s,lno,line)1065*7dd7cddfSDavid du Colombier histinsert(s, lno, line)
1066*7dd7cddfSDavid du Colombier 	Source *s;
1067*7dd7cddfSDavid du Colombier 	int lno;
1068*7dd7cddfSDavid du Colombier 	unsigned char *line;
1069*7dd7cddfSDavid du Colombier {
1070*7dd7cddfSDavid du Colombier 	register char **hp;
1071*7dd7cddfSDavid du Colombier 
1072*7dd7cddfSDavid du Colombier 	if (lno >= s->line-(histptr-history) && lno <= s->line) {
1073*7dd7cddfSDavid du Colombier 		hp = &histptr[lno-s->line];
1074*7dd7cddfSDavid du Colombier 		if (*hp)
1075*7dd7cddfSDavid du Colombier 			afree((void*)*hp, APERM);
1076*7dd7cddfSDavid du Colombier 		*hp = str_save((char *)line, APERM);
1077*7dd7cddfSDavid du Colombier 	}
1078*7dd7cddfSDavid du Colombier }
1079*7dd7cddfSDavid du Colombier 
1080*7dd7cddfSDavid du Colombier /*
1081*7dd7cddfSDavid du Colombier  *	write a command to the end of the history file
1082*7dd7cddfSDavid du Colombier  *	This *MAY* seem easy but it's also necessary to check
1083*7dd7cddfSDavid du Colombier  *	that the history file has not changed in size.
1084*7dd7cddfSDavid du Colombier  *	If it has - then some other shell has written to it
1085*7dd7cddfSDavid du Colombier  *	and we should read those commands to update our history
1086*7dd7cddfSDavid du Colombier  */
1087*7dd7cddfSDavid du Colombier static void
writehistfile(lno,cmd)1088*7dd7cddfSDavid du Colombier writehistfile(lno, cmd)
1089*7dd7cddfSDavid du Colombier 	int lno;
1090*7dd7cddfSDavid du Colombier 	char *cmd;
1091*7dd7cddfSDavid du Colombier {
1092*7dd7cddfSDavid du Colombier 	int	sizenow;
1093*7dd7cddfSDavid du Colombier 	unsigned char	*base;
1094*7dd7cddfSDavid du Colombier 	unsigned char	*new;
1095*7dd7cddfSDavid du Colombier 	int	bytes;
1096*7dd7cddfSDavid du Colombier 	char	hdr[5];
1097*7dd7cddfSDavid du Colombier 
1098*7dd7cddfSDavid du Colombier 	(void) flock(histfd, LOCK_EX);
1099*7dd7cddfSDavid du Colombier 	sizenow = lseek(histfd, 0L, SEEK_END);
1100*7dd7cddfSDavid du Colombier 	if (sizenow != hsize) {
1101*7dd7cddfSDavid du Colombier 		/*
1102*7dd7cddfSDavid du Colombier 		 *	Things have changed
1103*7dd7cddfSDavid du Colombier 		 */
1104*7dd7cddfSDavid du Colombier 		if (sizenow > hsize) {
1105*7dd7cddfSDavid du Colombier 			/* someone has added some lines */
1106*7dd7cddfSDavid du Colombier 			bytes = sizenow - hsize;
1107*7dd7cddfSDavid du Colombier 			base = (unsigned char *)mmap(0, sizenow, PROT_READ, MAP_FLAGS, histfd, 0);
1108*7dd7cddfSDavid du Colombier 			if ((int)base == -1)
1109*7dd7cddfSDavid du Colombier 				goto bad;
1110*7dd7cddfSDavid du Colombier 			new = base + hsize;
1111*7dd7cddfSDavid du Colombier 			if (*new != COMMAND) {
1112*7dd7cddfSDavid du Colombier 				munmap((caddr_t)base, sizenow);
1113*7dd7cddfSDavid du Colombier 				goto bad;
1114*7dd7cddfSDavid du Colombier 			}
1115*7dd7cddfSDavid du Colombier 			hist_source->line--;
1116*7dd7cddfSDavid du Colombier 			histload(hist_source, new, bytes);
1117*7dd7cddfSDavid du Colombier 			hist_source->line++;
1118*7dd7cddfSDavid du Colombier 			lno = hist_source->line;
1119*7dd7cddfSDavid du Colombier 			munmap((caddr_t)base, sizenow);
1120*7dd7cddfSDavid du Colombier 			hsize = sizenow;
1121*7dd7cddfSDavid du Colombier 		} else {
1122*7dd7cddfSDavid du Colombier 			/* it has shrunk */
1123*7dd7cddfSDavid du Colombier 			/* but to what? */
1124*7dd7cddfSDavid du Colombier 			/* we'll give up for now */
1125*7dd7cddfSDavid du Colombier 			goto bad;
1126*7dd7cddfSDavid du Colombier 		}
1127*7dd7cddfSDavid du Colombier 	}
1128*7dd7cddfSDavid du Colombier 	/*
1129*7dd7cddfSDavid du Colombier 	 *	we can write our bit now
1130*7dd7cddfSDavid du Colombier 	 */
1131*7dd7cddfSDavid du Colombier 	hdr[0] = COMMAND;
1132*7dd7cddfSDavid du Colombier 	hdr[1] = (lno>>24)&0xff;
1133*7dd7cddfSDavid du Colombier 	hdr[2] = (lno>>16)&0xff;
1134*7dd7cddfSDavid du Colombier 	hdr[3] = (lno>>8)&0xff;
1135*7dd7cddfSDavid du Colombier 	hdr[4] = lno&0xff;
1136*7dd7cddfSDavid du Colombier 	(void) write(histfd, hdr, 5);
1137*7dd7cddfSDavid du Colombier 	(void) write(histfd, cmd, strlen(cmd)+1);
1138*7dd7cddfSDavid du Colombier 	hsize = lseek(histfd, 0L, SEEK_END);
1139*7dd7cddfSDavid du Colombier 	(void) flock(histfd, LOCK_UN);
1140*7dd7cddfSDavid du Colombier 	return;
1141*7dd7cddfSDavid du Colombier bad:
1142*7dd7cddfSDavid du Colombier 	hist_finish();
1143*7dd7cddfSDavid du Colombier }
1144*7dd7cddfSDavid du Colombier 
1145*7dd7cddfSDavid du Colombier void
hist_finish()1146*7dd7cddfSDavid du Colombier hist_finish()
1147*7dd7cddfSDavid du Colombier {
1148*7dd7cddfSDavid du Colombier 	(void) flock(histfd, LOCK_UN);
1149*7dd7cddfSDavid du Colombier 	(void) close(histfd);
1150*7dd7cddfSDavid du Colombier 	histfd = 0;
1151*7dd7cddfSDavid du Colombier }
1152*7dd7cddfSDavid du Colombier 
1153*7dd7cddfSDavid du Colombier /*
1154*7dd7cddfSDavid du Colombier  *	add magic to the history file
1155*7dd7cddfSDavid du Colombier  */
1156*7dd7cddfSDavid du Colombier static int
sprinkle(fd)1157*7dd7cddfSDavid du Colombier sprinkle(fd)
1158*7dd7cddfSDavid du Colombier 	int fd;
1159*7dd7cddfSDavid du Colombier {
1160*7dd7cddfSDavid du Colombier 	static char mag[] = { HMAGIC1, HMAGIC2 };
1161*7dd7cddfSDavid du Colombier 
1162*7dd7cddfSDavid du Colombier 	return(write(fd, mag, 2) != 2);
1163*7dd7cddfSDavid du Colombier }
1164*7dd7cddfSDavid du Colombier 
1165*7dd7cddfSDavid du Colombier # endif
1166*7dd7cddfSDavid du Colombier #else /* HISTORY */
1167*7dd7cddfSDavid du Colombier 
1168*7dd7cddfSDavid du Colombier /* No history to be compiled in: dummy routines to avoid lots more ifdefs */
1169*7dd7cddfSDavid du Colombier void
init_histvec()1170*7dd7cddfSDavid du Colombier init_histvec()
1171*7dd7cddfSDavid du Colombier {
1172*7dd7cddfSDavid du Colombier }
1173*7dd7cddfSDavid du Colombier void
hist_init(s)1174*7dd7cddfSDavid du Colombier hist_init(s)
1175*7dd7cddfSDavid du Colombier 	Source *s;
1176*7dd7cddfSDavid du Colombier {
1177*7dd7cddfSDavid du Colombier }
1178*7dd7cddfSDavid du Colombier void
hist_finish()1179*7dd7cddfSDavid du Colombier hist_finish()
1180*7dd7cddfSDavid du Colombier {
1181*7dd7cddfSDavid du Colombier }
1182*7dd7cddfSDavid du Colombier void
histsave(lno,cmd,dowrite)1183*7dd7cddfSDavid du Colombier histsave(lno, cmd, dowrite)
1184*7dd7cddfSDavid du Colombier 	int lno;
1185*7dd7cddfSDavid du Colombier 	const char *cmd;
1186*7dd7cddfSDavid du Colombier 	int dowrite;
1187*7dd7cddfSDavid du Colombier {
1188*7dd7cddfSDavid du Colombier 	errorf("history not enabled");
1189*7dd7cddfSDavid du Colombier }
1190*7dd7cddfSDavid du Colombier #endif /* HISTORY */
1191