libs-steptalk/Languages/Smalltalk/STBlock.m
Wolfgang Lux 5e87e76a79 Fix bug where the cached context of a block could be reused prematurely.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/steptalk/trunk@38146 72102866-910b-0410-8b05-ffd578937521
2014-11-01 18:26:39 +00:00

256 lines
6.2 KiB
Objective-C

/**
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 "Externs.h"
#import "STBlock.h"
#import "STLiterals.h"
#import "STBlockContext.h"
#import "STBytecodeInterpreter.h"
#import "STStack.h"
#import <StepTalk/STExterns.h>
#import <Foundation/NSArray.h>
#import <Foundation/NSDebug.h>
#import <Foundation/NSException.h>
Class STBlockContextClass = nil;
@implementation STBlock
+ (void)initialize
{
STBlockContextClass = [STBlockContext class];
}
- initWithInterpreter:(STBytecodeInterpreter *)anInterpreter
homeContext:(STMethodContext *)context
initialIP:(NSUInteger)ptr
argumentCount:(NSUInteger)count
stackSize:(NSUInteger)size
{
if ((self = [super init]) != nil)
{
homeContext = RETAIN(context);
argCount = count;
stackSize = size;
initialIP = ptr;
interpreter = RETAIN(anInterpreter);
}
return self;
}
- (void)dealloc
{
RELEASE(homeContext);
RELEASE(interpreter);
RELEASE(cachedContext);
[super dealloc];
}
- (NSUInteger)argumentCount
{
return argCount;
}
- value
{
if(argCount != 0)
{
[NSException raise:STScriptingException
format:@"Block needs %lu arguments",
(unsigned long)argCount];
return nil;
}
return [self valueWithArgs:(id*)0 count:0];
}
- value:arg
{
return [self valueWithArguments:[NSArray arrayWithObject:arg]];
}
- value:arg1 value:arg2
{
return [self valueWithArguments:[NSArray arrayWithObjects:arg1,arg2,nil]];
}
- value:arg1 value:arg2 value:arg3
{
return [self valueWithArguments:[NSArray arrayWithObjects:arg1,arg2,arg3,nil]];
}
- valueWith:arg
{
id args[1] = {arg};
return [self valueWithArgs:args count:1];
}
- valueWith:arg1 with:arg2
{
id args[2] = {arg1,arg2};
return [self valueWithArgs:args count:2];
}
- valueWith:arg1 with:arg2 with:arg3
{
id args[3] = {arg1,arg2,arg3};
return [self valueWithArgs:args count:3];
}
- valueWith:arg1 with:arg2 with:arg3 with:arg4
{
id args[4] = {arg1,arg2,arg3,arg4};
return [self valueWithArgs:args count:4];
}
- valueWithArgs:(id *)args count:(NSUInteger)count
{
NSArray *arguments = [NSArray arrayWithObjects:args count:count];
return [self valueWithArguments:arguments];
}
- valueWithArguments:(NSArray *)arguments
{
STExecutionContext *parentContext;
STBlockContext *context;
STStack *stack;
NSUInteger i, count;
id retval;
count = [arguments count];
if (argCount != count)
{
[NSException raise:STScriptingException
format:@"Invalid block argument count %lu, "
@"wants to be %lu", (unsigned long)count,
(unsigned long)argCount];
return nil;
}
if (!usingCachedContext)
{
/* In case of recursive block nesting */
usingCachedContext = YES;
if (!cachedContext)
{
cachedContext = [[STBlockContextClass alloc]
initWithInitialIP:initialIP
stackSize:stackSize];
}
/* Avoid allocation */
context = cachedContext;
[[context stack] empty];
[context resetInstructionPointer];
}
else
{
/* Create new context */
context = [[STBlockContextClass alloc] initWithInitialIP:initialIP
stackSize:stackSize];
AUTORELEASE(context);
}
/* push block arguments to the stack */
stack = [context stack];
for (i = 0; i<count; i++)
{
[stack push:[arguments objectAtIndex:i]];
}
[context setHomeContext:homeContext];
parentContext = [interpreter context];
[interpreter setContext:context];
NS_DURING
retval = [interpreter interpret];
NS_HANDLER
if (context == cachedContext)
usingCachedContext = NO;
[localException raise];
NS_ENDHANDLER
[interpreter setContext:parentContext];
/* Release cached context */
if (context == cachedContext)
{
if (usingCachedContext)
usingCachedContext = NO;
else
[NSException raise:STInternalInconsistencyException
format:@"%@: using cached context %@",
@" but usingCachedContext is not set",
self, cachedContext];
}
return retval;
}
- handler:(STBlock *)handlerBlock
{
STExecutionContext *context;
id retval;
/* save the execution context for the case of exception */
context = [interpreter context];
NS_DURING
retval = [self value];
NS_HANDLER
/* don't handle the internal STInterpreterReturnException exception */
if ([[localException name] isEqualToString:STInterpreterReturnException])
[localException raise];
retval = [handlerBlock value:localException];
/* restore the execution context */
[interpreter setContext:context];
NS_ENDHANDLER
return retval;
}
- whileTrue:(STBlock *)doBlock
{
id retval = nil;
while([[self value] boolValue])
{
retval = [doBlock value];
}
return retval;
}
- whileFalse:(STBlock *)doBlock
{
id retval = nil;
while(! [[self value] boolValue])
{
retval = [doBlock value];
}
return retval;
}
@end