• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Licensed to the Apache Software Foundation (ASF) under one or more
3  * contributor license agreements.  See the NOTICE file distributed with
4  * this work for additional information regarding copyright ownership.
5  * The ASF licenses this file to You under the Apache License, Version 2.0
6  * (the "License"); you may not use this file except in compliance with
7  * the License.  You may obtain a copy of the License at
8  *
9  *     http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  */
17 
18 package java.lang;
19 
20 import java.io.Serializable;
21 
22 /**
23  * A representation of a single stack frame. Arrays of {@code StackTraceElement}
24  * are stored in {@link Throwable} objects to represent the whole state of the
25  * call stack at the time a {@code Throwable} gets thrown.
26  *
27  * @see Throwable#getStackTrace()
28  * @since Android 1.0
29  */
30 public final class StackTraceElement implements Serializable {
31 
32     private static final long serialVersionUID = 6992337162326171013L;
33 
34     // BEGIN android-added
35     private static final int NATIVE_LINE_NUMBER = -2;
36     // END android-added
37 
38     String declaringClass;
39 
40     String methodName;
41 
42     String fileName;
43 
44     int lineNumber;
45 
46     /**
47      * Constructs a new {@code StackTraceElement} for a specified execution
48      * point.
49      *
50      * @param cls
51      *            the fully qualified name of the class where execution is at.
52      * @param method
53      *            the name of the method where execution is at.
54      * @param file
55      *            The name of the file where execution is at or {@code null}.
56      * @param line
57      *            the line of the file where execution is at, a negative number
58      *            if unknown or {@code -2} if the execution is in a native
59      *            method.
60      * @throws NullPointerException
61      *             if {@code cls} or {@code method} is {@code null}.
62      * @since Android 1.0
63      */
StackTraceElement(String cls, String method, String file, int line)64     public StackTraceElement(String cls, String method, String file, int line) {
65         super();
66         if (cls == null || method == null) {
67             throw new NullPointerException();
68         }
69         declaringClass = cls;
70         methodName = method;
71         fileName = file;
72         lineNumber = line;
73     }
74 
75     /**
76      * <p>
77      * Private, nullary constructor for VM use only.
78      * </p>
79      */
StackTraceElement()80     private StackTraceElement() {
81         super();
82     }
83 
84     /**
85      * Compares this instance with the specified object and indicates if they
86      * are equal. In order to be equal, the following conditions must be
87      * fulfilled:
88      * <ul>
89      * <li>{@code obj} must be a stack trace element,</li>
90      * <li>the method names of this stack trace element and of {@code obj} must
91      * not be {@code null},</li>
92      * <li>the class, method and file names as well as the line number of this
93      * stack trace element and of {@code obj} must be equal.</li>
94      * </ul>
95      *
96      * @param obj
97      *            the object to compare this instance with.
98      * @return {@code true} if the specified object is equal to this
99      *         {@code StackTraceElement}; {@code false} otherwise.
100      * @see #hashCode
101      * @since Android 1.0
102      */
103     @Override
equals(Object obj)104     public boolean equals(Object obj) {
105         if (!(obj instanceof StackTraceElement)) {
106             return false;
107         }
108         StackTraceElement castObj = (StackTraceElement) obj;
109 
110         /*
111          * Unknown methods are never equal to anything (not strictly to spec,
112          * but spec does not allow null method/class names)
113          */
114         if ((methodName == null) || (castObj.methodName == null)) {
115             return false;
116         }
117 
118         if (!getMethodName().equals(castObj.getMethodName())) {
119             return false;
120         }
121         if (!getClassName().equals(castObj.getClassName())) {
122             return false;
123         }
124         String localFileName = getFileName();
125         if (localFileName == null) {
126             if (castObj.getFileName() != null) {
127                 return false;
128             }
129         } else {
130             if (!localFileName.equals(castObj.getFileName())) {
131                 return false;
132             }
133         }
134         if (getLineNumber() != castObj.getLineNumber()) {
135             return false;
136         }
137 
138         return true;
139     }
140 
141     /**
142      * Returns the fully qualified name of the class belonging to this
143      * {@code StackTraceElement}.
144      *
145      * @return the fully qualified type name of the class
146      * @since Android 1.0
147      */
getClassName()148     public String getClassName() {
149         return (declaringClass == null) ? "<unknown class>" : declaringClass;
150     }
151 
152     /**
153      * Returns the name of the Java source file containing class belonging to
154      * this {@code StackTraceElement}.
155      *
156      * @return the name of the file, or {@code null} if this information is not
157      *         available.
158      * @since Android 1.0
159      */
getFileName()160     public String getFileName() {
161         return fileName;
162     }
163 
164     /**
165      * Returns the line number in the source for the class belonging to this
166      * {@code StackTraceElement}.
167      *
168      * @return the line number, or a negative number if this information is not
169      *         available.
170      * @since Android 1.0
171      */
getLineNumber()172     public int getLineNumber() {
173         return lineNumber;
174     }
175 
176     /**
177      * Returns the name of the method belonging to this {@code
178      * StackTraceElement}.
179      *
180      * @return the name of the method, or "<unknown method>" if this information
181      *         is not available.
182      * @since Android 1.0
183      */
getMethodName()184     public String getMethodName() {
185         return (methodName == null) ? "<unknown method>" : methodName;
186     }
187 
188     @Override
hashCode()189     public int hashCode() {
190         /*
191          * Either both methodName and declaringClass are null, or neither are
192          * null.
193          */
194         if (methodName == null) {
195             // all unknown methods hash the same
196             return 0;
197         }
198         // declaringClass never null if methodName is non-null
199         return methodName.hashCode() ^ declaringClass.hashCode();
200     }
201 
202     /**
203      * Indicates if the method name returned by {@link #getMethodName()} is
204      * implemented as a native method.
205      *
206      * @return {@code true} if the method in which this stack trace element is
207      *         executing is a native method; {@code false} otherwise.
208      * @since Android 1.0
209      */
isNativeMethod()210     public boolean isNativeMethod() {
211         // BEGIN android-changed
212         return lineNumber == NATIVE_LINE_NUMBER;
213         // END android-changed
214     }
215 
216     @Override
toString()217     public String toString() {
218         StringBuilder buf = new StringBuilder(80);
219 
220         buf.append(getClassName());
221         buf.append('.');
222         buf.append(getMethodName());
223 
224         if (isNativeMethod()) {
225             buf.append("(Native Method)");
226         } else {
227             String fName = getFileName();
228 
229             if (fName == null) {
230                 buf.append("(Unknown Source)");
231             } else {
232                 int lineNum = getLineNumber();
233 
234                 buf.append('(');
235                 buf.append(fName);
236                 if (lineNum >= 0) {
237                     buf.append(':');
238                     buf.append(lineNum);
239                 }
240                 buf.append(')');
241             }
242         }
243         return buf.toString();
244     }
245 }
246