All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.luaj.vm2.lib.DebugLib Maven / Gradle / Ivy

/*******************************************************************************
* Copyright (c) 2009 Luaj.org. All rights reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
* 
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
******************************************************************************/
package org.luaj.vm2.lib;

import org.luaj.vm2.Lua;
import org.luaj.vm2.LuaBoolean;
import org.luaj.vm2.LuaClosure;
import org.luaj.vm2.LuaError;
import org.luaj.vm2.LuaFunction;
import org.luaj.vm2.LuaNil;
import org.luaj.vm2.LuaNumber;
import org.luaj.vm2.LuaString;
import org.luaj.vm2.LuaTable;
import org.luaj.vm2.LuaThread;
import org.luaj.vm2.LuaValue;
import org.luaj.vm2.Print;
import org.luaj.vm2.Prototype;
import org.luaj.vm2.Varargs;

public class DebugLib extends VarArgFunction {
	public static final boolean CALLS = (null != System.getProperty("CALLS"));
	public static final boolean TRACE = (null != System.getProperty("TRACE"));

	// leave this unset to allow obfuscators to 
	// remove it in production builds
	public static boolean DEBUG_ENABLED;

	static final String[] NAMES = {
		"debug",
		"getfenv",
		"gethook",
		"getinfo",
		"getlocal",
		"getmetatable",
		"getregistry",
		"getupvalue",
		"setfenv",
		"sethook",
		"setlocal",
		"setmetatable",
		"setupvalue",
		"traceback",
	};
	
	private static final int INIT        	= 0;
	private static final int DEBUG        	= 1;
	private static final int GETFENV        = 2;
	private static final int GETHOOK        = 3;
	private static final int GETINFO        = 4;
	private static final int GETLOCAL       = 5;
	private static final int GETMETATABLE 	= 6;
	private static final int GETREGISTRY    = 7;
	private static final int GETUPVALUE    	= 8;
	private static final int SETFENV        = 9;
	private static final int SETHOOK        = 10;
	private static final int SETLOCAL 		= 11;
	private static final int SETMETATABLE   = 12;
	private static final int SETUPVALUE    	= 13;
	private static final int TRACEBACK    	= 14;

	/* maximum stack for a Lua function */
	private static final int MAXSTACK = 250;
	
	private static final LuaString LUA        = valueOf("Lua");  
	private static final LuaString JAVA       = valueOf("Java");  
	private static final LuaString QMARK      = valueOf("?");  
	private static final LuaString GLOBAL     = valueOf("global");  
	private static final LuaString LOCAL      = valueOf("local");  
	private static final LuaString METHOD     = valueOf("method");  
	private static final LuaString UPVALUE    = valueOf("upvalue");  
	private static final LuaString FIELD      = valueOf("field");
	private static final LuaString CALL       = valueOf("call");  
	private static final LuaString LINE       = valueOf("line");  
	private static final LuaString COUNT      = valueOf("count");  
	private static final LuaString RETURN     = valueOf("return");  
	private static final LuaString TAILRETURN = valueOf("tail return");
	
	private static final LuaString FUNC            = valueOf("func");  
	private static final LuaString NUPS            = valueOf("nups");  
	private static final LuaString NAME            = valueOf("name");  
	private static final LuaString NAMEWHAT        = valueOf("namewhat");  
	private static final LuaString WHAT            = valueOf("what");  
	private static final LuaString SOURCE          = valueOf("source");  
	private static final LuaString SHORT_SRC       = valueOf("short_src");  
	private static final LuaString LINEDEFINED     = valueOf("linedefined");  
	private static final LuaString LASTLINEDEFINED = valueOf("lastlinedefined");  
	private static final LuaString CURRENTLINE     = valueOf("currentline");  
	private static final LuaString ACTIVELINES     = valueOf("activelines");  

	public DebugLib() {
	}
	
	private LuaTable init() {
		DEBUG_ENABLED = true;
		LuaTable t = new LuaTable();
		bind(t, DebugLib.class, NAMES, DEBUG);
		env.set("debug", t);
		PackageLib.instance.LOADED.set("debug", t);
		return t;
	}
	
	public Varargs invoke(Varargs args) {
		switch ( opcode ) {
		case INIT:         return init();
		case DEBUG:        return _debug(args);
		case GETFENV:      return _getfenv(args);
		case GETHOOK:      return _gethook(args);
		case GETINFO:      return _getinfo(args,this);
		case GETLOCAL:     return _getlocal(args);
		case GETMETATABLE: return _getmetatable(args);
		case GETREGISTRY:  return _getregistry(args);
		case GETUPVALUE:   return _getupvalue(args);
		case SETFENV:      return _setfenv(args);
		case SETHOOK:      return _sethook(args);
		case SETLOCAL:     return _setlocal(args);
		case SETMETATABLE: return _setmetatable(args);
		case SETUPVALUE:   return _setupvalue(args);
		case TRACEBACK:    return _traceback(args);
		default:           return NONE;
		}
	}

	// ------------------------ Debug Info management --------------------------
	// 
	// when DEBUG_ENABLED is set to true, these functions will be called 
	// by Closure instances as they process bytecodes.
	//
	// Each thread will get a DebugState attached to it by the debug library
	// which will track function calls, hook functions, etc.
	// 
	static class DebugInfo {
		LuaValue func;
		LuaClosure closure;
		LuaValue[] stack;
		Varargs varargs, extras;
		int pc, top;
		
		private DebugInfo() {			
			func = NIL;
		}
		private DebugInfo(LuaValue func) {
			pc = -1;
			setfunction( func );
		}
		void setargs(Varargs varargs, LuaValue[] stack) {
			this.varargs = varargs;
			this.stack = stack;
		}
		void setfunction( LuaValue func ) {
			this.func = func;
			this.closure = (func instanceof LuaClosure? (LuaClosure) func: null);
		}
		void clear() {
			func = NIL;
			closure = null;
			stack = null;
			varargs = extras = null;
			pc = top = 0;
		}
		public void bytecode(int pc, Varargs extras, int top) {
			this.pc = pc;
			this.top = top;
			this.extras = extras;
		}
		public int currentline() {
			if ( closure == null ) return -1;
			int[] li = closure.p.lineinfo;
			return li==null || pc<0 || pc>=li.length? -1: li[pc]; 
		}
		public LuaString[] getfunckind() {
			if ( closure == null || pc<0 ) return null;
			int stackpos = (closure.p.code[pc] >> 6) & 0xff; 
			return getobjname(this, stackpos);
		}
		public String sourceline() {
			if ( closure == null ) return func.tojstring();
			String s = closure.p.source.tojstring();
			int line = currentline();
			return (s.startsWith("@")||s.startsWith("=")? s.substring(1): s) + ":" + line;
		}
		public String tracename() {
			// if ( func != null )
			// 	return func.tojstring();
			LuaString[] kind = getfunckind();
			if ( kind == null )
				return "function ?";
			return "function "+kind[0].tojstring();
		}
		public LuaString getlocalname(int index) {
			if ( closure == null ) return null;
			return closure.p.getlocalname(index, pc);
		}
		public String tojstring() {
			return tracename()+" "+sourceline();
		}
	}
	
	/** DebugState is associated with a Thread */
	static class DebugState {
		private final LuaThread thread;
		private int debugCalls = 0;
		private DebugInfo[] debugInfo = new DebugInfo[LuaThread.MAX_CALLSTACK+1];
		private LuaValue hookfunc;
		private boolean hookcall,hookline,hookrtrn,inhook;
		private int hookcount,hookcodes;
		private int line;
		DebugState(LuaThread thread) {
			this.thread = thread;
		}
		public DebugInfo nextInfo() {
			DebugInfo di = debugInfo[debugCalls];
			if ( di == null ) 
				debugInfo[debugCalls] = di = new DebugInfo();
			return di;
		}
		public DebugInfo pushInfo( int calls ) {
			while ( debugCalls < calls ) {
				nextInfo();
				++debugCalls;
			}
			return debugInfo[debugCalls-1];
		}
		public void popInfo(int calls) {
			while ( debugCalls > calls )
				debugInfo[--debugCalls].clear();
		}
		void callHookFunc(DebugState ds, LuaString type, LuaValue arg) {
			if ( inhook || hookfunc == null )
				return;
			inhook = true;
			try {
				int n = debugCalls;
				ds.nextInfo().setargs( arg, null );
				ds.pushInfo(n+1).setfunction(hookfunc);
				try {
					hookfunc.call(type,arg);
				} finally {
					ds.popInfo(n);
				}
			} catch ( Throwable  t ) {
				t.printStackTrace();
			} finally {
				inhook = false;
			}
		}
		public void sethook(LuaValue func, boolean call, boolean line, boolean rtrn, int count) {
			this.hookcount = count;
			this.hookcall = call;
			this.hookline = line;
			this.hookrtrn = rtrn;
			this.hookfunc = func;
		}
		DebugInfo getDebugInfo() {
			try {
				return debugInfo[debugCalls-1];
			} catch ( Throwable t ) {
				if ( debugCalls <= 0 )
					return debugInfo[debugCalls++] = new DebugInfo();
				return null;
			}
		}
		DebugInfo getDebugInfo(int level) {
			return level < 0 || level >= debugCalls? null: debugInfo[debugCalls-level-1];
		}
		public DebugInfo findDebugInfo(LuaValue func) {			
			for ( int i=debugCalls; --i>=0; ) {
				if ( debugInfo[i].func == func ) {
					return debugInfo[i];
				}
			}
			return new DebugInfo(func);
		}
		public String tojstring() {
			return DebugLib.traceback(thread, 0);
		}
	}
	
	static DebugState getDebugState( LuaThread thread ) {
		if ( thread.debugState == null )
			thread.debugState = new DebugState(thread);
		return (DebugState) thread.debugState;
	}
	
	static DebugState getDebugState() {
		return getDebugState( LuaThread.getRunning() );
	}
	
	/** Called by Closures to set up stack and arguments to next call */
	public static void debugSetupCall(Varargs args, LuaValue[] stack) {
		DebugState ds = getDebugState();
		if ( ds.inhook )
			return;
		ds.nextInfo().setargs( args, stack );
	}
	
	/** Called by Closures and recursing java functions on entry
	 * @param thread the thread for the call 
	 * @param calls the number of calls in the call stack
	 * @param func the function called
	 */
	public static void debugOnCall(LuaThread thread, int calls, LuaFunction func) {
		DebugState ds = getDebugState();
		if ( ds.inhook )
			return;
		DebugInfo di = ds.pushInfo(calls);
		di.setfunction( func );
		if(CALLS)System.out.println("calling "+func);		
		if ( ds.hookcall )
			ds.callHookFunc( ds, CALL, LuaValue.NIL );
	}
	
	/** Called by Closures and recursing java functions on return 
	 * @param thread the thread for the call 
	 * @param calls the number of calls in the call stack
	 */
	public static void debugOnReturn(LuaThread thread, int calls) {
		DebugState ds = getDebugState(thread);
		if ( ds.inhook )
			return;
		if(CALLS)System.out.println("returning");		
		try {
			if ( ds.hookrtrn )
				ds.callHookFunc( ds, RETURN, LuaValue.NIL );
		} finally {
			getDebugState().popInfo(calls);
		}
	}
	
	/** Called by Closures on bytecode execution */
	public static void debugBytecode( int pc, Varargs extras, int top ) {
		DebugState ds = getDebugState();
		if ( ds.inhook )
			return;
		DebugInfo di = ds.getDebugInfo();
		if(TRACE)Print.printState(di.closure, pc, di.stack, top, di.varargs);		
		di.bytecode( pc, extras, top );
		if ( ds.hookcount > 0 ) {
			if ( ++ds.hookcodes >= ds.hookcount ) {
				ds.hookcodes = 0;
				ds.callHookFunc( ds, COUNT, LuaValue.NIL );
			}
		}
		if ( ds.hookline ) {
			int newline = di.currentline();
			if ( newline != ds.line ) {
				int c = di.closure.p.code[pc];
				if ( (c&0x3f) != Lua.OP_JMP || ((c>>>14)-0x1ffff) >= 0 ) {
					ds.line = newline;
					ds.callHookFunc( ds, LINE, LuaValue.valueOf(newline) );
				}
			}
		}
	}

	// ------------------- library function implementations -----------------
	
	// j2se subclass may wish to override and provide actual console here. 
	// j2me platform has not System.in to provide console.
	static Varargs _debug(Varargs args) {
		return NONE;
	}
	
	static Varargs _gethook(Varargs args) {
		int a=1;
		LuaThread thread = args.isthread(a)? args.checkthread(a++): LuaThread.getRunning(); 
		DebugState ds = getDebugState(thread);
		return varargsOf(
				ds.hookfunc,
				valueOf((ds.hookcall?"c":"")+(ds.hookline?"l":"")+(ds.hookrtrn?"r":"")),
				valueOf(ds.hookcount));
	}

	static Varargs _sethook(Varargs args) {
		int a=1;
		LuaThread thread = args.isthread(a)? args.checkthread(a++): LuaThread.getRunning(); 
		LuaValue func    = args.optfunction(a++, null);
		String str       = args.optjstring(a++,"");
		int count        = args.optint(a++,0);
		boolean call=false,line=false,rtrn=false;
		for ( int i=0; i0? 
				ds.getDebugInfo(level-1):
				new DebugInfo( level0func );
		} else {			
			di = ds.findDebugInfo( func.checkfunction() );
		}
		if ( di == null )
			return NIL;

		// start a table
		LuaTable info = new LuaTable();
		LuaClosure c = di.closure;
		for (int i = 0, j = what.length(); i < j; i++) {
			switch (what.charAt(i)) {
				case 'S': {
					if ( c != null ) {
						Prototype p = c.p;
						info.set(WHAT, LUA);
						info.set(SOURCE, p.source);
						info.set(SHORT_SRC, valueOf(sourceshort(p)));
						info.set(LINEDEFINED, valueOf(p.linedefined));
						info.set(LASTLINEDEFINED, valueOf(p.lastlinedefined));
					} else {
						String shortName = di.func.tojstring();
						LuaString name = LuaString.valueOf("[Java] "+shortName);
						info.set(WHAT, JAVA);
						info.set(SOURCE, name);
						info.set(SHORT_SRC, valueOf(shortName));
						info.set(LINEDEFINED, LuaValue.MINUSONE);
						info.set(LASTLINEDEFINED, LuaValue.MINUSONE);
					}
					break;
				}
				case 'l': {
					int line = di.currentline();
					info.set( CURRENTLINE, valueOf(line) );
					break;
				}
				case 'u': {
					info.set(NUPS, valueOf(c!=null? c.p.nups: 0));
					break;
				}
				case 'n': {
					LuaString[] kind = di.getfunckind();
					info.set(NAME, kind!=null? kind[0]: QMARK);
					info.set(NAMEWHAT, kind!=null? kind[1]: EMPTYSTRING);
					break;
				}
				case 'f': {
					info.set( FUNC, di.func );
					break;
				}
				case 'L': {
					LuaTable lines = new LuaTable();
					info.set(ACTIVELINES, lines);
//					if ( di.luainfo != null ) {
//						int line = di.luainfo.currentline();
//						if ( line >= 0 )
//							lines.set(1, IntValue.valueOf(line));
//					}
					break;
				}
			}
		}
		return info;
	}

	public static String sourceshort(Prototype p) {
		String name = p.source.tojstring();
        if ( name.startsWith("@") || name.startsWith("=") )
			name = name.substring(1);
		else if ( name.startsWith("\033") )
			name = "binary string";
        return name;
	}
	
	static Varargs _getlocal(Varargs args) {
		int a=1;
		LuaThread thread = args.isthread(a)? args.checkthread(a++): LuaThread.getRunning(); 
		int level = args.checkint(a++);
		int local = args.checkint(a++);
		
		DebugState ds = getDebugState(thread); 
		DebugInfo di = ds.getDebugInfo(level-1);
		LuaString name = (di!=null? di.getlocalname(local): null);
		if ( name != null ) {
			LuaValue value = di.stack[local-1];
			return varargsOf( name, value );
		} else {
			return NIL;
		}
	}

	static Varargs _setlocal(Varargs args) {
		int a=1;
		LuaThread thread = args.isthread(a)? args.checkthread(a++): LuaThread.getRunning(); 
		int level = args.checkint(a++);
		int local = args.checkint(a++);
		LuaValue value = args.arg(a++);
		
		DebugState ds = getDebugState(thread); 
		DebugInfo di = ds.getDebugInfo(level-1);
		LuaString name = (di!=null? di.getlocalname(local): null);
		if ( name != null ) {
			di.stack[local-1] = value;
			return name;
		} else {
			return NIL;
		}
	}

	static LuaValue _getmetatable(Varargs args) {
		LuaValue object = args.arg(1);
		LuaValue mt = object.getmetatable();
		return mt!=null? mt: NIL;
	}

	static Varargs _setmetatable(Varargs args) {
		LuaValue object = args.arg(1);
		try {
			LuaValue mt = args.opttable(2, null);
			switch ( object.type() ) {
				case TNIL:      LuaNil.s_metatable      = mt; break;
				case TNUMBER:   LuaNumber.s_metatable   = mt; break;
				case TBOOLEAN:  LuaBoolean.s_metatable  = mt; break;
				case TSTRING:   LuaString.s_metatable   = mt; break;
				case TFUNCTION: LuaFunction.s_metatable = mt; break;
				case TTHREAD:   LuaThread.s_metatable   = mt; break;
				default: object.setmetatable( mt );
			}
			return LuaValue.TRUE;
		} catch ( LuaError e ) {
			return varargsOf(FALSE, valueOf(e.toString()));
		}
	}

	static Varargs _getregistry(Varargs args) {
		return new LuaTable();
	}

	static LuaString findupvalue(LuaClosure c, int up) {
		if ( c.upValues != null && up > 0 && up <= c.upValues.length ) {
			if ( c.p.upvalues != null && up <= c.p.upvalues.length )
				return c.p.upvalues[up-1];
			else
				return LuaString.valueOf( "."+up );
		}
		return null;
	}

	static Varargs _getupvalue(Varargs args) {
		LuaValue func = args.checkfunction(1);
		int up = args.checkint(2);
		if ( func instanceof LuaClosure ) {
			LuaClosure c = (LuaClosure) func;
			LuaString name = findupvalue(c, up);
			if ( name != null ) {
				return varargsOf(name, c.upValues[up-1].getValue() );
			}
		}
		return NIL;
	}

	static LuaValue _setupvalue(Varargs args) {
		LuaValue func = args.checkfunction(1);
		int up = args.checkint(2);
		LuaValue value = args.arg(3);
		if ( func instanceof LuaClosure ) {
			LuaClosure c = (LuaClosure) func;
			LuaString name = findupvalue(c, up);
			if ( name != null ) {
				c.upValues[up-1].setValue(value);
				return name;
			}
		}
		return NIL;
	}

	static LuaValue _traceback(Varargs args) {
		int a=1;
		LuaThread thread = args.isthread(a)? args.checkthread(a++): LuaThread.getRunning(); 
		String message = args.optjstring(a++, null);
		int level = args.optint(a++,1);
		String tb = DebugLib.traceback(thread, level-1);
		return valueOf(message!=null? message+"\n"+tb: tb);
	}
	
	// =================== public utilities ====================
	
	/** 
	 * Get a traceback as a string for the current thread 
	 */
	public static String traceback(int level) {
		return traceback(LuaThread.getRunning(), level);
	}
	
	/**
	 * Get a traceback for a particular thread.
	 * @param thread LuaThread to provide stack trace for
	 * @param level 0-based level to start reporting on
	 * @return String containing the stack trace.
	 */
	public static String traceback(LuaThread thread, int level) {
		StringBuffer sb = new StringBuffer();
		DebugState ds = getDebugState(thread);
		sb.append( "stack traceback:" );
		DebugInfo di = ds.getDebugInfo(level);
		if ( di != null ) {
			sb.append( "\n\t" );
			sb.append( di.sourceline() );
			sb.append( " in " );
			while ( (di = ds.getDebugInfo(++level)) != null ) {
				sb.append( di.tracename() );
				sb.append( "\n\t" );
				sb.append( di.sourceline() );
				sb.append( " in " );
			}
			sb.append( "main chunk" );
		}
		return sb.toString();
	}


	/**
	 * Get file and line for the nearest calling closure.
	 * @return String identifying the file and line of the nearest lua closure,
	 * or the function name of the Java call if no closure is being called.
	 */
	public static String fileline() {
		DebugState ds = getDebugState(LuaThread.getRunning());
		DebugInfo di;
		for ( int i=0, n=ds.debugCalls; i 0) {
						/* cannot jump to a setlist count */
						int d = pt.code[dest - 1];
						if ((Lua.GET_OPCODE(d) == Lua.OP_SETLIST && Lua.GETARG_C(d) == 0)) return 0;
					}
				}
				break;
			}
			}
			if (Lua.testAMode(op)) {
				if (a == reg)
					last = pc; /* change register `a' */
			}
			if (Lua.testTMode(op)) {
				if (!(pc + 2 < pt.code.length)) return 0; /* check skip */
				if (!(Lua.GET_OPCODE(pt.code[pc + 1]) == Lua.OP_JMP)) return 0;
			}
			switch (op) {
			case Lua.OP_LOADBOOL: {
				if (!(c == 0 || pc + 2 < pt.code.length)) return 0; /* check its jump */
				break;
			}
			case Lua.OP_LOADNIL: {
				if (a <= reg && reg <= b)
					last = pc; /* set registers from `a' to `b' */
				break;
			}
			case Lua.OP_GETUPVAL:
			case Lua.OP_SETUPVAL: {
				if (!(b < pt.nups)) return 0;
				break;
			}
			case Lua.OP_GETGLOBAL:
			case Lua.OP_SETGLOBAL: {
				if (!(pt.k[b].isstring())) return 0;
				break;
			}
			case Lua.OP_SELF: {
				if (!checkreg(pt, a + 1)) return 0;
				if (reg == a + 1)
					last = pc;
				break;
			}
			case Lua.OP_CONCAT: {
				if (!(b < c)) return 0; /* at least two operands */
				break;
			}
			case Lua.OP_TFORLOOP: {
				if (!(c >= 1)) return 0; /* at least one result (control variable) */
				if (!checkreg(pt, a + 2 + c)) return 0; /* space for results */
				if (reg >= a + 2)
					last = pc; /* affect all regs above its base */
				break;
			}
			case Lua.OP_FORLOOP:
			case Lua.OP_FORPREP:
				if (!checkreg(pt, a + 3)) return 0;
				/* go through */
			case Lua.OP_JMP: {
				int dest = pc + 1 + b;
				/* not full check and jump is forward and do not skip `lastpc'? */
				if (reg != Lua.NO_REG && pc < dest && dest <= lastpc)
					pc += b; /* do the jump */
				break;
			}
			case Lua.OP_CALL:
			case Lua.OP_TAILCALL: {
				if (b != 0) {
					if (!checkreg(pt, a + b - 1)) return 0;
				}
				c--; /* c = num. returns */
				if (c == Lua.LUA_MULTRET) {
					if (!(checkopenop(pt, pc))) return 0;
				} else if (c != 0)
					if (!checkreg(pt, a + c - 1)) return 0;
				if (reg >= a)
					last = pc; /* affect all registers above base */
				break;
			}
			case Lua.OP_RETURN: {
				b--; /* b = num. returns */
				if (b > 0)
					if (!checkreg(pt, a + b - 1)) return 0;
				break;
			}
			case Lua.OP_SETLIST: {
				if (b > 0)
					if (!checkreg(pt, a + b)) return 0;
				if (c == 0)
					pc++;
				break;
			}
			case Lua.OP_CLOSURE: {
				int nup, j;
				if (!(b < pt.p.length)) return 0;
				nup = pt.p[b].nups;
				if (!(pc + nup < pt.code.length)) return 0;
				for (j = 1; j <= nup; j++) {
					int op1 = Lua.GET_OPCODE(pt.code[pc + j]);
					if (!(op1 == Lua.OP_GETUPVAL || op1 == Lua.OP_MOVE)) return 0;
				}
				if (reg != Lua.NO_REG) /* tracing? */
					pc += nup; /* do not 'execute' these pseudo-instructions */
				break;
			}
			case Lua.OP_VARARG: {
				if (!((pt.is_vararg & Lua.VARARG_ISVARARG) != 0
						&& (pt.is_vararg & Lua.VARARG_NEEDSARG) == 0)) return 0;
				b--;
				if (b == Lua.LUA_MULTRET)
					if (!(checkopenop(pt, pc))) return 0;
				if (!checkreg(pt, a + b - 1)) return 0;
				break;
			}
			default:
				break;
			}
		}
		return pt.code[last];
	}
	
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy