1 2 package java_cup; 3 4 import java.util.Enumeration; 5 import java.util.Hashtable; 6 7 /** This class represents a production in the grammar. It contains 8 * a LHS non terminal, and an array of RHS symbols. As various 9 * transformations are done on the RHS of the production, it may shrink. 10 * As a result a separate length is always maintained to indicate how much 11 * of the RHS array is still valid.<p> 12 * 13 * I addition to construction and manipulation operations, productions provide 14 * methods for factoring out actions (see remove_embedded_actions()), for 15 * computing the nullability of the production (i.e., can it derive the empty 16 * string, see check_nullable()), and operations for computing its first 17 * set (i.e., the set of terminals that could appear at the beginning of some 18 * string derived from the production, see check_first_set()). 19 * 20 * @see java_cup.production_part 21 * @see java_cup.symbol_part 22 * @see java_cup.action_part 23 * @version last updated: 11/25/95 24 * @author Scott Hudson 25 */ 26 27 public class production { 28 29 /*-----------------------------------------------------------*/ 30 /*--- Constructor(s) ----------------------------------------*/ 31 /*-----------------------------------------------------------*/ 32 33 /** Full constructor. This constructor accepts a LHS non terminal, 34 * an array of RHS parts (including terminals, non terminals, and 35 * actions), and a string for a final reduce action. It does several 36 * manipulations in the process of creating a production object. 37 * After some validity checking it translates labels that appear in 38 * actions into code for accessing objects on the runtime parse stack. 39 * It them merges adjacent actions if they appear and moves any trailing 40 * action into the final reduce actions string. Next it removes any 41 * embedded actions by factoring them out with new action productions. 42 * Finally it assigns a unique index to the production.<p> 43 * 44 * Factoring out of actions is accomplished by creating new "hidden" 45 * non terminals. For example if the production was originally: <pre> 46 * A ::= B {action} C D 47 * </pre> 48 * then it is factored into two productions:<pre> 49 * A ::= B X C D 50 * X ::= {action} 51 * </pre> 52 * (where X is a unique new non terminal). This has the effect of placing 53 * all actions at the end where they can be handled as part of a reduce by 54 * the parser. 55 */ production( non_terminal lhs_sym, production_part rhs_parts[], int rhs_l, String action_str)56 public production( 57 non_terminal lhs_sym, 58 production_part rhs_parts[], 59 int rhs_l, 60 String action_str) 61 throws internal_error 62 { 63 int i; 64 action_part tail_action; 65 66 /* remember the length */ 67 if (rhs_l >= 0) 68 _rhs_length = rhs_l; 69 else if (rhs_parts != null) 70 _rhs_length = rhs_parts.length; 71 else 72 _rhs_length = 0; 73 74 /* make sure we have a valid left-hand-side */ 75 if (lhs_sym == null) 76 throw new internal_error( 77 "Attempt to construct a production with a null LHS"); 78 79 /* translate labels appearing in action strings */ 80 action_str = translate_labels( 81 rhs_parts, rhs_l, action_str, lhs_sym.stack_type()); 82 83 /* count use of lhs */ 84 lhs_sym.note_use(); 85 86 /* create the part for left-hand-side */ 87 _lhs = new symbol_part(lhs_sym); 88 89 /* merge adjacent actions (if any) */ 90 _rhs_length = merge_adjacent_actions(rhs_parts, _rhs_length); 91 92 /* strip off any trailing action */ 93 tail_action = strip_trailing_action(rhs_parts, _rhs_length); 94 if (tail_action != null) _rhs_length--; 95 96 /* allocate and copy over the right-hand-side */ 97 _rhs = new production_part[_rhs_length]; 98 for (i=0; i<_rhs_length; i++) 99 _rhs[i] = rhs_parts[i]; 100 101 /* count use of each rhs symbol */ 102 for (i=0; i<_rhs_length; i++) 103 if (!_rhs[i].is_action()) 104 ((symbol_part)_rhs[i]).the_symbol().note_use(); 105 106 /* merge any trailing action with action string parameter */ 107 if (action_str == null) action_str = ""; 108 if (tail_action != null && tail_action.code_string() != null) 109 action_str = tail_action.code_string() + action_str; 110 111 /* stash the action */ 112 _action = new action_part(action_str); 113 114 /* rewrite production to remove any embedded actions */ 115 remove_embedded_actions(); 116 117 /* assign an index */ 118 _index = next_index++; 119 120 /* put us in the global collection of productions */ 121 _all.put(new Integer(_index),this); 122 123 /* put us in the production list of the lhs non terminal */ 124 lhs_sym.add_production(this); 125 } 126 127 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 128 129 /** Constructor with no action string. */ production( non_terminal lhs_sym, production_part rhs_parts[], int rhs_l)130 public production( 131 non_terminal lhs_sym, 132 production_part rhs_parts[], 133 int rhs_l) 134 throws internal_error 135 { 136 this(lhs_sym,rhs_parts,rhs_l,null); 137 } 138 139 /*-----------------------------------------------------------*/ 140 /*--- (Access to) Static (Class) Variables ------------------*/ 141 /*-----------------------------------------------------------*/ 142 143 /** Table of all productions. Elements are stored using their index as 144 * the key. 145 */ 146 protected static Hashtable _all = new Hashtable(); 147 148 /** Access to all productions. */ all()149 public static Enumeration all() {return _all.elements();}; 150 151 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 152 153 /** Total number of productions. */ number()154 public static int number() {return _all.size();}; 155 156 /** Static counter for assigning unique index numbers. */ 157 protected static int next_index; 158 159 /*-----------------------------------------------------------*/ 160 /*--- (Access to) Instance Variables ------------------------*/ 161 /*-----------------------------------------------------------*/ 162 163 /** The left hand side non-terminal. */ 164 protected symbol_part _lhs; 165 166 /** The left hand side non-terminal. */ lhs()167 public symbol_part lhs() {return _lhs;} 168 169 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 170 171 /** A collection of parts for the right hand side. */ 172 protected production_part _rhs[]; 173 174 /** Access to the collection of parts for the right hand side. */ rhs(int indx)175 public production_part rhs(int indx) throws internal_error 176 { 177 if (indx >= 0 && indx < _rhs_length) 178 return _rhs[indx]; 179 else 180 throw new internal_error( 181 "Index out of range for right hand side of production"); 182 } 183 184 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 185 186 /** How much of the right hand side array we are presently using. */ 187 protected int _rhs_length; 188 189 /** How much of the right hand side array we are presently using. */ rhs_length()190 public int rhs_length() {return _rhs_length;} 191 192 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 193 194 /** An action_part containing code for the action to be performed when we 195 * reduce with this production. 196 */ 197 protected action_part _action; 198 199 /** An action_part containing code for the action to be performed when we 200 * reduce with this production. 201 */ action()202 public action_part action() {return _action;} 203 204 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 205 206 /** Index number of the production. */ 207 protected int _index; 208 209 /** Index number of the production. */ index()210 public int index() {return _index;} 211 212 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 213 214 /** Count of number of reductions using this production. */ 215 protected int _num_reductions = 0; 216 217 /** Count of number of reductions using this production. */ num_reductions()218 public int num_reductions() {return _num_reductions;} 219 220 /** Increment the count of reductions with this non-terminal */ note_reduction_use()221 public void note_reduction_use() {_num_reductions++;} 222 223 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 224 225 /** Is the nullability of the production known or unknown? */ 226 protected boolean _nullable_known = false; 227 228 /** Is the nullability of the production known or unknown? */ nullable_known()229 public boolean nullable_known() {return _nullable_known;} 230 231 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 232 233 /** Nullability of the production (can it derive the empty string). */ 234 protected boolean _nullable = false; 235 236 /** Nullability of the production (can it derive the empty string). */ nullable()237 public boolean nullable() {return _nullable;} 238 239 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 240 241 /** First set of the production. This is the set of terminals that 242 * could appear at the front of some string derived from this production. 243 */ 244 protected terminal_set _first_set = new terminal_set(); 245 246 /** First set of the production. This is the set of terminals that 247 * could appear at the front of some string derived from this production. 248 */ first_set()249 public terminal_set first_set() {return _first_set;} 250 251 /*-----------------------------------------------------------*/ 252 /*--- Static Methods ----------------------------------------*/ 253 /*-----------------------------------------------------------*/ 254 255 /** Determine if a given character can be a label id starter. 256 * @param c the character in question. 257 */ is_id_start(char c)258 protected static boolean is_id_start(char c) 259 { 260 return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c == '_'); 261 262 //later need to handle non-8-bit chars here 263 } 264 265 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 266 267 /** Determine if a character can be in a label id. 268 * @param c the character in question. 269 */ is_id_char(char c)270 protected static boolean is_id_char(char c) 271 { 272 return is_id_start(c) || (c >= '0' && c <= '9'); 273 } 274 275 /*-----------------------------------------------------------*/ 276 /*--- General Methods ---------------------------------------*/ 277 /*-----------------------------------------------------------*/ 278 279 /** Determine the translation for one label id found within a code_string. 280 * Symbols appearing in the RHS correspond to objects found on the parse 281 * stack at runtime. The code to access them, becomes code to access an 282 * object at the appropriate offset from the top of the stack, and then 283 * cast that to the proper type. 284 * 285 * @param id_str the name of the id to be translated. 286 * @param act_pos the original position of the action it appears in. 287 * @param label_map a hash table mapping labels to positions in the RHS. 288 * @param type_map a hash table mapping labels to declared symbol types. 289 */ label_translate( String id_str, int act_pos, Hashtable label_map, Hashtable label_types)290 protected String label_translate( 291 String id_str, /* the id string we are (possibly) translating */ 292 int act_pos, /* position of the action */ 293 Hashtable label_map, /* map from labels to positions in the RHS */ 294 Hashtable label_types)/* map from labels to stack types */ 295 { 296 Integer label_pos; 297 String label_type; 298 int offset; 299 300 /* look up the id */ 301 label_pos = (Integer)label_map.get(id_str); 302 303 /* if we don't find it, just return the id */ 304 if (label_pos == null) return id_str; 305 306 /* extract the type of the labeled symbol */ 307 label_type = (String)label_types.get(id_str); 308 309 /* is this for the LHS? */ 310 if (label_pos.intValue() == -1) 311 { 312 /* return the result object cast properly */ 313 return "((" + label_type + ")" + emit.pre("result") + ")"; 314 } 315 316 /* its a RHS label */ 317 318 /* if the label appears after the action, we have an error */ 319 if (label_pos.intValue() > act_pos) 320 { 321 /* emit an error message */ 322 System.err.println("*** Label \"" + id_str + 323 "\" appears in action before it appears in production"); 324 lexer.error_count++; 325 326 // later need to print the production this is in 327 328 /* just return the id unchanged */ 329 return id_str; 330 } 331 332 /* calculate the stack offset as the difference in position from 333 label to action minus one */ 334 offset = (act_pos - label_pos.intValue())-1; 335 336 /* translation is properly cast element at that offset from TOS */ 337 return "(/*"+id_str+"*/("+label_type+")" + 338 emit.pre("stack") + ".elementAt(" + emit.pre("top") +"-"+ offset + "))"; 339 340 } 341 342 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 343 344 /** Translate all the label names within an action string to appropriate code. 345 * @param act_string the string to be translated 346 * @param act_pos the position that the action originally held in the 347 * production. 348 * @param label_map a hash table mapping labels to positions in the RHS. 349 * @param type_map a hash table mapping labels to declared symbol types. 350 */ action_translate( String act_string, int act_pos, Hashtable label_map, Hashtable label_types)351 protected String action_translate( 352 String act_string, /* the action string */ 353 int act_pos, /* the position of the action on the RHS */ 354 Hashtable label_map, /* map from labels to RHS positions */ 355 Hashtable label_types) /* map from labels to symbol stack types */ 356 { 357 int id_start; 358 int pos; 359 int len; 360 String id_str; 361 boolean in_id; 362 StringBuffer result; 363 char buffer[]; 364 365 /* if we have no string we are done */ 366 if (act_string == null || act_string.length()== 0) return act_string; 367 368 len = act_string.length(); 369 370 /* set up a place to put the result */ 371 result = new StringBuffer(len + 50); 372 373 /* extract string into array */ 374 buffer = new char[len + 1]; 375 act_string.getChars(0, len, buffer, 0); 376 377 /* put terminator in buffer so we can look one past the end */ 378 buffer[len] = '\0'; 379 380 /* walk down the input buffer looking for identifiers */ 381 in_id = false; 382 for (pos = id_start = 0; pos <= len; pos++) 383 { 384 /* are we currently working on an id? */ 385 if (in_id) 386 { 387 /* does this end the id? */ 388 if (!is_id_char(buffer[pos])) 389 { 390 /* extract the id string and translate it */ 391 id_str = new String(buffer, id_start, pos - id_start); 392 result.append( 393 label_translate(id_str, act_pos, label_map,label_types)); 394 395 /* copy over the ending character */ 396 if (buffer[pos] != '\0') 397 result.append(buffer, pos, 1); 398 399 /* and we are done with this id */ 400 in_id = false; 401 } 402 else 403 { 404 /* we are still in the id, so just keep going */ 405 } 406 } 407 else /* we are not inside an id */ 408 { 409 /* start a new id? */ 410 if (is_id_start(buffer[pos])) 411 { 412 /* start keeping these chars as an id */ 413 in_id = true; 414 id_start = pos; 415 } 416 else 417 { 418 /* just copy over the char */ 419 if (buffer[pos] != '\0') 420 result.append(buffer, pos, 1); 421 } 422 } 423 } 424 425 /* return the accumulated result */ 426 return result.toString(); 427 } 428 429 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 430 431 /** Translate label names to appropriate code within all action strings. 432 * @param rhs array of RHS parts. 433 * @param rhs_len how much of rhs to consider valid. 434 * @param final_action the final action string of the production. 435 * @param lhs_type the object type associated with the LHS symbol. 436 */ translate_labels( production_part rhs[], int rhs_len, String final_action, String lhs_type)437 protected String translate_labels( 438 production_part rhs[], 439 int rhs_len, 440 String final_action, 441 String lhs_type) 442 { 443 Hashtable label_map = new Hashtable(11); 444 Hashtable label_types = new Hashtable(11); 445 symbol_part part; 446 action_part act_part; 447 int pos; 448 449 /* walk down the parts and extract the labels */ 450 for (pos = 0; pos < rhs_len; pos++) 451 { 452 if (!rhs[pos].is_action()) 453 { 454 part = (symbol_part)rhs[pos]; 455 456 /* if it has a label enter it in the tables */ 457 if (part.label() != null) 458 { 459 label_map.put(part.label(), new Integer(pos)); 460 label_types.put(part.label(), part.the_symbol().stack_type()); 461 } 462 } 463 } 464 465 /* add a label for the LHS */ 466 label_map.put("RESULT", new Integer(-1)); 467 label_types.put("RESULT", lhs_type); 468 469 /* now walk across and do each action string */ 470 for (pos = 0; pos < rhs_len; pos++) 471 { 472 if (rhs[pos].is_action()) 473 { 474 act_part = (action_part)rhs[pos]; 475 act_part.set_code_string( 476 action_translate( 477 act_part.code_string(), pos, label_map, label_types)); 478 } 479 } 480 481 /* now do the final action string at the position after the last */ 482 return action_translate(final_action, rhs_len, label_map, label_types); 483 484 } 485 486 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 487 488 /** Helper routine to merge adjacent actions in a set of RHS parts 489 * @param rhs_parts array of RHS parts. 490 * @param len amount of that array that is valid. 491 * @return remaining valid length. 492 */ merge_adjacent_actions(production_part rhs_parts[], int len)493 protected int merge_adjacent_actions(production_part rhs_parts[], int len) 494 { 495 int from_loc, to_loc, merge_cnt; 496 497 /* bail out early if we have no work to do */ 498 if (rhs_parts == null || len == 0) return 0; 499 500 merge_cnt = 0; 501 to_loc = -1; 502 for (from_loc=0; from_loc<len; from_loc++) 503 { 504 /* do we go in the current position or one further */ 505 if (to_loc < 0 || !rhs_parts[to_loc].is_action() 506 || !rhs_parts[from_loc].is_action()) 507 { 508 /* next one */ 509 to_loc++; 510 511 /* clear the way for it */ 512 if (to_loc != from_loc) rhs_parts[to_loc] = null; 513 } 514 515 /* if this is not trivial? */ 516 if (to_loc != from_loc) 517 { 518 /* do we merge or copy? */ 519 if (rhs_parts[to_loc] != null && rhs_parts[to_loc].is_action() && 520 rhs_parts[from_loc].is_action()) 521 { 522 /* merge */ 523 rhs_parts[to_loc] = new action_part( 524 ((action_part)rhs_parts[to_loc]).code_string() + 525 ((action_part)rhs_parts[from_loc]).code_string()); 526 merge_cnt++; 527 } 528 else 529 { 530 /* copy */ 531 rhs_parts[to_loc] = rhs_parts[from_loc]; 532 } 533 } 534 } 535 536 /* return the used length */ 537 return len - merge_cnt; 538 } 539 540 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 541 542 /** Helper routine to strip a trailing action off rhs and return it 543 * @param rhs_parts array of RHS parts. 544 * @param len how many of those are valid. 545 * @return the removed action part. 546 */ strip_trailing_action( production_part rhs_parts[], int len)547 protected action_part strip_trailing_action( 548 production_part rhs_parts[], 549 int len) 550 { 551 action_part result; 552 553 /* bail out early if we have nothing to do */ 554 if (rhs_parts == null || len == 0) return null; 555 556 /* see if we have a trailing action */ 557 if (rhs_parts[len-1].is_action()) 558 { 559 /* snip it out and return it */ 560 result = (action_part)rhs_parts[len-1]; 561 rhs_parts[len-1] = null; 562 return result; 563 } 564 else 565 return null; 566 } 567 568 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 569 570 /** Remove all embedded actions from a production by factoring them 571 * out into individual action production using new non terminals. 572 * if the original production was: <pre> 573 * A ::= B {action1} C {action2} D 574 * </pre> 575 * then it will be factored into: <pre> 576 * A ::= B NT$1 C NT$2 D 577 * NT$1 ::= {action1} 578 * NT$2 ::= {action2} 579 * </pre> 580 * where NT$1 and NT$2 are new system created non terminals. 581 */ remove_embedded_actions()582 protected void remove_embedded_actions() throws internal_error 583 { 584 non_terminal new_nt; 585 production new_prod; 586 587 /* walk over the production and process each action */ 588 for (int act_loc = 0; act_loc < rhs_length(); act_loc++) 589 if (rhs(act_loc).is_action()) 590 { 591 /* create a new non terminal for the action production */ 592 new_nt = non_terminal.create_new(); 593 594 /* create a new production with just the action */ 595 new_prod = new action_production(this, new_nt, null, 0, 596 ((action_part)rhs(act_loc)).code_string()); 597 598 /* replace the action with the generated non terminal */ 599 _rhs[act_loc] = new symbol_part(new_nt); 600 } 601 } 602 603 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 604 605 /** Check to see if the production (now) appears to be nullable. 606 * A production is nullable if its RHS could derive the empty string. 607 * This results when the RHS is empty or contains only non terminals 608 * which themselves are nullable. 609 */ check_nullable()610 public boolean check_nullable() throws internal_error 611 { 612 production_part part; 613 symbol sym; 614 int pos; 615 616 /* if we already know bail out early */ 617 if (nullable_known()) return nullable(); 618 619 /* if we have a zero size RHS we are directly nullable */ 620 if (rhs_length() == 0) 621 { 622 /* stash and return the result */ 623 return set_nullable(true); 624 } 625 626 /* otherwise we need to test all of our parts */ 627 for (pos=0; pos<rhs_length(); pos++) 628 { 629 part = rhs(pos); 630 631 /* only look at non-actions */ 632 if (!part.is_action()) 633 { 634 sym = ((symbol_part)part).the_symbol(); 635 636 /* if its a terminal we are definitely not nullable */ 637 if (!sym.is_non_term()) 638 return set_nullable(false); 639 /* its a non-term, is it marked nullable */ 640 else if (!((non_terminal)sym).nullable()) 641 /* this one not (yet) nullable, so we aren't */ 642 return false; 643 } 644 } 645 646 /* if we make it here all parts are nullable */ 647 return set_nullable(true); 648 } 649 650 /** set (and return) nullability */ set_nullable(boolean v)651 boolean set_nullable(boolean v) 652 { 653 _nullable_known = true; 654 _nullable = v; 655 return v; 656 } 657 658 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 659 660 /** Update (and return) the first set based on current NT firsts. 661 * This assumes that nullability has already been computed for all non 662 * terminals and productions. 663 */ check_first_set()664 public terminal_set check_first_set() throws internal_error 665 { 666 int part; 667 symbol sym; 668 669 /* walk down the right hand side till we get past all nullables */ 670 for (part=0; part<rhs_length(); part++) 671 { 672 /* only look at non-actions */ 673 if (!rhs(part).is_action()) 674 { 675 sym = ((symbol_part)rhs(part)).the_symbol(); 676 677 /* is it a non-terminal?*/ 678 if (sym.is_non_term()) 679 { 680 /* add in current firsts from that NT */ 681 _first_set.add(((non_terminal)sym).first_set()); 682 683 /* if its not nullable, we are done */ 684 if (!((non_terminal)sym).nullable()) 685 break; 686 } 687 else 688 { 689 /* its a terminal -- add that to the set */ 690 _first_set.add((terminal)sym); 691 692 /* we are done */ 693 break; 694 } 695 } 696 } 697 698 /* return our updated first set */ 699 return first_set(); 700 } 701 702 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 703 704 /** Equality comparison. */ equals(production other)705 public boolean equals(production other) 706 { 707 if (other == null) return false; 708 return other._index == _index; 709 } 710 711 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 712 713 /** Generic equality comparison. */ equals(Object other)714 public boolean equals(Object other) 715 { 716 if (!(other instanceof production)) 717 return false; 718 else 719 return equals((production)other); 720 } 721 722 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 723 724 /** Produce a hash code. */ hashCode()725 public int hashCode() 726 { 727 /* just use a simple function of the index */ 728 return _index*13; 729 } 730 731 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 732 733 /** Convert to a string. */ toString()734 public String toString() 735 { 736 String result; 737 738 /* catch any internal errors */ 739 try { 740 result = "production [" + index() + "]: "; 741 result += ((lhs() != null) ? lhs().toString() : "$$NULL-LHS$$"); 742 result += " :: = "; 743 for (int i = 0; i<rhs_length(); i++) 744 result += rhs(i) + " "; 745 result += ";"; 746 if (action() != null && action().code_string() != null) 747 result += " {" + action().code_string() + "}"; 748 749 if (nullable_known()) 750 if (nullable()) 751 result += "[NULLABLE]"; 752 else 753 result += "[NOT NULLABLE]"; 754 } catch (internal_error e) { 755 /* crash on internal error since we can't throw it from here (because 756 superclass does not throw anything. */ 757 e.crash(); 758 result = null; 759 } 760 761 return result; 762 } 763 764 /*. . . . . . . . . . . . . . . . . . . . . . . . . . . . . .*/ 765 766 /** Convert to a simpler string. */ to_simple_string()767 public String to_simple_string() throws internal_error 768 { 769 String result; 770 771 result = ((lhs() != null) ? lhs().the_symbol().name() : "NULL_LHS"); 772 result += " ::= "; 773 for (int i = 0; i < rhs_length(); i++) 774 if (!rhs(i).is_action()) 775 result += ((symbol_part)rhs(i)).the_symbol().name() + " "; 776 777 return result; 778 } 779 780 /*-----------------------------------------------------------*/ 781 782 }; 783