v23i080:  ABC interactive programming environment, Part01/25
    Rich Salz 
    rsalz at bbn.com
       
    Tue Dec 18 05:34:13 AEST 1990
    
    
  
Submitted-by: Steven Pemberton <steven at cwi.nl>
Posting-number: Volume 23, Issue 80
Archive-name: abc/part01
This is a posting of an implementation of ABC, a new interactive
programming language. Versions for Unix, the Atari ST, the Macintosh,
and MS-DOS are being posted this week to the net.
ABC is an imperative language originally designed as a replacement for
BASIC: interactive, very easy to learn, but structured, high-level,
and easy to use.
It is suitable for general everyday programming, the sort of
programming that you would use BASIC, Pascal, or AWK for. It is not a
systems-programming language. It is an excellent teaching language,
and because it is interactive, excellent for prototyping. It is much
faster than 'bc' for doing quick calculations.
ABC programs are typically very compact, around a quarter to a fifth
the size of the equivalent Pascal or C program. However, this is not
at the cost of readability, on the contrary in fact.
ABC is simple to learn due to the small number of types in the
language (five). If you already know Pascal or something similar you
can learn the whole language in an hour or so.  It is easy to use
because the data-types are very high-level.
Fuller documentation, including examples, is in the file abcintro.doc
#! /bin/sh
# This is a shell archive.  Remove anything before this line, then feed it
# into a shell via "sh file" or similar.  To overwrite existing files,
# type "sh file -c".
# The tool that generated this appeared in the comp.sources.unix newsgroup;
# send mail to comp-sources-unix at uunet.uu.net if you want that tool.
# Contents:  MANIFEST abc abc/b abc/bed abc/bhdrs abc/bint1 abc/bint2
#   abc/bint3 abc/bint3/i3sou.c abc/bio abc/boot abc/btr abc/doc
#   abc/ehdrs abc/ex abc/ex/generate abc/ex/hanoi abc/ex/pi abc/ex/try
#   abc/ex/xref abc/ihdrs abc/keys abc/lin abc/lin/i1obj.c abc/scripts
#   abc/stc abc/tc abc/uhdrs abc/ukeys abc/unix
# Wrapped by rsalz at litchi.bbn.com on Mon Dec 17 13:27:50 1990
PATH=/bin:/usr/bin:/usr/ucb ; export PATH
echo If this archive is complete, you will see the following message:
echo '          "shar: End of archive 1 (of 25)."'
if test -f 'MANIFEST' -a "${1}" != "-c" ; then 
  echo shar: Will not clobber existing file \"'MANIFEST'\"
else
  echo shar: Extracting \"'MANIFEST'\" \(8839 characters\)
  sed "s/^X//" >'MANIFEST' <<'END_OF_FILE'
X   File Name		Archive #	Description
X----------------------------------------------------------
XMANIFEST                   1	
Xabc                        1	
Xabc/Makefile.unix         12	
Xabc/Problems              11	
Xabc/README                23	
Xabc/README2               23	
Xabc/Setup                  9	
Xabc/abc.1                  4	
Xabc/abc.hlp                7	
Xabc/abc.msg               10	
Xabc/b                      1	
Xabc/b/DEP                 24	
Xabc/b/MF                  25	
Xabc/b/b1file.c            23	
Xabc/b/b1grab.c            21	
Xabc/b/b1memo.c            22	
Xabc/b/b1mess.c            22	
Xabc/b/b1outp.c            21	
Xabc/b/getopt.c            23	
Xabc/bed                    1	
Xabc/bed/DEP               15	
Xabc/bed/MF                25	
Xabc/bed/e1cell.c          18	
Xabc/bed/e1code.c          24	
Xabc/bed/e1comm.c          21	
Xabc/bed/e1deco.c          11	
Xabc/bed/e1edit.c          19	
Xabc/bed/e1edoc.c          10	
Xabc/bed/e1erro.c          20	
Xabc/bed/e1eval.c          20	
Xabc/bed/e1getc.c           8	
Xabc/bed/e1goto.c          19	
Xabc/bed/e1gram.c          18	
Xabc/bed/e1ins2.c          18	
Xabc/bed/e1inse.c          17	
Xabc/bed/e1lexi.c          24	
Xabc/bed/e1line.c          20	
Xabc/bed/e1move.c          17	
Xabc/bed/e1node.c          14	
Xabc/bed/e1outp.c          17	
Xabc/bed/e1que1.c          13	
Xabc/bed/e1que2.c           6	
Xabc/bed/e1save.c          23	
Xabc/bed/e1scrn.c          14	
Xabc/bed/e1spos.c          21	
Xabc/bed/e1sugg.c           9	
Xabc/bed/e1supr.c           8	
Xabc/bed/e1tabl.c           4	
Xabc/bed/e1term.c          25	
Xabc/bed/e1wide.c          19	
Xabc/bhdrs                  1	
Xabc/bhdrs/b.h             22	
Xabc/bhdrs/b0lan.h         22	
Xabc/bhdrs/bcom.h          25	
Xabc/bhdrs/bedi.h          24	
Xabc/bhdrs/bfil.h          24	
Xabc/bhdrs/bgfx.h          25	
Xabc/bhdrs/bmem.h          25	
Xabc/bhdrs/bobj.h          21	
Xabc/bhdrs/getopt.h        25	
Xabc/bhdrs/release.h       25	
Xabc/bint1                  1	
Xabc/bint1/DEP             13	
Xabc/bint1/MF              25	
Xabc/bint1/i1fun.c         10	
Xabc/bint1/i1nua.c         14	
Xabc/bint1/i1nuc.c         16	
Xabc/bint1/i1nug.c         18	
Xabc/bint1/i1nui.c         17	
Xabc/bint1/i1num.c          2	
Xabc/bint1/i1nur.c         20	
Xabc/bint1/i1nut.c         22	
Xabc/bint2                  1	
Xabc/bint2/DEP             22	
Xabc/bint2/MF              25	
Xabc/bint2/i2ana.c         16	
Xabc/bint2/i2cmd.c         15	
Xabc/bint2/i2dis.c         19	
Xabc/bint2/i2exp.c          9	
Xabc/bint2/i2fix.c         21	
Xabc/bint2/i2gen.c          7	
Xabc/bint2/i2syn.c         11	
Xabc/bint2/i2tar.c         24	
Xabc/bint2/i2tes.c         21	
Xabc/bint2/i2uni.c         15	
Xabc/bint3                  1	
Xabc/bint3/MF              25	
Xabc/bint3/i3bws.c          7	
Xabc/bint3/i3com.c         22	
Xabc/bint3/i3env.c         21	
Xabc/bint3/i3err.c         16	
Xabc/bint3/i3fil.c         20	
Xabc/bint3/i3fpr.c         18	
Xabc/bint3/i3gfx.c         17	
Xabc/bint3/i3imm.c         22	
Xabc/bint3/i3in2.c         22	
Xabc/bint3/i3ini.c         22	
Xabc/bint3/i3int.c         15	
Xabc/bint3/i3loc.c         13	
Xabc/bint3/i3scr.c         13	
Xabc/bint3/i3sou.c          1	
Xabc/bint3/i3sta.c          8	
Xabc/bint3/i3typ.c         19	
Xabc/bio                    1	
Xabc/bio/DEP                5	
Xabc/bio/MF                25	
Xabc/bio/i4bio.c           24	
Xabc/bio/i4bio.h           24	
Xabc/bio/i4fil.c           20	
Xabc/bio/i4grp.c           23	
Xabc/bio/i4inp.c           25	
Xabc/bio/i4lis.c           25	
Xabc/bio/i4out.c           24	
Xabc/bio/i4rec.c           19	
Xabc/boot                   1	
Xabc/boot/Makefile         20	
Xabc/boot/Makefile.bsd     23	
Xabc/boot/README           23	
Xabc/boot/alloc.c          25	
Xabc/boot/comp.c           21	
Xabc/boot/dump.c            6	
Xabc/boot/grammar.abc      12	
Xabc/boot/lang.h           24	
Xabc/boot/main.h           22	
Xabc/boot/read.c           11	
Xabc/btr                    1	
Xabc/btr/DEP               23	
Xabc/btr/MF                25	
Xabc/btr/e1etex.c          24	
Xabc/btr/etex.h            25	
Xabc/btr/i1btr.c           21	
Xabc/btr/i1btr.h           19	
Xabc/btr/i1lta.c            5	
Xabc/btr/i1obj.c           14	
Xabc/btr/i1tex.c           12	
Xabc/btr/i1tlt.c           14	
Xabc/btr/i1tlt.h           25	
Xabc/ch_all                24	
Xabc/ch_clean              25	
Xabc/ch_config             10	
Xabc/ch_depend             25	
Xabc/ch_install            24	
Xabc/ch_makefiles          25	
Xabc/ch_messages           24	
Xabc/doc                    1	
Xabc/doc/ABCproject        23	
Xabc/doc/BugReport         24	
Xabc/doc/Structure         24	
Xabc/doc/abcintro.doc      16	
Xabc/ehdrs                  1	
Xabc/ehdrs/cell.h          24	
Xabc/ehdrs/code.h          25	
Xabc/ehdrs/erro.h          24	
Xabc/ehdrs/getc.h          23	
Xabc/ehdrs/gram.h          25	
Xabc/ehdrs/keys.h          22	
Xabc/ehdrs/node.h          22	
Xabc/ehdrs/queu.h          25	
Xabc/ehdrs/supr.h          23	
Xabc/ehdrs/tabl.h          15	
Xabc/ehdrs/trm.h           24	
Xabc/ex                     1	
Xabc/ex/DoExamples         25	
Xabc/ex/README             24	
Xabc/ex/TryEditor          24	
Xabc/ex/generate            1	
Xabc/ex/generate.in        25	
Xabc/ex/generate.out       24	
Xabc/ex/generate/analyze.cmd 25	
Xabc/ex/generate/enders.cts 25	
Xabc/ex/generate/fill.cmd  25	
Xabc/ex/generate/follower.cts 23	
Xabc/ex/generate/generate.cmd 25	
Xabc/ex/generate/perm.abc  25	
Xabc/ex/generate/start.cmd 25	
Xabc/ex/generate/starters.cts 25	
Xabc/ex/generate/suggest.abc 25	
Xabc/ex/hanoi               1	
Xabc/ex/hanoi.in           25	
Xabc/ex/hanoi.out          25	
Xabc/ex/hanoi/hanoi.cmd    25	
Xabc/ex/hanoi/perm.abc     25	
Xabc/ex/hanoi/suggest.abc   2	
Xabc/ex/pi                  1	
Xabc/ex/pi.in              25	
Xabc/ex/pi.out             25	
Xabc/ex/pi/perm.abc        25	
Xabc/ex/pi/pi.cmd          25	
Xabc/ex/pi/suggest.abc     25	
Xabc/ex/try                 1	
Xabc/ex/try/analyze.cmd    25	
Xabc/ex/try/enders.cts     25	
Xabc/ex/try/fill.cmd       25	
Xabc/ex/try/follower.cts   23	
Xabc/ex/try/generate.cmd   25	
Xabc/ex/try/perm.abc       25	
Xabc/ex/try/position.abc    7	
Xabc/ex/try/start.cmd      25	
Xabc/ex/try/starters.cts   25	
Xabc/ex/try/suggest.abc    25	
Xabc/ex/wsgroup.abc         4	
Xabc/ex/xref                1	
Xabc/ex/xref.in            25	
Xabc/ex/xref.out           25	
Xabc/ex/xref/alphabet.mpd  25	
Xabc/ex/xref/output.cmd    25	
Xabc/ex/xref/perm.abc      25	
Xabc/ex/xref/save.cmd      25	
Xabc/ex/xref/suggest.abc   25	
Xabc/ex/xref/text.cts      25	
Xabc/ex/xref/words.mfd     25	
Xabc/ex/xref/xref.cmd      25	
Xabc/ex/xref/xtab.cts      25	
Xabc/ihdrs                  1	
Xabc/ihdrs/i0err.h         23	
Xabc/ihdrs/i1num.h         20	
Xabc/ihdrs/i2exp.h         24	
Xabc/ihdrs/i2gen.h         25	
Xabc/ihdrs/i2nod.h         18	
Xabc/ihdrs/i2par.h         16	
Xabc/ihdrs/i3bws.h         25	
Xabc/ihdrs/i3env.h         24	
Xabc/ihdrs/i3in2.h         25	
Xabc/ihdrs/i3int.h         24	
Xabc/ihdrs/i3scr.h         25	
Xabc/ihdrs/i3sou.h         23	
Xabc/ihdrs/i3sta.h         25	
Xabc/ihdrs/i3typ.h         25	
Xabc/keys                   1	
Xabc/keys/DEP              22	
Xabc/keys/Makefile         23	
Xabc/keys/keydef.c          3	
Xabc/keys/keydef.h         23	
Xabc/keys/keyhlp.c         20	
Xabc/lin                    1	
Xabc/lin/etex.h            25	
Xabc/lin/i1lta.c           16	
Xabc/lin/i1obj.c            1	
Xabc/lin/i1tex.c           21	
Xabc/lin/i1tlt.c           12	
Xabc/lin/i1tlt.h           17	
Xabc/mkconfig.c            13	
Xabc/scripts                1	
Xabc/scripts/Change        24	
Xabc/scripts/Collect       24	
Xabc/scripts/mkdep.gen      9	
Xabc/stc                    1	
Xabc/stc/DEP               24	
Xabc/stc/MF                25	
Xabc/stc/i2stc.h           22	
Xabc/stc/i2tca.c            3	
Xabc/stc/i2tce.c           17	
Xabc/stc/i2tcp.c           18	
Xabc/stc/i2tcu.c           20	
Xabc/tc                     1	
Xabc/tc/Makefile           24	
Xabc/tc/README             23	
Xabc/tc/tc1.c              25	
Xabc/tc/tc2.c              24	
Xabc/tc/tc3.c              23	
Xabc/tc/termcap             6	
Xabc/tc/termcap.5           5	
Xabc/tc/termcap.c          19	
Xabc/tc/tgoto.c            21	
Xabc/tc/tputs.c            22	
Xabc/uhdrs                  1	
Xabc/uhdrs/args.h          25	
Xabc/uhdrs/conf.h          24	
Xabc/uhdrs/defs.h          25	
Xabc/uhdrs/dir.h           24	
Xabc/uhdrs/feat.h          23	
Xabc/uhdrs/os.h.gen        23	
Xabc/uhdrs/osconf.h        25	
Xabc/ukeys                  1	
Xabc/ukeys/abckeys_2621    24	
Xabc/ukeys/abckeys_2640b   25	
Xabc/ukeys/abckeys_5620    24	
Xabc/ukeys/abckeys_5620-2  24	
Xabc/ukeys/abckeys_5620-e  24	
Xabc/ukeys/abckeys_924     21	
Xabc/ukeys/abckeys_adm31   24	
Xabc/unix                   1	
Xabc/unix/DEP              23	
Xabc/unix/MF               25	
Xabc/unix/u1dir.c          23	
Xabc/unix/u1edit.c         24	
Xabc/unix/u1file.c         20	
Xabc/unix/u1keys.c         15	
Xabc/unix/u1os.c           24	
Xabc/unix/u1time.c         23	
Xabc/unix/u1trm.c           2	
END_OF_FILE
  if test 8839 -ne `wc -c <'MANIFEST'`; then
    echo shar: \"'MANIFEST'\" unpacked with wrong size!
  fi
  # end of 'MANIFEST'
fi
if test ! -d 'abc' ; then
    echo shar: Creating directory \"'abc'\"
    mkdir 'abc'
fi
if test ! -d 'abc/b' ; then
    echo shar: Creating directory \"'abc/b'\"
    mkdir 'abc/b'
fi
if test ! -d 'abc/bed' ; then
    echo shar: Creating directory \"'abc/bed'\"
    mkdir 'abc/bed'
fi
if test ! -d 'abc/bhdrs' ; then
    echo shar: Creating directory \"'abc/bhdrs'\"
    mkdir 'abc/bhdrs'
fi
if test ! -d 'abc/bint1' ; then
    echo shar: Creating directory \"'abc/bint1'\"
    mkdir 'abc/bint1'
fi
if test ! -d 'abc/bint2' ; then
    echo shar: Creating directory \"'abc/bint2'\"
    mkdir 'abc/bint2'
fi
if test ! -d 'abc/bint3' ; then
    echo shar: Creating directory \"'abc/bint3'\"
    mkdir 'abc/bint3'
fi
if test -f 'abc/bint3/i3sou.c' -a "${1}" != "-c" ; then 
  echo shar: Will not clobber existing file \"'abc/bint3/i3sou.c'\"
else
  echo shar: Extracting \"'abc/bint3/i3sou.c'\" \(29957 characters\)
  sed "s/^X//" >'abc/bint3/i3sou.c' <<'END_OF_FILE'
X/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1986. */
X
X/* Sources: maintaining units and values on external files */
X
X#include "b.h"
X#include "bint.h"
X#include "feat.h"
X#include "bmem.h"
X#include "bobj.h"
X#include "bfil.h"
X#include "i2par.h"
X#include "i2nod.h"
X#include "i3env.h"
X#include "i3scr.h"
X#include "i3in2.h"
X#include "i3sou.h"
X
X#ifdef TYPE_CHECK
Xvalue stc_code();
X#endif
X#ifdef unix
X#define CK_WS_WRITABLE
X#endif
X
XVisible value b_perm= Vnil;
X	/* The table that maps tags to their file names */
XVisible value b_units= Vnil;
X	/* The table that maps tags to their internal repr. */
X
X#define Is_filed(v) (Is_indirect(v))
X
X#define t_exists(name, aa)	(in_env(prmnv->tab, name, aa))
X
XVisible Procedure def_target(name, t) value name, t; {
X	e_replace(t, &prmnv->tab, name);
X}
X
X#define free_target(name)	(e_delete(&prmnv->tab, name))
X
X/************************** UNITS ************************************/
X
X#define Is_funprd(u)		(Is_function(u) || Is_predicate(u))
X#define Is_predefined(u)	(Is_funprd(u) && Funprd(u)->pre != Use)
X
X#define USR_ALL		'1'
X#define USR_PARSED	'2'
X
XHidden Procedure freeunits(which) literal which; {
X	intlet k, len;
X	value vkey, vassoc;
X	
X	len= length(b_units);
X	for (k= len-1; k >= 0; --k) {
X		/* Reverse loop so deletions don't affect the numbering! */
X		vkey= *key(b_units, k);
X		vassoc= *assoc(b_units, k);
X		switch (which) {
X		case USR_ALL:
X			if (!Is_predefined(vassoc)) free_unit(vkey);
X			break;
X		case USR_PARSED:
X			if (!Is_predefined(vassoc) &&
X					!How_to(vassoc)->unparsed)
X				free_unit(vkey);
X			break;
X		}
X	}
X}
X
XVisible Procedure rem_unit(u) parsetree u; {
X	value pname= get_pname(u);
X	free_unit(pname);
X	release(pname);
X}
X
X/********************************************************************** */
X
XVisible value permkey(name, type) value name; literal type; {
X	char t[2];
X	value v, w;
X	
X	if (!Valid(name))
X		return Vnil;
X	t[0]= type; t[1]= '\0';
X	w= mk_text(t);
X	v= concat(w, name); release(w);
X	return v;
X}
X
XVisible string lastunitname() {
X	value *aa;
X	
X	if (p_exists(last_unit, &aa))
X		return sstrval(Permname(*aa));
X	return NULL;
X}
X
X#define CANTGETFNAME	MESS(4000, "cannot create file name for %s")
X
XHidden value get_ufname(pname, silently) value pname; bool silently; {
X	value fname;
X	value *aa;
X	
X	if (p_exists(pname, &aa))
X		fname= copy(*aa);
X	else {
X		value name= Permname(pname);
X		literal type= Permtype(pname);
X		
X		fname= new_fname(name, type);
X		if (Valid(fname))
X			def_perm(pname, fname);
X		else if (!silently)
X			interrV(CANTGETFNAME, name);
X		release(name);
X	}
X	return fname;
X}
X
XHidden bool p_version(name, type, pname) value name, *pname; literal type; {
X	value *aa;
X	*pname= permkey(name, type);
X	if (p_exists(*pname, &aa)) return Yes;
X	release(*pname); *pname= Vnil;
X	return No;
X}
X
XHidden bool u_version(name, type, pname) value name, *pname; literal type; {
X	value *aa;
X	*pname= permkey(name, type);
X	if (u_exists(*pname, &aa)) return Yes;
X	release(*pname); *pname= Vnil;
X	return No;
X}
X
XHidden bool tar_version(name, pname) value name, *pname; {
X	value *aa;
X	if (p_version(name, Tar, pname))
X		return Yes;
X	else if (t_exists(name, &aa)) {
X		*pname= permkey(name, Tar);
X		return Yes;
X	}
X	else return No;
X}
X
XHidden Procedure del_perm(pname) value pname; {
X	value *aa;
X	if (p_exists(pname, &aa)) {
X		f_delete(*aa);
X		idelpos(*aa);	/* delete file from positions file */
X		free_perm(pname);
X	}
X}
X
X/***********************************************************************/
X
XHidden bool is_loaded(pname, aa) value pname, **aa; {
X	value u= Vnil, npname= Vnil, *a, get_unit();
X	if (u_exists(pname, &a)) {
X		if (Is_predefined(*a) && p_exists(pname, aa)) {
X			/* loading userdefined over predefined */;
X		}
X		else {
X			*aa= a; 
X			return Yes; /* already loaded */
X		}
X	}
X	else if (!p_exists(pname, aa)) {
X		return No;
X	}
X	ifile= fopen(strval(**aa), "r");
X	if (ifile == NULL) {
X		vs_ifile();
X		return No;
X	}
X	Eof= No;
X	first_ilev();
X	u= get_unit(&npname, Yes, No);
X	if (still_ok) def_unit(npname, u);
X	fclose(ifile);
X	vs_ifile();
X	Eof= No;
X	if (still_ok && !u_exists(pname, aa)) {
X		value name= Permname(pname);; 
X		release(uname); uname= copy(pname);
X		curline= How_to(u)->unit; curlino= one;
X		interrV(MESS(4001, "filename and how-to name incompatible for %s"), name);
X		release(name);
X	}
X	release(u); release(npname);
X	return still_ok;
X}
X
X/* Does the unit exist without faults? */
X
XVisible bool is_unit(name, type, aa) value name, **aa; literal type; {
X	value pname;
X	context c; bool is;
X	sv_context(&c);
X	cntxt= In_unit;
X	pname= permkey(name, type);
X	is= is_loaded(pname, aa);
X	release(pname);
X	set_context(&c);
X	return is;
X}
X
X/***********************************************************************/
X
X#define CANT_WRITE	MESS(4002, "cannot create file %s; need write permission in directory")
X
X#define CANT_READ	MESS(4003, "unable to find file")
X
XHidden Procedure u_name_type(v, name, type) parsetree v; value *name;
X		literal *type; {
X	intlet adic;
X	switch (Nodetype(v)) {
X		case HOW_TO:	*type= Cmd; break;
X		case YIELD:	adic= intval(*Branch(v, FPR_ADICITY));
X				*type= adic==0 ? Zfd : adic==1 ? Mfd : Dfd;
X				break;
X		case TEST:	adic= intval(*Branch(v, FPR_ADICITY));
X				*type= adic==0 ? Zpd : adic==1 ? Mpd : Dpd;
X				break;
X		default:	syserr(MESS(4004, "wrong nodetype of how-to"));
X	}
X	*name= copy(*Branch(v, UNIT_NAME));
X}
X
XHidden value get_unit(pname, filed, editing) value *pname; bool filed, editing;
X{
X	value name; literal type;
X	parsetree u= unit(No, editing);
X	if (u == NilTree)
X		return Vnil;
X	u_name_type(u, &name, &type);
X	*pname= permkey(name, type);
X	release(name);
X	switch (Nodetype(u)) {
X		case HOW_TO:	return mk_how(u, filed);
X		case YIELD:	return mk_fun(type, Use, u, filed);
X		case TEST:	return mk_prd(type, Use, u, filed);
X		default:	return Vnil; /* Keep lint happy */
X	}
X}
X
XVisible value get_pname(v) parsetree v; {
X	value pname, name; literal type;
X	u_name_type(v, &name, &type);
X	pname= permkey(name, type);
X	release(name);
X	return pname;
X}
X
XHidden Procedure get_heading(h, pname) parsetree *h; value *pname; {
X	*h= unit(Yes, No);
X	*pname= still_ok ? get_pname(*h) : Vnil;
X}
X
X/********************************************************************** */
X
X/* Check for certain types of name conflicts.
X   The checks made are:
X   - unit with the same name
X   - function and predicate with the same name (and different or same
X     adicity)
X   - function or predicate with the same name as a target
X   - zeroadic and monadic unit with the same name
X   - zeroadic and dyadic unit with the same name.
X*/
X
X#define CR_EXIST	MESS(4005, "there is already a how-to with this name")
X
X#define CR_TAR		MESS(4006, "there is already a permanent location with this name")
X
X#define ED_EXIST	MESS(4007, "*** the how-to name is already in use;\n*** should the old how-to be discarded?\n*** (if not you have to change the how-to name)\n")
X
X#define ED_TAR		MESS(4008, "*** the how-to name is already in use for a permanent location;\n*** should that location be deleted?\n*** (if not you have to change the how-to name)\n")
X
X/* name_conflict() is called if a unit is created (HOW TO ? : command) */
X
XHidden bool name_conflict(pname) value pname; {
X	value npname;
X	if (smash(pname, &npname)) {
X		interr(Permtype(npname) == Tar ? CR_TAR : CR_EXIST);
X		if (Permtype(pname) != Tar)
X			def_perm(last_unit, npname);
X		release(npname);
X		return Yes;
X	}
X	return No;
X}
X
X/* name_clash() is called if a unit is edited through the ':' command */
X
XHidden bool name_clash(pname) value pname; {
X	value npname;
X	
X	if (!Valid(pname))
X		return No;
X	while (smash(pname, &npname)) {
X		if (!do_discard(npname)) {
X			release(npname);
X			return Yes;
X		}
X		/* continue: there can be both a monadic and a	*/
X		/* 	     dyadic version 			*/
X		release(npname); npname= Vnil;
X	}
X	return No;
X}
X
XHidden bool do_discard(pname) value pname; {
X	bool istarg= Permtype(pname) == Tar;
X	
X	if (is_intended(istarg ? ED_TAR : ED_EXIST)) {
X		if (istarg) {
X			value name= Permname(pname);
X			del_target(name);
X			release(name);
X		}
X		else {
X			free_unit(pname);
X			del_perm(pname);
X		}
X		return Yes;
X	}
X	return No;
X}
X
XHidden bool smash(pname, npname) value pname, *npname; {
X	value name, *aa;
X	literal u_type, v_type;
X	bool sm;
X
X	if (p_exists(pname, &aa)) {
X		*npname= copy(pname);
X		return Yes;
X	}
X	u_type= Permtype(pname);
X	if (u_type == Cmd) {
X		*npname= Vnil;
X		return No;
X	}
X	name= Permname(pname);
X	sm= p_version(name, Zfd, npname) ||
X		p_version(name, Mfd, npname) ||
X		p_version(name, Dfd, npname) ||
X		p_version(name, Zpd, npname) ||
X		p_version(name, Mpd, npname) ||
X		p_version(name, Dpd, npname) ||
X		tar_version(name, npname);
X	release(name);
X	if (!sm) {
X		release(*npname); *npname= Vnil;
X		return No;
X	}
X	v_type= Permtype(*npname);
X	switch (u_type) {
X		case Mfd: sm= v_type != Dfd; break;
X		case Dfd: sm= v_type != Mfd; break;
X		case Mpd: sm= v_type != Dpd; break;
X		case Dpd: sm= v_type != Mpd; break;
X		default: sm= Yes; break;
X	}
X	if (!sm) {
X		release(*npname); *npname= Vnil;
X		return No;
X	}
X	return Yes;
X}
X
X/***********************************************************************/
X
X/* Create a unit via the editor or from the input stream. */
X
XVisible Procedure create_unit() {
X	value pname= Vnil; parsetree heading= NilTree;
X	if (!interactive) {
X		value v= get_unit(&pname, No, No);
X		if (still_ok) def_unit(pname, v);
X		release(v); release(pname);
X		return;
X	}
X	get_heading(&heading, &pname);
X	curline= heading; curlino= one; /* For all error messages */
X	if (still_ok && !name_conflict(pname)) {
X		value fname= get_ufname(pname, No);
X
X		if (Valid(fname)) {
X			FILE *fp= fopen(strval(fname), "w");
X			if (fp == NULL)
X				interrV(CANT_WRITE, fname);
X			else {
X				txptr tp= fcol();
X				do { fputc(Char(tp), fp); }
X				while (Char(tp++) != '\n');
X				fputc('\n', fp);
X				f_close(fp);
X				ed_unit(&pname, &fname, Yes);
X			}
X		}
X		release(fname);
X	}
X	release(pname); release(heading);
X}
X
X
X/***********************************************************************/
X
X/* Edit a unit. The name of the unit is either given, or is defaulted
X   to the last unit edited or the last unit that gave an error, whichever
X   was most recent.
X   It is possible for the user to mess things up with the w command, for
X   instance, but this is not checked. It is allowed to rename the unit though,
X   or delete it completely. If the file is empty, the unit is disposed of.
X   Otherwise, the name and adicity are determined and if these have changed,
X   the new unit is written out to a new file, and the original deleted.
X   Thus the original is not saved.
X
X   The function edit_unit parses the command line and does some
X   high-level bookkeeping; ed_unit does the lower-level bookkeeping;
X   f_edit is called to pass control to the editor and wait till it
X   finishes its job.  Note that the editor reads the unit from the file
X   and writes it back (if changed); there is no sharing of data
X   structures such as parse trees in this version of the system.
X
X   Renaming, deleting, or changing the adicity of a test or yield
X   unfortunately requires all other units to be thrown away internally
X   (by freeunits), since the unit parse trees may be wrong. For instance,
X   consider the effect on the following of making a formerly monadic
X   function f, into a zeroadic function:
X	WRITE f root 2
X*/
X
X#define CANT_EDIT	MESS(4009, "I find nothing editible here")
X
XVisible value last_unit= Vnil;
X
XVisible Procedure edit_unit() {
X	value name= Vnil, pname= Vnil; 
X	value fname, *aa;
X	value which_funprd();
X	char *kw;
X
X	if (Ceol(tx)) {
X		if (!p_exists(last_unit, &aa))
X			parerr(MESS(4010, "no current how-to"));
X		else pname= copy(*aa);
X	}
X	else if (is_cmdname(ceol, &kw)) {
X		name= mk_text(kw);
X		pname= permkey(name, Cmd);
X	}
X	else if (is_tag(&name))
X		pname= which_funprd(name);
X	else
X		parerr(CANT_EDIT);
X
X	if (still_ok && ens_filed(pname, &fname)) {
X		ed_unit(&pname, &fname, No);
X		release(fname);
X	}
X	release(name); release(pname);
X}
X
X#define ED_MONDYA	MESS(4011, "*** do you want to visit the version with %c or %c operands?\n")
X#define ONE_PAR '1'
X#define TWO_PAR '2'
X
XHidden value which_funprd(name) value name; {
X	/* There may be two units with the same name (functions
X	   or predicates of different adicity).  Check if this
X	   is the case, and if so, ask which one is meant.
X	*/
X	value pname, v= Vnil;
X	char qans;
X	
X	if (p_version(name, Zfd, &pname) || p_version(name, Zpd, &pname))
X		return pname;
X	if (p_version(name, Mfd, &pname) || p_version(name, Mpd, &pname)) {
X		if (p_version(name, Dfd, &v) || p_version(name, Dpd, &v)) {
X			qans= q_answer(ED_MONDYA, ONE_PAR, TWO_PAR);
X			if (qans == ONE_PAR) {
X				release(v);
X				return pname;
X			}
X			else if (qans == TWO_PAR) {
X				release(pname);
X				return copy(v);
X			}
X			else {
X				/* interrupted */
X				still_ok = No;
X				return pname;
X			}
X		}
X		else {
X			release(v);
X			return pname;
X		}
X	}
X	if (p_version(name, Dfd, &pname))
X		return pname;
X	if (p_version(name, Dpd, &pname))
X		return pname;
X
X	/* be prepared to find at least one not-filed how-to;
X	 * this does not find all of them;
X	 * and it doesn't allow any conflicting with already existing ones.
X	 */
X	
X	if (u_version(name, Zfd, &pname) ||
X	    u_version(name, Mfd, &pname) ||
X	    u_version(name, Dfd, &pname) ||
X	    u_version(name, Zpd, &pname) ||
X	    u_version(name, Mpd, &pname) ||
X	    u_version(name, Dpd, &pname)
X	)
X		return pname;
X
X	return permkey(name, Dpd);
X	/* If it doesn't exist, ens_filed will complain. */
X}
X	
X#define NO_U_WRITE	MESS(4012, "*** you have no write permission in this workspace:\n*** you may not change the how-to\n*** do you still want to display the how-to?\n")
X
X/* Edit a unit.  Parameters are the prmnv key and the file name.
X   This is called in response to the ':' command and when a new unit is
X   created (the header of the new unit must already be written to the
X   file).
X   Side effects are many, e.g. on prmnv: the unit may be deleted or
X   renamed.  When renamed, the original unit is lost.
X   The unit is reparsed after editing.  A check is made for illegal
X   name conflicts (e.g., a zeroadic and a monadic unit of the same
X   name), and this is resolved by forcing the user to edit the unit
X   again. In that case the edit is done on a temporary file.
X   The new unit name is kept as the current unit name; when the unit is
X   deleted the current unit name is set to Vnil. */
X
XHidden bool clash;
X
X#define First_edit (!clash)
X
X#ifdef TYPE_CHECK
XHidden value old_typecode= Vnil;
X#define Sametypes(old, new) ((!Valid(old) && !Valid(new)) || \
X		(Valid(old) && Valid(new) && compare(old, new) == 0))
X#endif
X
XHidden Procedure ed_unit(pname, fname, creating) value *pname, *fname;
X		bool creating;
X{
X#ifdef CK_WS_WRITABLE
X	if (!wsp_writable() && !is_intended(NO_U_WRITE)) return;
X#endif
X#ifdef CLEAR_MEM
X	clear_perm();
X		/* To give the editor as much space as possible, remove
X		   all parse trees and target values from memory.
X		   (targets that have been modified are first written
X		   out, of course).
X		*/
X#endif
X	clash= No;
X#ifdef TYPE_CHECK
X	old_typecode= stc_code(*pname);
X	if (!creating) del_types();
X#endif
X	do edunit(pname, fname, creating); while (clash);
X#ifdef SAVE_PERM
X	put_perm(b_perm);
X#endif
X#ifdef TYPE_CHECK
X	release(old_typecode);
X#endif
X}
X
XHidden Procedure edunit(p_pname, p_fname, creating) value *p_pname, *p_fname;
X		bool creating; {
X	value pname= *p_pname, fname= *p_fname;
X	value npname= Vnil, u;
X	bool new_def, changed, samehead;
X#ifdef TYPE_CHECK
X	value new_typecode;
X#endif
X
X	release(uname); uname= copy(pname);
X	changed= f_edit(fname, err_line(pname), ':', creating && First_edit)
X		 || creating;
X	errlino= 0;
X	if (First_edit && !changed) {
X		/* Remember it as current unit: */
X		def_perm(last_unit, pname);
X#ifdef TYPE_CHECK
X		if (!creating) adjust_types(Yes);
X#endif
X		return;
X	}
X	if (!still_there(fname)) {
X		free_original(pname);
X#ifdef TYPE_CHECK
X		if (!creating) adjust_types(No);
X#endif
X		idelpos(fname);	/* delete file from positions file */
X		free_perm(last_unit);
X		clash= No;
X		return;
X	}
X	first_ilev();
X	u= get_unit(&npname, Yes, Yes);
X		/* the second Yes means the user may edit the heading;
X		 * therefore no type check now in unit() */
X	fclose(ifile); vs_ifile(); Eof= No;
X	
X	if (First_edit && same_heading(pname, npname, u)) {
X		new_def= Yes;
X		samehead= Yes;
X	}
X	else {
X		samehead= No;
X		free_original(pname);
X		if (!name_clash(npname) && rnm_file(fname, npname))
X			clash= No;
X		else {
X			/* edit again with npname and temp fname */
X			release(*p_pname);
X			*p_pname= copy(npname);
X			if (First_edit) {
X				value tfile= mk_text(temp1file);
X				f_rename(fname, tfile);
X				imovpos(fname, tfile);
X				/* move position in positions file */
X				release(*p_fname);
X				*p_fname= tfile;
X			}
X			clash= Yes;
X		}
X		new_def= !clash;
X	}
X	if (new_def) {
X		/* changed heading now def_perm()'ed, so now typecheck */
X#ifdef TYPE_CHECK
X		type_check((Is_funprd(u) ? Funprd(u)->unit : How_to(u)->unit));
X		new_typecode= stc_code(npname);
X		if (!creating)
X			adjust_types(samehead &&
X				     Sametypes(old_typecode, new_typecode));
X		release(new_typecode);
X#endif
X		if (still_ok) def_unit(npname, u);
X		else free_unit(npname);
X		def_perm(last_unit, npname);
X	}
X	release(npname); release(u);
X}
X
XHidden Procedure free_original(pname) value pname; {
X	if (First_edit) {
X		free_unit(pname); 
X		free_perm(pname);
X		freeunits(USR_PARSED);
X	}
X}
X
X#define cmd_unit(pname)	(Permtype(pname) == Cmd)
X
XHidden bool same_heading(pname, npname, u_new) value pname, npname, u_new; {
X	value *aa;
X	
X	if (!Valid(u_new) || !Valid(npname))
X		return No;
X	else if (compare(pname, npname) != 0)
X		return No;
X	else if (!cmd_unit(pname))
X		return Yes;
X	else if (!u_exists(pname, &aa))
X		return Yes;
X	else {
X		parsetree old= How_to(*aa)->unit;
X		parsetree new= How_to(u_new)->unit;
X		parsetree old_kw, old_fml, old_next;
X		parsetree new_kw, new_fml, new_next;
X		
X		old= *Branch(old, HOW_FORMALS);
X		new= *Branch(new, HOW_FORMALS);
X		do {
X			old_kw= *Branch(old, FML_KEYW);
X			old_fml= *Branch(old, FML_TAG);
X			old_next= *Branch(old, FML_NEXT);
X			new_kw= *Branch(new, FML_KEYW);
X			new_fml= *Branch(new, FML_TAG);
X			new_next= *Branch(new, FML_NEXT);
X			
X			if (compare(old_kw, new_kw) != 0)
X				return No;
X			else if (old_fml == NilTree && new_fml != NilTree)
X				return No;
X			else if (old_fml != NilTree && new_fml == NilTree)
X				return No;
X			else if (old_next == NilTree && new_next != NilTree)
X				return No;
X			else if (old_next != NilTree && new_next == NilTree)
X				return No;
X			old= old_next;
X			new= new_next;
X		}
X		while (old != NilTree);
X		return Yes;
X	}
X}
X
X#define CANT_GET_FNAME	MESS(4013, "*** cannot create file name;\n*** you have to change the how-to name\n")
X
XHidden bool rnm_file(fname, pname) value fname, pname; {
X	value nfname;
X	
X	nfname= (Valid(pname) ? get_ufname(pname, Yes) : Vnil);
X	
X	if (Valid(nfname)) {
X		f_rename(fname, nfname);
X		imovpos(fname, nfname); /* move position in positions file */
X		release(nfname);
X		return Yes;
X	}
X	else {
X		putmess(errfile, CANT_GET_FNAME);
X		return No;
X	}
X}
X
X/* Find out if the file exists, and is not empty. Some editors don't
X   allow a file to be edited to empty, but insist it should be at least
X   one empty line.  Therefore, a file with one, empty, line is also
X   considered empty.
X   As a side effect, if the file is 'still there', ifile is set to it
X   and it remains open, positioned at the beginning.
X   (A previous version of this function would leave it positioned after
X   an initial \n, if there was one; this version just rewinds the file.)
X   */
X
XHidden bool still_there(fname) value fname; {
X	int k;
X
X	ifile= fopen(strval(fname), "r");
X	if (ifile == NULL) {
X		vs_ifile();
X		return No;
X	} else {
X		if ((k= getc(ifile)) == EOF ||
X				(k == '\n' && (k= getc(ifile)) == EOF)) {
X			fclose(ifile);
X			f_delete(fname);
X			vs_ifile();
X			return No;
X		}
X		rewind(ifile);
X		return Yes;
X	}
X}
X
X/* Ensure the unit is filed. If the unit was read non-interactively (eg passed
X   as a parameter to abc), it is only held in store.
X   Editing it puts it into a file. This is the safest way to copy a unit from
X   one workspace to another.
X*/
X
X#define NO_HOWTO MESS(4014, "%s isn't a how-to in this workspace")
X
XHidden bool ens_filed(pname, fname) value pname, *fname; {
X	value *aa;
X	if (p_exists(pname, &aa)) {
X		*fname= copy(*aa);
X		return Yes;
X	} else if (!u_exists(pname, &aa) || How_to(*aa)->unit == NilTree) {
X		value name= Permname(pname);
X		pprerrV(NO_HOWTO, name);
X		release(name);
X		return No;
X	} else {
X		how *du= How_to(*aa); FILE *fp;
X		if (du->filed == Yes) {
X			syserr(MESS(4015, "ens_filed()"));
X			return No;
X		}
X		*fname= get_ufname(pname, No);
X		if (!Valid(*fname))
X			return No;
X		fp= fopen(strval(*fname), "w");
X		if (!fp) {
X			interrV(CANT_WRITE, *fname);
X			release(*fname);
X			return No;
X		} else {
X			display(fp, du->unit, No);
X			f_close(fp);
X			du->filed= Yes;
X			return Yes;
X		}
X	}
X}
X
XHidden int err_line(pname) value pname; {
X	value *aa;
X	if (!p_exists(last_unit, &aa) || compare(*aa, pname) != 0)
X		return 0;
X	else
X		return errlino;
X}
X
X/************************** VALUES ***************************************/
X/* The permanent environment in the old format was kept as a single file */
X/* but this caused slow start ups if the file was big.			 */
X/* Thus the new version stores each permanent target on a separate file, */
X/* that furthermore is only loaded on demand.				 */
X/* To achieve this, a directory is kept of the permanent tags and their  */
X/* file names. Care has to be taken that disaster occurring in		 */
X/* the middle of an update of this directory does the least harm.	 */
X/* Having the directory refer to a non-existent file is considered less  */
X/* harmful than leaving a file around that can never be accessed, for	 */
X/* instance, so a file is deleted before its directory entry,		 */
X/* and so forth.							 */
X/*************************************************************************/
X
XVisible value errtname= Vnil;
X
XHidden Procedure tarfiled(name, v) value name, v; {
X	value p= mk_indirect(v);
X	def_target(name, p);
X	release(p);
X}
X
XVisible value last_target= Vnil; /* last edited target */
X
XVisible Procedure del_target(name) value name; {
X	value pname= permkey(name, Tar);
X	value *aa;
X	free_target(name);
X	del_perm(pname);
X	if (p_exists(last_target, &aa) && (compare(name, *aa) == 0))
X		free_perm(last_target);
X	release(pname);
X}
X
XHidden value get_tfname(name) value name; {
X	value fname;
X	value pname= permkey(name, Tar);
X	value *aa;
X	
X	if (p_exists(pname, &aa))
X		fname= copy(*aa);
X	else {
X		fname= new_fname(name, Tar);
X		if (Valid(fname))
X			def_perm(pname, fname);
X		else
X			interrV(CANTGETFNAME, name);
X	}
X	release(pname);
X	return fname;
X}
X
XVisible Procedure edit_target() {
X	value name= Vnil;
X	value fname, *aa;
X	if (Ceol(tx)) {
X		if (!p_exists(last_target, &aa))
X			parerr(MESS(4016, "no current location"));
X		else
X			name= copy(*aa);
X	} else if (!is_tag(&name))
X		parerr(CANT_EDIT);
X	if (still_ok && ens_tfiled(name, &fname)) {
X		ed_target(name, fname);
X		release(fname);
X	}
X	release(name);
X}
X
X#define NO_T_WRITE	MESS(4017, "*** you have no write permission in this workspace:\n*** you may not change the location\n*** do you still want to display the location?\n")
X
X/* Edit a target. The value in the target is written to the file,
X   and then removed from the internal permanent environment so that
X   if a syntax error occurs when reading the value back, the value is
X   absent from the internal permanent environment.
X   Thus when editing the file to correct the syntax error, the
X   file doesn't get overwritten.
X   The contents may be completely deleted in which case the target is
X   deleted. */
X
XHidden Procedure ed_target(name, fname) value name, fname; {
X	value v;
X
X#ifdef CK_WS_WRITABLE
X	if (!wsp_writable() && !is_intended(NO_T_WRITE)) return;
X#endif
X#ifdef CLEAR_MEM
X	clear_perm(); /* To give the editor as much space as possible */
X#endif
X	def_perm(last_target, name);
X	if (!f_edit(fname, 0, '=', No))
X		/* File is unchanged */
X		return;
X	if (!still_there(fname)) {
X		del_target(name);
X#ifdef SAVE_PERM
X		put_perm(b_perm);
X#endif
X		return;
X	}
X	fclose(ifile); /* Since still_there leaves it open */
X	/* vs_ifile(); ? */
X	v= getval(fname, In_edval);
X	if (still_ok) def_target(name, v);
X	release(v);
X}
X
X#define NO_TARGET MESS(4018, "%s isn't a location in this workspace")
X
XVisible bool ens_tfiled(name, fname) value name, *fname; {
X	value *aa;
X	if (!t_exists(name, &aa)) {
X		pprerrV(NO_TARGET, name);
X		return No;
X	} else {
X		*fname= get_tfname(name);
X		if (!Valid(*fname))
X			return No;
X		if (!Is_filed(*aa)) {
X			release(errtname); errtname= copy(name);
X			putval(*fname, *aa, No, In_tarval);
X			tarfiled(name, *aa);
X		}
X		return Yes;
X	}
X}
X
X/***************************** Values on files ****************************/
X
XVisible value getval(fname, ct) value fname; literal ct; {
X	char *buf; int k; parsetree w, code= NilTree; value v= Vnil;
X	ifile= fopen(strval(fname), "r");
X	if (ifile) {
X		txptr fcol_save= first_col, tx_save= tx; context c;
X		sv_context(&c);
X		cntxt= ct;
X		buf= (char *) getmem((unsigned)(f_size(ifile)+2)*sizeof(char));
X		first_col= tx= ceol= buf;
X		while ((k= getc(ifile)) != EOF)
X			if (k != '\n') *ceol++= k;
X		*ceol= '\n';
X		fclose(ifile); vs_ifile();
X		w= expr(ceol);
X		if (still_ok) fix_nodes(&w, &code);
X		curline= w; curlino= one;
X		v= evalthread(code); 
X		if (!env_ok(v)) {
X			release(v);
X			v= Vnil;
X		}
X		curline= Vnil;
X		release(w);
X		freemem((ptr) buf);
X		set_context(&c);
X		first_col= fcol_save; tx= tx_save;
X	} else {
X		interr(CANT_READ);
X		vs_ifile();
X	}
X	return v;
X}
X
XHidden bool env_ok(v) value v; {
X	if (cntxt == In_prmnv || cntxt == In_wsgroup) {
X		if (!Is_table(v)) {
X			interr(MESS(4019, "value is not a table"));
X			return No;
X		}
X		else if (!Is_ELT(v) && !Is_text(*key(v, 0))) {
X			interr(MESS(4020, "in t[k], k is not a text"));
X			return No;
X		}
X	}
X	return Yes;
X}
X
XVisible bool permchanges;
X
XVisible Procedure initperm() {
X	if (F_exists(permfile)) {
X		value fn, name;
X		intlet k, len;
X		value v, pname;
X		
X		fn= mk_text(permfile);
X		v= getval(fn, In_prmnv);
X		release(fn);
X		if (Valid(v)) {
X			release(b_perm);
X			b_perm= v;
X		}
X		len= length(b_perm);
X		for (k= 0; k < len; k++) {
X			pname= *key(b_perm, k);
X			if (Permtype(pname) == Tar) {
X				name= Permname(pname);
X				tarfiled(name, Vnil);
X				release(name);
X			}
X		}
X	}
X	permchanges= No;
X}
X
XVisible Procedure putval(fname, v, silently, ct) value fname, v;
X		bool silently; literal ct; {
X	value fn= copy(fname);
X	FILE *fp;
X	bool was_ok= still_ok;
X	context c;
X
X	sv_context(&c);
X	cntxt= ct;
X	curline= Vnil;
X	curlino= one;
X#ifdef unix
X	release(fn); fn= mk_text(tempfile);
X#endif
X	fp= fopen(strval(fn), "w");
X	if (fp != NULL) {
X		redirect(fp);
X		still_ok= Yes;
X		wri(v, No, No, Yes); newline();
X		f_close(fp);
X		redirect(stdout);
X#ifdef unix
X		if (still_ok) f_rename(fn, fname);
X#endif
X	}
X	else if (!silently) interrV(CANT_WRITE, fn);
X	still_ok= was_ok;
X	release(fn);
X	set_context(&c);
X}
X
XVisible Procedure endperm() {
X	static bool active;
X	bool was_ok= still_ok;
X	
X	if (active)
X		return;
X	active= Yes;
X	still_ok= Yes;
X	put_targs();
X	put_perm(b_perm);
X	still_ok= was_ok;
X	active= No;
X}
X
XHidden Procedure put_targs() {
X	int k, len;
X	value v, name;
X	
X	len= Valid(prmnv->tab) ? length(prmnv->tab) : 0;
X	for (k= 0; k < len; k++) {
X		v= copy(*assoc(prmnv->tab, k));
X		name= copy(*key(prmnv->tab, k));
X		if (!Is_filed(v)) {
X			value fname= get_tfname(name);
X			if (Valid(fname)) {
X				release(errtname); errtname= copy(name);
X				putval(fname, v, Yes, In_tarval);
X			}
X			release(fname);
X		}
X		tarfiled(name, Vnil);
X		release(v); release(name);
X	}
X}
X
XVisible Procedure put_perm(v) value v; {
X	value fn;
X	intlet len;
X	
X	if (!permchanges || !Valid(v))
X		return;
X	fn= mk_text(permfile);
X	/* Remove the file if the permanent environment is empty */
X	len= length(v);
X	if (len == 0)
X		f_delete(fn);
X	else
X		putval(fn, v, Yes, In_prmnv);
X	release(fn);
X	permchanges= No;
X}
X
XVisible Procedure clear_perm() {
X	freeunits(USR_ALL);
X	endperm();
X}
X
XVisible Procedure initsou() {
X	release(b_units); b_units= mk_elt();
X	release(last_unit); last_unit= mk_text(":");
X	release(last_target); last_target= mk_text("=");
X	release(b_perm); b_perm= mk_elt();
X}
X
XVisible Procedure endsou() {
X	if (terminated)
X		return;	/* hack; to prevent seemingly endless QUIT */
X	release(b_units); b_units= Vnil;
X	release(b_perm); b_perm= Vnil;
X	release(last_unit); last_unit= Vnil;
X	release(last_target); last_target= Vnil;
X}
X
X/*
X * lst_uhds() displays the first line of the unit without a possible
X * present simple command
X */
X 
X#define MORE MESS(4021, "Press [SPACE] for more, [RETURN] to exit list")
Xextern int winheight;
Xbool ask_for();
X
XVisible Procedure lst_uhds() {
X	intlet k, len;
X	value pname, *aa;
X	how *u;
X	int nprinted= 0;
X	bool more= Yes;
X	
X	len= length(b_perm);
X	for (k= 0; k<len && still_ok && more; ++k) {
X		pname= *key(b_perm, k);
X		if (!Is_text(pname) || Permtype(pname) == Tar) 
X			continue;
X		/* reduce disk access: */
X		if (u_exists(pname, &aa) && !Is_predefined(*aa))
X			display(stdout, How_to(*aa)->unit, Yes);
X		else
X			lst_fileheading(*assoc(b_perm, k));
X		fflush(stdout);
X		if (++nprinted >= winheight) {
X			more= ask_for(MORE);
X			nprinted= 0;
X		}
X	}
X	/* not interactive units */
X	len= length(b_units);
X	for (k= 0; k<len && still_ok && more; ++k) {
X		u= How_to(*assoc(b_units, k));
X		if (u -> filed == No && !p_exists(*key(b_units, k), &aa)) {
X			display(stdout, u -> unit, Yes);
X			fflush(stdout);
X			if (++nprinted >= winheight) {
X				more= ask_for(MORE);
X				nprinted= 0;
X			}
X		}
X
X	}
X}
X
XHidden Procedure lst_fileheading(v) value v; {
X	FILE *fn;
X	char *line;
X	char *pcolon, *pc;
X
X	if (!Is_text(v))
X		return;
X	fn= fopen(strval(v), "r");
X	if (!fn)
X		return;
X	if ((line= f_getline(fn)) != NULL) {
X		pcolon= strchr(line, C_COLON);
X		if (pcolon != NULL) {
X			pc= ++pcolon;
X			while (Space(*pc)) ++pc;
X			if (*pc != C_COMMENT && *pc != '\n') {
X				/* single command after colon;
X				 * don't show it.
X				 */
X				*(pcolon+1)= '\n';
X				*(pcolon+2)= '\0';
X			}
X		}
X		putstr(stdout, line);
X		freestr(line);
X	}
X	fclose(fn);
X}
END_OF_FILE
  if test 29957 -ne `wc -c <'abc/bint3/i3sou.c'`; then
    echo shar: \"'abc/bint3/i3sou.c'\" unpacked with wrong size!
  fi
  # end of 'abc/bint3/i3sou.c'
fi
if test ! -d 'abc/bio' ; then
    echo shar: Creating directory \"'abc/bio'\"
    mkdir 'abc/bio'
fi
if test ! -d 'abc/boot' ; then
    echo shar: Creating directory \"'abc/boot'\"
    mkdir 'abc/boot'
fi
if test ! -d 'abc/btr' ; then
    echo shar: Creating directory \"'abc/btr'\"
    mkdir 'abc/btr'
fi
if test ! -d 'abc/doc' ; then
    echo shar: Creating directory \"'abc/doc'\"
    mkdir 'abc/doc'
fi
if test ! -d 'abc/ehdrs' ; then
    echo shar: Creating directory \"'abc/ehdrs'\"
    mkdir 'abc/ehdrs'
fi
if test ! -d 'abc/ex' ; then
    echo shar: Creating directory \"'abc/ex'\"
    mkdir 'abc/ex'
fi
if test ! -d 'abc/ex/generate' ; then
    echo shar: Creating directory \"'abc/ex/generate'\"
    mkdir 'abc/ex/generate'
fi
if test ! -d 'abc/ex/hanoi' ; then
    echo shar: Creating directory \"'abc/ex/hanoi'\"
    mkdir 'abc/ex/hanoi'
fi
if test ! -d 'abc/ex/pi' ; then
    echo shar: Creating directory \"'abc/ex/pi'\"
    mkdir 'abc/ex/pi'
fi
if test ! -d 'abc/ex/try' ; then
    echo shar: Creating directory \"'abc/ex/try'\"
    mkdir 'abc/ex/try'
fi
if test ! -d 'abc/ex/xref' ; then
    echo shar: Creating directory \"'abc/ex/xref'\"
    mkdir 'abc/ex/xref'
fi
if test ! -d 'abc/ihdrs' ; then
    echo shar: Creating directory \"'abc/ihdrs'\"
    mkdir 'abc/ihdrs'
fi
if test ! -d 'abc/keys' ; then
    echo shar: Creating directory \"'abc/keys'\"
    mkdir 'abc/keys'
fi
if test ! -d 'abc/lin' ; then
    echo shar: Creating directory \"'abc/lin'\"
    mkdir 'abc/lin'
fi
if test -f 'abc/lin/i1obj.c' -a "${1}" != "-c" ; then 
  echo shar: Will not clobber existing file \"'abc/lin/i1obj.c'\"
else
  echo shar: Extracting \"'abc/lin/i1obj.c'\" \(7180 characters\)
  sed "s/^X//" >'abc/lin/i1obj.c' <<'END_OF_FILE'
X/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1986. */
X
X/* Generic routines for all values */
X
X#include "b.h"
X#include "bint.h"
X#include "bmem.h"
X#include "bobj.h"
X#include "i1tlt.h"
X#include "i3typ.h"
X
X#define Len (len < 200 ? len : ((len-1)/8+1)*8)
X
XVisible unsigned tltsyze(type, len, nptrs) 
X	literal type;
X	intlet len;
X	int *nptrs;
X{
X	register unsigned syze= 0;
X	*nptrs= 0;
X	switch (type) {
X	case Tex: syze= (len+1)*sizeof(char); *nptrs= 0; break;
X	case ELT:
X	case Lis:
X	case Ran:
X	case Tab: syze= Len*sizeof(value); *nptrs= len; break;
X	}
X	return syze;
X}
X
XVisible Procedure rel_subvalues(v) value v; {
X	rrelease(v);
X}
X
X#define INCOMP	MESS(500, "incompatible types %s and %s")
X
XHidden Procedure incompatible(v, w) value v, w; {
X	value m1, m2, m3, m;
X	string s1, s2;
X	
X	m1= convert(m3= (value) valtype(v), No, No); release(m3);
X	m2= convert(m3= (value) valtype(w), No, No); release(m3);
X	s1= sstrval(m1);
X	s2= sstrval(m2);
X	sprintf(messbuf, getmess(INCOMP), s1, s2);
X	m= mk_text(messbuf);
X	interrV(-1, m);
X
X	fstrval(s1); fstrval(s2);
X	release(m1); release(m2);
X	release(m);
X}
X
XVisible bool comp_ok;
X
X#define Sgn(d) (d)
X
XVisible relation compare(v, w) value v, w; {
X	literal vt= Type(v), wt= Type(w);
X	register intlet vlen, wlen, len, k;
X
X	comp_ok= Yes;
X	vlen= IsSmallInt(v) ? 0 : Length(v);
X	wlen= IsSmallInt(w) ? 0 : Length(w);
X	if (v == w) return 0;
X	if (!(vt == wt && !(vt == Com && vlen != wlen) ||
X			    vt == Ran && (wt == Lis || wt == ELT) ||
X			    wt == Ran && (vt == Lis || vt == ELT) ||
X			    vt == ELT && (wt == Lis || wt == Tab) ||
X			    wt == ELT && (vt == Lis || vt == Tab))) {
X		incompatible(v, w);
X		comp_ok= No;
X		return -1;
X	}
X	if (vt != Num && (vlen == 0 || wlen == 0))
X		return Sgn(vlen-wlen);
X	if (vt == Ran || wt == Ran)
X		return range_comp(v, w);
X	switch (vt) {
X	case Num: return numcomp(v, w);
X	case Tex: return strcmp(Str(v), Str(w));
X
X	case Com:
X	case Lis:
X	case Tab:
X	case ELT:
X		{value *vp= Ats(v), *wp= Ats(w);
X		 relation c;
X			len= vlen < wlen ? vlen : wlen;
X			for (k= 0; k < len; k++)
X				if ((c= compare(*vp++, *wp++)) != 0)
X					return c;
X			return Sgn(vlen-wlen);
X		}
X	default:
X		syserr(MESS(501, "comparison of unknown types"));
X		/* NOTREACHED */
X	}
X}
X
XVisible double hash(v) value v; {
X	if (Is_number(v))
X		return numhash(v);
X	else {
X		literal t= Type(v); intlet len= Length(v), k; 
X		double d= t+.404*len;
X		switch (t) {
X		case Tex:
X			{string vp= Str(v);
X				for (k= 0; k < len; k++)
X					d= .987*d+.277*(*vp++);
X				return d;
X			}
X		case Com:
X		case Lis:
X		case Ran:
X		case Tab:
X		case ELT:
X			{value *vp= Ats(v);
X				if (len == 0) return .909;
X				for (k= 0; k < len; k++)
X					d= .874*d+.310*hash(*vp++);
X				return d;
X			}
X		default:
X			syserr(MESS(502, "hash called with unknown type"));
X			/* NOTREACHED */
X		}
X	}
X}
X
X/* For reasons of efficiency, wri does not always call convert but writes
X   directly on the standard output. Modifications in convert should
X   be mirrored by changes in wri and vice versa. */
X
X#ifdef RANGEPRINT
XHidden Procedure conc_vals(pt, l, u) value *pt; value l, u; {
X	value x;
X	if (compare(l, u) == 0)
X		concato(pt, x= convert(l, No, No));
X	else if (is_increment(u, l)) {
X		concato(pt, x= convert(l, No, No)); release(x);
X		concato(pt, x= mk_text("; ")); release(x);
X		concato(pt, x= convert(u, No, No));
X	}
X	else {
X		concato(pt, x= convert(l, No, No)); release(x);
X		concato(pt, x= mk_text("..")); release(x);
X		concato(pt, x= convert(u, No, No));
X	}
X	release(x);
X}
X#endif /* RANGEPRINT */
X
X#define Last(k, len)	((k) == (len)-1)
X
XVisible value convert(v, coll, outer) value v; bool coll, outer; {
X	value t, quote, c, cv, sep, th, open, close, i, s;
X	int k, len; char ch; relation r;
X	switch (Type(v)) {
X	case Num:
X		return mk_text(convnum(v));
X	case Tex:
X		if (outer) return copy(v);
X		quote= mk_text("\"");
X		len= length(v);
X		t= copy(quote);
X		for (k=1; k<=len; k++) {
X			c= thof(k, v);
X			ch= charval(c);
X			concato(&t, c);
X			if (ch == '"' || ch == '`') concato(&t, c);
X			release(c);
X		}
X		concato(&t, quote);
X		release(quote);
X		break;
X	case Com:
X		len= Nfields(v);
X		outer&= coll;
X		sep= mk_text(outer ? " " : ", ");
X		t= mk_text(coll ? "" : "(");
X		for (k= 0; k < len; k++) {
X			concato(&t, cv= convert(*Field(v, k), No, outer));
X			release(cv);
X			if (!Last(k, len)) concato(&t, sep);
X		}
X		release(sep);
X		if (!coll) {
X			concato(&t, cv= mk_text(")"));
X			release(cv);
X		}
X		break;
X	case Ran:
X	case Lis:
X	case ELT:
X		t= mk_text("{");
X		sep= mk_text("; ");
X#ifndef RANGEPRINT
X		i= copy(one); s= size(v); 
X		while ((r=numcomp(i, s)) <= 0) {
X			th= item(v, i);
X			concato(&t, cv= convert(th, No, No));
X			if (r < 0) {
X				concato(&t, sep);
X			}
X			release(cv); release(th);
X			i= sum(th=i, one);
X			release(th);
X		}
X		release(i); release(s);
X#else /* RANGEPRINT */
X		{
X			value lwb, upb;
X			bool first= Yes;
X			i= copy(one); s= size(v);
X			while (numcomp(i, s) <= 0) {
X				th= item(v, i);
X				if (first) {
X					lwb= copy(th);
X					upb= copy(th);
X					first= No;
X				}
X				else if (is_increment(th, upb)) {
X					release(upb);
X					upb= copy(th);
X				}
X				else {
X					conc_vals(&t, lwb, upb) ;
X					concato(&t, sep);
X					release(lwb); release(upb);
X					lwb= copy(th); upb= copy(th);
X				}
X				release(th);
X				i= sum(th=i, one);
X				release(th);
X			}
X			if (!first) {
X				conc_vals(&t, lwb, upb);
X				release(lwb); release(upb);
X			}
X			release(i); release(s);
X		}
X#endif /* RANGEPRINT */
X		concato(&t, cv= mk_text("}"));
X		release(cv); release(sep);
X		break;
X	case Tab:
X		len= length(v);
X		open= mk_text("[");
X		close= mk_text("]: ");
X		sep= mk_text("; ");
X		t= mk_text("{");
X		for (k= 0; k < len; k++) {
X			concato(&t, open);
X			concato(&t, cv= convert(*key(v, k), Yes, No));
X			release(cv);
X			concato(&t, close);
X			concato(&t, cv= convert(*assoc(v, k), No, No));
X			release(cv);
X			if (!Last(k, len)) concato(&t, sep);
X		}
X		concato(&t, cv= mk_text("}")); release(cv);
X		release(open); release(close); release(sep);
X		break;
X	default:
X		syserr(MESS(503, "unknown type in convert"));
X	}
X	return t;
X}
X
X#define Left 'L'
X#define Right 'R'
X#define Centre 'C'
X
X#define ADJLEFT_NUM	MESS(504, "in t<<n, n is not a number")
X#define ADJRIGHT_NUM	MESS(505, "in t><n, n is not a number")
X#define CENTRE_NUM	MESS(506, "in t>>n, n is not a number")
X
XHidden value adj(x, y, side) value x, y; literal side; {
X	value r, v= convert(x, Yes, Yes); int i;
X	intlet lv, la, k, ls, rs;
X	string rp, vp;
X
X	if (!Is_number(y)) {
X		switch (side) {
X		case Left:	interr(ADJLEFT_NUM); break;
X		case Centre:	interr(ADJRIGHT_NUM); break;
X		case Right:	interr(CENTRE_NUM); break;
X		}
X		return v;
X	}
X	i= intval(y);
X	lv= Length(v);
X	la= propintlet(i) - lv;
X	if (la <= 0) return v;
X	r= grab(Tex, lv+la); rp= Str(r); vp= Str(v);
X
X	if (side == Left) { ls= 0; rs= la; }
X	else if (side == Centre) { ls= la/2; rs= (la+1)/2; }
X	else { ls= la; rs= 0; }
X
X	for (k= 0; k < ls; k++) *rp++= ' ';
X	for (k= 0; k < lv; k++) *rp++= *vp++;
X	for (k= 0; k < rs; k++) *rp++= ' ';
X	*rp= 0;
X	release(v);
X	return r;
X}
X
XVisible value adjleft(x, y) value x, y; {
X	return adj(x, y, Left);
X}
X
XVisible value centre(x, y) value x, y; {
X	return adj(x, y, Centre);
X}
X
XVisible value adjright(x, y) value x, y; {
X	return adj(x, y, Right);
X}
X
X
END_OF_FILE
  if test 7180 -ne `wc -c <'abc/lin/i1obj.c'`; then
    echo shar: \"'abc/lin/i1obj.c'\" unpacked with wrong size!
  fi
  # end of 'abc/lin/i1obj.c'
fi
if test ! -d 'abc/scripts' ; then
    echo shar: Creating directory \"'abc/scripts'\"
    mkdir 'abc/scripts'
fi
if test ! -d 'abc/stc' ; then
    echo shar: Creating directory \"'abc/stc'\"
    mkdir 'abc/stc'
fi
if test ! -d 'abc/tc' ; then
    echo shar: Creating directory \"'abc/tc'\"
    mkdir 'abc/tc'
fi
if test ! -d 'abc/uhdrs' ; then
    echo shar: Creating directory \"'abc/uhdrs'\"
    mkdir 'abc/uhdrs'
fi
if test ! -d 'abc/ukeys' ; then
    echo shar: Creating directory \"'abc/ukeys'\"
    mkdir 'abc/ukeys'
fi
if test ! -d 'abc/unix' ; then
    echo shar: Creating directory \"'abc/unix'\"
    mkdir 'abc/unix'
fi
echo shar: End of archive 1 \(of 25\).
cp /dev/null ark1isdone
MISSING=""
for I in 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 ; do
    if test ! -f ark${I}isdone ; then
	MISSING="${MISSING} ${I}"
    fi
done
if test "${MISSING}" = "" ; then
    echo You have unpacked all 25 archives.
    rm -f ark[1-9]isdone ark[1-9][0-9]isdone
else
    echo You still must unpack the following archives:
    echo "        " ${MISSING}
fi
exit 0
-- 
Please send comp.sources.unix-related mail to rsalz at uunet.uu.net.
Use a domain-based address or give alternate paths, or you may lose out.
    
    
More information about the Comp.sources.unix
mailing list