mirror of
https://github.com/gnustep/libs-steptalk.git
synced 2025-02-22 19:11:05 +00:00
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/steptalk/trunk@38146 72102866-910b-0410-8b05-ffd578937521
256 lines
6.2 KiB
Objective-C
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
|