view core/src/luan/impl/Closure.java @ 195:24ede40ee0aa

make MetatableGetter DeepCloneable, scoped, and secure git-svn-id: https://luan-java.googlecode.com/svn/trunk@196 21e917c8-12df-6dd8-5cb6-c86387c605b9
author fschmidt@gmail.com <fschmidt@gmail.com@21e917c8-12df-6dd8-5cb6-c86387c605b9>
date Thu, 03 Jul 2014 08:19:48 +0000
parents 3dcb0f9bee82
children 5ba136769034
line wrap: on
line source

package luan.impl;

import luan.Luan;
import luan.LuanFunction;
import luan.LuanState;
import luan.LuanElement;
import luan.LuanException;
import luan.DeepCloner;
import luan.DeepCloneable;


final class Closure extends LuanFunction implements DeepCloneable<Closure> {
	private final FnDef fnDef;
	private MtGetterLink mtGetterLink;
	private UpValue[] upValues;

	Closure(LuanStateImpl luan,FnDef fnDef,MtGetterLink mtGetterLink) throws LuanException {
		this.fnDef = fnDef;
		this.mtGetterLink = mtGetterLink;
		UpValue.Getter[] upValueGetters = fnDef.upValueGetters;
		this.upValues = new UpValue[upValueGetters.length];
		for( int i=0; i<upValues.length; i++ ) {
			upValues[i] = upValueGetters[i].get(luan);
		}
	}

	private Closure(Closure c) {
		this.fnDef = c.fnDef;
	}

	@Override public Closure shallowClone() {
		return new Closure(this);
	}

	@Override public void deepenClone(Closure clone,DeepCloner cloner) {
		clone.mtGetterLink = cloner.deepClone(mtGetterLink);
		clone.upValues = cloner.deepClone(upValues);
	}

	MtGetterLink mtGetterLink() {
		return mtGetterLink;
	}

	UpValue[] upValues() {
		return upValues;
	}

	@Override public Object call(LuanState luan,Object[] args) throws LuanException {
		return call(this,(LuanStateImpl)luan,args);
	}

	private static Object call(Closure closure,LuanStateImpl luan,Object[] args) throws LuanException {
		while(true) {
			FnDef fnDef = closure.fnDef;
			Object[] varArgs = null;
			if( fnDef.isVarArg ) {
				if( args.length > fnDef.numArgs ) {
					varArgs = new Object[ args.length - fnDef.numArgs ];
					for( int i=0; i<varArgs.length; i++ ) {
						varArgs[i] = args[fnDef.numArgs+i];
					}
				} else {
					varArgs = LuanFunction.NOTHING;
				}
			}
			Object[] stack = luan.newFrame(closure,fnDef.stackSize,varArgs);
			final int n = Math.min(args.length,fnDef.numArgs);
			for( int i=0; i<n; i++ ) {
				stack[i] = args[i];
			}
			Object returnValues;
			try {
				fnDef.block.eval(luan);
			} catch(ReturnException e) {
			} finally {
				returnValues = luan.returnValues;
				closure = luan.tailFn;
				luan.popFrame();
			}
			if( closure == null )
				return returnValues;
			args = Luan.array(returnValues);
		}
	}

}