4.3BSD/usr/contrib/B/src/bsmall/b2loc.c

Compare this file to the similar file:
Show the results in this format:

/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1984. */
/* $Header: b2loc.c,v 1.1 84/06/28 00:49:16 timo Exp $ */

/* B locations and environments */
#include "b.h"
#include "b0con.h"
#include "b1obj.h"
#include "b2env.h" /* for bndtgs */
#include "b2sem.h"

Hidden value* location(l) loc l; {
	value *ll;
	if (Is_simploc(l)) {
		simploc *sl= Simploc(l);
		if (!in_env(sl->e->tab, sl->i, &ll)) error("target still empty");
		return ll;
	} else if (Is_tbseloc(l)) {
		tbseloc *tl= Tbseloc(l);
		ll= adrassoc(*location(tl->R), tl->K);
		if (ll == Pnil) error("key not in table");
		return ll;
	} else {
		syserr("call of location with improper type");
		return (value *) Dummy;
	}
}

Hidden Procedure uniquify(l) loc l; {
	if (Is_simploc(l)) {
		simploc *sl= Simploc(l);
		value *ta= &(sl->e->tab), ke= sl->i;
		uniql(ta);
		check_location(l);
		uniq_assoc(*ta, ke);
	} else if (Is_tbseloc(l)) {
		tbseloc *tl= Tbseloc(l);
		value t, ke;
		uniquify(tl->R);
		t= *location(tl->R); ke= tl->K;
		if (!Is_table(t)) error("selection on non-table");
		if (empty(t)) error("selection on empty table");
		check_location(l);
		uniq_assoc(t, ke);
	} else if (Is_trimloc(l)) { syserr("uniquifying trimloc");
	} else if (Is_compound(l)) { syserr("uniquifying comploc");
	} else syserr("uniquifying non-location");
}

Visible Procedure check_location(l) loc l; {
	VOID location(l);
	/* location may produce an error message */
}

Visible value content(l) loc l; {
	return copy(*location(l));
}

Visible loc trim_loc(R, B, C) loc R; intlet B, C; {
	if (Is_trimloc(R)) {
		trimloc *rr= Trimloc(R);
		return mk_trimloc(rr->R, B, C);
	} else if (Is_simploc(R) || Is_tbseloc(R)) {
		return mk_trimloc(R, B, C);
	} else {
		error("trim (@ or |) on target of improper type");
		/* NOTREACHED */
	}
}

Visible loc tbsel_loc(R, K) loc R; value K; {
	if (Is_simploc(R) || Is_tbseloc(R)) return mk_tbseloc(R, K);
	else error("selection on target of improper type");
	/* NOTREACHED */
}

Visible loc local_loc(i) basidf i; { return mk_simploc(i, curnv); }

Visible loc global_loc(i) basidf i; { return mk_simploc(i, prmnv); }

Visible Procedure put(v, l) value v; loc l; {
	if (Is_simploc(l)) {
		simploc *sl= Simploc(l);
		e_replace(v, &(sl->e->tab), sl->i);
	} else if (Is_trimloc(l)) {
		trimloc *tl= Trimloc(l);
		value rr, nn, head, tail, part;
		intlet B= tl->B, C= tl->C, len;
		rr= *location(tl->R);
		if (!Is_text(rr)) error("trim target contains no text");
		if (!Is_text(v))
		    error("putting non-text in trim(@ or|) on text location");
		len= length(rr);
		if (B < 0 || C < 0 || B+C > len)
		    error("trim (@ or |) on text location out of bounds");
		head= trim(rr, 0, len-B); /* rr|B */
		tail= trim(rr, len-C, 0); /* rr@(#rr-C+1) */
		part= concat(head, v);
		nn= concat(part, tail);
		put(nn, tl->R);
		release(nn); release(head); release(tail); release(part);
	} else if (Is_compound(l)) {
		intlet k, len= Nfields(l);
		if (!Is_compound(v))
		    error("putting non-compound in compound location");
		if (Nfields(v) != Nfields(l))
		    error("putting compound in compound location of different length");
		k_Overfields { put(*field(v, k), *field(l, k)); }
	} else if (Is_tbseloc(l)) {
		tbseloc *tl= Tbseloc(l);
		uniquify(tl->R);
		replace(v, location(tl->R), tl->K);
	} else error("putting in non-target");
}

Hidden bool l_exists(l) loc l; {
	if (Is_simploc(l)) {
		simploc *sl= Simploc(l);
		return in_keys(sl->i, sl->e->tab);
	} else if (Is_trimloc(l)) {
		error("deleting trimmed (@ or |) target");
		return No;
	} else if (Is_compound(l)) {
		intlet k, len= Nfields(l);
		k_Overfields { if (!l_exists(*field(l, k))) return No; }
		return Yes;
	} else if (Is_tbseloc(l)) {
		tbseloc *tl= Tbseloc(l);
		uniquify(tl->R);
		return in_keys(tl->K, *location(tl->R));
	} else {
		error("deleting non-target");
		return No;
	}
}

Hidden Procedure l_del(l) loc l; {
	if (Is_simploc(l)) {
		simploc *sl= Simploc(l);
		if (in_keys(sl->i, sl->e->tab)) {
			uniql(&(sl->e->tab)); /*no need?: see delete*/
			e_delete(&(sl->e->tab), sl->i);
		}
	} else if (Is_trimloc(l)) {
		error("deleting trimmed (@ or |) target");
	} else if (Is_compound(l)) {
		intlet k, len= Nfields(l);
		k_Overfields { l_del(*field(l, k)); }
	} else if (Is_tbseloc(l)) {
		tbseloc *tl= Tbseloc(l);
		value *lc;
		uniquify(tl->R);
		lc= location(tl->R);
		if (in_keys(tl->K, *lc)) delete(lc, tl->K);
	} else error("deleting non-target");
}

Visible Procedure l_delete(l) loc l; {
	if (l_exists(l)) l_del(l);
	else error("deleting non-existent target");
}

Visible Procedure l_insert(v, l) value v; loc l; {
	value *ll;
	uniquify(l);
	ll= location(l);
	insert(v, ll);
}

Visible Procedure l_remove(v, l) value v; loc l; {
	uniquify(l);
	remove(v, location(l));
}

Visible Procedure choose(l, v) loc l; value v; {
	value w, s, r;
	if (!Is_tlt(v)) error("choosing from non-text, -list or -table");
	s= size(v);
	if (compare(s, zero) == 0)
		error("choosing from empty text, list or table");
	/* PUT (floor(random*#v) + 1) th'of v IN l */
	r= prod(w= random(), s); release(w); release(s);
	w= floorf(r); release(r);
	r= sum(w, one); release(w);
	put(w= th_of(r, v), l); release(w);
}

Visible Procedure draw(l) loc l; {
	value r= random();
	put(r, l);
	release(r);
}

Visible Procedure bind(l) loc l; {
	if (Is_simploc(l)) {
		simploc *ll= Simploc(l);
		if (!in(ll->i, *bndtgs)) /* kludge */
			insert(ll->i, bndtgs);
	} else if (Is_compound(l)) {
		intlet k, len= Nfields(l);
		k_Overfields { bind(*field(l, k)); }
	} else if (Is_trimloc(l)) {
		pprerr("t@p or t|p not allowed in ranger", "");
	} else if (Is_tbseloc(l)) {
		pprerr("t[e] not allowed in ranger", "");
	} else error("binding non-identifier");
}