X-Git-Url: http://git.megacz.com/?a=blobdiff_plain;f=src%2Fedu%2Fberkeley%2Fsbp%2Fmeta%2FAnnotationGrammarBindings.java;fp=src%2Fedu%2Fberkeley%2Fsbp%2Fmeta%2FAnnotationGrammarBindings.java;h=b7d89a3212012416921fef4c4c35c90f7ac457df;hb=2ca60a30895e90a0458361974402cee001498784;hp=0000000000000000000000000000000000000000;hpb=b1a5513197f7c776706fd5b5618574906f72b707;p=sbp.git diff --git a/src/edu/berkeley/sbp/meta/AnnotationGrammarBindings.java b/src/edu/berkeley/sbp/meta/AnnotationGrammarBindings.java new file mode 100644 index 0000000..b7d89a3 --- /dev/null +++ b/src/edu/berkeley/sbp/meta/AnnotationGrammarBindings.java @@ -0,0 +1,84 @@ +package edu.berkeley.sbp.meta; +import edu.berkeley.sbp.util.*; +import edu.berkeley.sbp.*; +import edu.berkeley.sbp.chr.*; +import edu.berkeley.sbp.misc.*; +import edu.berkeley.sbp.bind.*; +import java.util.*; +import java.lang.annotation.*; +import java.lang.reflect.*; +import java.io.*; + +// FIXME: non-static methods +public class AnnotationGrammarBindings extends Grammar.Bindings { + + private static boolean harsh = true; + + private final Class _cl; + private HashMap _inner = new HashMap(); + private HashMap _allMethods = new HashMap(); + + public AnnotationGrammarBindings(Class c) { + this._cl = c; + add(c, ""); + } + + public void add(Class c, String prefix) { + + ArrayList alc = new ArrayList(); + if (_inner.get(prefix) != null) + for(Class cc : _inner.get(prefix)) + alc.add(cc); + + ArrayList alm = new ArrayList(); + if (_allMethods.get(prefix) != null) + for(Method m : _allMethods.get(prefix)) + alm.add(m); + + add(c, alc, alm, prefix); + this._inner.put(prefix, (Class[])alc.toArray(new Class[0])); + this._allMethods.put(prefix, (Method[])alm.toArray(new Method[0])); + } + + public Object repeatTag() { return new Tree.ArrayBuildingTreeFunctor(); } + + public Sequence tryResolveTag(Production p) { + + String key = p.tag==null?p.nonTerminal:p.tag; + if (key==null) return null; + + String prefix = key.indexOf('.')==-1 ? "" : key.substring(0, key.lastIndexOf('.')); + String suffix = key.indexOf('.')==-1 ? key : key.substring(key.lastIndexOf('.')+1); + + p = new Production(suffix, p.elements, p.drops); + for(Method m : _allMethods.get(prefix)) + if (p.isCompatible(m)) + return p.makeSequence(m); + for(Class c : _inner.get(prefix)) + for(Constructor con : c.getConstructors()) + if (p.isCompatible(con)) + return p.makeSequence(con); + for(Class c : _inner.get(prefix)) + if (p.isCompatible(c)) + return p.makeSequence(c); + + return null; + + } + + + // helper + + private static void add(Class cl, ArrayList alc, ArrayList alm, String prefix) { + if (cl==null) return; + for(Method m : cl.getDeclaredMethods()) + alm.add(m); + for(Class c : cl.getDeclaredClasses()) { + alc.add(c); + add(c, alc, alm, prefix); + } + if (cl.getSuperclass() != Object.class) + add(cl.getSuperclass(), alc, alm, prefix); + } + +}