⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 tl_cache.c

📁 这是一个同样来自贝尔实验室的和UNIX有着渊源的操作系统, 其简洁的设计和实现易于我们学习和理解
💻 C
字号:
/***** tl_spin: tl_cache.c *****//* Copyright (c) 1995-2003 by Lucent Technologies, Bell Laboratories.     *//* All Rights Reserved.  This software is for educational purposes only.  *//* No guarantee whatsoever is expressed or implied by the distribution of *//* this code.  Permission is given to distribute this code provided that  *//* this introductory message is not removed and no monies are exchanged.  *//* Software written by Gerard J. Holzmann.  For tool documentation see:   *//*             http://spinroot.com/                                       *//* Send all bug-reports and/or questions to: bugs@spinroot.com            *//* Based on the translation algorithm by Gerth, Peled, Vardi, and Wolper, *//* presented at the PSTV Conference, held in 1995, Warsaw, Poland 1995.   */#include "tl.h"typedef struct Cache {	Node *before;	Node *after;	int same;	struct Cache *nxt;} Cache;static Cache	*stored = (Cache *) 0;static unsigned long	Caches, CacheHits;static int	ismatch(Node *, Node *);extern void fatal(char *, char *);int	sameform(Node *, Node *);#if 0voidcache_dump(void){	Cache *d; int nr=0;	printf("\nCACHE DUMP:\n");	for (d = stored; d; d = d->nxt, nr++)	{	if (d->same) continue;		printf("B%3d: ", nr); dump(d->before); printf("\n");		printf("A%3d: ", nr); dump(d->after); printf("\n");	}	printf("============\n");}#endifNode *in_cache(Node *n){	Cache *d; int nr=0;	for (d = stored; d; d = d->nxt, nr++)		if (isequal(d->before, n))		{	CacheHits++;			if (d->same && ismatch(n, d->before)) return n;			return dupnode(d->after);		}	return ZN;}Node *cached(Node *n){	Cache *d;	Node *m;	if (!n) return n;	if ((m = in_cache(n)) != ZN)		return m;	Caches++;	d = (Cache *) tl_emalloc(sizeof(Cache));	d->before = dupnode(n);	d->after  = Canonical(n); /* n is released */	if (ismatch(d->before, d->after))	{	d->same = 1;		releasenode(1, d->after);		d->after = d->before;	}	d->nxt = stored;	stored = d;	return dupnode(d->after);}voidcache_stats(void){	printf("cache stores     : %9ld\n", Caches);	printf("cache hits       : %9ld\n", CacheHits);}voidreleasenode(int all_levels, Node *n){	if (!n) return;	if (all_levels)	{	releasenode(1, n->lft);		n->lft = ZN;		releasenode(1, n->rgt);		n->rgt = ZN;	}	tfree((void *) n);}Node *tl_nn(int t, Node *ll, Node *rl){	Node *n = (Node *) tl_emalloc(sizeof(Node));	n->ntyp = (short) t;	n->lft  = ll;	n->rgt  = rl;	return n;}Node *getnode(Node *p){	Node *n;	if (!p) return p;	n =  (Node *) tl_emalloc(sizeof(Node));	n->ntyp = p->ntyp;	n->sym  = p->sym; /* same name */	n->lft  = p->lft;	n->rgt  = p->rgt;	return n;}Node *dupnode(Node *n){	Node *d;	if (!n) return n;	d = getnode(n);	d->lft = dupnode(n->lft);	d->rgt = dupnode(n->rgt);	return d;}intone_lft(int ntyp, Node *x, Node *in){	if (!x)  return 1;	if (!in) return 0;	if (sameform(x, in))		return 1;	if (in->ntyp != ntyp)		return 0;	if (one_lft(ntyp, x, in->lft))		return 1;	return one_lft(ntyp, x, in->rgt);}intall_lfts(int ntyp, Node *from, Node *in){	if (!from) return 1;	if (from->ntyp != ntyp)		return one_lft(ntyp, from, in);	if (!one_lft(ntyp, from->lft, in))		return 0;	return all_lfts(ntyp, from->rgt, in);}intsametrees(int ntyp, Node *a, Node *b){	/* toplevel is an AND or OR */	/* both trees are right-linked, but the leafs */	/* can be in different places in the two trees */	if (!all_lfts(ntyp, a, b))		return 0;	return all_lfts(ntyp, b, a);}int	/* a better isequal() */sameform(Node *a, Node *b){	if (!a && !b) return 1;	if (!a || !b) return 0;	if (a->ntyp != b->ntyp) return 0;	if (a->sym	&&  b->sym	&&  strcmp(a->sym->name, b->sym->name) != 0)		return 0;	switch (a->ntyp) {	case TRUE:	case FALSE:		return 1;	case PREDICATE:		if (!a->sym || !b->sym) fatal("sameform...", (char *) 0);		return !strcmp(a->sym->name, b->sym->name);	case NOT:#ifdef NXT	case NEXT:#endif		return sameform(a->lft, b->lft);	case U_OPER:	case V_OPER:		if (!sameform(a->lft, b->lft))			return 0;		if (!sameform(a->rgt, b->rgt))			return 0;		return 1;	case AND:	case OR:	/* the hard case */		return sametrees(a->ntyp, a, b);	default:		printf("type: %d\n", a->ntyp);		fatal("cannot happen, sameform", (char *) 0);	}	return 0;}intisequal(Node *a, Node *b){	if (!a && !b)		return 1;	if (!a || !b)	{	if (!a)		{	if (b->ntyp == TRUE)				return 1;		} else		{	if (a->ntyp == TRUE)				return 1;		}		return 0;	}	if (a->ntyp != b->ntyp)		return 0;	if (a->sym	&&  b->sym	&&  strcmp(a->sym->name, b->sym->name) != 0)		return 0;	if (isequal(a->lft, b->lft)	&&  isequal(a->rgt, b->rgt))		return 1;	return sameform(a, b);}static intismatch(Node *a, Node *b){	if (!a && !b) return 1;	if (!a || !b) return 0;	if (a->ntyp != b->ntyp) return 0;	if (a->sym	&&  b->sym	&&  strcmp(a->sym->name, b->sym->name) != 0)		return 0;	if (ismatch(a->lft, b->lft)	&&  ismatch(a->rgt, b->rgt))		return 1;	return 0;}intany_term(Node *srch, Node *in){	if (!in) return 0;	if (in->ntyp == AND)		return	any_term(srch, in->lft) ||			any_term(srch, in->rgt);	return isequal(in, srch);}intany_and(Node *srch, Node *in){	if (!in) return 0;	if (srch->ntyp == AND)		return	any_and(srch->lft, in) &&			any_and(srch->rgt, in);	return any_term(srch, in);}intany_lor(Node *srch, Node *in){	if (!in) return 0;	if (in->ntyp == OR)		return	any_lor(srch, in->lft) ||			any_lor(srch, in->rgt);	return isequal(in, srch);}intanywhere(int tok, Node *srch, Node *in){	if (!in) return 0;	switch (tok) {	case AND:	return any_and(srch, in);	case  OR:	return any_lor(srch, in);	case   0:	return any_term(srch, in);	}	fatal("cannot happen, anywhere", (char *) 0);	return 0;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -