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

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

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

#include "b.h"
#include "b1obj.h"
#include "B1tlt.h"

Visible value mk_elt() { return grab_elt(); }

Visible value size(x) value x; { /* monadic # operator */
	if (!Is_tlt(x)) error("in #t, t is not a text, list or table");
	return mk_integer((int) Length(x));
}

#define Lisent(tp,k) (*(tp+(k)))

Visible value size2(v, t) value v, t; { /* Dyadic # operator */
	intlet len= Length(t), n= 0, k; value *tp= Ats(t);
	if (!Is_tlt(t)) error("in e#t, t is not a text, list or table");
	switch (t->type) {
	case Tex:
		{string cp= (string)tp; char c;
			if (v->type != Tex)
				error("in e#t, t is a text but e is not");
			if (Length(v) != 1) error(
				"in e#t, e is a text but not a character");
			c= *Str(v);
			Overall if (*cp++ == c) n++;
		} break;
	case ELT:
		break;
	case Lis:
		{intlet lo= -1, mi, xx, mm, hi= len; relation c;
		bins:	if (hi-lo < 2) break;
			mi= (lo+hi)/2;
			if ((c= compare(v, Lisent(tp,mi))) == 0) goto some;
			if (c < 0) hi= mi; else lo= mi;
			goto bins;
		some:	xx= mi;
			while (xx-lo > 1) {
				mm= (lo+xx)/2;
				if (compare(v, Lisent(tp,mm)) == 0) xx= mm;
				else lo= mm;
			}
			xx= mi;
			while (hi-xx > 1) {
				mm= (xx+hi)/2;
				if (compare(v, Lisent(tp,mm)) == 0) xx= mm;
				else hi= mm;
			}
			n= hi-lo-1;
		} break;
	case Tab:
		Overall if (compare(v, Dts(*tp++)) == 0) n++;
		break;
	default:
		syserr("e#t with non text, list or table");
		break;
	}
	return mk_integer((int) n);
}

Hidden bool less(r) relation r;    { return r<0; }
Hidden bool greater(r) relation r; { return r>0; }

Hidden value mm1(t, rel) value t; bool (*rel)(); {
	intlet len= Length(t), k; value m, *tp= Ats(t);
	switch (t->type) {
	case Tex:
		{string cp= (string) tp; char mc= '\0', mm[2];
			Overall {
				if (mc == '\0' || ((*rel)(*cp < mc ? -1 : (*cp > mc ? 1 : 0))))
					mc= *cp;
				cp++;
			}
			mm[0]= mc; mm[1]= '\0';
			m= mk_text(mm);
		} break;
	case Lis:
		if ((*rel)(-1)) /*min*/ m= copy(*Ats(t));
		else m= copy(*(Ats(t)+len-1));
		break;
	case Tab:
		{value dm= Vnil;
			Overall {
				if (dm == Vnil || (*rel)(compare(Dts(*tp), dm)))
					dm= Dts(*tp);
				tp++;
			}
			m= copy(dm);
		} break;
	default:
		syserr("min or max t, with non text, list or table");
	}
	return m;
}

Hidden value mm2(v, t, rel) value v, t; bool (*rel)(); {
	intlet len= Length(t), k; value m= Vnil, *tp= Ats(t);
	switch (t->type) {
	case Tex:
		{string cp= (string) tp; char c, mc= '\0', mm[2];
			c= *Str(v);
			Overall {
				if ((*rel)(c < *cp ? -1 : c > *cp ? 1 : 0)) {
					if (mc == '\0' || (*rel)(*cp < mc ? -1 : *cp>mc ? 1 : 0))
						mc= *cp;
				}
				cp++;
			}
			if (mc != '\0') {
				mm[0]= mc; mm[1]= '\0';
				m= mk_text(mm);
			}
		} break;
	case Lis:
		{intlet lim1, mid, lim2;
			if ((*rel)(-1)) { /*min*/
				lim1= 1; lim2= len-1;
			} else {
				lim2= 1; lim1= len-1;
			}
			if (!(*rel)(compare(v, Lisent(tp,lim2)))) break;
			if (len == 1 || (*rel)(compare(v, Lisent(tp,lim1)))) {
				m= copy(Lisent(tp,lim1));
				break;
			}
			/* v rel tp[lim2] && !(v rel tp[lim1]) */
			while (abs(lim2-lim1) > 1) {
				mid= (lim1+lim2)/2;
				if ((*rel)(compare(v, Lisent(tp,mid)))) lim2= mid;
				else lim1= mid;
			}
			m= copy(Lisent(tp,lim2));
		} break;
	case Tab:
		{value dm= Vnil;
			Overall {
				if ((*rel)(compare(v, Dts(*tp)))) {
					if (dm == Vnil ||
						(*rel)(compare(Dts(*tp), dm)))
						dm= Dts(*tp);
				}
				tp++;
			}
			if (dm != Vnil) m= copy(dm);
		} break;
	default:
		syserr("min2 or max2 with non text, list or table");
		break;
	}
	return m;
}

Visible value min1(t) value t; { /* Monadic min */
	if (!Is_tlt(t)) error("in min t, t is not a text, list or table");
	if (Length(t) == 0) error("in min t, t is empty");
	return mm1(t, less);
}

Visible value min2(v, t) value v, t; {
	value m;
	if (!Is_tlt(t)) error("in e min t, t is not a text, list or table");
	if (Length(t) == 0) error("in e min t, t is empty");
	if (Is_text(t)) {
		if (!Is_text(v)) error("in e min t, t is a text but e is not");
		if (Length(v) != 1) error("in e min t, e is a text but not a character");
	}
	m= mm2(v, t, less);
	if (m == Vnil) error("in e min t, no element of t exceeds e");
	return m;
}

Visible value max1(t) value t; {
	if (!Is_tlt(t)) error("in max t, t is not a text, list or table");
	if (Length(t) == 0) error("in max t, t is empty");
	return mm1(t, greater);
}

Visible value max2(v, t) value v, t; {
	value m;
	if (!Is_tlt(t)) error("in e max t, t is not a text, list or table");
	if (Length(t) == 0) error("in e max t, t is empty");
	if (Is_text(t)) {
		if (!Is_text(v)) error("in e max t, t is a text but e is not");
		if (Length(v) != 1) error("in e max t, e is a text but not a character");
	}
	m= mm2(v, t, greater);
	if (m == Vnil) error("in e max t, no element of t is less than e");
	return m;
}

Visible value th_of(n, t) value n, t; {
	return thof(intval(n), t);
}

Visible value thof(n, t) int n; value t; {
	intlet len= Length(t); value w;
	if (!Is_tlt(t)) error("in n th'of t, t is not a text, list or table");
	if (n <= 0 || n > len) error("in n th'of t, n is out of bounds");
	switch (t->type) {
	case Tex:
		{char ww[2];
			ww[0]= *(Str(t)+n-1); ww[1]= '\0';
			w= mk_text(ww);
		} break;
	case Lis:
		w= copy(*(Ats(t)+n-1));
		break;
	case Tab:
		w= copy(Dts(*(Ats(t)+n-1)));
		break;
	default:
		syserr("th'of with non text, list or table");
	}
	return w;
}

Visible bool found(elem, v, probe, where)
	value (*elem)(), v, probe; intlet *where;
	/* think of elem(v,lo-1) as -Infinity and elem(v,hi+1) as +Infinity.
	   found and where at the end satisfy:
	   SELECT:
	       SOME k IN {lo..hi} HAS probe = elem(v,k):
	           found = Yes AND where = k
	       ELSE: found = No AND elem(v,where-1) < probe < elem(v,where).
	*/
{relation c; intlet lo=0, hi= Length(v)-1;
	if (lo > hi) { *where= lo; return No; }
	if ((c= compare(probe, (*elem)(v, lo))) == 0) {*where= lo; return Yes; }
	if (c < 0) { *where=lo; return No; }
	if (lo == hi) { *where=hi+1; return No; }
	if ((c= compare(probe, (*elem)(v, hi))) == 0) { *where=hi; return Yes; }
	if (c > 0) { *where=hi+1; return No; }
	/* elem(lo) < probe < elem(hi) */
	while (hi-lo > 1) {
		if ((c= compare(probe, (*elem)(v, (lo+hi)/2))) == 0) {
			*where= (lo+hi)/2; return Yes;
		}
		if (c < 0) hi= (lo+hi)/2; else lo= (lo+hi)/2;
	}
	*where= hi; return No;
}

Visible bool in(v, t) value v, t; {
	intlet where, k, len= Length(t); value *tp= Ats(t);
	if (!Is_tlt(t)) error("in the test e in t, t is not a text, list or table");
	switch (t->type) {
	case Tex:
		if (v->type != Tex)
			error("in the test e in t, t is a text but e is not");
		if (Length(v) != 1)
			error("in the test e in t, e is a text but not a character");
		return index((string) tp, *Str(v)) != 0;
	case ELT:
		return No;
	case Lis:
		return found(list_elem, t, v, &where);
	case Tab:
		Overall if (compare(v, Dts(*tp++)) == 0) return Yes;
		return No;
	default:
		syserr("e in t with non text, list or table");
		return No;
	}
}