X-Git-Url: http://git.megacz.com/?p=sbp.git;a=blobdiff_plain;f=src%2Fedu%2Fberkeley%2Fsbp%2FSequence.java;h=d0c3dfd7c63345f4028abcd7a555b3d5199b913c;hp=c41be4ff9d62da6ba0537477b13c3db22aaed42d;hb=HEAD;hpb=2afdfe14e78fa0597186614937c679a09d74ecdf diff --git a/src/edu/berkeley/sbp/Sequence.java b/src/edu/berkeley/sbp/Sequence.java index c41be4f..d0c3dfd 100644 --- a/src/edu/berkeley/sbp/Sequence.java +++ b/src/edu/berkeley/sbp/Sequence.java @@ -1,4 +1,4 @@ -// Copyright 2006 all rights reserved; see LICENSE file for BSD-style license +// Copyright 2006-2007 all rights reserved; see LICENSE file for BSD-style license package edu.berkeley.sbp; import edu.berkeley.sbp.util.*; @@ -10,9 +10,9 @@ import java.lang.reflect.*; import java.lang.ref.*; /** juxtaposition; zero or more adjacent Elements; can specify a rewriting */ -public abstract class Sequence implements Iterable, SequenceOrElement { +public abstract class Sequence implements Iterable, SequenceOrElement, Serializable { - protected final Element[] elements; + protected transient final Element[] elements; boolean needed_or_hated = false; boolean in_a_union = false; @@ -24,9 +24,25 @@ public abstract class Sequence implements Iterable, SequenceOrElement { HashMap canNeed = new HashMap(); HashMap canKill = new HashMap(); - final Position firstp; + final Position firstp; + + transient Atom follow = null; + + private static int global_sernum = 0; + private int sernum = global_sernum++; + int[] needs_int() { + int[] ret = new int[needs.size()]; + int i = 0; + for(Sequence s : needs) ret[i++] = s.sernum; + return ret; + } + int[] hates_int() { + int[] ret = new int[hates.size()]; + int i = 0; + for(Sequence s : hates) ret[i++] = s.sernum; + return ret; + } - Atom follow = null; // Static Constructors ////////////////////////////////////////////////////////////////////////////// @@ -34,10 +50,18 @@ public abstract class Sequence implements Iterable, SequenceOrElement { public static Sequence create(Element e) { return create(new Element[] { e }, 0); } /** create a sequence which drops the result of all but one of its element */ - public static Sequence create(Element[] e, int which) { return new Singleton(e, which); } + public static Sequence create(Element[] e, int which) { + return new Singleton(e, which); } /** create a sequence which always evaluates to a constant result */ - public static Sequence create(Element[] e, Object result) { return new Constant(e, result); } + public static Sequence create(Object result, Element[] e) { + return new RewritingSequence(result, e, trues(e.length)); } + + private static boolean[] trues(int length) { + boolean[] ret = new boolean[length]; + for(int i=0; i, SequenceOrElement { * @param e the elements to match * @param drop only elements of e whose corresponding boolean in drops * is false will be included in the output tree - * @param foster if true, all children of the last child (ie - * grandchildren) are promoted to children of this - * node; this is very useful for matching repetitions + * @param lifts which (if any) child trees to lift **/ - public static Sequence create(Object head, Element[] e, boolean[] drop, boolean foster) { - return foster - ? new Unwrap(e, head, drop) - : new RewritingSequence(head, e, drop); - } - public static Sequence createLeft(Object head, Element[] e, boolean[] drop, boolean foster) { - return foster - ? new UnwrapLeft(e, head, drop) - : new RewritingSequence(head, e, drop); + public static Sequence create(Object head, Element[] e, boolean[] drop) { + return create(head, e, drop, new boolean[e.length]); } + public static Sequence create(Object head, Element[] e, boolean[] drop, boolean[] lifts) { + if (lifts==null) lifts = new boolean[e.length]; + return new RewritingSequence(head, e, drop, lifts); } /** return a new sequence identical to this one, but with a positive conjunct s */ @@ -97,8 +115,8 @@ public abstract class Sequence implements Iterable, SequenceOrElement { Iterable needs() { return needs; } Iterable hates() { return hates; } - Position firstp() { return firstp; } - Position lastp() { return firstp().last(); } + Pos firstp() { return firstp; } + Pos lastp() { return firstp().last(); } public Iterator iterator() { return new ArrayIterator(elements); } protected Sequence(Element[] elements) { @@ -106,48 +124,91 @@ public abstract class Sequence implements Iterable, SequenceOrElement { for(int i=0; i Forest postReduce(Input.Region loc, Forest[] args, Position p); // Position ////////////////////////////////////////////////////////////////////////////// - /** the imaginary position before or after an element of a sequence; corresponds to an "LR item" */ - class Position implements IntegerMappable { + static abstract class Pos implements IntegerMappable, Comparable, Serializable { public int ord = -1; + private Sequence owner; + + public int ord() { return ord; } + + final Forest[] holder; + + Pos(int len, Sequence owner) { + this.owner = owner; + this.holder = new Forest[len]; + } - private Forest zero = null; + public abstract int provides(); + public abstract int[] needs(); + public abstract int[] hates(); + public abstract boolean owner_needed_or_hated(); + + public abstract boolean isFirst(); + public abstract boolean isLast(); + public abstract Pos last(); + public abstract Pos prev(); + public abstract Pos next(); + + /** the element which produces the sequence to which this Position belongs */ + public Sequence owner() { return owner; } + + abstract Element element(); + + public abstract int numPops(); + public abstract Forest rewrite(Input.Region loc); + } + + /** the imaginary position before or after an element of a sequence; corresponds to an "LR item" */ + private static class Position extends Pos implements IntegerMappable { /* - public Forest zero(Input.Region reg) { - if (zero != null) return zero; - if (pos > 0) throw new RuntimeException("Position.zero(): pos>0"); - return zero = rewrite(reg); + public Pos getPos() { + return new DumbPos(elements.length, provides(), needs(), hates(), owner_needed_or_hated(), numPops(), + public int provides(); + public int[] needs(); + public int[] hates(); + public boolean owner_needed_or_hated(); + public int numPops(); + public Forest rewrite(Input.Region loc) + }; } */ - final int pos; - private final Position next; - private final Position prev; - final Forest[] holder; + public int numPops() { return pos; } + + final int pos; + private final Position next; + private final Position prev; + + public int provides() { return owner().sernum; } + public int[] needs() { return owner().needs_int(); } + public int[] hates() { return owner().hates_int(); } + public boolean owner_needed_or_hated() { return owner().needed_or_hated; } - private Position(int pos, Position prev) { + private Position(Sequence owner, int pos, Position prev) { + super(owner.elements.length,owner); this.pos = pos; - this.next = pos==elements.length ? null : new Position(pos+1, this); - this.holder = new Forest[elements.length]; + this.next = pos==owner.elements.length ? null : new Position(owner, pos+1, this); this.prev = prev; } - boolean isFirst() { return pos==0; } + public int compareTo(Pos p) { + return ord - ((Position)p).ord; + } - /** the element immediately after this Position, or null if this is the last Position */ - public Element element() { return pos>=elements.length ? null : elements[pos]; } + public boolean isFirst() { return pos==0; } + public int pos() { return pos; } - /** the element which produces the sequence to which this Position belongs */ - public Sequence owner() { return Sequence.this; } + /** the element immediately after this Position, or null if this is the last Position */ + public Element element() { return pos>=owner().elements.length ? null : owner().elements[pos]; } /** the next Position (the Position after this.element()) */ public Position next() { return next; } @@ -159,20 +220,24 @@ public abstract class Sequence implements Iterable, SequenceOrElement { // Position ///////////////////////////////////////////////////////////////////////////////// - final Forest rewrite(Input.Region loc, Grammar cache) { - if (this==firstp()) epsilonForm(loc, cache); + public final Forest rewrite(Input.Region loc) { + if (isFirst()) return owner().epsilonForm(loc); for(int i=0; i, SequenceOrElement { } private static int master_position_idx = 0; + // toString ////////////////////////////////////////////////////////////////////////////// public String toString() { return toString(new StringBuffer(), false).toString(); } @@ -210,96 +276,45 @@ public abstract class Sequence implements Iterable, SequenceOrElement { return sb; } - // Specialized Subclasses ////////////////////////////////////////////////////////////////////////////// - static class Constant extends Sequence { - private final Object result; - public Constant(Element[] e, Object result) { - super(e); - if (result==null) throw new Error("constant sequences may not have result==null"); - this.result = result; - } - Sequence _clone() { return new Constant(elements, result); } - public Forest postReduce(Input.Region loc, Forest[] args, Position p) { - return (Forest)Forest.create(loc, result, null, false); - } - Forest epsilonForm(Input.Region loc, Grammar cache) { - return Forest.create(loc, result, null, false); - } - } - static class Singleton extends Sequence { private final int idx; public Singleton(Element e) { this(new Element[] { e }, 0); } public Singleton(Element[] e, int idx) { super(e); this.idx = idx; } public Forest postReduce(Input.Region loc, Forest[] args, Position p) { return args[idx]; } Sequence _clone() { return new Singleton(elements,idx); } - Forest epsilonForm(Input.Region loc, Grammar cache) { - return ((Union)elements[idx]).epsilonForm(loc, cache); - } - } - - static class Unwrap extends Sequence { - private boolean[] drops; - private final Object tag; - public Unwrap(Element[] e, Object tag) { this(e, tag, null); } - public Unwrap(Element[] e, Object tag, boolean[] drops) { super(e); this.drops = drops; this.tag = tag; } - Sequence _clone() { return new Unwrap(elements, tag, drops); } - public Forest postReduce(Input.Region loc, Forest[] args, Position p) { - for(int i=0; i[] args2 = new Forest[count]; - int j = 0; - for(int i=0; i Forest postReduce(Input.Region loc, Forest[] args, Position p) { - for(int i=0; i[] args2 = new Forest[count]; - int j = 0; - for(int i=0; i Forest postReduce(Input.Region loc, Forest[] args, Position p) { - Forest[] args2 = new Forest[count]; + Forest[] args2 = new Forest[lifts.length]; int j = 0; for(int i=0; i, SequenceOrElement { if (spacing) for(int i=0; i<50-len; i++) sb.append(' '); return sb; } - Forest epsilonForm(Input.Region loc, Grammar cache) { - return Forest.create(loc, tag, new Forest[0], false); + Forest epsilonForm(Input.Region loc) { + return Forest.create(loc, tag, new Forest[0], lifts); } } - }