1//
2//  BufferedTreeNodeStream.m
3//  ANTLR
4//
5// [The "BSD licence"]
6// Copyright (c) 2010 Ian Michell 2010 Alan Condit
7// All rights reserved.
8//
9// Redistribution and use in source and binary forms, with or without
10// modification, are permitted provided that the following conditions
11// are met:
12// 1. Redistributions of source code must retain the above copyright
13//    notice, this list of conditions and the following disclaimer.
14// 2. Redistributions in binary form must reproduce the above copyright
15//    notice, this list of conditions and the following disclaimer in the
16//    documentation and/or other materials provided with the distribution.
17// 3. The name of the author may not be used to endorse or promote products
18//    derived from this software without specific prior written permission.
19//
20// THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
21// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
22// OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
23// IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
24// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
25// NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
29// THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31#import "BufferedTreeNodeStream.h"
32#import "StreamEnumerator.h"
33#import "CommonTreeAdaptor.h"
34
35extern NSInteger debug;
36
37#ifdef DONTUSENOMO
38@implementation TreeStreamIterator
39+ newTreeStreamIteratorWithNodes:(BufferedTreeNodeStream *)theStream
40{
41    return[[TreeStreamIterator alloc] initWithStream:theStream];
42}
43
44- (id) initWithStream:(BufferedTreeNodeStream *)theStream
45{
46    if ((self = [super init]) != nil) {
47        idx = 0;
48        input = theStream;
49        nodes = [theStream getNodes];
50    }
51    return self;
52}
53
54- (BOOL) hasNext
55{
56    return idx < [nodes count];
57}
58
59- (id) next
60{
61    NSInteger current = idx;
62    idx++;
63    if (current < [nodes count]) {
64    }
65    return [nodes getEof];
66}
67
68- (void) remove
69{
70	@throw [RuntimeException newException:@"cannot remove nodes from stream"];
71}
72
73@end
74#endif
75
76@implementation BufferedTreeNodeStream
77
78@synthesize up;
79@synthesize down;
80@synthesize eof;
81@synthesize nodes;
82@synthesize root;
83@synthesize tokens;
84@synthesize adaptor;
85@synthesize uniqueNavigationNodes;
86@synthesize index;
87@synthesize lastMarker;
88@synthesize calls;
89@synthesize e;
90@synthesize currentSymbol;
91
92+ (BufferedTreeNodeStream *) newBufferedTreeNodeStream:(CommonTree *) aTree
93{
94    return [((BufferedTreeNodeStream *)[BufferedTreeNodeStream alloc]) initWithTree:(CommonTree *)aTree];
95}
96
97+ (BufferedTreeNodeStream *) newBufferedTreeNodeStream:(id<TreeAdaptor>)adaptor Tree:(CommonTree *)aTree
98{
99    return [[BufferedTreeNodeStream alloc] initWithTreeAdaptor:adaptor Tree:(CommonTree *)aTree];
100}
101
102+ (BufferedTreeNodeStream *) newBufferedTreeNodeStream:(id<TreeAdaptor>)adaptor Tree:(CommonTree *)aTree withBufferSize:(NSInteger)initialBufferSize
103{
104    return [[BufferedTreeNodeStream alloc] initWithTreeAdaptor:adaptor Tree:(CommonTree *)aTree WithBufferSize:initialBufferSize];
105}
106
107-(BufferedTreeNodeStream *) init
108{
109	self = [super init];
110	if (self) {
111		index = -1;
112		uniqueNavigationNodes = NO;
113        root = [[CommonTree alloc] init];
114        //		tokens = tree;
115        adaptor = [[[CommonTreeAdaptor alloc] init] retain];
116        nodes = [[AMutableArray arrayWithCapacity:DEFAULT_INITIAL_BUFFER_SIZE] retain];
117        down = [[adaptor createTree:TokenTypeDOWN Text:@"DOWN"] retain];
118        up = [[adaptor createTree:TokenTypeUP Text:@"UP"] retain];
119        eof = [[adaptor createTree:TokenTypeEOF Text:@"EOF"] retain];
120    }
121	return self;
122}
123
124- (BufferedTreeNodeStream *)initWithTree:(CommonTree *) aTree
125{
126	self = [super init];
127	if (self) {
128		index = -1;
129		uniqueNavigationNodes = NO;
130        root = aTree;
131        //		tokens = aTree;
132        adaptor = [[[CommonTreeAdaptor alloc] init] retain];
133        nodes = [[AMutableArray arrayWithCapacity:DEFAULT_INITIAL_BUFFER_SIZE] retain];
134        down = [[adaptor createTree:TokenTypeDOWN Text:@"DOWN"] retain];
135        up = [[adaptor createTree:TokenTypeUP Text:@"UP"] retain];
136        eof = [[adaptor createTree:TokenTypeEOF Text:@"EOF"] retain];
137    }
138	return self;
139}
140
141-(BufferedTreeNodeStream *) initWithTreeAdaptor:(CommonTreeAdaptor *)anAdaptor Tree:(CommonTree *)aTree
142{
143	self = [super init];
144	if (self) {
145		index = -1;
146		uniqueNavigationNodes = NO;
147        root = aTree;
148        //		tokens = aTree;
149        adaptor = [anAdaptor retain];
150        nodes = [[AMutableArray arrayWithCapacity:DEFAULT_INITIAL_BUFFER_SIZE] retain];
151        down = [[adaptor createTree:TokenTypeDOWN Text:@"DOWN"] retain];
152        up = [[adaptor createTree:TokenTypeUP Text:@"UP"] retain];
153        eof = [[adaptor createTree:TokenTypeEOF Text:@"EOF"] retain];
154    }
155	return self;
156}
157
158-(BufferedTreeNodeStream *) initWithTreeAdaptor:(CommonTreeAdaptor *)anAdaptor Tree:(CommonTree *)aTree WithBufferSize:(NSInteger)bufferSize
159{
160	self = [super init];
161	if (self) {
162        //		down = [adaptor createToken:TokenTypeDOWN withText:@"DOWN"];
163        //		up = [adaptor createToken:TokenTypeDOWN withText:@"UP"];
164        //		eof = [adaptor createToken:TokenTypeDOWN withText:@"EOF"];
165		index = -1;
166		uniqueNavigationNodes = NO;
167        root = aTree;
168        //		tokens = aTree;
169        adaptor = [anAdaptor retain];
170        nodes = [[AMutableArray arrayWithCapacity:bufferSize] retain];
171        down = [[adaptor createTree:TokenTypeDOWN Text:@"DOWN"] retain];
172        up = [[adaptor createTree:TokenTypeUP Text:@"UP"] retain];
173        eof = [[adaptor createTree:TokenTypeEOF Text:@"EOF"] retain];
174	}
175	return self;
176}
177
178- (void)dealloc
179{
180#ifdef DEBUG_DEALLOC
181    NSLog( @"called dealloc in BufferedTreeNodeStream" );
182#endif
183    if ( adaptor ) [adaptor release];
184    if ( nodes ) [nodes release];
185    if ( root ) [root release];
186    if ( down ) [down release];
187    if ( up ) [up release];
188    if ( eof ) [eof release];
189	[super dealloc];
190}
191
192- (id) copyWithZone:(NSZone *)aZone
193{
194    BufferedTreeNodeStream *copy;
195
196    copy = [[[self class] allocWithZone:aZone] init];
197    if ( up )
198        copy.up = [up copyWithZone:aZone];
199    if ( down )
200        copy.down = [down copyWithZone:aZone];
201    if ( eof )
202        copy.eof = [eof copyWithZone:aZone];
203    if ( nodes )
204        copy.nodes = [nodes copyWithZone:aZone];
205    if ( root )
206        copy.root = [root copyWithZone:aZone];
207    if ( tokens )
208        copy.tokens = [tokens copyWithZone:aZone];
209    if ( adaptor )
210        copy.adaptor = [adaptor copyWithZone:aZone];
211    copy.uniqueNavigationNodes = self.uniqueNavigationNodes;
212    copy.index = self.index;
213    copy.lastMarker = self.lastMarker;
214    if ( calls )
215        copy.calls = [calls copyWithZone:aZone];
216    return copy;
217}
218
219// protected methods. DO NOT USE
220#pragma mark Protected Methods
221-(void) fillBuffer
222{
223	[self fillBufferWithTree:root];
224	// if (debug > 1) NSLog("revIndex=%@", tokenTypeToStreamIndexesMap);
225	index = 0; // buffer of nodes intialized now
226}
227
228-(void) fillBufferWithTree:(CommonTree *) aTree
229{
230	BOOL empty = [adaptor isNil:(id<BaseTree>)aTree];
231	if (!empty) {
232		[nodes addObject:aTree];
233	}
234	NSInteger n = [adaptor getChildCount:aTree];
235	if (!empty && n > 0) {
236		[self addNavigationNode:TokenTypeDOWN];
237	}
238	for (NSInteger c = 0; c < n; c++) {
239		id child = [adaptor getChild:aTree At:c];
240		[self fillBufferWithTree:child];
241	}
242	if (!empty && n > 0) {
243		[self addNavigationNode:TokenTypeUP];
244	}
245}
246
247-(NSInteger) getNodeIndex:(CommonTree *) node
248{
249	if (index == -1) {
250		[self fillBuffer];
251	}
252	for (NSUInteger i = 0; i < [nodes count]; i++) {
253		id t = [nodes objectAtIndex:i];
254		if (t == node) {
255			return i;
256		}
257	}
258	return -1;
259}
260
261-(void) addNavigationNode:(NSInteger) type
262{
263	id navNode = nil;
264	if (type == TokenTypeDOWN) {
265		if (self.uniqueNavigationNodes) {
266			navNode = [adaptor createToken:TokenTypeDOWN Text:@"DOWN"];
267		}
268		else {
269			navNode = down;
270		}
271
272	}
273	else {
274		if (self.uniqueNavigationNodes) {
275			navNode = [adaptor createToken:TokenTypeUP Text:@"UP"];
276		}
277		else {
278			navNode = up;
279		}
280	}
281	[nodes addObject:navNode];
282}
283
284-(id) get:(NSUInteger) i
285{
286	if (index == -1) {
287		[self fillBuffer];
288	}
289	return [nodes objectAtIndex:i];
290}
291
292-(id) LT:(NSInteger) k
293{
294	if (index == -1) {
295		[self fillBuffer];
296	}
297	if (k == 0) {
298		return nil;
299	}
300	if (k < 0) {
301		return [self LB:-k];
302	}
303	if ((index + k - 1) >= [nodes count]) {
304		return eof;
305	}
306	return [nodes objectAtIndex:(index + k - 1)];
307}
308
309-(id) getCurrentSymbol
310{
311	return [self LT:1];
312}
313
314-(id) LB:(NSInteger) k
315{
316	if (k == 0) {
317		return nil;
318	}
319	if ((index - k) < 0) {
320		return nil;
321	}
322	return [nodes objectAtIndex:(index - k)];
323}
324
325- (CommonTree *)getTreeSource
326{
327    return root;
328}
329
330-(NSString *)getSourceName
331{
332	return [[self getTokenStream] getSourceName];
333}
334
335- (id<TokenStream>)getTokenStream
336{
337    return tokens;
338}
339
340- (void) setTokenStream:(id<TokenStream>)newtokens
341{
342    tokens = newtokens;
343}
344
345- (id<TreeAdaptor>)getTreeAdaptor
346{
347    return adaptor;
348}
349
350- (void) setTreeAdaptor:(id<TreeAdaptor>)anAdaptor
351{
352    adaptor = anAdaptor;
353}
354
355- (BOOL)getUniqueNavigationNodes
356{
357    return uniqueNavigationNodes;
358}
359
360- (void) setUniqueNavigationNodes:(BOOL)aVal
361{
362    uniqueNavigationNodes = aVal;
363}
364
365-(void) consume
366{
367	if (index == -1) {
368		[self fillBuffer];
369	}
370	index++;
371}
372
373-(NSInteger) LA:(NSInteger) i
374{
375	return [adaptor getType:[self LT:i]];
376}
377
378-(NSInteger) mark
379{
380	if (index == -1) {
381		[self fillBuffer];
382	}
383	lastMarker = self.index;
384	return lastMarker;
385}
386
387-(void) release:(NSInteger) marker
388{
389	// do nothing
390}
391
392-(void) rewind:(NSInteger) marker
393{
394	[self seek:marker];
395}
396
397-(void) rewind
398{
399	[self seek:lastMarker];
400}
401
402-(void) seek:(NSInteger) i
403{
404	if (index == -1) {
405		[self fillBuffer];
406	}
407	index = i;
408}
409
410-(void) push:(NSInteger) i
411{
412	if (calls == nil) {
413		calls = [IntArray newArrayWithLen:INITIAL_CALL_STACK_SIZE];
414	}
415	[calls push:index];
416	[self seek:i];
417}
418
419-(NSInteger) pop
420{
421	NSInteger ret = [calls pop];
422	[self seek:ret];
423	return ret;
424}
425
426-(void) reset
427{
428	index = 0;
429	lastMarker = 0;
430	if (calls != nil) {
431		[calls reset];
432	}
433}
434
435-(NSUInteger) count
436{
437	if (index == -1) {
438		[self fillBuffer];
439	}
440	return [nodes count];
441}
442
443-(NSUInteger) size
444{
445	return [self count];
446}
447
448-(NSEnumerator *) objectEnumerator
449{
450	if (e == nil) {
451		e = [[StreamEnumerator alloc] initWithNodes:nodes andEOF:eof];
452	}
453	return e;
454}
455
456-(void) replaceChildren:(CommonTree *) parent From:(NSInteger)startIdx To:(NSInteger)stopIdx With:(CommonTree *)aTree
457{
458	if (parent != nil) {
459		[adaptor replaceChildren:parent From:startIdx To:stopIdx With:aTree];
460	}
461}
462
463-(NSString *) toTokenTypeString
464{
465	if (index == -1)
466	{
467		[self fillBuffer];
468	}
469	NSMutableString *buf = [NSMutableString stringWithCapacity:10];
470	for (NSUInteger i= 0; i < [nodes count]; i++) {
471		CommonTree * aTree = (CommonTree *)[self get:i];
472		[buf appendFormat:@" %d", [adaptor getType:aTree]];
473	}
474	return buf;
475}
476
477-(NSString *) toTokenString:(NSInteger)aStart ToEnd:(NSInteger)aStop
478{
479	if (index == -1) {
480		[self fillBuffer];
481	}
482	NSMutableString *buf = [NSMutableString stringWithCapacity:10];
483	for (NSUInteger i = aStart; i < [nodes count] && i <= aStop; i++) {
484		CommonTree * t = (CommonTree *)[self get:i];
485		[buf appendFormat:@" %d", [adaptor getType:t]];
486	}
487	return buf;
488}
489
490-(NSString *) toStringFromNode:(id)aStart ToNode:(id)aStop
491{
492	if (aStart == nil || aStop == nil) {
493		return nil;
494	}
495	if (index == -1) {
496		[self fillBuffer];
497	}
498
499	// if we have a token stream, use that to dump text in order
500	if ([self getTokenStream] != nil) {
501		NSInteger beginTokenIndex = [adaptor getTokenStartIndex:aStart];
502		NSInteger endTokenIndex = [adaptor getTokenStopIndex:aStop];
503
504		if ([adaptor getType:aStop] == TokenTypeUP) {
505			endTokenIndex = [adaptor getTokenStopIndex:aStart];
506		}
507		else if ([adaptor getType:aStop] == TokenTypeEOF) {
508			endTokenIndex = [self count] - 2; //don't use EOF
509		}
510        [tokens toStringFromStart:beginTokenIndex ToEnd:endTokenIndex];
511	}
512	// walk nodes looking for aStart
513	CommonTree * aTree = nil;
514	NSUInteger i = 0;
515	for (; i < [nodes count]; i++) {
516		aTree = [nodes objectAtIndex:i];
517		if (aTree == aStart) {
518			break;
519		}
520	}
521	NSMutableString *buf = [NSMutableString stringWithCapacity:10];
522	aTree = [nodes objectAtIndex:i]; // why?
523	while (aTree != aStop) {
524		NSString *text = [adaptor getText:aTree];
525		if (text == nil) {
526			text = [NSString stringWithFormat:@" %d", [adaptor getType:aTree]];
527		}
528		[buf appendString:text];
529		i++;
530		aTree = [nodes objectAtIndex:i];
531	}
532	NSString *text = [adaptor getText:aStop];
533	if (text == nil) {
534		text = [NSString stringWithFormat:@" %d", [adaptor getType:aStop]];
535	}
536	[buf appendString:text];
537	return buf;
538}
539
540// getters and setters
541- (AMutableArray *) getNodes
542{
543    return nodes;
544}
545
546- (id) eof
547{
548    return eof;
549}
550
551- (void) setEof:(id)theEOF
552{
553    eof = theEOF;
554}
555
556@end
557