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