JCStress arbiter question
Aleksey Shipilev
shade at redhat.com
Fri Jul 7 13:22:09 UTC 2017
On 06/20/2017 07:05 PM, Семухин Дмитрий wrote:
> @JCStressTest
> @Outcome(id = "0", expect = Expect.ACCEPTABLE_INTERESTING, desc = "Both
> results not visible")
> @Outcome(id = "1", expect = Expect.ACCEPTABLE_INTERESTING, desc = "Single
> result only visible")
> @Outcome(id = "2", expect = Expect.ACCEPTABLE, desc = "Normal outcome")
> @State
> public class UnsafeCountingFactorizerTest {
> private UnsafeCountingFactorizer toTest = new
> UnsafeCountingFactorizer();
>
> @Actor
> public void actor1() {
> toTest.service();
> }
>
> @Actor
> public void actor2() {
> toTest.service();
> }
>
> @Arbiter
> public void arbiter(J_Result r) {
> r.r1 = toTest.getCount();
> }
>
> public class UnsafeCountingFactorizer {
> private long count = 0;
>
> public long getCount() {
> return count;
> }
>
> public void service() {
> count++;
> }
> }
> }
>
> Test results are only outcome=1 and outcome=2 but outcome=0 is valid in
> JMM. Therefore I had a theory that threads are being reused during test
> operatins which has a chance for us not to see outcome=0. I decided to
> write this test:
Except that @Arbiter is specified as "All memory effects from {@link Actor}s are
visible in Arbiter". Which means outcome=0 is forbidden, because it will pick up
the state from one of the actors.
> Obviously it failed with forbidden outcome but I noticed the only 2 classes
> of outcome are x,y,x and x,y,y. This is one of the evidences that arbiter
> logic is executed in one of the threads already used by actors.
Yes, @Arbiter is executed by one of the @Actor threads. This is mostly done as
runtime optimization.
> 1)Am I doing something totally wrong?
Not really, but you think @Arbiter is weaker than it really is.
> 2)Was such behaviour encountered before?
See above.
> 3)If not - should I implement this or any other solution you can think of?
There seem to be contradictory goals here: make @Arbiter observe actor-set state
via the race, and make it happen after both actors.
The mechanics of getting @Arbiter to happen after the actors would preclude
races (mostly), be it running in the same thread, using a Thread.join, using
volatile vars to synchronize the access, etc. If you want something that runs
concurrently with @Actor-s, then it should be @Actor itself. Just running
@Arbiter in a separate thread would not help to get new behaviors.
Thanks,
-Aleksey
More information about the jcstress-dev
mailing list