Skip to content
This repository was archived by the owner on Aug 26, 2022. It is now read-only.

Commit 2b56ad2

Browse files
authored
Fixed race condition in periodic timers (#438)
1 parent 8517d39 commit 2b56ad2

File tree

3 files changed

+135
-1
lines changed

3 files changed

+135
-1
lines changed

Source/Core/Runtime/Timers/MachineTimer.cs

+7-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,13 @@ public MachineTimer(TimerInfo info, Machine owner)
4444
this.Owner = owner;
4545

4646
this.TimeoutEvent = new TimerElapsedEvent(this.Info);
47-
this.InternalTimer = new Timer(this.HandleTimeout, null, this.Info.DueTime, Timeout.InfiniteTimeSpan);
47+
48+
// To avoid a race condition between assigning the field of the timer
49+
// and HandleTimeout accessing the field before the assignment happens,
50+
// we first create a timer that cannot get triggered, then assign it to
51+
// the field, and finally we start the timer.
52+
this.InternalTimer = new Timer(this.HandleTimeout, null, Timeout.InfiniteTimeSpan, Timeout.InfiniteTimeSpan);
53+
this.InternalTimer.Change(this.Info.DueTime, Timeout.InfiniteTimeSpan);
4854
}
4955

5056
/// <summary>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
// ------------------------------------------------------------------------------------------------
2+
// Copyright (c) Microsoft Corporation. All rights reserved.
3+
// Licensed under the MIT License (MIT). See License.txt in the repo root for license information.
4+
// ------------------------------------------------------------------------------------------------
5+
6+
using System;
7+
using System.Collections.Generic;
8+
using System.Threading.Tasks;
9+
using Microsoft.PSharp.Timers;
10+
using Xunit;
11+
using Xunit.Abstractions;
12+
13+
namespace Microsoft.PSharp.Core.Tests
14+
{
15+
public class TimerStressTest : BaseTest
16+
{
17+
public TimerStressTest(ITestOutputHelper output)
18+
: base(output)
19+
{
20+
}
21+
22+
private class SetupEvent : Event
23+
{
24+
public TaskCompletionSource<bool> Tcs;
25+
26+
public SetupEvent(TaskCompletionSource<bool> tcs)
27+
{
28+
this.Tcs = tcs;
29+
}
30+
}
31+
32+
private class T1 : Machine
33+
{
34+
private TaskCompletionSource<bool> Tcs;
35+
36+
[Start]
37+
[OnEntry(nameof(InitOnEntry))]
38+
[OnEventDoAction(typeof(TimerElapsedEvent), nameof(HandleTimeout))]
39+
private class Init : MachineState
40+
{
41+
}
42+
43+
private void InitOnEntry()
44+
{
45+
this.Tcs = (this.ReceivedEvent as SetupEvent).Tcs;
46+
47+
// Start a regular timer.
48+
this.StartTimer(TimeSpan.FromTicks(1));
49+
}
50+
51+
private void HandleTimeout()
52+
{
53+
this.Tcs.SetResult(true);
54+
this.Raise(new Halt());
55+
}
56+
}
57+
58+
[Fact(Timeout= 6000)]
59+
public async Task TestTimerLifetime()
60+
{
61+
await this.RunAsync(async r =>
62+
{
63+
int numTimers = 1000;
64+
var awaiters = new Task[numTimers];
65+
for (int i = 0; i < numTimers; i++)
66+
{
67+
var tcs = new TaskCompletionSource<bool>();
68+
r.CreateMachine(typeof(T1), new SetupEvent(tcs));
69+
awaiters[i] = tcs.Task;
70+
}
71+
72+
Task task = Task.WhenAll(awaiters);
73+
await WaitAsync(task);
74+
});
75+
}
76+
77+
private class T2 : Machine
78+
{
79+
private TaskCompletionSource<bool> Tcs;
80+
private int Counter;
81+
82+
[Start]
83+
[OnEntry(nameof(InitOnEntry))]
84+
[OnEventDoAction(typeof(TimerElapsedEvent), nameof(HandleTimeout))]
85+
private class Init : MachineState
86+
{
87+
}
88+
89+
private void InitOnEntry()
90+
{
91+
this.Tcs = (this.ReceivedEvent as SetupEvent).Tcs;
92+
this.Counter = 0;
93+
94+
// Start a periodic timer.
95+
this.StartPeriodicTimer(TimeSpan.FromTicks(1), TimeSpan.FromTicks(1));
96+
}
97+
98+
private void HandleTimeout()
99+
{
100+
this.Counter++;
101+
if (this.Counter == 10)
102+
{
103+
this.Tcs.SetResult(true);
104+
this.Raise(new Halt());
105+
}
106+
}
107+
}
108+
109+
[Fact(Timeout = 6000)]
110+
public async Task TestPeriodicTimerLifetime()
111+
{
112+
await this.RunAsync(async r =>
113+
{
114+
int numTimers = 1000;
115+
var awaiters = new Task[numTimers];
116+
for (int i = 0; i < numTimers; i++)
117+
{
118+
var tcs = new TaskCompletionSource<bool>();
119+
r.CreateMachine(typeof(T2), new SetupEvent(tcs));
120+
awaiters[i] = tcs.Task;
121+
}
122+
123+
Task task = Task.WhenAll(awaiters);
124+
await WaitAsync(task);
125+
});
126+
}
127+
}
128+
}

0 commit comments

Comments
 (0)