The wait algorithm is turning out to be trickier than expected. I think I have answers for the two questions I posed a few days ago. Let’s go in reverse order, actually:
- When one thread has to wait inside
compactWait
, which is a synchronized method, can other threads even enter it?
I’m almost embarrassed I had to ask this. Of course, other threads cannot enter compactWait
at the same time. The current thread relinquishes control of the object it is waiting on (either the one in the wait array or the one for a new buffer), but it doesn’t reliquish control of all monitors it has. This, of course, means that the entire method cannot be synchronized. Certain portions, however, do have to be synchronized, as explained below.
- When threads are notified that a new buffer has been loaded, they might all work with the list of sync points simultaneously. This might create race conditions. Are these important?
It’s true that the sync points are unique: There will (or at least should) never be two threads looking for the same sync point. However, all threads are manipulating the indices into the sync point and wait arrays concurrently, so I think this actually is an issue.
When threads resume after waiting for a new buffer, they should not concurrently run through the method; otherwise, they might, for example, dispatch a thread that’s waiting twice, or attempt to do so but cause a NullPointerException
because the entry in the wait array was non-null, but another thread made it null before the first thread could call notify()
. Another really important thing is that buffer reloads are synchronized; otherwise, two threads could concurrently try to reload the buffer, and nasty things would happen.
This pretty much leads to the follwing structure:
- 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, reload buffer and notify all threads waiting for a buffer update.
- If there’s a thread waiting for this wait array entry, notify it and continue with the next sync point.
- If the current sync point in the array is the right one…
- Advance indices.
- Allow the thread to exit the “RETRY” loop.
- Otherwise…
- Look for the sync points in the remaining part of the array.
- If it could be found…
- Insert a
new Object()
into 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.
- 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).
- Begin of synchronized block
- …until the thread is allowed to exit the loop (end “RETRY” loop).
The most important change here is that the entire method is not synchronized anymore, but the entire block that does notification and scanning is. The waiting happens outside the synchronized block. This allows other threads to run through the block when threads are waiting.
I’m writing a piece of code that simulates the behavior outside of the actual program right now. I should probably write unit test, hm?