1*16467b97STreehugger Robot// [The "BSD licence"] 2*16467b97STreehugger Robot// Copyright (c) 2006-2007 Kay Roepke 2010 Alan Condit 3*16467b97STreehugger Robot// All rights reserved. 4*16467b97STreehugger Robot// 5*16467b97STreehugger Robot// Redistribution and use in source and binary forms, with or without 6*16467b97STreehugger Robot// modification, are permitted provided that the following conditions 7*16467b97STreehugger Robot// are met: 8*16467b97STreehugger Robot// 1. Redistributions of source code must retain the above copyright 9*16467b97STreehugger Robot// notice, this list of conditions and the following disclaimer. 10*16467b97STreehugger Robot// 2. Redistributions in binary form must reproduce the above copyright 11*16467b97STreehugger Robot// notice, this list of conditions and the following disclaimer in the 12*16467b97STreehugger Robot// documentation and/or other materials provided with the distribution. 13*16467b97STreehugger Robot// 3. The name of the author may not be used to endorse or promote products 14*16467b97STreehugger Robot// derived from this software without specific prior written permission. 15*16467b97STreehugger Robot// 16*16467b97STreehugger Robot// THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 17*16467b97STreehugger Robot// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 18*16467b97STreehugger Robot// OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 19*16467b97STreehugger Robot// IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 20*16467b97STreehugger Robot// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 21*16467b97STreehugger Robot// NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 22*16467b97STreehugger Robot// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 23*16467b97STreehugger Robot// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 24*16467b97STreehugger Robot// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 25*16467b97STreehugger Robot// THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 26*16467b97STreehugger Robot 27*16467b97STreehugger Robot#import "CommonTree.h" 28*16467b97STreehugger Robot 29*16467b97STreehugger Robot 30*16467b97STreehugger Robot@implementation CommonTree 31*16467b97STreehugger Robot 32*16467b97STreehugger Robot+ (CommonTree *)INVALID_NODE 33*16467b97STreehugger Robot{ 34*16467b97STreehugger Robot return [[CommonTree alloc] initWithToken:[CommonToken invalidToken]]; 35*16467b97STreehugger Robot} 36*16467b97STreehugger Robot 37*16467b97STreehugger Robot+ (CommonTree *)invalidNode 38*16467b97STreehugger Robot{ 39*16467b97STreehugger Robot // Had to cast to CommonTree * here, because GCC is dumb. 40*16467b97STreehugger Robot return [[CommonTree alloc] initWithToken:CommonToken.INVALID_TOKEN]; 41*16467b97STreehugger Robot} 42*16467b97STreehugger Robot 43*16467b97STreehugger Robot+ (CommonTree *)newTree 44*16467b97STreehugger Robot{ 45*16467b97STreehugger Robot return [[CommonTree alloc] init]; 46*16467b97STreehugger Robot} 47*16467b97STreehugger Robot 48*16467b97STreehugger Robot+ (CommonTree *)newTreeWithTree:(CommonTree *)aTree 49*16467b97STreehugger Robot{ 50*16467b97STreehugger Robot return [[CommonTree alloc] initWithTreeNode:aTree]; 51*16467b97STreehugger Robot} 52*16467b97STreehugger Robot 53*16467b97STreehugger Robot+ (CommonTree *)newTreeWithToken:(id<Token>)aToken 54*16467b97STreehugger Robot{ 55*16467b97STreehugger Robot return [[CommonTree alloc] initWithToken:aToken]; 56*16467b97STreehugger Robot} 57*16467b97STreehugger Robot 58*16467b97STreehugger Robot+ (CommonTree *)newTreeWithTokenType:(NSInteger)aTType 59*16467b97STreehugger Robot{ 60*16467b97STreehugger Robot return [[CommonTree alloc] initWithTokenType:(NSInteger)aTType]; 61*16467b97STreehugger Robot} 62*16467b97STreehugger Robot 63*16467b97STreehugger Robot+ (CommonTree *)newTreeWithTokenType:(NSInteger)aTType Text:(NSString *)theText 64*16467b97STreehugger Robot{ 65*16467b97STreehugger Robot return [[CommonTree alloc] initWithTokenType:(NSInteger)aTType Text:theText]; 66*16467b97STreehugger Robot} 67*16467b97STreehugger Robot 68*16467b97STreehugger Robot- (id)init 69*16467b97STreehugger Robot{ 70*16467b97STreehugger Robot self = (CommonTree *)[super init]; 71*16467b97STreehugger Robot if ( self != nil ) { 72*16467b97STreehugger Robot token = nil; 73*16467b97STreehugger Robot startIndex = -1; 74*16467b97STreehugger Robot stopIndex = -1; 75*16467b97STreehugger Robot parent = nil; 76*16467b97STreehugger Robot childIndex = -1; 77*16467b97STreehugger Robot } 78*16467b97STreehugger Robot return (CommonTree *)self; 79*16467b97STreehugger Robot} 80*16467b97STreehugger Robot 81*16467b97STreehugger Robot- (id)initWithTreeNode:(CommonTree *)aNode 82*16467b97STreehugger Robot{ 83*16467b97STreehugger Robot self = (CommonTree *)[super init]; 84*16467b97STreehugger Robot if ( self != nil ) { 85*16467b97STreehugger Robot token = aNode.token; 86*16467b97STreehugger Robot if ( token ) [token retain]; 87*16467b97STreehugger Robot startIndex = aNode.startIndex; 88*16467b97STreehugger Robot stopIndex = aNode.stopIndex; 89*16467b97STreehugger Robot parent = nil; 90*16467b97STreehugger Robot childIndex = -1; 91*16467b97STreehugger Robot } 92*16467b97STreehugger Robot return self; 93*16467b97STreehugger Robot} 94*16467b97STreehugger Robot 95*16467b97STreehugger Robot- (id)initWithToken:(id<Token>)aToken 96*16467b97STreehugger Robot{ 97*16467b97STreehugger Robot self = (CommonTree *)[super init]; 98*16467b97STreehugger Robot if ( self != nil ) { 99*16467b97STreehugger Robot token = aToken; 100*16467b97STreehugger Robot if ( token ) [token retain]; 101*16467b97STreehugger Robot startIndex = -1; 102*16467b97STreehugger Robot stopIndex = -1; 103*16467b97STreehugger Robot parent = nil; 104*16467b97STreehugger Robot childIndex = -1; 105*16467b97STreehugger Robot } 106*16467b97STreehugger Robot return self; 107*16467b97STreehugger Robot} 108*16467b97STreehugger Robot 109*16467b97STreehugger Robot- (id)initWithTokenType:(NSInteger)aTokenType 110*16467b97STreehugger Robot{ 111*16467b97STreehugger Robot self = (CommonTree *)[super init]; 112*16467b97STreehugger Robot if ( self != nil ) { 113*16467b97STreehugger Robot token = [[CommonToken newToken:aTokenType] retain]; 114*16467b97STreehugger Robot// startIndex = token.startIndex; 115*16467b97STreehugger Robot startIndex = -1; 116*16467b97STreehugger Robot// stopIndex = token.stopIndex; 117*16467b97STreehugger Robot stopIndex = -1; 118*16467b97STreehugger Robot parent = nil; 119*16467b97STreehugger Robot childIndex = -1; 120*16467b97STreehugger Robot } 121*16467b97STreehugger Robot return self; 122*16467b97STreehugger Robot} 123*16467b97STreehugger Robot 124*16467b97STreehugger Robot- (id) initWithTokenType:(NSInteger)aTokenType Text:(NSString *)theText 125*16467b97STreehugger Robot{ 126*16467b97STreehugger Robot self = (CommonTree *)[super init]; 127*16467b97STreehugger Robot if ( self != nil ) { 128*16467b97STreehugger Robot token = [[CommonToken newToken:aTokenType Text:theText] retain]; 129*16467b97STreehugger Robot// startIndex = token.startIndex; 130*16467b97STreehugger Robot startIndex = -1; 131*16467b97STreehugger Robot// stopIndex = token.stopIndex; 132*16467b97STreehugger Robot stopIndex = -1; 133*16467b97STreehugger Robot parent = nil; 134*16467b97STreehugger Robot childIndex = -1; 135*16467b97STreehugger Robot } 136*16467b97STreehugger Robot return self; 137*16467b97STreehugger Robot} 138*16467b97STreehugger Robot 139*16467b97STreehugger Robot- (void) dealloc 140*16467b97STreehugger Robot{ 141*16467b97STreehugger Robot if ( token ) { 142*16467b97STreehugger Robot [token release]; 143*16467b97STreehugger Robot token = nil; 144*16467b97STreehugger Robot } 145*16467b97STreehugger Robot if ( parent ) { 146*16467b97STreehugger Robot [parent release]; 147*16467b97STreehugger Robot parent = nil; 148*16467b97STreehugger Robot } 149*16467b97STreehugger Robot [super dealloc]; 150*16467b97STreehugger Robot} 151*16467b97STreehugger Robot 152*16467b97STreehugger Robot- (id) copyWithZone:(NSZone *)aZone 153*16467b97STreehugger Robot{ 154*16467b97STreehugger Robot CommonTree *copy; 155*16467b97STreehugger Robot 156*16467b97STreehugger Robot // copy = [[[self class] allocWithZone:aZone] init]; 157*16467b97STreehugger Robot copy = [super copyWithZone:aZone]; // allocation occurs in BaseTree 158*16467b97STreehugger Robot if ( self.token ) 159*16467b97STreehugger Robot copy.token = [self.token copyWithZone:aZone]; 160*16467b97STreehugger Robot copy.startIndex = startIndex; 161*16467b97STreehugger Robot copy.stopIndex = stopIndex; 162*16467b97STreehugger Robot copy.parent = (CommonTree *)[self.parent copyWithZone:aZone]; 163*16467b97STreehugger Robot copy.childIndex = childIndex; 164*16467b97STreehugger Robot return copy; 165*16467b97STreehugger Robot} 166*16467b97STreehugger Robot 167*16467b97STreehugger Robot- (BOOL) isNil 168*16467b97STreehugger Robot{ 169*16467b97STreehugger Robot return token == nil; 170*16467b97STreehugger Robot} 171*16467b97STreehugger Robot 172*16467b97STreehugger Robot- (CommonToken *) getToken 173*16467b97STreehugger Robot{ 174*16467b97STreehugger Robot return token; 175*16467b97STreehugger Robot} 176*16467b97STreehugger Robot 177*16467b97STreehugger Robot- (void) setToken:(CommonToken *) aToken 178*16467b97STreehugger Robot{ 179*16467b97STreehugger Robot if ( token != aToken ) { 180*16467b97STreehugger Robot if ( token ) [token release]; 181*16467b97STreehugger Robot [aToken retain]; 182*16467b97STreehugger Robot token = aToken; 183*16467b97STreehugger Robot } 184*16467b97STreehugger Robot} 185*16467b97STreehugger Robot 186*16467b97STreehugger Robot- (CommonTree *) dupNode 187*16467b97STreehugger Robot{ 188*16467b97STreehugger Robot return [CommonTree newTreeWithTree:self ]; 189*16467b97STreehugger Robot} 190*16467b97STreehugger Robot 191*16467b97STreehugger Robot- (NSInteger)type 192*16467b97STreehugger Robot{ 193*16467b97STreehugger Robot if (token) 194*16467b97STreehugger Robot return token.type; 195*16467b97STreehugger Robot return TokenTypeInvalid; 196*16467b97STreehugger Robot} 197*16467b97STreehugger Robot 198*16467b97STreehugger Robot- (NSString *)text 199*16467b97STreehugger Robot{ 200*16467b97STreehugger Robot if (token) 201*16467b97STreehugger Robot return token.text; 202*16467b97STreehugger Robot return nil; 203*16467b97STreehugger Robot} 204*16467b97STreehugger Robot 205*16467b97STreehugger Robot- (NSUInteger)line 206*16467b97STreehugger Robot{ 207*16467b97STreehugger Robot if (token) 208*16467b97STreehugger Robot return token.line; 209*16467b97STreehugger Robot return 0; 210*16467b97STreehugger Robot} 211*16467b97STreehugger Robot 212*16467b97STreehugger Robot- (void) setLine:(NSUInteger)aLine 213*16467b97STreehugger Robot{ 214*16467b97STreehugger Robot if (token) 215*16467b97STreehugger Robot token.line = aLine; 216*16467b97STreehugger Robot} 217*16467b97STreehugger Robot 218*16467b97STreehugger Robot- (NSUInteger)charPositionInLine 219*16467b97STreehugger Robot{ 220*16467b97STreehugger Robot if (token) 221*16467b97STreehugger Robot return token.charPositionInLine; 222*16467b97STreehugger Robot return 0; 223*16467b97STreehugger Robot} 224*16467b97STreehugger Robot 225*16467b97STreehugger Robot- (void) setCharPositionInLine:(NSUInteger)pos 226*16467b97STreehugger Robot{ 227*16467b97STreehugger Robot if (token) 228*16467b97STreehugger Robot token.charPositionInLine = pos; 229*16467b97STreehugger Robot} 230*16467b97STreehugger Robot 231*16467b97STreehugger Robot- (NSInteger) getTokenStartIndex 232*16467b97STreehugger Robot{ 233*16467b97STreehugger Robot if ( startIndex == -1 && token != nil ) { 234*16467b97STreehugger Robot return [token getTokenIndex]; 235*16467b97STreehugger Robot } 236*16467b97STreehugger Robot return startIndex; 237*16467b97STreehugger Robot} 238*16467b97STreehugger Robot 239*16467b97STreehugger Robot- (void) setTokenStartIndex: (NSInteger) aStartIndex 240*16467b97STreehugger Robot{ 241*16467b97STreehugger Robot startIndex = aStartIndex; 242*16467b97STreehugger Robot} 243*16467b97STreehugger Robot 244*16467b97STreehugger Robot- (NSInteger) getTokenStopIndex 245*16467b97STreehugger Robot{ 246*16467b97STreehugger Robot if ( stopIndex == -1 && token != nil ) { 247*16467b97STreehugger Robot return [token getTokenIndex]; 248*16467b97STreehugger Robot } 249*16467b97STreehugger Robot return stopIndex; 250*16467b97STreehugger Robot} 251*16467b97STreehugger Robot 252*16467b97STreehugger Robot- (void) setTokenStopIndex: (NSInteger) aStopIndex 253*16467b97STreehugger Robot{ 254*16467b97STreehugger Robot stopIndex = aStopIndex; 255*16467b97STreehugger Robot} 256*16467b97STreehugger Robot 257*16467b97STreehugger Robot#ifdef DONTUSENOMO 258*16467b97STreehugger Robot- (NSString *) treeDescription 259*16467b97STreehugger Robot{ 260*16467b97STreehugger Robot if (children) { 261*16467b97STreehugger Robot NSMutableString *desc = [NSMutableString stringWithString:@"(^"]; 262*16467b97STreehugger Robot [desc appendString:[self description]]; 263*16467b97STreehugger Robot unsigned int childIdx; 264*16467b97STreehugger Robot for (childIdx = 0; childIdx < [children count]; childIdx++) { 265*16467b97STreehugger Robot [desc appendFormat:@"%@", [[children objectAtIndex:childIdx] treeDescription]]; 266*16467b97STreehugger Robot } 267*16467b97STreehugger Robot [desc appendString:@")"]; 268*16467b97STreehugger Robot return desc; 269*16467b97STreehugger Robot } else { 270*16467b97STreehugger Robot return [self description]; 271*16467b97STreehugger Robot } 272*16467b97STreehugger Robot} 273*16467b97STreehugger Robot#endif 274*16467b97STreehugger Robot 275*16467b97STreehugger Robot/** For every node in this subtree, make sure it's start/stop token's 276*16467b97STreehugger Robot * are set. Walk depth first, visit bottom up. Only updates nodes 277*16467b97STreehugger Robot * with at least one token index < 0. 278*16467b97STreehugger Robot */ 279*16467b97STreehugger Robot- (void) setUnknownTokenBoundaries 280*16467b97STreehugger Robot{ 281*16467b97STreehugger Robot if ( children == nil ) { 282*16467b97STreehugger Robot if ( startIndex < 0 || stopIndex < 0 ) { 283*16467b97STreehugger Robot startIndex = stopIndex = [token getTokenIndex]; 284*16467b97STreehugger Robot } 285*16467b97STreehugger Robot return; 286*16467b97STreehugger Robot } 287*16467b97STreehugger Robot for (NSUInteger i=0; i < [children count]; i++) { 288*16467b97STreehugger Robot [[children objectAtIndex:i] setUnknownTokenBoundaries]; 289*16467b97STreehugger Robot } 290*16467b97STreehugger Robot if ( startIndex >= 0 && stopIndex >= 0 ) 291*16467b97STreehugger Robot return; // already set 292*16467b97STreehugger Robot if ( [children count] > 0 ) { 293*16467b97STreehugger Robot CommonTree *firstChild = (CommonTree *)[children objectAtIndex:0]; 294*16467b97STreehugger Robot CommonTree *lastChild = (CommonTree *)[children objectAtIndex:[children count]-1]; 295*16467b97STreehugger Robot startIndex = [firstChild getTokenStartIndex]; 296*16467b97STreehugger Robot stopIndex = [lastChild getTokenStopIndex]; 297*16467b97STreehugger Robot } 298*16467b97STreehugger Robot} 299*16467b97STreehugger Robot 300*16467b97STreehugger Robot- (NSInteger) getChildIndex 301*16467b97STreehugger Robot{ 302*16467b97STreehugger Robot return childIndex; 303*16467b97STreehugger Robot} 304*16467b97STreehugger Robot 305*16467b97STreehugger Robot- (CommonTree *) getParent 306*16467b97STreehugger Robot{ 307*16467b97STreehugger Robot return parent; 308*16467b97STreehugger Robot} 309*16467b97STreehugger Robot 310*16467b97STreehugger Robot- (void) setParent:(CommonTree *) t 311*16467b97STreehugger Robot{ 312*16467b97STreehugger Robot parent = t; 313*16467b97STreehugger Robot} 314*16467b97STreehugger Robot 315*16467b97STreehugger Robot- (void) setChildIndex:(NSInteger) anIndex 316*16467b97STreehugger Robot{ 317*16467b97STreehugger Robot childIndex = anIndex; 318*16467b97STreehugger Robot} 319*16467b97STreehugger Robot 320*16467b97STreehugger Robot- (NSString *) description 321*16467b97STreehugger Robot{ 322*16467b97STreehugger Robot if ( [self isNil] ) { 323*16467b97STreehugger Robot return @"nil"; 324*16467b97STreehugger Robot } 325*16467b97STreehugger Robot if ( [self type] == TokenTypeInvalid ) { 326*16467b97STreehugger Robot return @"<errornode>"; 327*16467b97STreehugger Robot } 328*16467b97STreehugger Robot if ( token==nil ) { 329*16467b97STreehugger Robot return nil; 330*16467b97STreehugger Robot } 331*16467b97STreehugger Robot return token.text; 332*16467b97STreehugger Robot} 333*16467b97STreehugger Robot 334*16467b97STreehugger Robot- (NSString *) toString 335*16467b97STreehugger Robot{ 336*16467b97STreehugger Robot return [self description]; 337*16467b97STreehugger Robot} 338*16467b97STreehugger Robot 339*16467b97STreehugger Robot@synthesize token; 340*16467b97STreehugger Robot@synthesize startIndex; 341*16467b97STreehugger Robot@synthesize stopIndex; 342*16467b97STreehugger Robot@synthesize parent; 343*16467b97STreehugger Robot@synthesize childIndex; 344*16467b97STreehugger Robot 345*16467b97STreehugger Robot@end 346