File: v1_design.xml

package info (click to toggle)
libgroboutils-java 5-3
  • links: PTS, VCS
  • area: main
  • in suites: stretch
  • size: 9,396 kB
  • ctags: 11,186
  • sloc: java: 59,748; xml: 12,762; sh: 377; perl: 104; makefile: 20
file content (207 lines) | stat: -rw-r--r-- 8,676 bytes parent folder | download | duplicates (3)
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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
<?xml version="1.0" encoding="ISO-8859-1"?>
<document>
<head>
    <name>Code Coverage Design (V1)</name>
    <doc-version>$Date: 2003/05/04 06:40:10 $</doc-version>
    <author>Matt Albrecht</author>
</head>
<body>

<section>Overview</section>
<P>
The goal of the codecoverage package is to capture the lines of code that were
executed in a Java VM.  Thanks to the JPDA (Java Platform Debugger
Architecture), Java code can now monitor Java VMs to discover which lines
of code were executed.
</P>
<P>
This package breaks the general code-coverage problem into three parts:
<UL>
    <LI><B>tracker:</B> tracks the remote VM as the code is executed. This
        is written to a file or other data source for later parsing. </LI>
    <LI><B>counter:</B> discovers the actual number of lines in each
        class. This output can be written to a file or other data source
        for later parsing. </LI>
    <LI><B>collator:</B> parses the tracker and counter output, and creates
        output which displays the actual number of lines, and those which
        were covered.  This depends upon the other two parts.</LI>
</UL>
</P>
<P>
This package <i>must</i> be classloader aware.  That is, multiple classes with
the same name may be loaded in a single VM.  Currently, this is handled by
calculating the CRC of each class.  This not only detects different class
files, but also allows the package to detect if there are differences between
a tracked class and a counted class.
</P>

<section>Definitions</section>

<P>
As the code coverage package deals with multiple VMs, I need to establish some
lingo such that others, as well as myself, may understand more clearly the
ideas I am attempting to state.  First, I will state the common design.
Then, I will assign definitions to the entities in the design.
</P>
<P>
The general flow of the application is:
    <OL>
        <LI>
        Either the code coverage tool or another tool starts a Java Virtual
        Machine (VM), which is listening for Java Debug Wire Protocol
        (JDWP) connections; this VM will execute the Java bytecode which
        will be analyzed for the code it executes.  Let's call this the
        <em>Remote VM</em>, since it is remote to the code coverage tool.
        </LI>
        <LI>
        The code coverage tool is executed within another Virtual Machine.
        Since this document analyzes the process flow from the perspective
        of the code coverage tool, let us call this the <em>Local VM</em>.
        </LI>
    </OL>
</P>

<section>Current Design Assumptions</section>

<P>The current design assumes that the bytecode retrieved by BCEL is identical
to the bytecode that JPDA returns.  If this assumption fails, then the class
checksums won't match, causing the actual class lines not to be found in the
counter.</P>

<P>Recent experiments found that this assumption is wrong.  There is a temporary
work-around (via ignoring checksums during collation), however a better
solution will need to be found.</P>

<section>Current Poorly Designed Parts</section>

<P>
The IMethodLinesRecord and IClassLinesRecord both merge the marshalling and
unmarshalling code together.  It is useful in that one can observe from within
the same file that one's logic is the reverse of the other.  However, this
is a poor design choice in that a code section can't tell if a record can
read or write.  This needs to be split apart (however, the marshall/unmarshall
code can remain in the same class, since it will be implementing interfaces).
</P>

<section>Dealing with JPDA Problems</section>

<P>
I have found JPDA to be incredibly flakey as of JDK 1.4:
    <OL>
        <LI>
        There is a chance, on entry to any native code, that inspection of the
        StepEvent encountered by the Local VM will cause the Remote VM
        JPDA implementation will not understand the stack, causing a VM Death
        in the Remote VM.
        <P>
        I have discovered this to occur commonly in calls to
        <tt>System.exit( int )</tt>, but it has also occured within
        java.util.jar.Attributes$Name.isValid:
<source>
     [java] FATAL ERROR in native method: JDWP "stepControl.c" (Feb  7 2002), li
ne 152: Unable to get frame location, error code = 32 (JVMDI_ERROR_OPAQUE_FRAME)

     [java]
     [java]     at java.util.jar.Attributes$Name.isValid(Attributes.java:410)
     [java]     at java.util.jar.Attributes$Name.isValid(Attributes.java:402)
     [java]     at java.util.jar.Attributes$Name.&lt;init&gt;(Attributes.java:390)
     [java]     at java.util.jar.Attributes.putValue(Attributes.java:144)
     [java]     at java.util.jar.Attributes.read(Attributes.java:362)
     [java]     at java.util.jar.Manifest.read(Manifest.java:162)
     [java]     at java.util.jar.Manifest.&lt;init&gt;(Manifest.java:52)
     [java]     at java.util.jar.JarFile.getManifest(JarFile.java:147)
     [java]     at sun.misc.URLClassPath$JarLoader.getClassPath(URLClassPath.jav
a:712)
     [java]     at sun.misc.URLClassPath.getLoader(URLClassPath.java:224)
     [java]     - locked &lt;02A937C8&gt; (a sun.misc.URLClassPath)
     [java]     at sun.misc.URLClassPath.getResource(URLClassPath.java:134)
     [java]     at sun.misc.URLClassPath.getResource(URLClassPath.java:144)
     [java]     at java.lang.ClassLoader.getBootstrapResource(ClassLoader.java:8
80)
     [java]     at java.lang.ClassLoader.getResource(ClassLoader.java:784)
     [java]     at java.lang.ClassLoader.getResource(ClassLoader.java:782)
     [java]     at java.lang.ClassLoader.getResourceAsStream(ClassLoader.java:94
8)
     [java]     at java.lang.Class.getResourceAsStream(Class.java:1284)
     [java]     at org.apache.tools.ant.Main.getAntVersion(Main.java:564)
     [java]     - locked &lt;06B0E6F8&gt; (a java.lang.Class)
     [java]     at org.apache.tools.ant.Main.printVersion(Main.java:555)
     [java]     at org.apache.tools.ant.Main.&lt;init&gt;(Main.java:200)
     [java]     at org.apache.tools.ant.Main.start(Main.java:138)
     [java]     at org.apache.tools.ant.Main.main(Main.java:176)
     [java] Java Result: 1
     [echo] Stopping JUnit.
</source>
        </P>
        </LI>
        <LI>
        There are times when the JPDA events received by the Local VM from
        the Remote VM do not properly portray the suspended state of the
        Remote VM (this may actually be a result of insufficiently detailed
        documentation describing what events actually suspend), causing
        the Local VM to not receive events.  
        </LI>
        <LI>
        At times, the <tt>StepRequest.setEnabled( true )</tt> method will
        hang.  This is currently under investigation.
        </LI>
        <LI>
        The "includes" and "excludes" methods for StepRequests fail to
        work as described.
        </LI>
    </OL>
Here are the current work-arounds for each part:
    <OL>
        <LI>
        The exception is trapped, and interpreted to mean the equivalent
        of a VM Death notice.  <em>Not implemented yet.</em>
        </LI>
        <LI>
        If no events have been received over a period of time, then
        attempts will be made to resume each thread in the Remote VM.
        "Over a period of time" means that the pull events method does not
        return any events after a specified number of attempts, and each
        attempt with a certain time-out period.
        </LI>
        <LI>
        A proposed solution will execute the <tt>setEnabled</tt> call in
        a separate thread, and kill that thread if it does not return
        after a period of several seconds.  <em>Not implemented yet.</em>
        </LI>
    </OL>
In the future, these should be abstracted out to per-JPDA implementation
classes for a better design.
</P>

<section>JPDA Bugs of Interest</section>

<UL>
    <LI>
    <javabug bugid="4490152" /> - JPDA: vm-wide suspend does not suspend all
    threads (zombie, dead, new, ...)
    <P>
In all likelyhood the thread in question is
a zombie thread.
</P>
<P>
HotSpot will return a thread status of unknown and
a zero suspend status for threads it doesn't know about
(which includes threads that haven't run yet, threads being
initialized or threads that have completed running).
HotSpot will return a thread status of zombie and
a zero suspend status for threads being shutdown.
The code of ThreadReference has code in it to do it's own
accounting for suspends of zombies (this code is currently
broken).
</P>
<P>
It may be that the correct approach is to spec in all
interfaces (JVMDI/JDWP/JDI) that new/zombie threads cannot
be suspended (or "may" be incapable of being suspended).
</P>
    </LI> 
</UL>

</body>
</document>