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

📄 syntax.c

📁 早期freebsd实现
💻 C
📖 第 1 页 / 共 2 页
字号:
		    {		    case Scharquote:		    case Sescape:		      from++;		    }		  from++;		}	      from++;	      if (!depth && sexpflag) goto done;	      break;	    }	}      /* Reached end of buffer.  Error if within object, return nil if between */      if (depth) goto lose;      immediate_quit = 0;      return Qnil;      /* End of object reached */    done:      count--;    }  while (count < 0)    {      stop = BEGV;      while (from > stop)	{	  from--;	  if (quoted = char_quoted (from))	    from--;	  c = FETCH_CHAR (from);	  code = SYNTAX (c);	  if (from > stop && SYNTAX_COMEND_SECOND (c)	      && SYNTAX_COMEND_FIRST (FETCH_CHAR (from - 1))	      && !char_quoted (from - 1)	      && parse_sexp_ignore_comments)	    code = Sendcomment, from--;#ifdef SWITCH_ENUM_BUG	  switch ((int) (quoted ? Sword : code))#else	  switch (quoted ? Sword : code)#endif	    {	    case Sword:	    case Ssymbol:	      if (depth || !sexpflag) break;	      /* This word counts as a sexp; count object finished after passing it. */	      while (from > stop)		{		  if (quoted = char_quoted (from - 1))		    from--;		  if (! (quoted || SYNTAX(FETCH_CHAR (from - 1)) == Sword ||			 SYNTAX(FETCH_CHAR (from - 1)) == Ssymbol ||			 SYNTAX(FETCH_CHAR (from - 1)) == Squote))            	    goto done2;		  from--;		}	      goto done2;	    case Smath:	      if (!sexpflag)		break;	      if (from != stop && c == FETCH_CHAR (from - 1))		from--;	      if (mathexit)		{		  mathexit = 0;		  goto open2;		}	      mathexit = 1;	    case Sclose:	      if (!++depth) goto done2;	      break;	    case Sopen:	    open2:	      if (!--depth) goto done2;	      if (depth < min_depth)		error ("Containing expression ends prematurely");	      break;	    case Sendcomment:	      if (!parse_sexp_ignore_comments) break;	      if (from != stop) from--;	      while (1)		{		  if (SYNTAX (c = FETCH_CHAR (from)) == Scomment)		    break;		  if (from == stop) goto done;		  from--;		  if (SYNTAX_COMSTART_SECOND (c)		      && SYNTAX_COMSTART_FIRST (FETCH_CHAR (from))		      && !char_quoted (from))		    break;		}	      break;	    case Sstring:	      stringterm = FETCH_CHAR (from);	      while (1)		{		  if (from == stop) goto lose;		  if (!char_quoted (from - 1)		      && stringterm == FETCH_CHAR (from - 1))		    break;		  from--;		}	      from--;	      if (!depth && sexpflag) goto done2;	      break;	    }	}      /* Reached start of buffer.  Error if within object, return nil if between */      if (depth) goto lose;      immediate_quit = 0;      return Qnil;    done2:      count++;    }  immediate_quit = 0;  XFASTINT (val) = from;  return val; lose:  error ("Unbalanced parentheses");  /* NOTREACHED */}char_quoted (pos)     register int pos;{  register enum syntaxcode code;  register int beg = BEGV;  register int quoted = 0;  while (pos > beg &&	 ((code = SYNTAX (FETCH_CHAR (pos - 1))) == Scharquote	  || code == Sescape))    pos--, quoted = !quoted;  return quoted;}DEFUN ("scan-lists", Fscan_lists, Sscan_lists, 3, 3, 0,  "Scan from character number FROM by COUNT lists.\n\Returns the character number of the position thus found.\n\\n\If DEPTH is nonzero, paren depth begins counting from that value,\n\only places where the depth in parentheses becomes zero\n\are candidates for stopping; COUNT such places are counted.\n\Thus, a positive value for DEPTH means go out levels.\n\\n\Comments are ignored if parse-sexp-ignore-comments is non-nil.\n\\n\If the beginning or end of (the visible part of) the buffer is reached\n\and the depth is wrong, an error is signaled.\n\If the depth is right but the count is not used up, nil is returned.")  (from, count, depth)     Lisp_Object from, count, depth;{  CHECK_NUMBER (from, 0);  CHECK_NUMBER (count, 1);  CHECK_NUMBER (depth, 2);  return scan_lists (XINT (from), XINT (count), XINT (depth), 0);}DEFUN ("scan-sexps", Fscan_sexps, Sscan_sexps, 2, 2, 0,  "Scan from character number FROM by COUNT balanced expressions.\n\Returns the character number of the position thus found.\n\\n\Comments are ignored if parse-sexp-ignore-comments is non-nil.\n\\n\If the beginning or end of (the visible part of) the buffer is reached\n\in the middle of a parenthetical grouping, an error is signaled.\n\If the beginning or end is reached between groupings but before count is used up,\n\nil is returned.")  (from, count)     Lisp_Object from, count;{  CHECK_NUMBER (from, 0);  CHECK_NUMBER (count, 1);  return scan_lists (XINT (from), XINT (count), 0, 1);}DEFUN ("backward-prefix-chars", Fbackward_prefix_chars, Sbackward_prefix_chars,  0, 0, 0,  "Move point backward over any number of chars with syntax \"prefix\".")  (){  int beg = BEGV;  int pos = point;  while (pos > beg && !char_quoted (pos - 1) && SYNTAX (FETCH_CHAR (pos - 1)) == Squote)    pos--;  SET_PT (pos);  return Qnil;}struct lisp_parse_state  {    int depth;		/* Depth at end of parsing */    int instring;	/* -1 if not within string, else desired terminator. */    int incomment;	/* Nonzero if within a comment at end of parsing */    int quoted;		/* Nonzero if just after an escape char at end of parsing */    int thislevelstart;	/* Char number of most recent start-of-expression at current level */    int prevlevelstart; /* Char number of start of containing expression */    int location;	/* Char number at which parsing stopped. */    int mindepth;	/* Minimum depth seen while scanning.  */  };/* Parse forward from FROM to END,   assuming that FROM is the start of a function,    and return a description of the state of the parse at END. */struct lisp_parse_state val_scan_sexps_forward;struct lisp_parse_state *scan_sexps_forward (from, end, targetdepth, stopbefore, oldstate)     register int from;     int end, targetdepth, stopbefore;     Lisp_Object oldstate;{  struct lisp_parse_state state;  register enum syntaxcode code;  struct level { int last, prev; };  struct level levelstart[100];  register struct level *curlevel = levelstart;  struct level *endlevel = levelstart + 100;  char prev;  register int depth;	/* Paren depth of current scanning location.			   level - levelstart equals this except			   when the depth becomes negative.  */  int mindepth;		/* Lowest DEPTH value seen.  */  int start_quoted = 0;		/* Nonzero means starting after a char quote */  Lisp_Object tem;  immediate_quit = 1;  QUIT;  if (NULL (oldstate))    {      depth = 0;      state.instring = -1;      state.incomment = 0;    }  else    {      tem = Fcar (oldstate);      if (!NULL (tem))	depth = XINT (tem);      else	depth = 0;      oldstate = Fcdr (oldstate);      oldstate = Fcdr (oldstate);      oldstate = Fcdr (oldstate);      tem = Fcar (oldstate);      state.instring = !NULL (tem) ? XINT (tem) : -1;      oldstate = Fcdr (oldstate);      tem = Fcar (oldstate);      state.incomment = !NULL (tem);      oldstate = Fcdr (oldstate);      tem = Fcar (oldstate);      start_quoted = !NULL (tem);    }  state.quoted = 0;  mindepth = depth;  curlevel->prev = -1;  curlevel->last = -1;  /* Enter the loop at a place appropriate for initial state. */  if (state.incomment) goto startincomment;  if (state.instring >= 0)    {      if (start_quoted) goto startquotedinstring;      goto startinstring;    }  if (start_quoted) goto startquoted;  while (from < end)    {      code = SYNTAX(FETCH_CHAR (from));      from++;      if (from < end && SYNTAX_COMSTART_FIRST (FETCH_CHAR (from - 1))	   && SYNTAX_COMSTART_SECOND (FETCH_CHAR (from)))	code = Scomment, from++;#ifdef SWITCH_ENUM_BUG      switch ((int) code)#else      switch (code)#endif	{	case Sescape:	case Scharquote:	  if (stopbefore) goto stop;  /* this arg means stop at sexp start */	  curlevel->last = from - 1;	startquoted:	  if (from == end) goto endquoted;	  from++;	  goto symstarted;	  /* treat following character as a word constituent */	case Sword:	case Ssymbol:	  if (stopbefore) goto stop;  /* this arg means stop at sexp start */	  curlevel->last = from - 1;	symstarted:	  while (from < end)	    {#ifdef SWITCH_ENUM_BUG	      switch ((int) SYNTAX(FETCH_CHAR (from)))#else	      switch (SYNTAX(FETCH_CHAR (from)))#endif		{		case Scharquote:		case Sescape:		  from++;		  if (from == end) goto endquoted;		  break;		case Sword:		case Ssymbol:		case Squote:		  break;		default:		  goto symdone;		}	      from++;	    }	symdone:	  curlevel->prev = curlevel->last;	  break;	case Scomment:	  state.incomment = 1;	startincomment:	  while (1)	    {	      if (from == end) goto done;	      if (SYNTAX (prev = FETCH_CHAR (from)) == Sendcomment)		break;	      from++;	      if (from < end && SYNTAX_COMEND_FIRST (prev)		   && SYNTAX_COMEND_SECOND (FETCH_CHAR (from)))		{ from++; break; }	    }	  state.incomment = 0;	  break;	case Sopen:	  if (stopbefore) goto stop;  /* this arg means stop at sexp start */	  depth++;	  /* curlevel++->last ran into compiler bug on Apollo */	  curlevel->last = from - 1;	  if (++curlevel == endlevel)	    error ("Nesting too deep for parser");	  curlevel->prev = -1;	  curlevel->last = -1;	  if (!--targetdepth) goto done;	  break;	case Sclose:	  depth--;	  if (depth < mindepth)	    mindepth = depth;	  if (curlevel != levelstart)	    curlevel--;	  curlevel->prev = curlevel->last;	  if (!++targetdepth) goto done;	  break;	case Sstring:	  if (stopbefore) goto stop;  /* this arg means stop at sexp start */	  curlevel->last = from - 1;	  state.instring = FETCH_CHAR (from - 1);	startinstring:	  while (1)	    {	      if (from >= end) goto done;	      if (FETCH_CHAR (from) == state.instring) break;#ifdef SWITCH_ENUM_BUG	      switch ((int) SYNTAX(FETCH_CHAR (from)))#else	      switch (SYNTAX(FETCH_CHAR (from)))#endif		{		case Scharquote:		case Sescape:		  from++;		startquotedinstring:		  if (from >= end) goto endquoted;		}	      from++;	    }	  state.instring = -1;	  curlevel->prev = curlevel->last;	  from++;	  break;	case Smath:	  break;	}    }  goto done; stop:   /* Here if stopping before start of sexp. */  from--;    /* We have just fetched the char that starts it; */  goto done; /* but return the position before it. */ endquoted:  state.quoted = 1; done:  state.depth = depth;  state.mindepth = mindepth;  state.thislevelstart = curlevel->prev;  state.prevlevelstart    = (curlevel == levelstart) ? -1 : (curlevel - 1)->last;  state.location = from;  immediate_quit = 0;  val_scan_sexps_forward = state;  return &val_scan_sexps_forward;}/* This comment supplies the doc string for parse-partial-sexp,   for make-docfile to see.  We cannot put this in the real DEFUN   due to limits in the Unix cpp.DEFUN ("parse-partial-sexp", Ffoo, Sfoo, 0, 0, 0,  "Parse Lisp syntax starting at FROM until TO; return status of parse at TO.\n\Parsing stops at TO or when certain criteria are met;\n\ point is set to where parsing stops.\n\If fifth arg STATE is omitted or nil,\n\ parsing assumes that FROM is the beginning of a function.\n\Value is a list of seven elements describing final state of parsing:\n\ 1. depth in parens.\n\ 2. character address of start of innermost containing list; nil if none.\n\ 3. character address of start of last complete sexp terminated.\n\ 4. non-nil if inside a string.\n\    (it is the character that will terminate the string.)\n\ 5. t if inside a comment.\n\ 6. t if following a quote character.\n\ 7. the minimum paren-depth encountered during this scan.\n\If third arg TARGETDEPTH is non-nil, parsing stops if the depth\n\in parentheses becomes equal to TARGETDEPTH.\n\Fourth arg STOPBEFORE non-nil means stop when come to\n\ any character that starts a sexp.\n\Fifth arg STATE is a seven-list like what this function returns.\n\It is used to initialize the state of the parse.")*/DEFUN ("parse-partial-sexp", Fparse_partial_sexp, Sparse_partial_sexp, 2, 5, 0,  0 /* See immediately above */)  (from, to, targetdepth, stopbefore, oldstate)     Lisp_Object from, to, targetdepth, stopbefore, oldstate;{  struct lisp_parse_state state;  int target;  if (!NULL (targetdepth))    {      CHECK_NUMBER (targetdepth, 3);      target = XINT (targetdepth);    }  else    target = -100000;		/* We won't reach this depth */  validate_region (&from, &to);  state = *scan_sexps_forward (XINT (from), XINT (to),			       target, !NULL (stopbefore), oldstate);  SET_PT (state.location);    return Fcons (make_number (state.depth),	   Fcons (state.prevlevelstart < 0 ? Qnil : make_number (state.prevlevelstart),	     Fcons (state.thislevelstart < 0 ? Qnil : make_number (state.thislevelstart),	       Fcons (state.instring >= 0 ? make_number (state.instring) : Qnil,		 Fcons (state.incomment ? Qt : Qnil,		   Fcons (state.quoted ? Qt : Qnil,			  Fcons (make_number (state.mindepth), Qnil)))))));}init_syntax_once (){  register int i;  register struct Lisp_Vector *v;  /* Set this now, so first buffer creation can refer to it. */  /* Make it nil before calling copy-syntax-table    so that copy-syntax-table will know not to try to copy from garbage */  Vstandard_syntax_table = Qnil;  Vstandard_syntax_table = Fcopy_syntax_table (Qnil);  v = XVECTOR (Vstandard_syntax_table);  for (i = 'a'; i <= 'z'; i++)    XFASTINT (v->contents[i]) = (int) Sword;  for (i = 'A'; i <= 'Z'; i++)    XFASTINT (v->contents[i]) = (int) Sword;  for (i = '0'; i <= '9'; i++)    XFASTINT (v->contents[i]) = (int) Sword;  XFASTINT (v->contents['$']) = (int) Sword;  XFASTINT (v->contents['%']) = (int) Sword;  XFASTINT (v->contents['(']) = (int) Sopen + (')' << 8);  XFASTINT (v->contents[')']) = (int) Sclose + ('(' << 8);  XFASTINT (v->contents['[']) = (int) Sopen + (']' << 8);  XFASTINT (v->contents[']']) = (int) Sclose + ('[' << 8);  XFASTINT (v->contents['{']) = (int) Sopen + ('}' << 8);  XFASTINT (v->contents['}']) = (int) Sclose + ('{' << 8);  XFASTINT (v->contents['"']) = (int) Sstring;  XFASTINT (v->contents['\\']) = (int) Sescape;  for (i = 0; i < 10; i++)    XFASTINT (v->contents["_-+*/&|<>="[i]]) = (int) Ssymbol;  for (i = 0; i < 12; i++)    XFASTINT (v->contents[".,;:?!#@~^'`"[i]]) = (int) Spunct;}syms_of_syntax (){  Qsyntax_table_p = intern ("syntax-table-p");  staticpro (&Qsyntax_table_p);  DEFVAR_BOOL ("parse-sexp-ignore-comments", &parse_sexp_ignore_comments,    "Non-nil means forward-sexp, etc., should treat comments as whitespace.\n\Non-nil works only when the comment terminator is something like *\/,\n\and appears only when it ends a comment.\n\If comments are terminated by newlines,\n\you must make this variable nil.");  defsubr (&Ssyntax_table_p);  defsubr (&Ssyntax_table);  defsubr (&Sstandard_syntax_table);  defsubr (&Scopy_syntax_table);  defsubr (&Sset_syntax_table);  defsubr (&Schar_syntax);  defsubr (&Smodify_syntax_entry);  defsubr (&Sdescribe_syntax);  defsubr (&Sforward_word);  defsubr (&Sscan_lists);  defsubr (&Sscan_sexps);  defsubr (&Sbackward_prefix_chars);  defsubr (&Sparse_partial_sexp);}

⌨️ 快捷键说明

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