Skip to content

Commit

Permalink
Merge pull request #26 from Timothy-LiuXuefeng/dev
Browse files Browse the repository at this point in the history
fix: data race when used in concurrent environment
  • Loading branch information
Timothy-Liuxf authored Mar 25, 2022
2 parents 5bce6e9 + 3d70a85 commit 2e758ec
Show file tree
Hide file tree
Showing 12 changed files with 233 additions and 106 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/publish.yml
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ jobs:
- name: Fetch code
uses: actions/checkout@v3
- name: Setup dotnet
uses: actions/setup-dotnet@v1
uses: actions/setup-dotnet@v2
with:
dotnet-version: |
3.1.x
Expand Down
7 changes: 5 additions & 2 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ jobs:
- name: Fetch code
uses: actions/checkout@v3
- name: Setup dotnet
uses: actions/setup-dotnet@v1
uses: actions/setup-dotnet@v2
with:
dotnet-version: |
3.1.x
Expand All @@ -22,4 +22,7 @@ jobs:
- name: Build FrameRateTask
run: dotnet build "./CSharp/FrameRateTask/FrameRateTask.csproj" -c Release
- name: Run test code
run: dotnet build "./CSharp/Test/Test.csproj" -c Release
run: |
dotnet build "./CSharp/Test/Test.csproj" -c Release
dotnet build "./CSharp/TestConcurrency/TestConcurrency.csproj" -c Release
dotnet run --project "./CSharp/TestConcurrency/TestConcurrency.csproj" -c Release
32 changes: 0 additions & 32 deletions CSharp/FrameRateTask/FrameRateTask - Backup.csproj

This file was deleted.

78 changes: 51 additions & 27 deletions CSharp/FrameRateTask/FrameRateTask.cs
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@

namespace Timothy.FrameRateTask
{

/// <summary>
/// The class intends to execute a task that need to be executed repeatedly every less than one second and need to be accurate.
/// </summary>
Expand All @@ -22,32 +23,41 @@ public class FrameRateTaskExecutor<TResult>
/// <summary>
/// The actual framerate recently.
/// </summary>
public uint FrameRate { get; private set; }
public uint FrameRate
{
get => (uint)Interlocked.CompareExchange(ref frameRate, 0, 0);
private set => Interlocked.Exchange(ref frameRate, value);
}
private long frameRate;

/// <summary>
/// Gets a value indicating whether or not the task has finished.
/// </summary>
/// <returns>
/// true if the task has finished; otherwise, false.
/// </returns>
public bool Finished { get; private set; } = false;
public bool Finished
{
get => Interlocked.CompareExchange(ref finished, 0, 0) != 0;
set => Interlocked.Exchange(ref finished, value ? 1 : 0);
}
private int finished = 0;

/// <summary>
/// Gets a value indicating whether or not the task has started.
/// </summary>
/// <returns>
/// true if the task has started; otherwise, false.
/// </returns>
public bool HasExecuted { get; private set; } = false;
private object hasExecutedLock = new object();
public bool HasExecuted { get => Interlocked.CompareExchange(ref hasExecuted, 0, 0) != 0; }
private int hasExecuted = 0;
private bool TrySetExecute()
{
lock (hasExecutedLock)
if (Interlocked.Exchange(ref hasExecuted, 1) != 0)
{
if (HasExecuted) return false;
HasExecuted = true;
return true;
return false;
}
return true;
}

private TResult result;
Expand All @@ -67,41 +77,54 @@ public TResult Result
private set => result = value;
}


/// <summary>
/// Gets or sets whether it allows time exceeding.
/// </summary>
/// <remarks>
/// If this property is false, the task will throw a TimeExceed exception when the task cannot finish in the given time.
/// The default value is true.
/// </remarks>
public bool AllowTimeExceed { get; set; } = true;
public bool AllowTimeExceed
{
get;
#if NET5_0_OR_GREATER
init;
#else
set;
#endif
} = true;

/// <summary>
/// It will be called once time exceeds if AllowTimeExceed is set true.
/// </summary>
/// <remarks>
/// parameter bool: If it is called because of the number of time exceeding is greater than MaxTimeExceedCount, the argument is true; if it is called because of exceeding once, the argument is false.
/// </remarks>
public Action<bool> TimeExceedAction { get; set; } = callByExceed => { };

/// <summary>
/// The TickCount when beginning the loop,
/// </summary>
public long BeginTickCount { get; private set; } = 0L;

/// <summary>
/// The TickCount should be when ending last loop.
/// </summary>
public long LastLoopEndingTickCount { get; private set; }
public Action<bool> TimeExceedAction
{
get;
#if NET5_0_OR_GREATER
init;
#else
set;
#endif
} = callByExceed => { };

/// <summary>
/// Gets or sets the maximum count of time exceeding continuously.
/// </summary>
/// <remarks>
/// The value is 5 for default.
/// </remarks>
public ulong MaxTolerantTimeExceedCount { get; set; } = 5;
public ulong MaxTolerantTimeExceedCount
{
get;
#if NET5_0_OR_GREATER
init;
#else
set;
#endif
} = 5;

/// <summary>
/// Start this task synchronously.
Expand Down Expand Up @@ -155,12 +178,13 @@ public FrameRateTaskExecutor
loopFunc = () =>
{
ulong timeExceedCount = 0UL;
long lastLoopEndingTickCount, beginTickCount;

var nextTime = (LastLoopEndingTickCount = BeginTickCount = Environment.TickCount64) + timeInterval;
var endTime = BeginTickCount < long.MaxValue - maxTotalDuration ? BeginTickCount + maxTotalDuration : long.MaxValue;
var nextTime = (lastLoopEndingTickCount = beginTickCount = Environment.TickCount64) + timeInterval;
var endTime = beginTickCount < long.MaxValue - maxTotalDuration ? beginTickCount + maxTotalDuration : long.MaxValue;

uint loopCnt = 0;
var nextCntTime = BeginTickCount + 1000L;
var nextCntTime = beginTickCount + 1000L;

while (loopCondition() && nextTime <= endTime)
{
Expand Down Expand Up @@ -191,7 +215,7 @@ public FrameRateTaskExecutor
else if (AllowTimeExceed) TimeExceedAction(false);
}

LastLoopEndingTickCount = nextTime;
lastLoopEndingTickCount = nextTime;
nextTime += timeInterval;
++loopCnt;
if (Environment.TickCount64 >= nextCntTime)
Expand All @@ -203,9 +227,9 @@ public FrameRateTaskExecutor
}

result = finallyReturn();
Interlocked.MemoryBarrierProcessWide();
Finished = true;
};

}

/// <summary>
Expand Down
14 changes: 12 additions & 2 deletions CSharp/FrameRateTask/FrameRateTask.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,9 @@

<PropertyGroup>
<TargetFrameworks>netcoreapp3.1;net5.0;net6.0</TargetFrameworks>
<FileVersion>1.0.0.0</FileVersion>
<AssemblyVersion>1.0.0.0</AssemblyVersion>
<FileVersion>1.1.0.0</FileVersion>
<AssemblyVersion>1.1.0.0</AssemblyVersion>
<VersionPrefix>1.1.0</VersionPrefix>
<RepositoryUrl>https://github.com/Timothy-LiuXuefeng/FrameRateTask</RepositoryUrl>
<PackageReadmeFile>README.md</PackageReadmeFile>
<Authors>TimothyLiuXuefeng</Authors>
Expand All @@ -13,6 +14,7 @@
<PackageRequireLicenseAcceptance>True</PackageRequireLicenseAcceptance>
<PackageLicenseFile>LICENSE.txt</PackageLicenseFile>
<PackageProjectUrl>https://github.com/Timothy-LiuXuefeng/FrameRateTask</PackageProjectUrl>
<Nullable>disable</Nullable>
</PropertyGroup>

<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'">
Expand All @@ -23,6 +25,14 @@
<DocumentationFile>.\FrameRateTask.xml</DocumentationFile>
</PropertyGroup>

<PropertyGroup Condition="'$(Configuration)|$(TargetFramework)|$(Platform)'=='Debug|net6.0|AnyCPU'">
<WarningLevel>5</WarningLevel>
</PropertyGroup>

<PropertyGroup Condition="'$(Configuration)|$(TargetFramework)|$(Platform)'=='Release|net6.0|AnyCPU'">
<WarningLevel>5</WarningLevel>
</PropertyGroup>

<ItemGroup>
<None Include="..\..\README.md">
<Pack>True</Pack>
Expand Down
38 changes: 19 additions & 19 deletions CSharp/Test/Program.cs
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ static void Demo1() // The most common circumstance
{
Random r = new Random();
int cnt = 0;
FrameRateTaskExecutor<int> frt = new FrameRateTaskExecutor<int>
FrameRateTaskExecutor<int> executor = new FrameRateTaskExecutor<int>
(
() => true,
() =>
Expand All @@ -31,21 +31,21 @@ static void Demo1() // The most common circumstance
(
() =>
{
while (!frt.Finished)
while (!executor.Finished)
{
Console.WriteLine($"Now framerate: { frt.FrameRate }");
Console.WriteLine($"Now framerate: { executor.FrameRate }");
Thread.Sleep(1000);
}
}
);
frt.Start();
executor.Start();
}

static void Demo2() // The most common circumstance.
{
int i = 1, sum = 0;
Random r = new Random();
FrameRateTaskExecutor<int> frt = new FrameRateTaskExecutor<int>
FrameRateTaskExecutor<int> executor = new FrameRateTaskExecutor<int>
(
() => i <= 10,
() =>
Expand All @@ -64,8 +64,8 @@ static void Demo2() // The most common circumstance.
{
MaxTolerantTimeExceedCount = ulong.MaxValue // Set it to ulong.MaxValue in case time exceeding could cause the result incorrect.
};
frt.Start();
Console.WriteLine($"result: {frt.Result}");
executor.Start();
Console.WriteLine($"result: {executor.Result}");
}
static void Demo3()
{
Expand Down Expand Up @@ -109,7 +109,7 @@ static void Demo4()
{
Random r = new Random();
int tm = 0;
FrameRateTaskExecutor<int> frt = new FrameRateTaskExecutor<int>
FrameRateTaskExecutor<int> executor = new FrameRateTaskExecutor<int>
(
() => true,
() =>
Expand All @@ -127,14 +127,14 @@ static void Demo4()
(
() =>
{
while (!frt.Finished)
while (!executor.Finished)
{
Console.WriteLine($"Now framerate: { frt.FrameRate }");
Console.WriteLine($"Now framerate: { executor.FrameRate }");
Thread.Sleep(1000);
}
}
);
frt.Start();
executor.Start();
}

static void Demo5()
Expand All @@ -143,7 +143,7 @@ static void Demo5()
Random r = new Random();
int tm = 0;
int cnt = 0;
FrameRateTaskExecutor<int> frt = new FrameRateTaskExecutor<int>
FrameRateTaskExecutor<int> executor = new FrameRateTaskExecutor<int>
(
() => true,
() =>
Expand All @@ -166,14 +166,14 @@ static void Demo5()
(
() =>
{
while (!frt.Finished)
while (!executor.Finished)
{
Console.WriteLine($"Now framerate: { frt.FrameRate }");
Console.WriteLine($"Now framerate: { executor.FrameRate }");
Thread.Sleep(1000);
}
}
);
frt.Start();
executor.Start();
}
Console.WriteLine("You can see it is stable after it becomes clear.");

Expand All @@ -185,7 +185,7 @@ static void Demo5()
Random r = new Random();
int tm = 0;
int cnt = 0;
FrameRateTaskExecutor<int> frt = new FrameRateTaskExecutor<int>
FrameRateTaskExecutor<int> executor = new FrameRateTaskExecutor<int>
(
() => true,
() =>
Expand All @@ -208,14 +208,14 @@ static void Demo5()
(
() =>
{
while (!frt.Finished)
while (!executor.Finished)
{
Console.WriteLine($"Now framerate: { frt.FrameRate }");
Console.WriteLine($"Now framerate: { executor.FrameRate }");
Thread.Sleep(1000);
}
}
);
frt.Start();
executor.Start();
Console.WriteLine("You can see it is NOT stable after it becomes clear.");
}
}
Expand Down
Loading

0 comments on commit 2e758ec

Please sign in to comment.