You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			263 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| %{
 | |
| using System.Text;
 | |
| using System.IO;
 | |
| using System;
 | |
| using System.Linq;
 | |
| using System.Collections.Generic;
 | |
| 
 | |
| namespace Monodoc.Ecma
 | |
| {
 | |
| 	public class EcmaUrlParser
 | |
| 	{
 | |
|         int yacc_verbose_flag = 0;
 | |
| 
 | |
|         public void IsValid (string input)
 | |
|         {
 | |
| 			var lexer = new EcmaUrlTokenizer (input);
 | |
| 			this.yyparse (lexer);
 | |
|         }
 | |
| 
 | |
|         public EcmaDesc Parse (string input)
 | |
|         {
 | |
| 			var lexer = new EcmaUrlTokenizer (input);
 | |
| 			return (EcmaDesc)this.yyparse (lexer);
 | |
|         }
 | |
| 
 | |
|         public bool TryParse (string input, out EcmaDesc desc)
 | |
|         {
 | |
|             desc = null;
 | |
|             try {
 | |
|                 desc = Parse (input);
 | |
|             } catch {
 | |
|                 return false;
 | |
|             }
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         EcmaDesc SetEcmaDescType (object result, EcmaDesc.Kind kind)
 | |
|         {
 | |
|             var desc = result as EcmaDesc;
 | |
|             desc.DescKind = kind;
 | |
|             return desc;
 | |
|         }
 | |
| 
 | |
|         List<T> SafeReverse<T> (List<T> input)
 | |
|         {
 | |
|             if (input == null)
 | |
|                return null;
 | |
|             input.Reverse ();
 | |
|             return input;
 | |
|         }
 | |
| %}
 | |
| 
 | |
| %token ERROR
 | |
| %token IDENTIFIER
 | |
| %token DIGIT
 | |
| %token DOT
 | |
| %token COMMA
 | |
| %token COLON
 | |
| %token INNER_TYPE_SEPARATOR
 | |
| %token OP_GENERICS_LT
 | |
| %token OP_GENERICS_GT
 | |
| %token OP_GENERICS_BACKTICK
 | |
| %token OP_OPEN_PAREN
 | |
| %token OP_CLOSE_PAREN
 | |
| %token OP_ARRAY_OPEN
 | |
| %token OP_ARRAY_CLOSE
 | |
| %token SLASH_SEPARATOR
 | |
| %token STAR
 | |
| %token REF_ARG
 | |
| %token OUT_ARG
 | |
| %token EXPLICIT_IMPL_SEP
 | |
| 
 | |
| %start expression
 | |
| 
 | |
| %%
 | |
| 
 | |
| expression
 | |
|         : 'T' COLON type_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Type); }
 | |
|         | 'N' COLON namespace_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Namespace); }
 | |
|         | 'M' COLON method_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Method); }
 | |
|         | 'F' COLON simple_member_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Field); }
 | |
|         | 'C' COLON constructor_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Constructor); }
 | |
|         | 'P' COLON property_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Property); }
 | |
|         | 'E' COLON simple_member_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Event); }
 | |
|         | 'O' COLON operator_expression { $$ = SetEcmaDescType ($3, EcmaDesc.Kind.Operator); }
 | |
| 
 | |
| /* i.e. id.id.id or id */
 | |
| dot_expression
 | |
|         : IDENTIFIER { $$ = new List<string> { (string)$1 }; }
 | |
|         | IDENTIFIER DOT dot_expression { ((ICollection<string>)$3).Add ((string)$1); $$ = $3; }
 | |
| 
 | |
| namespace_expression
 | |
|         : dot_expression { $$ = new EcmaDesc { Namespace = string.Join (".", ((IEnumerable<string>)$1).Reverse ()) }; }
 | |
| 
 | |
| type_expression
 | |
|         : dot_expression type_expression_suffix {
 | |
|                          var dotExpr = ((List<string>)$1);
 | |
|                          dotExpr.Reverse ();
 | |
|                          var desc = $2 as EcmaDesc;
 | |
|                          desc.DescKind = EcmaDesc.Kind.Type;
 | |
|                          desc.Namespace = string.Join (".", dotExpr.Take (dotExpr.Count - 1));
 | |
|                          desc.TypeName = dotExpr.Last ();
 | |
|                          $$ = desc;
 | |
|                      }
 | |
| 
 | |
| /* To be used in types with no namespaces attached to them like an inner type*/
 | |
| reduced_type_expression
 | |
|         : IDENTIFIER type_expression_suffix {
 | |
|                          var desc = $2 as EcmaDesc;
 | |
|                          desc.DescKind = EcmaDesc.Kind.Type;
 | |
|                          desc.TypeName = $1 as string;
 | |
|                          $$ = desc;
 | |
|                      }
 | |
| 
 | |
| type_expression_suffix
 | |
|         : opt_generic_type_suffix opt_inner_type_description opt_array_definition opt_etc {
 | |
|                          bool nestedDescHasEtc = $2 != null && ((EcmaDesc)$2).IsEtc;
 | |
|                          EcmaDesc nestedType = (EcmaDesc)$2;
 | |
|                          $$ = new EcmaDesc {
 | |
|                             GenericTypeArguments = $1 as List<EcmaDesc>,
 | |
|                             NestedType = nestedType,
 | |
|                             ArrayDimensions = SafeReverse ($3 as List<int>),
 | |
|                             Etc = $4 != null ? ((Tuple<char, string>)$4).Item1 : nestedDescHasEtc ? nestedType.Etc : (char)0,
 | |
|                             EtcFilter = $4 != null ? ((Tuple<char, string>)$4).Item2 : nestedDescHasEtc ? nestedType.EtcFilter : null
 | |
|                          };
 | |
|                          if (nestedDescHasEtc) {
 | |
|                             nestedType.Etc = (char)0;
 | |
|                             nestedType.EtcFilter = null;
 | |
|                          }
 | |
|                      }
 | |
| 
 | |
| opt_inner_type_description
 | |
|         : /* empty */ { $$ = null; }
 | |
|         | INNER_TYPE_SEPARATOR reduced_type_expression { $$ = $2; }
 | |
| 
 | |
| opt_generic_type_suffix
 | |
|         : /* empty */ { $$ = null; }
 | |
|         | OP_GENERICS_BACKTICK DIGIT { $$ = Enumerable.Repeat<EcmaDesc> (null, (int)$2).ToList (); }
 | |
|         | OP_GENERICS_LT generic_type_arg_list OP_GENERICS_GT { $$ = $2; }
 | |
| 
 | |
| generic_type_arg_list
 | |
|         : type_expression { $$ = new List<EcmaDesc> () { (EcmaDesc)$1 }; }
 | |
|         | generic_type_arg_list COMMA type_expression { ((List<EcmaDesc>)$1).Add ((EcmaDesc)$3); $$ = $1; }
 | |
| 
 | |
| opt_array_definition
 | |
|         : /* empty */ { $$ = null; }
 | |
|         | OP_ARRAY_OPEN opt_array_definition_list OP_ARRAY_CLOSE opt_array_definition {
 | |
|                       var dims = ((IList<int>)$4) ?? new List<int> (2);
 | |
|                       dims.Add ((int)$2);
 | |
|                       $$ = dims;
 | |
|                 }
 | |
| 
 | |
| opt_array_definition_list
 | |
|         : /* empty */ { $$ = 1; }
 | |
|         | COMMA opt_array_definition_list { $$ = ((int)$2) + 1; }
 | |
| 
 | |
| opt_etc
 | |
|         : /* empty */ { $$ = null; }
 | |
|         | SLASH_SEPARATOR etc_identifier { $$ = Tuple.Create<char, string> (((string)$2)[0], null); }
 | |
|         | SLASH_SEPARATOR etc_identifier SLASH_SEPARATOR reduced_member_expression { $$ = Tuple.Create<char, string> (((string)$2)[0], (string)$4); }
 | |
| /*        | SLASH_SEPARATOR etc_identifier SLASH_SEPARATOR IDENTIFIER opt_generic_type_suffix { $$ = Tuple.Create<char, string> (((string)$2)[0], (string)$4 + ($5 == null ? string.Empty : "<" + string.Join (",", ((IEnumerable<EcmaDesc>)$5).Select (t => t.ToCompleteTypeName ())) + ">")); } */
 | |
| 
 | |
| etc_identifier
 | |
|         : STAR { $$ = "*"; }
 | |
|         | IDENTIFIER { $$ = $1; }
 | |
| 
 | |
| method_expression
 | |
|         : type_expression DOT IDENTIFIER opt_generic_type_suffix opt_arg_list_suffix {
 | |
|                       var desc = $1 as EcmaDesc;
 | |
|                       desc.MemberName = $3 as string;
 | |
|                       desc.GenericMemberArguments = $4 as List<EcmaDesc>;
 | |
|                       desc.MemberArguments = SafeReverse ($5 as List<EcmaDesc>);
 | |
|                       $$ = desc;
 | |
|                 }
 | |
|         | dot_expression opt_generic_type_suffix opt_arg_list_suffix {
 | |
|                       var dotExpr = ((List<string>)$1);
 | |
|                       $$ = new EcmaDesc {
 | |
|                            Namespace = string.Join (".", dotExpr.Skip (2).DefaultIfEmpty (string.Empty).Reverse ()),
 | |
|                            TypeName = dotExpr.Skip (1).First (),
 | |
|                            MemberName = dotExpr.First (),
 | |
|                            GenericMemberArguments = $2 as List<EcmaDesc>,
 | |
|                            MemberArguments = SafeReverse ($3 as List<EcmaDesc>)
 | |
|                       };
 | |
|                 }
 | |
|         | type_expression EXPLICIT_IMPL_SEP method_expression {
 | |
|                       var desc = $1 as EcmaDesc;
 | |
|                       desc.ExplicitImplMember = $3 as EcmaDesc;
 | |
|                       $$ = desc;
 | |
|                 }
 | |
| 
 | |
| /* To be used with members that may have no type/namespace attached */
 | |
| reduced_member_expression
 | |
|         : IDENTIFIER opt_generic_type_suffix { $$ = (string)$1 + ($2 == null ? string.Empty : "<" + string.Join (",", ((IEnumerable<EcmaDesc>)$2).Select (t => t.ToCompleteTypeName ())) + ">"); }
 | |
|         | IDENTIFIER opt_generic_type_suffix DOT reduced_member_expression {
 | |
|                       var existing = $4 as string;
 | |
|                       var expr = (string)$1 + ($2 == null ? string.Empty : "<" + string.Join (",", ((IEnumerable<EcmaDesc>)$2).Select (t => t.ToCompleteTypeName ())) + ">");
 | |
|                       $$ = expr + "." + existing;
 | |
|                 }
 | |
| 
 | |
| arg_type_expression
 | |
|         : type_expression opt_arg_type_suffix { var desc = (EcmaDesc)$1; desc.DescModifier = (EcmaDesc.Mod)$2; $$ = desc; }
 | |
| 
 | |
| opt_arg_type_suffix
 | |
|         : /* empty */ { $$ = EcmaDesc.Mod.Normal; }
 | |
|         | STAR { $$ = EcmaDesc.Mod.Pointer; }
 | |
|         | REF_ARG { $$ = EcmaDesc.Mod.Ref; }
 | |
|         | OUT_ARG { $$ = EcmaDesc.Mod.Out; }
 | |
| 
 | |
| type_expression_list
 | |
|         : /* empty */ { $$ = null; }
 | |
|         | arg_type_expression { $$ = new List<EcmaDesc> () { (EcmaDesc)$1 }; }
 | |
|         | arg_type_expression COMMA type_expression_list { ((List<EcmaDesc>)$3).Add ((EcmaDesc)$1); $$ = $3; }
 | |
| 
 | |
| simple_member_expression
 | |
|         : dot_expression {
 | |
|                  var dotExpr = ((List<string>)$1);
 | |
|                  dotExpr.Reverse ();
 | |
| 
 | |
|                  $$ = new EcmaDesc {
 | |
|                       Namespace = dotExpr.Count > 2 ? string.Join (".", dotExpr.Take (dotExpr.Count - 2)) : string.Empty,
 | |
|                       TypeName = dotExpr.Count > 1 ?  dotExpr[dotExpr.Count - 2] : string.Empty,
 | |
|                       MemberName = dotExpr[dotExpr.Count - 1]
 | |
|                  };
 | |
|              }
 | |
|         | type_expression DOT IDENTIFIER {
 | |
|                  var desc = $1 as EcmaDesc;
 | |
|                  desc.MemberName = $3 as string;
 | |
|                  $$ = desc;
 | |
|              }
 | |
|         | type_expression EXPLICIT_IMPL_SEP simple_member_expression {
 | |
|                  var desc = $1 as EcmaDesc;
 | |
|                  desc.ExplicitImplMember = $3 as EcmaDesc;
 | |
|                  $$ = desc;
 | |
|              }
 | |
| 
 | |
| constructor_expression
 | |
|         : method_expression { $$ = $1; }
 | |
| 
 | |
| operator_expression
 | |
|         : method_expression { $$ = $1; }
 | |
| 
 | |
| property_expression
 | |
|         : simple_member_expression opt_property_indexer {
 | |
|                  var desc = $1 as EcmaDesc;
 | |
|                  (desc.ExplicitImplMember ?? desc).MemberArguments = SafeReverse ($2 as List<EcmaDesc>);
 | |
|                  $$ = desc;
 | |
|              }
 | |
| 
 | |
| opt_property_indexer
 | |
|         : opt_arg_list_suffix { $$ = $1; }
 | |
| 
 | |
| /*simple_member_expression opt_arg_list_suffix { $$ = CopyFromEcmaDesc (new EcmaDesc {
 | |
|                            MemberArguments = SafeReverse ($2 as List<EcmaDesc>)
 | |
|                       }, (EcmaDesc)$1);
 | |
|                 }*/
 | |
| 
 | |
| opt_arg_list_suffix
 | |
|         : /* empty */ { $$ = null; }
 | |
|         | OP_OPEN_PAREN type_expression_list OP_CLOSE_PAREN { $$ = $2; }
 | |
| 
 | |
| %%
 | |
| 
 | |
| } |