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

📄 spin.y

📁 这是一个同样来自贝尔实验室的和UNIX有着渊源的操作系统, 其简洁的设计和实现易于我们学习和理解
💻 Y
📖 第 1 页 / 共 2 页
字号:
/***** spin: spin.y *****//* Copyright (c) 1989-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            */%{#include "spin.h"#include <stdarg.h>#define YYDEBUG	0#define Stop	nn(ZN,'@',ZN,ZN)extern  Symbol	*context, *owner;extern  int	u_sync, u_async, dumptab;extern	short	has_sorted, has_random, has_enabled, has_pcvalue, has_np;extern	short	has_code, has_state, has_io;extern	void	count_runs(Lextok *);extern	void	validref(Lextok *, Lextok *);extern	char	yytext[];int	Mpars = 0;	/* max nr of message parameters  */int	runsafe = 1;	/* 1 if all run stmnts are in init */int	Expand_Ok = 0, realread = 1, IArgs = 0, NamesNotAdded = 0;char	*claimproc = (char *) 0;char	*eventmap = (char *) 0;static	int	Embedded = 0, inEventMap = 0, has_ini = 0;%}%token	ASSERT PRINT PRINTM%token	C_CODE C_DECL C_EXPR C_STATE C_TRACK%token	RUN LEN ENABLED EVAL PC_VAL%token	TYPEDEF MTYPE INLINE LABEL OF%token	GOTO BREAK ELSE SEMI%token	IF FI DO OD SEP%token	ATOMIC NON_ATOMIC D_STEP UNLESS%token  TIMEOUT NONPROGRESS%token	ACTIVE PROCTYPE D_PROCTYPE%token	HIDDEN SHOW ISLOCAL%token	PRIORITY PROVIDED%token	FULL EMPTY NFULL NEMPTY%token	CONST TYPE XU			/* val */%token	NAME UNAME PNAME INAME		/* sym */%token	STRING CLAIM TRACE INIT		/* sym */%right	ASGN%left	SND O_SND RCV R_RCV /* SND doubles as boolean negation */%left	OR%left	AND%left	'|'%left	'^'%left	'&'%left	EQ NE%left	GT LT GE LE%left	LSHIFT RSHIFT%left	'+' '-'%left	'*' '/' '%'%left	INCR DECR%right	'~' UMIN NEG%left	DOT%%/** PROMELA Grammar Rules **/program	: units		{ yytext[0] = '\0'; }	;units	: unit	| units unit	;unit	: proc		/* proctype { }       */	| init		/* init { }           */	| claim		/* never claim        */	| events	/* event assertions   */	| one_decl	/* variables, chans   */	| utype		/* user defined types */	| c_fcts	/* c functions etc.   */	| ns		/* named sequence     */	| SEMI		/* optional separator */	| error	;proc	: inst		/* optional instantiator */	  proctype NAME	{			  setptype($3, PROCTYPE, ZN);			  setpname($3);			  context = $3->sym;			  context->ini = $2; /* linenr and file */			  Expand_Ok++; /* expand struct names in decl */			  has_ini = 0;			}	  '(' decl ')'	{ Expand_Ok--;			  if (has_ini)			  fatal("initializer in parameter list", (char *) 0);			}	  Opt_priority	  Opt_enabler	  body		{ ProcList *rl;			  rl = ready($3->sym, $6, $11->sq, $2->val, $10);			  if ($1 != ZN && $1->val > 0)			  {	int j;			  	for (j = 0; j < $1->val; j++)				runnable(rl, $9?$9->val:1, 1);				announce(":root:");				if (dumptab) $3->sym->ini = $1;			  }			  context = ZS;			}	;proctype: PROCTYPE	{ $$ = nn(ZN,CONST,ZN,ZN); $$->val = 0; }	| D_PROCTYPE	{ $$ = nn(ZN,CONST,ZN,ZN); $$->val = 1; }	;inst	: /* empty */	{ $$ = ZN; }	| ACTIVE	{ $$ = nn(ZN,CONST,ZN,ZN); $$->val = 1; }	| ACTIVE '[' CONST ']' {			  $$ = nn(ZN,CONST,ZN,ZN); $$->val = $3->val;			  if ($3->val > 255)				non_fatal("max nr of processes is 255\n", "");			}	| ACTIVE '[' NAME ']' {			  $$ = nn(ZN,CONST,ZN,ZN);			  $$->val = 0;			  if (!$3->sym->type)				non_fatal("undeclared variable %s",					$3->sym->name);			  else if ($3->sym->ini->ntyp != CONST)				non_fatal("need constant initializer for %s\n",					$3->sym->name);			  else				$$->val = $3->sym->ini->val;			}	;init	: INIT		{ context = $1->sym; }	  Opt_priority	  body		{ ProcList *rl;			  rl = ready(context, ZN, $4->sq, 0, ZN);			  runnable(rl, $3?$3->val:1, 1);			  announce(":root:");			  context = ZS;        		}	;claim	: CLAIM		{ context = $1->sym;			  if (claimproc)				non_fatal("claim %s redefined", claimproc);			  claimproc = $1->sym->name;			}	  body		{ (void) ready($1->sym, ZN, $3->sq, 0, ZN);        		  context = ZS;        		}	;events : TRACE		{ context = $1->sym;			  if (eventmap)				non_fatal("trace %s redefined", eventmap);			  eventmap = $1->sym->name;			  inEventMap++;			}	  body		{ (void) ready($1->sym, ZN, $3->sq, 0, ZN);        		  context = ZS;			  inEventMap--;			}	;utype	: TYPEDEF NAME		{ if (context)				   fatal("typedef %s must be global",						$2->sym->name);				   owner = $2->sym;				}	  '{' decl_lst '}'	{ setuname($5); owner = ZS; }	;nm	: NAME			{ $$ = $1; }	| INAME			{ $$ = $1;				  if (IArgs)				  fatal("invalid use of '%s'", $1->sym->name);				}	;ns	: INLINE nm '('		{ NamesNotAdded++; }	  args ')'		{ prep_inline($2->sym, $5);				  NamesNotAdded--;				}	;c_fcts	: ccode			{ /* leaves pseudo-inlines with sym of				   * type CODE_FRAG or CODE_DECL in global context				   */				}	| cstate	;cstate	: C_STATE STRING STRING	{				  c_state($2->sym, $3->sym, ZS);				  has_code = has_state = 1;				}	| C_TRACK STRING STRING {				  c_track($2->sym, $3->sym, ZS);				  has_code = has_state = 1;				}	| C_STATE STRING STRING	STRING {				  c_state($2->sym, $3->sym, $4->sym);				  has_code = has_state = 1;				}	| C_TRACK STRING STRING STRING {				  c_track($2->sym, $3->sym, $4->sym);				  has_code = has_state = 1;				}	;ccode	: C_CODE		{ Symbol *s;				  NamesNotAdded++;				  s = prep_inline(ZS, ZN);				  NamesNotAdded--;				  $$ = nn(ZN, C_CODE, ZN, ZN);				  $$->sym = s;				  has_code = 1;				}	| C_DECL		{ Symbol *s;				  NamesNotAdded++;				  s = prep_inline(ZS, ZN);				  NamesNotAdded--;				  s->type = CODE_DECL;				  $$ = nn(ZN, C_CODE, ZN, ZN);				  $$->sym = s;				  has_code = 1;				}	;cexpr	: C_EXPR		{ Symbol *s;				  NamesNotAdded++;				  s = prep_inline(ZS, ZN);				  NamesNotAdded--;				  $$ = nn(ZN, C_EXPR, ZN, ZN);				  $$->sym = s;				  no_side_effects(s->name);				  has_code = 1;				}	;body	: '{'			{ open_seq(1); }          sequence OS		{ add_seq(Stop); }          '}'			{ $$->sq = close_seq(0); }	;sequence: step			{ if ($1) add_seq($1); }	| sequence MS step	{ if ($3) add_seq($3); }	;step    : one_decl		{ $$ = ZN; }	| XU vref_lst		{ setxus($2, $1->val); $$ = ZN; }	| NAME ':' one_decl	{ fatal("label preceding declaration,", (char *)0); }	| NAME ':' XU		{ fatal("label predecing xr/xs claim,", 0); }	| stmnt			{ $$ = $1; }	| stmnt UNLESS stmnt	{ $$ = do_unless($1, $3); }	;vis	: /* empty */		{ $$ = ZN; }	| HIDDEN		{ $$ = $1; }	| SHOW			{ $$ = $1; }	| ISLOCAL		{ $$ = $1; }	;asgn:	/* empty */	| ASGN	;one_decl: vis TYPE var_list	{ setptype($3, $2->val, $1); $$ = $3; }	| vis UNAME var_list	{ setutype($3, $2->sym, $1);				  $$ = expand($3, Expand_Ok);				}	| vis TYPE asgn '{' nlst '}' {				  if ($2->val != MTYPE)					fatal("malformed declaration", 0);				  setmtype($5);				  if ($1)					non_fatal("cannot %s mtype (ignored)",						$1->sym->name);				  if (context != ZS)					fatal("mtype declaration must be global", 0);				}	;decl_lst: one_decl       	{ $$ = nn(ZN, ',', $1, ZN); }	| one_decl SEMI	  decl_lst		{ $$ = nn(ZN, ',', $1, $3); }	;decl    : /* empty */		{ $$ = ZN; }	| decl_lst      	{ $$ = $1; }	;vref_lst: varref		{ $$ = nn($1, XU, $1, ZN); }	| varref ',' vref_lst	{ $$ = nn($1, XU, $1, $3); }	;var_list: ivar           	{ $$ = nn($1, TYPE, ZN, ZN); }	| ivar ',' var_list	{ $$ = nn($1, TYPE, ZN, $3); }	;ivar    : vardcl           	{ $$ = $1;				  $1->sym->ini = nn(ZN,CONST,ZN,ZN);				  $1->sym->ini->val = 0;				}	| vardcl ASGN expr   	{ $1->sym->ini = $3; $$ = $1;				  trackvar($1,$3); has_ini = 1;				}	| vardcl ASGN ch_init	{ $1->sym->ini = $3;				  $$ = $1; has_ini = 1;				}	;ch_init : '[' CONST ']' OF	  '{' typ_list '}'	{ if ($2->val) u_async++;				  else u_sync++;        			  {	int i = cnt_mpars($6);					Mpars = max(Mpars, i);				  }        			  $$ = nn(ZN, CHAN, ZN, $6);				  $$->val = $2->val;        			}	;vardcl  : NAME  		{ $1->sym->nel = 1; $$ = $1; }	| NAME ':' CONST	{ $1->sym->nbits = $3->val;				  if ($3->val >= 8*sizeof(long))				  {	non_fatal("width-field %s too large",						$1->sym->name);					$3->val = 8*sizeof(long)-1;				  }				  $1->sym->nel = 1; $$ = $1;				}	| NAME '[' CONST ']'	{ $1->sym->nel = $3->val; $$ = $1; }	;varref	: cmpnd			{ $$ = mk_explicit($1, Expand_Ok, NAME); }	;pfld	: NAME			{ $$ = nn($1, NAME, ZN, ZN); }	| NAME			{ owner = ZS; }	  '[' expr ']'		{ $$ = nn($1, NAME, $4, ZN); }	;cmpnd	: pfld			{ Embedded++;				  if ($1->sym->type == STRUCT)					owner = $1->sym->Snm;				}	  sfld			{ $$ = $1; $$->rgt = $3;

⌨️ 快捷键说明

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