org.snapscript.tree.define.ClassDefinition Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of snap-all Show documentation
Show all versions of snap-all Show documentation
Dynamic scripting for the JVM
package org.snapscript.tree.define;
import static org.snapscript.core.result.Result.NORMAL;
import static org.snapscript.core.type.Category.CLASS;
import static org.snapscript.core.type.Phase.COMPILE;
import static org.snapscript.core.type.Phase.CREATE;
import static org.snapscript.core.type.Phase.DEFINE;
import java.util.concurrent.atomic.AtomicBoolean;
import org.snapscript.common.Progress;
import org.snapscript.core.Compilation;
import org.snapscript.core.Context;
import org.snapscript.core.Execution;
import org.snapscript.core.NoExecution;
import org.snapscript.core.Statement;
import org.snapscript.core.constraint.Constraint;
import org.snapscript.core.error.ErrorHandler;
import org.snapscript.core.module.Module;
import org.snapscript.core.module.Path;
import org.snapscript.core.scope.Scope;
import org.snapscript.core.trace.Trace;
import org.snapscript.core.trace.TraceInterceptor;
import org.snapscript.core.trace.TraceStatement;
import org.snapscript.core.type.Phase;
import org.snapscript.core.type.Type;
import org.snapscript.core.type.TypePart;
import org.snapscript.core.type.TypeState;
import org.snapscript.tree.annotation.AnnotationList;
public class ClassDefinition implements Compilation {
private final Statement definition;
public ClassDefinition(AnnotationList annotations, ClassName name, TypeHierarchy hierarchy, TypePart... parts) {
this.definition = new CompileResult(annotations, name, hierarchy, parts);
}
@Override
public Statement compile(Module module, Path path, int line) throws Exception {
Context context = module.getContext();
ErrorHandler handler = context.getHandler();
TraceInterceptor interceptor = context.getInterceptor();
Trace trace = Trace.getDefine(module, path, line);
return new TraceStatement(interceptor, handler, definition, trace);
}
private static class CompileResult extends Statement {
private final FunctionPropertyGenerator generator;
private final DefaultConstructor constructor;
private final TypeStateCollector collector;
private final TypeState constants;
private final AtomicBoolean compile;
private final AtomicBoolean define;
private final AtomicBoolean create;
private final ClassBuilder builder;
private final Execution execution;
private final TypePart[] parts;
public CompileResult(AnnotationList annotations, ClassName name, TypeHierarchy hierarchy, TypePart... parts) {
this.builder = new ClassBuilder(annotations, name, hierarchy, CLASS);
this.generator = new FunctionPropertyGenerator();
this.collector = new TypeStateCollector();
this.constructor = new DefaultConstructor(collector);
this.execution = new NoExecution(NORMAL);
this.compile = new AtomicBoolean(true);
this.define = new AtomicBoolean(true);
this.create = new AtomicBoolean(true);
this.constants = new StaticState();
this.parts = parts;
}
@Override
public void create(Scope outer) throws Exception {
if(!create.compareAndSet(false, true)) {
Type type = builder.create(collector, outer);
Progress progress = type.getProgress();
Scope scope = type.getScope();
try {
for(TypePart part : parts) {
part.create(collector, type, scope);
}
} finally {
progress.done(CREATE);
}
}
}
@Override
public boolean define(Scope outer) throws Exception {
if(!define.compareAndSet(false, true)) {
Type type = builder.define(collector, outer);
Progress progress = type.getProgress();
Scope scope = type.getScope();
try {
collector.update(constants); // collect static constants first
for(TypePart part : parts) {
TypeState state = part.define(collector, type, scope);
collector.update(state);
}
collector.update(constructor);
collector.define(scope, type);
generator.generate(type);
} finally {
progress.done(DEFINE);
}
}
return true;
}
@Override
public Execution compile(Scope outer, Constraint returns) throws Exception {
if(!compile.compareAndSet(false, true)) {
Type type = builder.compile(collector, outer);
Progress progress = type.getProgress();
Scope scope = type.getScope();
Scope local = scope.getStack(); // make it temporary
try {
collector.compile(local, type);
} finally {
progress.done(COMPILE);
}
}
return execution;
}
}
}