blob: 7e031454a44e90164f53eef4ffec393008f6911e (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
/*=============================================================================
**
**
** Purpose: Interface meant for CLR to participate in framework rundown.
** AppDomainPauseManager is the class that encapsulates all Fx rundown work.
**
**
=============================================================================*/
namespace System
{
using System;
using System.Threading;
using System.Security;
using System.Diagnostics.Contracts;
using System.Runtime.Versioning;
using System.Runtime.CompilerServices;
[System.Security.SecurityCritical]
internal class AppDomainPauseManager
{
[System.Security.SecurityCritical]
public AppDomainPauseManager()
{
isPaused = false;
}
[System.Security.SecurityCritical]
static AppDomainPauseManager()
{
}
static readonly AppDomainPauseManager instance = new AppDomainPauseManager();
internal static AppDomainPauseManager Instance
{
[System.Security.SecurityCritical]
get { return instance; }
}
// FAS: IAppDomainPauseConsumer interface implementation
// currently there is nothing we do here as the implementation
// of updating pause times have been moved to native CorHost2
[System.Security.SecurityCritical]
public void Pausing()
{
}
[System.Security.SecurityCritical]
public void Paused()
{
Contract.Assert(!isPaused);
if(ResumeEvent == null)
ResumeEvent = new ManualResetEvent(false);
else
ResumeEvent.Reset();
Timer.Pause();
// Set IsPaused at the last as other threads seeing it set, will expect a valid
// reset ResumeEvent. Also the requirement here is that only after Paused
// returns other threads should block on this event. So there is a race condition here.
isPaused = true;
}
[System.Security.SecurityCritical]
public void Resuming()
{
Contract.Assert(isPaused);
isPaused = false;
ResumeEvent.Set();
}
[System.Security.SecurityCritical]
public void Resumed()
{
Timer.Resume();
}
private static volatile bool isPaused;
internal static bool IsPaused
{
[System.Security.SecurityCritical]
get { return isPaused; }
}
internal static ManualResetEvent ResumeEvent
{
[System.Security.SecurityCritical]
get;
[System.Security.SecurityCritical]
set;
}
}
}
|