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