#include "sam.h" /* * Structure of Undo list: * The Undo structure follows any associated data, so the list * can be read backwards: read the structure, then read whatever * data is associated (insert string, file name) and precedes it. * The structure includes the previous value of the modify bit * and a sequence number; successive Undo structures with the * same sequence number represent simultaneous changes. */ typedef struct Undo Undo; typedef struct Merge Merge; struct Undo { short type; /* Delete, Insert, Filename, Dot, Mark */ short mod; /* modify bit */ uint seq; /* sequence number */ uint p0; /* location of change (unused in f) */ uint n; /* # runes in string or file name */ }; struct Merge { File *f; uint seq; /* of logged change */ uint p0; /* location of change (unused in f) */ uint n; /* # runes to delete */ uint nbuf; /* # runes to insert */ Rune buf[RBUFSIZE]; }; enum { Maxmerge = 50, Undosize = sizeof(Undo)/sizeof(Rune) }; static Merge merge; File* fileopen(void) { File *f; f = emalloc(sizeof(File)); f->dot.f = f; f->ndot.f = f; f->seq = 0; f->mod = FALSE; f->unread = TRUE; Strinit0(&f->name); return f; } int fileisdirty(File *f) { return f->seq != f->cleanseq; } static void wrinsert(Buffer *delta, int seq, int mod, uint p0, Rune *s, uint ns) { Undo u; u.type = Insert; u.mod = mod; u.seq = seq; u.p0 = p0; u.n = ns; bufinsert(delta, delta->nc, s, ns); bufinsert(delta, delta->nc, (Rune*)&u, Undosize); } static void wrdelete(Buffer *delta, int seq, int mod, uint p0, uint p1) { Undo u; u.type = Delete; u.mod = mod; u.seq = seq; u.p0 = p0; u.n = p1 - p0; bufinsert(delta, delta->nc, (Rune*)&u, Undosize); } void flushmerge(void) { File *f; f = merge.f; if(f == nil) return; if(merge.seq != f->seq) panic("flushmerge seq mismatch"); if(merge.n != 0) wrdelete(&f->epsilon, f->seq, TRUE, merge.p0, merge.p0+merge.n); if(merge.nbuf != 0) wrinsert(&f->epsilon, f->seq, TRUE, merge.p0+merge.n, merge.buf, merge.nbuf); merge.f = nil; merge.n = 0; merge.nbuf = 0; } void mergeextend(File *f, uint p0) { uint mp0n; mp0n = merge.p0+merge.n; if(mp0n != p0){ bufread(&f->b, mp0n, merge.buf+merge.nbuf, p0-mp0n); merge.nbuf += p0-mp0n; merge.n = p0-merge.p0; } } /* * like fileundelete, but get the data from arguments */ void loginsert(File *f, uint p0, Rune *s, uint ns) { if(f->rescuing) return; if(ns == 0) return; if(ns<0 || ns>STRSIZE) panic("loginsert"); if(f->seq < seq) filemark(f); if(p0 < f->hiposn) error(Esequence); if(merge.f != f || p0-(merge.p0+merge.n)>Maxmerge /* too far */ || merge.nbuf+((p0+ns)-(merge.p0+merge.n))>=RBUFSIZE) /* too long */ flushmerge(); if(ns>=RBUFSIZE){ if(!(merge.n == 0 && merge.nbuf == 0 && merge.f == nil)) panic("loginsert bad merge state"); wrinsert(&f->epsilon, f->seq, TRUE, p0, s, ns); }else{ if(merge.f != f){ merge.f = f; merge.p0 = p0; merge.seq = f->seq; } mergeextend(f, p0); /* append string to merge */ runemove(merge.buf+merge.nbuf, s, ns); merge.nbuf += ns; } f->hiposn = p0; if(!f->unread && !f->mod) state(f, Dirty); } void logdelete(File *f, uint p0, uint p1) { if(f->rescuing) return; if(p0 == p1) return; if(f->seq < seq) filemark(f); if(p0 < f->hiposn) error(Esequence); if(merge.f != f || p0-(merge.p0+merge.n)>Maxmerge /* too far */ || merge.nbuf+(p0-(merge.p0+merge.n))>=RBUFSIZE){ /* too long */ flushmerge(); merge.f = f; merge.p0 = p0; merge.seq = f->seq; } mergeextend(f, p0); /* add to deletion */ merge.n = p1-merge.p0; f->hiposn = p1; if(!f->unread && !f->mod) state(f, Dirty); } /* * like fileunsetname, but get the data from arguments */ void logsetname(File *f, String *s) { Undo u; Buffer *delta; if(f->rescuing) return; if(f->unread){ /* This is setting initial file name */ filesetname(f, s); return; } if(f->seq < seq) filemark(f); /* undo a file name change by restoring old name */ delta = &f->epsilon; u.type = Filename; u.mod = TRUE; u.seq = f->seq; u.p0 = 0; /* unused */ u.n = s->n; if(s->n) bufinsert(delta, delta->nc, s->s, s->n); bufinsert(delta, delta->nc, (Rune*)&u, Undosize); if(!f->unread && !f->mod) state(f, Dirty); } #ifdef NOTEXT File* fileaddtext(File *f, Text *t) { if(f == nil){ f = emalloc(sizeof(File)); f->unread = TRUE; } f->text = realloc(f->text, (f->ntext+1)*sizeof(Text*)); f->text[f->ntext++] = t; f->curtext = t; return f; } void filedeltext(File *f, Text *t) { int i; for(i=0; i<f->ntext; i++) if(f->text[i] == t) goto Found; panic("can't find text in filedeltext"); Found: f->ntext--; if(f->ntext == 0){ fileclose(f); return; } memmove(f->text+i, f->text+i+1, (f->ntext-i)*sizeof(Text*)); if(f->curtext == t) f->curtext = f->text[0]; } #endif void fileuninsert(File *f, Buffer *delta, uint p0, uint ns) { Undo u; /* undo an insertion by deleting */ u.type = Delete; u.mod = f->mod; u.seq = f->seq; u.p0 = p0; u.n = ns; bufinsert(delta, delta->nc, (Rune*)&u, Undosize); } void fileundelete(File *f, Buffer *delta, uint p0, uint p1) { Undo u; Rune *buf; uint i, n; /* undo a deletion by inserting */ u.type = Insert; u.mod = f->mod; u.seq = f->seq; u.p0 = p0; u.n = p1-p0; buf = fbufalloc(); for(i=p0; i<p1; i+=n){ n = p1 - i; if(n > RBUFSIZE) n = RBUFSIZE; bufread(&f->b, i, buf, n); bufinsert(delta, delta->nc, buf, n); } fbuffree(buf); bufinsert(delta, delta->nc, (Rune*)&u, Undosize); } int filereadc(File *f, uint q) { Rune r; if(q >= f->b.nc) return -1; bufread(&f->b, q, &r, 1); return r; } void filesetname(File *f, String *s) { if(!f->unread) /* This is setting initial file name */ fileunsetname(f, &f->delta); Strduplstr(&f->name, s); sortname(f); f->unread = TRUE; } void fileunsetname(File *f, Buffer *delta) { String s; Undo u; /* undo a file name change by restoring old name */ u.type = Filename; u.mod = f->mod; u.seq = f->seq; u.p0 = 0; /* unused */ Strinit(&s); Strduplstr(&s, &f->name); fullname(&s); u.n = s.n; if(s.n) bufinsert(delta, delta->nc, s.s, s.n); bufinsert(delta, delta->nc, (Rune*)&u, Undosize); Strclose(&s); } void fileunsetdot(File *f, Buffer *delta, Range dot) { Undo u; u.type = Dot; u.mod = f->mod; u.seq = f->seq; u.p0 = dot.p1; u.n = dot.p2 - dot.p1; bufinsert(delta, delta->nc, (Rune*)&u, Undosize); } void fileunsetmark(File *f, Buffer *delta, Range mark) { Undo u; u.type = Mark; u.mod = f->mod; u.seq = f->seq; u.p0 = mark.p1; u.n = mark.p2 - mark.p1; bufinsert(delta, delta->nc, (Rune*)&u, Undosize); } uint fileload(File *f, uint p0, int fd, int *nulls) { if(f->seq > 0) panic("undo in file.load unimplemented"); return bufload(&f->b, p0, fd, nulls); } int fileupdate(File *f, int notrans, int toterm) { uint p1, p2; int mod; if(f->rescuing) return FALSE; flushmerge(); /* * fix the modification bit * subtle point: don't save it away in the log. * * if another change is made, the correct f->mod * state is saved in the undo log by filemark * when setting the dot and mark. * * if the change is undone, the correct state is * saved from f in the fileun... routines. */ mod = f->mod; f->mod = f->prevmod; if(f == cmd) notrans = TRUE; else{ fileunsetdot(f, &f->delta, f->prevdot); fileunsetmark(f, &f->delta, f->prevmark); } f->dot = f->ndot; fileundo(f, FALSE, !notrans, &p1, &p2, toterm); f->mod = mod; if(f->delta.nc == 0) f->seq = 0; if(f == cmd) return FALSE; if(f->mod){ f->closeok = 0; quitok = 0; }else f->closeok = 1; return TRUE; } long prevseq(Buffer *b) { Undo u; uint up; up = b->nc; if(up == 0) return 0; up -= Undosize; bufread(b, up, (Rune*)&u, Undosize); return u.seq; } long undoseq(File *f, int isundo) { if(isundo) return f->seq; return prevseq(&f->epsilon); } void fileundo(File *f, int isundo, int canredo, uint *q0p, uint *q1p, int flag) { Undo u; Rune *buf; uint i, n, up; uint stop; Buffer *delta, *epsilon; if(isundo){ /* undo; reverse delta onto epsilon, seq decreases */ delta = &f->delta; epsilon = &f->epsilon; stop = f->seq; }else{ /* redo; reverse epsilon onto delta, seq increases */ delta = &f->epsilon; epsilon = &f->delta; stop = 0; /* don't know yet */ } raspstart(f); while(delta->nc > 0){ /* rasp and buffer are in sync; sync with wire if needed */ if(needoutflush()) raspflush(f); up = delta->nc-Undosize; bufread(delta, up, (Rune*)&u, Undosize); if(isundo){ if(u.seq < stop){ f->seq = u.seq; raspdone(f, flag); return; } }else{ if(stop == 0) stop = u.seq; if(u.seq > stop){ raspdone(f, flag); return; } } switch(u.type){ default: panic("undo unknown u.type"); break; case Delete: f->seq = u.seq; if(canredo) fileundelete(f, epsilon, u.p0, u.p0+u.n); f->mod = u.mod; bufdelete(&f->b, u.p0, u.p0+u.n); raspdelete(f, u.p0, u.p0+u.n, flag); *q0p = u.p0; *q1p = u.p0; break; case Insert: f->seq = u.seq; if(canredo) fileuninsert(f, epsilon, u.p0, u.n); f->mod = u.mod; up -= u.n; buf = fbufalloc(); for(i=0; i<u.n; i+=n){ n = u.n - i; if(n > RBUFSIZE) n = RBUFSIZE; bufread(delta, up+i, buf, n); bufinsert(&f->b, u.p0+i, buf, n); raspinsert(f, u.p0+i, buf, n, flag); } fbuffree(buf); *q0p = u.p0; *q1p = u.p0+u.n; break; case Filename: f->seq = u.seq; if(canredo) fileunsetname(f, epsilon); f->mod = u.mod; up -= u.n; Strinsure(&f->name, u.n+1); bufread(delta, up, f->name.s, u.n); f->name.s[u.n] = 0; f->name.n = u.n; fixname(&f->name); sortname(f); break; case Dot: f->seq = u.seq; if(canredo) fileunsetdot(f, epsilon, f->dot.r); f->mod = u.mod; f->dot.r.p1 = u.p0; f->dot.r.p2 = u.p0 + u.n; break; case Mark: f->seq = u.seq; if(canredo) fileunsetmark(f, epsilon, f->mark); f->mod = u.mod; f->mark.p1 = u.p0; f->mark.p2 = u.p0 + u.n; break; } bufdelete(delta, up, delta->nc); } if(isundo) f->seq = 0; raspdone(f, flag); } void filereset(File *f) { bufreset(&f->delta); bufreset(&f->epsilon); f->seq = 0; } void fileclose(File *f) { Strclose(&f->name); bufclose(&f->b); bufclose(&f->delta); bufclose(&f->epsilon); if(f->rasp) listfree(f->rasp); free(f); } void filemark(File *f) { if(f->unread) return; if(f->epsilon.nc) bufdelete(&f->epsilon, 0, f->epsilon.nc); if(f != cmd){ f->prevdot = f->dot.r; f->prevmark = f->mark; f->prevseq = f->seq; f->prevmod = f->mod; } f->ndot = f->dot; f->seq = seq; f->hiposn = 0; }