-
Notifications
You must be signed in to change notification settings - Fork 182
Expand file tree
/
Copy pathCallStack.java
More file actions
141 lines (123 loc) · 5.05 KB
/
CallStack.java
File metadata and controls
141 lines (123 loc) · 5.05 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
/*****************************************************************************
* Licensed to the Apache Software Foundation (ASF) under one *
* or more contributor license agreements. See the NOTICE file *
* distributed with this work for additional information *
* regarding copyright ownership. The ASF licenses this file *
* to you under the Apache License, Version 2.0 (the *
* "License"); you may not use this file except in compliance *
* with the License. You may obtain a copy of the License at *
* *
* http://www.apache.org/licenses/LICENSE-2.0 *
* *
* Unless required by applicable law or agreed to in writing, *
* software distributed under the License is distributed on an *
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY *
* KIND, either express or implied. See the License for the *
* specific language governing permissions and limitations *
* under the License. *
* *
* *
* This file is part of the BeanShell Java Scripting distribution. *
* Documentation and updates may be found at http://www.beanshell.org/ *
* Patrick Niemeyer ([email protected]) *
* Author of Learning Java, O'Reilly & Associates *
* *
*****************************************************************************/
package bsh;
import java.util.Stack;
import java.io.Serializable;
import java.util.EmptyStackException;
/**
A stack of NameSpaces representing the call path.
Each method invocation, for example, pushes a new NameSpace onto the stack.
The top of the stack is always the current namespace of evaluation.
<p>
This is used to support the this.caller magic reference and to print
script "stack traces" when evaluation errors occur.
<p>
Note: How can this be thread safe, you might ask? Wouldn't a thread
executing various beanshell methods be mutating the callstack? Don't we
need one CallStack per Thread in the interpreter? The answer is that we do.
Any java.lang.Thread enters our script via an external (hard) Java
reference via a This type interface, e.g. the Runnable interface
implemented by This or an arbitrary interface implemented by XThis.
In that case the This invokeMethod() method (called by any interface that
it exposes) creates a new CallStack for each external call.
<p>
*/
public final class CallStack implements Serializable {
/** default serial version id */
private static final long serialVersionUID = 1L;
private final Stack<NameSpace> stack = new Stack<>();
public CallStack() { }
public CallStack( NameSpace namespace ) {
push( namespace );
}
public void clear() {
stack.clear();
}
public void push( NameSpace ns ) {
stack.push( ns );
}
public NameSpace top() {
return stack.peek();
}
/**
zero based.
*/
public NameSpace get(int depth) {
int size = stack.size();
if ( depth >= size )
return NameSpace.JAVACODE;
return stack.toArray(new NameSpace[size])[size-1-depth];
}
/**
This is kind of crazy, but used by the setNameSpace command.
zero based.
*/
public synchronized void set(int depth, NameSpace ns) {
stack.set( stack.size()-1-depth, ns );
}
public NameSpace pop() {
try {
return stack.pop();
} catch(EmptyStackException e) {
throw new InterpreterError("pop on empty CallStack");
}
}
/**
Swap in the value as the new top of the stack and return the old
value.
*/
public NameSpace swap( NameSpace newTop ) {
NameSpace oldTop = stack.pop();
stack.push(newTop);
return oldTop;
}
public int depth() {
return stack.size();
}
/*
public NameSpace [] toArray() {
NameSpace [] nsa = new NameSpace [ depth() ];
stack.copyInto( nsa );
return nsa;
}
*/
public synchronized String toString() {
StringBuilder sb = new StringBuilder();
sb.append("CallStack:\n");
for( int i=stack.size()-1; i>=0; i-- )
sb.append("\t"+stack.get(i)+"\n");
return sb.toString();
}
/**
Occasionally we need to freeze the callstack for error reporting
purposes, etc.
*/
public CallStack copy() {
CallStack cs = new CallStack();
cs.stack.addAll(this.stack);
return cs;
}
}