mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			595 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			595 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Java
		
	
	
	
| /* XPathTokenizer.java --
 | |
|    Copyright (C) 2004 Free Software Foundation, Inc.
 | |
| 
 | |
| This file is part of GNU Classpath.
 | |
| 
 | |
| GNU Classpath is free software; you can redistribute it and/or modify
 | |
| it under the terms of the GNU General Public License as published by
 | |
| the Free Software Foundation; either version 2, or (at your option)
 | |
| any later version.
 | |
| 
 | |
| GNU Classpath is distributed in the hope that it will be useful, but
 | |
| WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
| General Public License for more details.
 | |
| 
 | |
| You should have received a copy of the GNU General Public License
 | |
| along with GNU Classpath; see the file COPYING.  If not, write to the
 | |
| Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | |
| 02110-1301 USA.
 | |
| 
 | |
| Linking this library statically or dynamically with other modules is
 | |
| making a combined work based on this library.  Thus, the terms and
 | |
| conditions of the GNU General Public License cover the whole
 | |
| combination.
 | |
| 
 | |
| As a special exception, the copyright holders of this library give you
 | |
| permission to link this library with independent modules to produce an
 | |
| executable, regardless of the license terms of these independent
 | |
| modules, and to copy and distribute the resulting executable under
 | |
| terms of your choice, provided that you also meet, for each linked
 | |
| independent module, the terms and conditions of the license of that
 | |
| module.  An independent module is a module which is not derived from
 | |
| or based on this library.  If you modify this library, you may extend
 | |
| this exception to your version of the library, but you are not
 | |
| obligated to do so.  If you do not wish to do so, delete this
 | |
| exception statement from your version. */
 | |
| 
 | |
| package gnu.xml.xpath;
 | |
| 
 | |
| import gnu.java.lang.CPStringBuilder;
 | |
| 
 | |
| import java.io.BufferedReader;
 | |
| import java.io.IOException;
 | |
| import java.io.Reader;
 | |
| import java.io.StringReader;
 | |
| import java.util.Map;
 | |
| import java.util.TreeMap;
 | |
| 
 | |
| /*import antlr.Token;
 | |
| import antlr.TokenStream;
 | |
| import antlr.TokenStreamException;
 | |
| import antlr.TokenStreamIOException;*/
 | |
| 
 | |
| /**
 | |
|  * XPath 1.0 expression tokenizer.
 | |
|  *
 | |
|  * @author <a href='mailto:dog@gnu.org'>Chris Burdess</a>
 | |
|  */
 | |
| public class XPathTokenizer
 | |
| implements XPathParser.yyInput
 | |
| //implements TokenStream
 | |
| {
 | |
| 
 | |
|   static class XPathToken
 | |
|   //extends Token
 | |
|   {
 | |
| 
 | |
|     int type;
 | |
|     String val;
 | |
| 
 | |
|     XPathToken (int type)
 | |
|     {
 | |
|       this (type, null);
 | |
|     }
 | |
| 
 | |
|     XPathToken (int type, String val)
 | |
|     {
 | |
|       //super (type);
 | |
|       this.type = type;
 | |
|       this.val = val;
 | |
|     }
 | |
| 
 | |
|     public String getText ()
 | |
|     {
 | |
|       return val;
 | |
|     }
 | |
| 
 | |
|     public String toString ()
 | |
|     {
 | |
|       return val;
 | |
|     }
 | |
| 
 | |
|   }
 | |
| 
 | |
|   static final Map<String,Integer> keywords = new TreeMap<String,Integer> ();
 | |
|   static
 | |
|   {
 | |
|     keywords.put ("ancestor", new Integer (XPathParser.ANCESTOR));
 | |
|     keywords.put ("ancestor-or-self", new Integer (XPathParser.ANCESTOR_OR_SELF));
 | |
|     keywords.put ("attribute", new Integer (XPathParser.ATTRIBUTE));
 | |
|     keywords.put ("child", new Integer (XPathParser.CHILD));
 | |
|     keywords.put ("descendant", new Integer (XPathParser.DESCENDANT));
 | |
|     keywords.put ("descendant-or-self", new Integer (XPathParser.DESCENDANT_OR_SELF));
 | |
|     keywords.put ("following", new Integer (XPathParser.FOLLOWING));
 | |
|     keywords.put ("following-sibling", new Integer (XPathParser.FOLLOWING_SIBLING));
 | |
|     keywords.put ("namespace", new Integer (XPathParser.NAMESPACE));
 | |
|     keywords.put ("parent", new Integer (XPathParser.PARENT));
 | |
|     keywords.put ("preceding", new Integer (XPathParser.PRECEDING));
 | |
|     keywords.put ("preceding-sibling", new Integer (XPathParser.PRECEDING_SIBLING));
 | |
|     keywords.put ("self", new Integer (XPathParser.SELF));
 | |
|     keywords.put ("div", new Integer (XPathParser.DIV));
 | |
|     keywords.put ("mod", new Integer (XPathParser.MOD));
 | |
|     keywords.put ("or", new Integer (XPathParser.OR));
 | |
|     keywords.put ("and", new Integer (XPathParser.AND));
 | |
|     keywords.put ("comment", new Integer (XPathParser.COMMENT));
 | |
|     keywords.put ("processing-instruction", new Integer (XPathParser.PROCESSING_INSTRUCTION));
 | |
|     keywords.put ("text", new Integer (XPathParser.TEXT));
 | |
|     keywords.put ("node", new Integer (XPathParser.NODE));
 | |
|   }
 | |
| 
 | |
|   Reader in;
 | |
|   XPathToken token;
 | |
|   XPathToken lastToken;
 | |
| 
 | |
|   public XPathTokenizer (String expr)
 | |
|   {
 | |
|     this (new StringReader (expr));
 | |
|   }
 | |
| 
 | |
|   XPathTokenizer (Reader in)
 | |
|   {
 | |
|     this.in = in.markSupported () ? in : new BufferedReader (in);
 | |
|   }
 | |
| 
 | |
|   /* Begin ANTLR specific *
 | |
| 
 | |
|   public Token nextToken ()
 | |
|     throws TokenStreamException
 | |
|   {
 | |
|     try
 | |
|       {
 | |
|         if (!advance ())
 | |
|           {
 | |
|             throw new TokenStreamException ("eof");
 | |
|           }
 | |
|         token ();
 | |
|         return token;
 | |
|       }
 | |
|     catch (IOException e)
 | |
|       {
 | |
|         throw new TokenStreamIOException (e);
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   * End ANTLR specific */
 | |
| 
 | |
|   public boolean advance ()
 | |
|     throws IOException
 | |
|   {
 | |
|     lastToken = token;
 | |
|     int c = in.read ();
 | |
|     switch (c)
 | |
|       {
 | |
|       case -1: // eof
 | |
|         return false;
 | |
|       case 0x20:
 | |
|       case 0x09:
 | |
|       case 0x0d:
 | |
|       case 0x0a: // skip whitespace
 | |
|         return advance ();
 | |
|       case 0x22: // "
 | |
|       case 0x27: // '
 | |
|         token = consume_literal (c);
 | |
|         break;
 | |
|       case 0x28: // (
 | |
|         token = new XPathToken (XPathParser.LP);
 | |
|         break;
 | |
|       case 0x29: // )
 | |
|         token = new XPathToken (XPathParser.RP);
 | |
|         break;
 | |
|       case 0x5b: // [
 | |
|         token = new XPathToken (XPathParser.LB);
 | |
|         break;
 | |
|       case 0x5d: // ]
 | |
|         token = new XPathToken (XPathParser.RB);
 | |
|         break;
 | |
|       case 0x2c: // ,
 | |
|         token = new XPathToken (XPathParser.COMMA);
 | |
|         break;
 | |
|       case 0x7c: // |
 | |
|         token = new XPathToken (XPathParser.PIPE);
 | |
|         break;
 | |
|       case 0x2f: // /
 | |
|         in.mark (1);
 | |
|         int d1 = in.read ();
 | |
|         if (d1 == 0x2f)
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.DOUBLE_SLASH);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             token = new XPathToken (XPathParser.SLASH);
 | |
|           }
 | |
|         break;
 | |
|       case 0x3d: // =
 | |
|         token = new XPathToken (XPathParser.EQ);
 | |
|         break;
 | |
|       case 0x21: // !
 | |
|         in.mark (1);
 | |
|         int d2 = in.read ();
 | |
|         if (d2 == 0x3d) // =
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.NE);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             token = new XPathToken (XPathParser.yyErrorCode);
 | |
|           }
 | |
|         break;
 | |
|       case 0x3e: // >
 | |
|         in.mark (1);
 | |
|         int d3 = in.read ();
 | |
|         if (d3 == 0x3d) // =
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.GTE);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             token = new XPathToken (XPathParser.GT);
 | |
|           }
 | |
|         break;
 | |
|       case 0x3c: // <
 | |
|         in.mark (1);
 | |
|         int d4 = in.read ();
 | |
|         if (d4 == 0x3d) // =
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.LTE);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             token = new XPathToken (XPathParser.LT);
 | |
|           }
 | |
|         break;
 | |
|       case 0x2b: // +
 | |
|         token = new XPathToken (XPathParser.PLUS);
 | |
|         break;
 | |
|       case 0x2d: // -
 | |
|         token = new XPathToken (XPathParser.MINUS);
 | |
|         break;
 | |
|       case 0x40: // @
 | |
|         token = new XPathToken (XPathParser.AT);
 | |
|         break;
 | |
|       case 0x2a: // *
 | |
|         token = new XPathToken (XPathParser.STAR);
 | |
|         break;
 | |
|       case 0x24: // $
 | |
|         token = new XPathToken (XPathParser.DOLLAR);
 | |
|         break;
 | |
|       case 0x3a: // :
 | |
|         in.mark (1);
 | |
|         int d5 = in.read ();
 | |
|         if (d5 == 0x3a)
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.DOUBLE_COLON);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             token = new XPathToken (XPathParser.COLON);
 | |
|           }
 | |
|         break;
 | |
|       case 0x2e: // .
 | |
|         in.mark (1);
 | |
|         int d6 = in.read ();
 | |
|         if (d6 == 0x2e)
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.DOUBLE_DOT);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             token = new XPathToken (XPathParser.DOT);
 | |
|           }
 | |
|         break;
 | |
|       default:
 | |
|         if (c >= 0x30 && c <= 0x39)
 | |
|           {
 | |
|             token = consume_digits (c);
 | |
|           }
 | |
|         else if (c == 0x5f || Character.isLetter ((char) c))
 | |
|           {
 | |
|             token = consume_name (c);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             token = new XPathToken (XPathParser.yyErrorCode);
 | |
|           }
 | |
|       }
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   public int token ()
 | |
|   {
 | |
|     return token.type;
 | |
|   }
 | |
| 
 | |
|   public Object value ()
 | |
|   {
 | |
|     return token.val;
 | |
|   }
 | |
| 
 | |
|   XPathToken consume_literal (int delimiter)
 | |
|     throws IOException
 | |
|   {
 | |
|     CPStringBuilder buf = new CPStringBuilder ();
 | |
|     while (true)
 | |
|       {
 | |
|         int c = in.read ();
 | |
|         if (c == -1)
 | |
|           {
 | |
|             return new XPathToken (XPathParser.yyErrorCode);
 | |
|           }
 | |
|         else if (c == delimiter)
 | |
|           {
 | |
|             return new XPathToken (XPathParser.LITERAL, buf.toString ());
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             buf.append ((char) c);
 | |
|           }
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   XPathToken consume_digits (int c)
 | |
|     throws IOException
 | |
|   {
 | |
|     CPStringBuilder buf = new CPStringBuilder ();
 | |
|     buf.append ((char) c);
 | |
|     while (true)
 | |
|       {
 | |
|         in.mark (1);
 | |
|         c = in.read ();
 | |
|         if (c >= 0x30 && c <= 0x39)
 | |
|           {
 | |
|             buf.append ((char) c);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             return new XPathToken (XPathParser.DIGITS, buf.toString ());
 | |
|           }
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   XPathToken consume_name (int c)
 | |
|     throws IOException
 | |
|   {
 | |
|     CPStringBuilder buf = new CPStringBuilder ();
 | |
|     buf.append ((char) c);
 | |
|     while (true)
 | |
|       {
 | |
|         in.mark (1);
 | |
|         c = in.read ();
 | |
|         if (isNameChar (c))
 | |
|           {
 | |
|             buf.append ((char) c);
 | |
|           }
 | |
|         else
 | |
|           {
 | |
|             in.reset ();
 | |
|             String name = buf.toString ();
 | |
|             Integer keyword = (Integer) keywords.get (name);
 | |
|             if (keyword == null)
 | |
|               {
 | |
|                 return new XPathToken (XPathParser.NAME, name);
 | |
|               }
 | |
|             else
 | |
|               {
 | |
|                 int val = keyword.intValue ();
 | |
|                 switch (val)
 | |
|                   {
 | |
|                   case XPathParser.NODE:
 | |
|                   case XPathParser.COMMENT:
 | |
|                   case XPathParser.TEXT:
 | |
|                   case XPathParser.PROCESSING_INSTRUCTION:
 | |
|                     // Consume subsequent (
 | |
|                     in.mark (1);
 | |
|                     do
 | |
|                       {
 | |
|                         c = in.read ();
 | |
|                       }
 | |
|                     while (c == 0x20 || c == 0x09);
 | |
|                     if (c != 0x28)
 | |
|                       {
 | |
|                         in.reset ();
 | |
|                         return new XPathToken (XPathParser.NAME, name);
 | |
|                       }
 | |
|                     break;
 | |
|                   case XPathParser.CHILD:
 | |
|                   case XPathParser.PARENT:
 | |
|                   case XPathParser.SELF:
 | |
|                   case XPathParser.DESCENDANT:
 | |
|                   case XPathParser.ANCESTOR:
 | |
|                   case XPathParser.DESCENDANT_OR_SELF:
 | |
|                   case XPathParser.ANCESTOR_OR_SELF:
 | |
|                   case XPathParser.ATTRIBUTE:
 | |
|                   case XPathParser.NAMESPACE:
 | |
|                   case XPathParser.FOLLOWING:
 | |
|                   case XPathParser.FOLLOWING_SIBLING:
 | |
|                   case XPathParser.PRECEDING:
 | |
|                   case XPathParser.PRECEDING_SIBLING:
 | |
|                     // Check that this is an axis specifier
 | |
|                     in.mark(1);
 | |
|                     do
 | |
|                       {
 | |
|                         c = in.read();
 | |
|                       }
 | |
|                     while (c == 0x20 || c == 0x09);
 | |
|                     if (c == 0x3a)
 | |
|                       {
 | |
|                         c = in.read();
 | |
|                         if (c == 0x3a)
 | |
|                           {
 | |
|                             in.reset();
 | |
|                             return new XPathToken(val);
 | |
|                           }
 | |
|                       }
 | |
|                     in.reset();
 | |
|                     return new XPathToken(XPathParser.NAME, name);
 | |
|                   case XPathParser.DIV:
 | |
|                   case XPathParser.MOD:
 | |
|                     // May be a name
 | |
|                     if (lastToken == null)
 | |
|                       {
 | |
|                         return new XPathToken(XPathParser.NAME, name);
 | |
|                       }
 | |
|                     switch (lastToken.type)
 | |
|                       {
 | |
|                       case XPathParser.LP:
 | |
|                       case XPathParser.LB:
 | |
|                       case XPathParser.COMMA:
 | |
|                       case XPathParser.PIPE:
 | |
|                       case XPathParser.EQ:
 | |
|                       case XPathParser.NE:
 | |
|                       case XPathParser.GT:
 | |
|                       case XPathParser.LT:
 | |
|                       case XPathParser.GTE:
 | |
|                       case XPathParser.LTE:
 | |
|                       case XPathParser.PLUS:
 | |
|                       case XPathParser.MINUS:
 | |
|                       case XPathParser.STAR:
 | |
|                       case XPathParser.AT:
 | |
|                       case XPathParser.DOLLAR:
 | |
|                       case XPathParser.COLON:
 | |
|                       case XPathParser.DOUBLE_COLON:
 | |
|                       case XPathParser.DIV:
 | |
|                       case XPathParser.MOD:
 | |
|                       case XPathParser.OR:
 | |
|                       case XPathParser.AND:
 | |
|                       case XPathParser.SLASH:
 | |
|                         return new XPathToken(XPathParser.NAME, name);
 | |
|                       }
 | |
|                     break;
 | |
|                   }
 | |
|                 return new XPathToken (val);
 | |
|               }
 | |
|           }
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   boolean isNameChar (int c)
 | |
|   {
 | |
|     /* Name */
 | |
|     return (c == 0x5f
 | |
|             || c == 0x2d
 | |
|             || c == 0x2e
 | |
|             || (c >= 0x30 && c <= 0x39)
 | |
|             /* CombiningChar */
 | |
|             || (c >= 0x0300 && c <= 0x0345)
 | |
|             || (c >= 0x0360 && c <= 0x0361)
 | |
|             || (c >= 0x0483 && c <= 0x0486)
 | |
|             || (c >= 0x0591 && c <= 0x05A1)
 | |
|             || (c >= 0x05A3 && c <= 0x05B9)
 | |
|             || (c >= 0x05BB && c <= 0x05BD)
 | |
|             || c == 0x05BF
 | |
|             || (c >= 0x05C1 && c <= 0x05C2)
 | |
|             || c == 0x05C4
 | |
|             || (c >= 0x064B && c <= 0x0652)
 | |
|             || c == 0x0670
 | |
|             || (c >= 0x06D6 && c <= 0x06DC)
 | |
|             || (c >= 0x06DD && c <= 0x06DF)
 | |
|             || (c >= 0x06E0 && c <= 0x06E4)
 | |
|             || (c >= 0x06E7 && c <= 0x06E8)
 | |
|             || (c >= 0x06EA && c <= 0x06ED)
 | |
|             || (c >= 0x0901 && c <= 0x0903)
 | |
|             || c == 0x093C
 | |
|             || (c >= 0x093E && c <= 0x094C)
 | |
|             || c == 0x094D
 | |
|             || (c >= 0x0951 && c <= 0x0954)
 | |
|             || (c >= 0x0962 && c <= 0x0963)
 | |
|             || (c >= 0x0981 && c <= 0x0983)
 | |
|             || c == 0x09BC
 | |
|             || c == 0x09BE
 | |
|             || c == 0x09BF
 | |
|             || (c >= 0x09C0 && c <= 0x09C4)
 | |
|             || (c >= 0x09C7 && c <= 0x09C8)
 | |
|             || (c >= 0x09CB && c <= 0x09CD)
 | |
|             || c == 0x09D7
 | |
|             || (c >= 0x09E2 && c <= 0x09E3)
 | |
|             || c == 0x0A02
 | |
|             || c == 0x0A3C
 | |
|             || c == 0x0A3E
 | |
|             || c == 0x0A3F
 | |
|             || (c >= 0x0A40 && c <= 0x0A42)
 | |
|             || (c >= 0x0A47 && c <= 0x0A48)
 | |
|             || (c >= 0x0A4B && c <= 0x0A4D)
 | |
|             || (c >= 0x0A70 && c <= 0x0A71)
 | |
|             || (c >= 0x0A81 && c <= 0x0A83)
 | |
|             || c == 0x0ABC
 | |
|             || (c >= 0x0ABE && c <= 0x0AC5)
 | |
|             || (c >= 0x0AC7 && c <= 0x0AC9)
 | |
|             || (c >= 0x0ACB && c <= 0x0ACD)
 | |
|             || (c >= 0x0B01 && c <= 0x0B03)
 | |
|             || c == 0x0B3C
 | |
|             || (c >= 0x0B3E && c <= 0x0B43)
 | |
|             || (c >= 0x0B47 && c <= 0x0B48)
 | |
|             || (c >= 0x0B4B && c <= 0x0B4D)
 | |
|             || (c >= 0x0B56 && c <= 0x0B57)
 | |
|             || (c >= 0x0B82 && c <= 0x0B83)
 | |
|             || (c >= 0x0BBE && c <= 0x0BC2)
 | |
|             || (c >= 0x0BC6 && c <= 0x0BC8)
 | |
|             || (c >= 0x0BCA && c <= 0x0BCD)
 | |
|             || c == 0x0BD7
 | |
|             || (c >= 0x0C01 && c <= 0x0C03)
 | |
|             || (c >= 0x0C3E && c <= 0x0C44)
 | |
|             || (c >= 0x0C46 && c <= 0x0C48)
 | |
|             || (c >= 0x0C4A && c <= 0x0C4D)
 | |
|             || (c >= 0x0C55 && c <= 0x0C56)
 | |
|             || (c >= 0x0C82 && c <= 0x0C83)
 | |
|             || (c >= 0x0CBE && c <= 0x0CC4)
 | |
|             || (c >= 0x0CC6 && c <= 0x0CC8)
 | |
|             || (c >= 0x0CCA && c <= 0x0CCD)
 | |
|             || (c >= 0x0CD5 && c <= 0x0CD6)
 | |
|             || (c >= 0x0D02 && c <= 0x0D03)
 | |
|             || (c >= 0x0D3E && c <= 0x0D43)
 | |
|             || (c >= 0x0D46 && c <= 0x0D48)
 | |
|             || (c >= 0x0D4A && c <= 0x0D4D)
 | |
|             || c == 0x0D57
 | |
|             || c == 0x0E31
 | |
|             || (c >= 0x0E34 && c <= 0x0E3A)
 | |
|             || (c >= 0x0E47 && c <= 0x0E4E)
 | |
|             || c == 0x0EB1
 | |
|             || (c >= 0x0EB4 && c <= 0x0EB9)
 | |
|             || (c >= 0x0EBB && c <= 0x0EBC)
 | |
|             || (c >= 0x0EC8 && c <= 0x0ECD)
 | |
|             || (c >= 0x0F18 && c <= 0x0F19)
 | |
|             || c == 0x0F35
 | |
|             || c == 0x0F37
 | |
|             || c == 0x0F39
 | |
|             || c == 0x0F3E
 | |
|             || c == 0x0F3F
 | |
|             || (c >= 0x0F71 && c <= 0x0F84)
 | |
|             || (c >= 0x0F86 && c <= 0x0F8B)
 | |
|             || (c >= 0x0F90 && c <= 0x0F95)
 | |
|             || c == 0x0F97
 | |
|             || (c >= 0x0F99 && c <= 0x0FAD)
 | |
|             || (c >= 0x0FB1 && c <= 0x0FB7)
 | |
|             || c == 0x0FB9
 | |
|             || (c >= 0x20D0 && c <= 0x20DC)
 | |
|             || c == 0x20E1
 | |
|             || (c >= 0x302A && c <= 0x302F)
 | |
|             || c == 0x3099
 | |
|             || c == 0x309A
 | |
|             /* Extender */
 | |
|             || c == 0x00B7
 | |
|             || c == 0x02D0
 | |
|             || c == 0x02D1
 | |
|             || c == 0x0387
 | |
|             || c == 0x0640
 | |
|             || c == 0x0E46
 | |
|             || c == 0x0EC6
 | |
|             || c == 0x3005
 | |
|             || (c >= 0x3031 && c <= 0x3035)
 | |
|             || (c >= 0x309D && c <= 0x309E)
 | |
|             || (c >= 0x30FC && c <= 0x30FE)
 | |
|             /* Name */
 | |
|             || Character.isLetter ((char) c));
 | |
|   }
 | |
| 
 | |
| }
 |