/** STBlock.m Wrapper for STBlockContext to make it invisible from the user Copyright (c) 2002 Free Software Foundation This file is part of the StepTalk project. This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #import "STBlock.h" #import "STLiterals.h" #import "STBlockContext.h" #import "STBytecodeInterpreter.h" #import "STStack.h" #import #import #import #import Class STBlockContextClass = nil; @implementation STBlock + (void)initialize { STBlockContextClass = [STBlockContext class]; } - initWithInterpreter:(STBytecodeInterpreter *)anInterpreter homeContext:(STMethodContext *)context initialIP:(unsigned)ptr argumentCount:(int)count stackSize:(int)size { homeContext = context; argCount = count; stackSize = size; initialIP = ptr; interpreter = anInterpreter; return [super init]; } - (unsigned)argumentCount { return argCount; } - value { if(argCount != 0) { [NSException raise:STScriptingException format:@"Block needs %i arguments", argCount]; return nil; } return [self valueWithArgs:(id*)0 count:0]; } - valueWith:arg { id args[1] = {arg}; id retval; retval = [self valueWithArgs:args count:1]; return retval; } - valueWith:arg1 with:arg2 { id args[2] = {arg1,arg2}; id retval; retval = [self valueWithArgs:args count:2]; return retval; } - valueWith:arg1 with:arg2 with:arg3 { id args[3] = {arg1,arg2,arg3}; id retval; retval = [self valueWithArgs:args count:3]; return retval; } - valueWith:arg1 with:arg2 with:arg3 with:arg4 { id args[4] = {arg1,arg2,arg3,arg4}; id retval; retval = [self valueWithArgs:args count:4]; return retval; } - valueWithArgs:(id *)args count:(unsigned)count; { STExecutionContext *parentContext; STBlockContext *context; STStack *stack; BOOL usingShared = NO; unsigned int i; id retval; if(argCount != count) { [NSException raise:STScriptingException format:@"Invalid block argument count %i, " @"wants to be %i", count, argCount]; return nil; } if(!usingCachedContext) { /* In case of recursive block nesting */ usingCachedContext = YES; if(!cachedContext) { cachedContext = [[STBlockContextClass alloc] initWithInterpreter:interpreter initialIP:initialIP stackSize:stackSize]; } /* Avoid allocation */ context = cachedContext; [[context stack] empty]; [context resetInstructionPointer]; } else { /* Create new context */ context = [[STBlockContextClass alloc] initWithInterpreter:interpreter initialIP:initialIP stackSize:stackSize]; AUTORELEASE(context); } /* push block arguments to the stack */ stack = [context stack]; for(i = 0; i