Debug.java 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. /* Copyright (C) 2004 Free Software Foundation
  2. This file is part of libgcj.
  3. This software is copyrighted work licensed under the terms of the
  4. Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
  5. details. */
  6. /* Utility methods that allow an object to be converted to a textual
  7. representation on an OutputStream. The intention here is that this
  8. class be used for debugging, so we provide information about all
  9. fields, public or otherwise. */
  10. package gnu.gcj.util;
  11. import java.lang.reflect.*;
  12. import java.io.*;
  13. import java.util.*;
  14. class Debug
  15. {
  16. private final PrintStream p;
  17. private final int maxdepth;
  18. private final int maxArrayLength;
  19. private final boolean printStaticFields;
  20. private int depth;
  21. Debug(PrintStream writer, int maxdepth, int maxArrayLength, boolean printStaticFields)
  22. {
  23. p = writer;
  24. this.maxdepth = maxdepth;
  25. this.maxArrayLength = maxArrayLength;
  26. this.printStaticFields = printStaticFields;
  27. }
  28. Debug(PrintStream writer)
  29. {
  30. this(writer, 0, 10, false);
  31. }
  32. Debug(int maxdepth, boolean printStaticFields)
  33. {
  34. this(new PrintStream
  35. (new FileOutputStream(FileDescriptor.err), true),
  36. maxdepth,
  37. maxdepth > 0 ? 1000 : 10, printStaticFields);
  38. }
  39. Debug(int maxdepth)
  40. {
  41. this(maxdepth, false);
  42. }
  43. Debug()
  44. {
  45. this(0, false);
  46. }
  47. private final void indent()
  48. {
  49. for (int i = 0; i < depth; i++)
  50. p.print(" ");
  51. }
  52. private final java.util.IdentityHashMap h =
  53. new java.util.IdentityHashMap();
  54. private static native Field[] getDeclaredFields(Class c);
  55. private static native Object getField(Object o, Field f);
  56. private static native long getAddr(Object o);
  57. // Return an array containing all the fields of a class and its
  58. // superclasses.
  59. private Field[] internalGetFields(Class c)
  60. {
  61. HashSet set = new HashSet();
  62. set.addAll(Arrays.asList(getDeclaredFields(c)));
  63. Class[] interfaces = c.getInterfaces();
  64. for (int i = 0; i < interfaces.length; i++)
  65. set.addAll(Arrays.asList(internalGetFields(interfaces[i])));
  66. Class superClass = c.getSuperclass();
  67. if (superClass != null)
  68. set.addAll(Arrays.asList(internalGetFields(superClass)));
  69. return (Field[])set.toArray(new Field[set.size()]);
  70. }
  71. // FIXME: We could just use getClass() here, but this is a
  72. // workaround for a C++ bug that is causing getClass() to be
  73. // miscompiled.
  74. static private Class getItsClass(Object O)
  75. {
  76. return O.getClass();
  77. }
  78. // Print a reasonably readable textual representation of an object
  79. // on our OutputStream. Objects are only printed once, no matter
  80. // how many references point to them.
  81. private void print(Object O)
  82. {
  83. int savedDepth = depth;
  84. h.put(O, O);
  85. try
  86. {
  87. Class C = getItsClass(O);
  88. p.print(C.getName() + "@");
  89. p.println(Long.toHexString(getAddr(O)));
  90. if (C.isArray())
  91. {
  92. indent(); p.println("{");
  93. depth++;
  94. indent();
  95. C = C.getComponentType();
  96. int len = Array.getLength(O);
  97. for (int i = 0; i < len; i++)
  98. {
  99. Object thing = Array.get(O, i);
  100. print0(thing, C);
  101. p.print(", ");
  102. if (i > maxArrayLength)
  103. {
  104. p.print("...");
  105. break;
  106. }
  107. }
  108. depth--;
  109. p.println();
  110. indent(); p.print("}");
  111. return;
  112. }
  113. indent(); p.println("{");
  114. depth++;
  115. if (C == java.lang.Class.class)
  116. {
  117. indent();
  118. p.println ("class = " + O.toString() + ",");
  119. }
  120. else if (C == java.lang.reflect.Field.class)
  121. {
  122. indent();
  123. p.println ("<field> = \"" + O.toString() + "\",");
  124. }
  125. else if (C == java.lang.String.class)
  126. {
  127. indent();
  128. p.println ("<string> = \"" + O.toString() + "\",");
  129. }
  130. Field[] f = internalGetFields(C);
  131. for (int i = 0; i < f.length; i++)
  132. {
  133. Class type = f[i].getType();
  134. boolean isStatic = (f[i].getModifiers() & Modifier.STATIC) != 0;
  135. if (isStatic && ! printStaticFields)
  136. continue;
  137. indent();
  138. if (isStatic)
  139. p.print("static ");
  140. p.print(type.getName() +" " +f[i].getName() + " = ");
  141. Object thing = getField(O, f[i]);
  142. print0(thing, type);
  143. p.println(",");
  144. }
  145. depth--;
  146. indent(); p.print("}");
  147. }
  148. catch (Throwable t)
  149. {
  150. p.print("error: 0x" + Long.toHexString(getAddr(O)) + ";");
  151. depth = savedDepth;
  152. }
  153. }
  154. private void print0(Object thing, Class C)
  155. {
  156. try
  157. {
  158. if (thing == null)
  159. {
  160. p.print("null");
  161. return;
  162. }
  163. else if (C == gnu.gcj.RawData.class ||
  164. C == gnu.gcj.RawDataManaged.class)
  165. {
  166. }
  167. else if (C.isPrimitive())
  168. {
  169. if (getItsClass(thing) == Character.class)
  170. p.print("'" + thing + "'");
  171. else
  172. p.print(thing);
  173. return;
  174. }
  175. else if (getItsClass(thing) == String.class)
  176. {
  177. p.print("\"" + thing + "\"");
  178. return;
  179. }
  180. else if (depth < maxdepth && h.get(thing) == null)
  181. {
  182. depth++;
  183. print(thing);
  184. depth--;
  185. return;
  186. }
  187. }
  188. catch (Throwable t)
  189. {
  190. }
  191. // The default action: just print the address.
  192. p.print("0x"+ Long.toHexString(getAddr(thing)));
  193. }
  194. // Print the textual representation of an object on System.err.
  195. public void write(Object O)
  196. {
  197. depth = 0;
  198. print(O);
  199. p.flush();
  200. }
  201. }