I’ve just realized there’s a potential problem with this algorithm too: Assume the second to last sync point of the schedule is being processed, and a thread is waiting for it. This thread will be woken up, and the current thread will wait for the last sync point to be hit. However, since it’s the last sync point in the schedule, compactWait will never be entered again. The last thread will never complete!
This isn’t really a problem in this application, though, since the there are these eight sync points produced by “DestroyJavaVM†(thread 9) that don’t show up in the recorded trace. Thread 9 will enter compactWait and wake up the last thread. But now thread 9 will look for its sync point, and it’s not in the schedule. So it will wait for a buffer update… which will never come.
Therefore, when a thread is woken up so it can continue, it needs to check if the next entry in the sync point array marks the end of the schedule (SP_END). If so, replay needs to be disabled and all threads waiting for a buffer update need to be woken up.
Another problem arises, unfortunately: If it’s the end of the buffer, compactWait cannot determine if the end of the schedule has been reached without reloading the buffer. But reloading the buffer needs to happen in a synchronized block.
It’s probably the easiest to have another static flag, LASTBUFFER, that the master VM sets if this is the last block. compactWait will disable scheduled replay if the index is at the end of the buffer and the LASTBUFFER flag is set, or if the next sync point marks the end. Note that this “or” has to short-circuit.
So, without further ado, another sketch of the algorithm:
- Repeat this… (“RETRY” loop)
- Begin of synchronized block
- If the buffer has never been loaded, or if the index is at the end of the buffer…
- If the LASTBUFFER flag is set…
- Disable scheduled replay.
- Wake up all threads waiting for a buffer update.
- Break out of the “RETRY” loop.
 
- Otherwise…
- Load the buffer and reset the indices.
- Wake up all threads waiting for a buffer update.
 
 
- If the LASTBUFFER flag is set…
- If the current sync point marks the end of the schedule…
- Disable scheduled replay.
- Wake up all threads waiting for a buffer update.
- Break out of the “RETRY” loop.
 
- If there’s a thread waiting for this wait array entry…
- Notify it
- Set a flag to scan ahead for the current sync point (“SCAN”).
- Do not advance indices.
 
- Otherwise…
- If the current sync point in the array is the right one…
- Advance indices.
- Allow the thread to exit the “RETRY” loop.
 
- Otherwise…
- Set a flag to scan ahead (“SCAN”).
 
 
- If the current sync point in the array is the right one…
- If the “SCAN” flag is set (i.e. either a thread was woken up or the current sync point did not match)…
- Look for the sync points in the remaining part of the array.
- If it could be found…
- Insert a new Object()into corresponding slot of the wait array.
- Set that Object()as “wait object” for this method.
- Allow the thread to exit the “RETRY” loop.
 
- Insert a 
- Otherwise…
- Set the new buffer wait object as “wait object” for this method.
- Do not allow the thread to exit the “RETRY” loop.
 
 
 
- If the buffer has never been loaded, or if the index is at the end of the buffer…
- End of synchronized block
- If a “wait object” has been set for this method…
- Call wait()on that object (make sure to continue waiting if interrupted).
- If scheduled replay is still enabled and the thread is allowed to exit the “RETRY” loop (i.e. it was not a “new buffer” wait)…
- Advance the indices.
- If the index is at the end of the buffer and the LASTBUFFER flag is set, or the sync point at the new index marks the end of the schedule (note: this must short-circuit)…
- Disable scheduled replay.
- Wake up all threads waiting for a buffer update.
- Break out of the “RETRY” loop.
 
 
 
- Call 
 
- Begin of synchronized block
- …while the thread is not allowed to exit the loop and scheduled replay is enabled (end “RETRY” loop).
