00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
#include "exp.h"
00027
00028
00029
00030
00031
00032 typedef struct _ETIMER {
00033 KTIMER KeTimer;
00034 KAPC TimerApc;
00035 KDPC TimerDpc;
00036 LIST_ENTRY
ActiveTimerListEntry;
00037 KSPIN_LOCK
Lock;
00038 LONG
Period;
00039 BOOLEAN
ApcAssociated;
00040 BOOLEAN
WakeTimer;
00041 LIST_ENTRY
WakeTimerListEntry;
00042 }
ETIMER, *
PETIMER;
00043
00044
00045
00046
00047
00048 KSPIN_LOCK
ExpWakeTimerListLock;
00049 LIST_ENTRY
ExpWakeTimerList;
00050
00051
00052
00053
00054
00055 POBJECT_TYPE ExTimerObjectType;
00056
00057
00058
00059
00060
00061
00062 GENERIC_MAPPING
ExpTimerMapping = {
00063 STANDARD_RIGHTS_READ |
00064 TIMER_QUERY_STATE,
00065 STANDARD_RIGHTS_WRITE |
00066 TIMER_MODIFY_STATE,
00067 STANDARD_RIGHTS_EXECUTE |
00068 SYNCHRONIZE,
00069 TIMER_ALL_ACCESS
00070 };
00071
00072
#ifdef ALLOC_PRAGMA
00073
#pragma alloc_text(INIT, ExpTimerInitialization)
00074
#pragma alloc_text(PAGE, NtCreateTimer)
00075
#pragma alloc_text(PAGE, NtOpenTimer)
00076
#pragma alloc_text(PAGE, NtQueryTimer)
00077
#pragma alloc_text(PAGELK, ExGetNextWakeTime)
00078
#endif
00079
00080
VOID
00081 ExpTimerApcRoutine (
00082 IN
PKAPC Apc,
00083 IN PKNORMAL_ROUTINE *NormalRoutine,
00084 IN PVOID *NormalContext,
00085 IN PVOID *SystemArgument1,
00086 IN PVOID *SystemArgument2
00087 )
00088
00089
00090
00091
00092
00093
00094
00095
00096
00097
00098
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115 {
00116
00117 BOOLEAN Dereference;
00118
PETHREAD ExThread;
00119
PETIMER ExTimer;
00120 KIRQL OldIrql1;
00121
00122
00123
00124
00125
00126 ExThread =
PsGetCurrentThread();
00127 ExTimer = CONTAINING_RECORD(Apc,
ETIMER, TimerApc);
00128
00129
00130
00131
00132
00133
00134
00135
00136
00137
00138
00139
00140
00141
00142 Dereference =
FALSE;
00143 ExAcquireSpinLock(&ExTimer->
Lock, &OldIrql1);
00144 ExAcquireSpinLockAtDpcLevel(&ExThread->
ActiveTimerListLock);
00145
if ((ExTimer->
ApcAssociated) && (&ExThread->
Tcb == ExTimer->
TimerApc.
Thread)) {
00146
if (ExTimer->
Period == 0) {
00147 RemoveEntryList(&ExTimer->
ActiveTimerListEntry);
00148 ExTimer->
ApcAssociated =
FALSE;
00149 Dereference =
TRUE;
00150 }
00151
00152 }
else {
00153 *NormalRoutine = (
PKNORMAL_ROUTINE)
NULL;
00154 }
00155
00156 ExReleaseSpinLockFromDpcLevel(&ExThread->
ActiveTimerListLock);
00157 ExReleaseSpinLock(&ExTimer->
Lock, OldIrql1);
00158
if (Dereference) {
00159
ObDereferenceObject((PVOID)ExTimer);
00160 }
00161
00162
return;
00163 }
00164
00165
VOID
00166 ExpTimerDpcRoutine (
00167 IN
PKDPC Dpc,
00168 IN PVOID DeferredContext,
00169 IN PVOID SystemArgument1,
00170 IN PVOID SystemArgument2
00171 )
00172
00173
00174
00175
00176
00177
00178
00179
00180
00181
00182
00183
00184
00185
00186
00187
00188
00189
00190
00191
00192
00193
00194
00195
00196
00197 {
00198
00199
PETIMER ExTimer;
00200
PKTIMER KeTimer;
00201 KIRQL OldIrql;
00202
00203
00204
00205
00206
00207 ExTimer = (
PETIMER)DeferredContext;
00208 KeTimer = &ExTimer->
KeTimer;
00209
00210
00211
00212
00213
00214
00215
00216
00217
00218
00219 ExAcquireSpinLock(&ExTimer->
Lock, &OldIrql);
00220
if (ExTimer->
ApcAssociated) {
00221
KeInsertQueueApc(&ExTimer->
TimerApc,
00222 SystemArgument1,
00223 SystemArgument2,
00224
TIMER_APC_INCREMENT);
00225 }
00226
00227 ExReleaseSpinLock(&ExTimer->
Lock, OldIrql);
00228
return;
00229 }
00230
00231
static VOID
00232 ExpDeleteTimer (
00233 IN PVOID Object
00234 )
00235
00236
00237
00238
00239
00240
00241
00242
00243
00244
00245
00246
00247
00248
00249
00250
00251
00252
00253 {
00254
PETIMER ExTimer;
00255 KIRQL OldIrql;
00256
00257 ExTimer = (
PETIMER) Object;
00258
00259
00260
00261
00262
00263
if (ExTimer->
WakeTimerListEntry.Flink) {
00264 ExAcquireSpinLock(&
ExpWakeTimerListLock, &OldIrql);
00265
if (ExTimer->
WakeTimerListEntry.Flink) {
00266 RemoveEntryList(&ExTimer->
WakeTimerListEntry);
00267 ExTimer->
WakeTimerListEntry.Flink =
NULL;
00268 }
00269 ExReleaseSpinLock(&
ExpWakeTimerListLock, OldIrql);
00270 }
00271
00272
00273
00274
00275
00276
KeCancelTimer(&ExTimer->
KeTimer);
00277
return;
00278 }
00279
00280 BOOLEAN
00281 ExpTimerInitialization (
00282 )
00283
00284
00285
00286
00287
00288
00289
00290
00291
00292
00293
00294
00295
00296
00297
00298
00299
00300
00301
00302
00303 {
00304
00305
OBJECT_TYPE_INITIALIZER ObjectTypeInitializer;
00306
NTSTATUS Status;
00307 UNICODE_STRING TypeName;
00308
00309
KeInitializeSpinLock (&
ExpWakeTimerListLock);
00310 InitializeListHead (&
ExpWakeTimerList);
00311
00312
00313
00314
00315
00316
RtlInitUnicodeString(&TypeName,
L"Timer");
00317
00318
00319
00320
00321
00322 RtlZeroMemory(&ObjectTypeInitializer,
sizeof(ObjectTypeInitializer));
00323 ObjectTypeInitializer.Length =
sizeof(ObjectTypeInitializer);
00324 ObjectTypeInitializer.InvalidAttributes = OBJ_OPENLINK;
00325 ObjectTypeInitializer.GenericMapping =
ExpTimerMapping;
00326 ObjectTypeInitializer.PoolType =
NonPagedPool;
00327 ObjectTypeInitializer.DefaultNonPagedPoolCharge =
sizeof(
ETIMER);
00328 ObjectTypeInitializer.ValidAccessMask = TIMER_ALL_ACCESS;
00329 ObjectTypeInitializer.DeleteProcedure =
ExpDeleteTimer;
00330
Status =
ObCreateObjectType(&TypeName,
00331 &ObjectTypeInitializer,
00332 (PSECURITY_DESCRIPTOR)
NULL,
00333 &
ExTimerObjectType);
00334
00335
00336
00337
00338
00339
00340
00341
00342
return (BOOLEAN)(
NT_SUCCESS(
Status));
00343 }
00344
00345
VOID
00346 ExTimerRundown (
00347 )
00348
00349
00350
00351
00352
00353
00354
00355
00356
00357
00358
00359
00360
00361
00362
00363
00364
00365
00366
00367
00368 {
00369
00370 BOOLEAN Dereference;
00371
PETHREAD ExThread;
00372
PETIMER ExTimer;
00373 PLIST_ENTRY NextEntry;
00374 KIRQL OldIrql1;
00375
00376
00377
00378
00379
00380 ExThread =
PsGetCurrentThread();
00381 ExAcquireSpinLock(&ExThread->
ActiveTimerListLock, &OldIrql1);
00382 NextEntry = ExThread->
ActiveTimerListHead.Flink;
00383
while (NextEntry != &ExThread->
ActiveTimerListHead) {
00384 ExTimer = CONTAINING_RECORD(NextEntry,
ETIMER, ActiveTimerListEntry);
00385
00386
00387
00388
00389
00390
00391
00392
00393
ObReferenceObject(ExTimer);
00394 ExReleaseSpinLock(&ExThread->
ActiveTimerListLock, OldIrql1);
00395
00396
00397
00398
00399
00400
00401
00402
00403
00404
00405
00406
00407
00408 ExAcquireSpinLock(&ExTimer->
Lock, &OldIrql1);
00409 ExAcquireSpinLockAtDpcLevel(&ExThread->
ActiveTimerListLock);
00410
if ((ExTimer->
ApcAssociated) && (&ExThread->
Tcb == ExTimer->
TimerApc.
Thread)) {
00411 RemoveEntryList(&ExTimer->
ActiveTimerListEntry);
00412 ExTimer->
ApcAssociated =
FALSE;
00413
KeCancelTimer(&ExTimer->
KeTimer);
00414
KeRemoveQueueDpc(&ExTimer->
TimerDpc);
00415
KeRemoveQueueApc(&ExTimer->
TimerApc);
00416 Dereference =
TRUE;
00417
00418 }
else {
00419 Dereference =
FALSE;
00420 }
00421
00422 ExReleaseSpinLockFromDpcLevel(&ExThread->
ActiveTimerListLock);
00423 ExReleaseSpinLock(&ExTimer->
Lock, OldIrql1);
00424
if (Dereference) {
00425
ObDereferenceObject((PVOID)ExTimer);
00426 }
00427
00428
ObDereferenceObject((PVOID)ExTimer);
00429
00430
00431
00432
00433
00434
00435 ExAcquireSpinLock(&ExThread->
ActiveTimerListLock, &OldIrql1);
00436 NextEntry = ExThread->
ActiveTimerListHead.Flink;
00437 }
00438
00439 ExReleaseSpinLock(&ExThread->
ActiveTimerListLock, OldIrql1);
00440
return;
00441 }
00442
00443
NTSTATUS
00444 NtCreateTimer (
00445 OUT PHANDLE TimerHandle,
00446 IN ACCESS_MASK DesiredAccess,
00447 IN POBJECT_ATTRIBUTES ObjectAttributes OPTIONAL,
00448 IN TIMER_TYPE TimerType
00449 )
00450
00451
00452
00453
00454
00455
00456
00457
00458
00459
00460
00461
00462
00463
00464
00465
00466
00467
00468
00469
00470
00471
00472
00473
00474
00475 {
00476
00477
PETIMER ExTimer;
00478 HANDLE
Handle;
00479
KPROCESSOR_MODE PreviousMode;
00480
NTSTATUS Status;
00481
00482
00483
00484
00485
00486
00487
00488
00489
try {
00490
00491
00492
00493
00494
00495
00496 PreviousMode = KeGetPreviousMode();
00497
if (PreviousMode !=
KernelMode) {
00498
ProbeForWriteHandle(
TimerHandle);
00499 }
00500
00501
00502
00503
00504
00505
if ((TimerType != NotificationTimer) &&
00506 (TimerType != SynchronizationTimer)) {
00507
return STATUS_INVALID_PARAMETER_4;
00508 }
00509
00510
00511
00512
00513
00514
Status =
ObCreateObject(PreviousMode,
00515
ExTimerObjectType,
00516
ObjectAttributes,
00517 PreviousMode,
00518
NULL,
00519
sizeof(
ETIMER),
00520 0,
00521 0,
00522 (PVOID *)&ExTimer);
00523
00524
00525
00526
00527
00528
00529
00530
if (
NT_SUCCESS(
Status)) {
00531
KeInitializeDpc(&ExTimer->TimerDpc,
00532
ExpTimerDpcRoutine,
00533 (PVOID)ExTimer);
00534
00535
KeInitializeTimerEx(&ExTimer->KeTimer, TimerType);
00536
KeInitializeSpinLock(&ExTimer->Lock);
00537 ExTimer->ApcAssociated =
FALSE;
00538 ExTimer->WakeTimer =
FALSE;
00539 ExTimer->WakeTimerListEntry.Flink =
NULL;
00540
Status =
ObInsertObject((PVOID)ExTimer,
00541
NULL,
00542 DesiredAccess,
00543 0,
00544 (PVOID *)
NULL,
00545 &
Handle);
00546
00547
00548
00549
00550
00551
00552
00553
00554
00555
if (
NT_SUCCESS(
Status)) {
00556
try {
00557 *
TimerHandle =
Handle;
00558 } except(
ExSystemExceptionFilter()) {
00559 }
00560 }
00561 }
00562
00563
00564
00565
00566
00567
00568
00569 } except(
ExSystemExceptionFilter()) {
00570
return GetExceptionCode();
00571 }
00572
00573
00574
00575
00576
00577
return Status;
00578 }
00579
00580
NTSTATUS
00581 NtOpenTimer (
00582 OUT PHANDLE TimerHandle,
00583 IN ACCESS_MASK DesiredAccess,
00584 IN POBJECT_ATTRIBUTES ObjectAttributes
00585 )
00586
00587
00588
00589
00590
00591
00592
00593
00594
00595
00596
00597
00598
00599
00600
00601
00602
00603
00604
00605
00606
00607
00608
00609 {
00610
00611 HANDLE
Handle;
00612
KPROCESSOR_MODE PreviousMode;
00613
NTSTATUS Status;
00614
00615
00616
00617
00618
00619
00620
00621
00622
try {
00623
00624
00625
00626
00627
00628
00629 PreviousMode = KeGetPreviousMode();
00630
if (PreviousMode !=
KernelMode) {
00631
ProbeForWriteHandle(
TimerHandle);
00632 }
00633
00634
00635
00636
00637
00638
Status =
ObOpenObjectByName(
ObjectAttributes,
00639
ExTimerObjectType,
00640 PreviousMode,
00641
NULL,
00642 DesiredAccess,
00643
NULL,
00644 &
Handle);
00645
00646
00647
00648
00649
00650
00651
00652
00653
if (
NT_SUCCESS(
Status)) {
00654
try {
00655 *
TimerHandle =
Handle;
00656
00657 } except(
ExSystemExceptionFilter()) {
00658 }
00659 }
00660
00661
00662
00663
00664
00665
00666
00667 } except(
ExSystemExceptionFilter()) {
00668
return GetExceptionCode();
00669 }
00670
00671
00672
00673
00674
00675
return Status;
00676 }
00677
00678
NTSTATUS
00679 NtCancelTimer (
00680 IN HANDLE TimerHandle,
00681 OUT PBOOLEAN CurrentState OPTIONAL
00682 )
00683
00684
00685
00686
00687
00688
00689
00690
00691
00692
00693
00694
00695
00696
00697
00698
00699
00700
00701
00702
00703 {
00704
00705 BOOLEAN Dereference;
00706
PETHREAD ExThread;
00707
PETIMER ExTimer;
00708 KIRQL OldIrql1;
00709
KPROCESSOR_MODE PreviousMode;
00710 BOOLEAN State;
00711
NTSTATUS Status;
00712
00713
00714
00715
00716
00717
00718
00719
00720
00721
try {
00722
00723
00724
00725
00726
00727
00728 PreviousMode = KeGetPreviousMode();
00729
if ((PreviousMode !=
KernelMode) && (ARGUMENT_PRESENT(CurrentState))) {
00730
ProbeForWriteBoolean(CurrentState);
00731 }
00732
00733
00734
00735
00736
00737
Status =
ObReferenceObjectByHandle(
TimerHandle,
00738 TIMER_MODIFY_STATE,
00739
ExTimerObjectType,
00740 PreviousMode,
00741 (PVOID *)&ExTimer,
00742
NULL);
00743
00744
00745
00746
00747
00748
00749
00750
00751
00752
if (
NT_SUCCESS(
Status)) {
00753 ExAcquireSpinLock(&ExTimer->Lock, &OldIrql1);
00754
if (ExTimer->ApcAssociated) {
00755 ExThread = CONTAINING_RECORD(ExTimer->TimerApc.Thread,
ETHREAD, Tcb);
00756 ExAcquireSpinLockAtDpcLevel(&ExThread->
ActiveTimerListLock);
00757 RemoveEntryList(&ExTimer->ActiveTimerListEntry);
00758 ExTimer->ApcAssociated =
FALSE;
00759 ExReleaseSpinLockFromDpcLevel(&ExThread->
ActiveTimerListLock);
00760
KeCancelTimer(&ExTimer->KeTimer);
00761
KeRemoveQueueDpc(&ExTimer->TimerDpc);
00762
KeRemoveQueueApc(&ExTimer->TimerApc);
00763 Dereference =
TRUE;
00764
00765 }
else {
00766
KeCancelTimer(&ExTimer->KeTimer);
00767 Dereference =
FALSE;
00768 }
00769
00770
if (ExTimer->WakeTimerListEntry.Flink) {
00771 ExAcquireSpinLockAtDpcLevel(&
ExpWakeTimerListLock);
00772
00773
00774
00775
00776
if (ExTimer->WakeTimerListEntry.Flink) {
00777 RemoveEntryList(&ExTimer->WakeTimerListEntry);
00778 ExTimer->WakeTimerListEntry.Flink =
NULL;
00779 }
00780 ExReleaseSpinLockFromDpcLevel(&
ExpWakeTimerListLock);
00781 }
00782
00783 ExReleaseSpinLock(&ExTimer->Lock, OldIrql1);
00784
if (Dereference) {
00785
ObDereferenceObject((PVOID)ExTimer);
00786 }
00787
00788
00789
00790
00791
00792
00793 State =
KeReadStateTimer(&ExTimer->KeTimer);
00794
ObDereferenceObject(ExTimer);
00795
if (ARGUMENT_PRESENT(CurrentState)) {
00796
try {
00797 *CurrentState = State;
00798
00799 } except(
ExSystemExceptionFilter()) {
00800 }
00801 }
00802 }
00803
00804
00805
00806
00807
00808
00809
00810 } except(
ExSystemExceptionFilter()) {
00811
return GetExceptionCode();
00812 }
00813
00814
00815
00816
00817
00818
return Status;
00819 }
00820
00821
NTSTATUS
00822 NtQueryTimer (
00823 IN HANDLE TimerHandle,
00824 IN TIMER_INFORMATION_CLASS TimerInformationClass,
00825 OUT PVOID TimerInformation,
00826 IN ULONG TimerInformationLength,
00827 OUT PULONG ReturnLength OPTIONAL
00828 )
00829
00830
00831
00832
00833
00834
00835
00836
00837
00838
00839
00840
00841
00842
00843
00844
00845
00846
00847
00848
00849
00850
00851
00852
00853
00854
00855
00856
00857
00858 {
00859
00860
PETIMER ExTimer;
00861
PKTIMER KeTimer;
00862
KPROCESSOR_MODE PreviousMode;
00863 BOOLEAN State;
00864
NTSTATUS Status;
00865 LARGE_INTEGER TimeToGo;
00866
00867
00868
00869
00870
00871
00872
00873
00874
00875
try {
00876
00877
00878
00879
00880
00881 PreviousMode = KeGetPreviousMode();
00882
if (PreviousMode !=
KernelMode) {
00883
ProbeForWrite(TimerInformation,
00884
sizeof(TIMER_BASIC_INFORMATION),
00885
sizeof(ULONG));
00886
00887
if (ARGUMENT_PRESENT(ReturnLength)) {
00888
ProbeForWriteUlong(ReturnLength);
00889 }
00890 }
00891
00892
00893
00894
00895
00896
if (TimerInformationClass != TimerBasicInformation) {
00897
return STATUS_INVALID_INFO_CLASS;
00898 }
00899
00900
if (TimerInformationLength !=
sizeof(TIMER_BASIC_INFORMATION)) {
00901
return STATUS_INFO_LENGTH_MISMATCH;
00902 }
00903
00904
00905
00906
00907
00908
Status =
ObReferenceObjectByHandle(
TimerHandle,
00909 TIMER_QUERY_STATE,
00910
ExTimerObjectType,
00911 PreviousMode,
00912 (PVOID *)&ExTimer,
00913
NULL);
00914
00915
00916
00917
00918
00919
00920
00921
00922
00923
00924
00925
if (
NT_SUCCESS(
Status)) {
00926 KeTimer = &ExTimer->KeTimer;
00927 State =
KeReadStateTimer(KeTimer);
00928 KiQueryInterruptTime(&TimeToGo);
00929 TimeToGo.QuadPart = KeTimer->
DueTime.QuadPart - TimeToGo.QuadPart;
00930
ObDereferenceObject(ExTimer);
00931
try {
00932 ((PTIMER_BASIC_INFORMATION)TimerInformation)->TimerState = State;
00933 ((PTIMER_BASIC_INFORMATION)TimerInformation)->RemainingTime = TimeToGo;
00934
if (ARGUMENT_PRESENT(ReturnLength)) {
00935 *ReturnLength =
sizeof(TIMER_BASIC_INFORMATION);
00936 }
00937
00938 } except(
ExSystemExceptionFilter()) {
00939 }
00940 }
00941
00942
00943
00944
00945
00946
00947
00948 } except(
ExSystemExceptionFilter()) {
00949
return GetExceptionCode();
00950 }
00951
00952
00953
00954
00955
00956
return Status;
00957 }
00958
00959
NTSTATUS
00960 NtSetTimer (
00961 IN HANDLE TimerHandle,
00962 IN PLARGE_INTEGER DueTime,
00963 IN PTIMER_APC_ROUTINE TimerApcRoutine OPTIONAL,
00964 IN PVOID TimerContext OPTIONAL,
00965 IN BOOLEAN WakeTimer,
00966 IN LONG Period OPTIONAL,
00967 OUT PBOOLEAN PreviousState OPTIONAL
00968 )
00969
00970
00971
00972
00973
00974
00975
00976
00977
00978
00979
00980
00981
00982
00983
00984
00985
00986
00987
00988
00989
00990
00991
00992
00993
00994
00995
00996
00997
00998
00999
01000
01001
01002
01003
01004
01005
01006
01007 {
01008
01009 BOOLEAN AssociatedApc;
01010 BOOLEAN Dereference;
01011
PETHREAD ExThread;
01012
PETIMER ExTimer;
01013 LARGE_INTEGER ExpirationTime;
01014 KIRQL OldIrql1;
01015
KPROCESSOR_MODE PreviousMode;
01016 BOOLEAN State;
01017
NTSTATUS Status;
01018
01019
01020
01021
01022
01023
01024
01025
01026
01027
try {
01028
01029
01030
01031
01032
01033
01034 PreviousMode = KeGetPreviousMode();
01035
if (PreviousMode !=
KernelMode) {
01036
if (ARGUMENT_PRESENT(PreviousState)) {
01037
ProbeForWriteBoolean(PreviousState);
01038 }
01039
01040
ProbeForRead(DueTime,
sizeof(LARGE_INTEGER),
sizeof(ULONG));
01041 }
01042
01043
01044
01045
01046
01047
if (Period < 0) {
01048
return STATUS_INVALID_PARAMETER_6;
01049 }
01050
01051
01052
01053
01054
01055 ExpirationTime = *DueTime;
01056
01057
01058
01059
01060
01061
Status =
ObReferenceObjectByHandle(
TimerHandle,
01062 TIMER_MODIFY_STATE,
01063
ExTimerObjectType,
01064 PreviousMode,
01065 (PVOID *)&ExTimer,
01066
NULL);
01067
01068
01069
01070
01071
01072
01073
if (
NT_SUCCESS(
Status) && WakeTimer && !
PoWakeTimerSupported()) {
01074
Status = STATUS_TIMER_RESUME_IGNORED;
01075 }
01076
01077
01078
01079
01080
01081
01082
01083
01084
01085
if (
NT_SUCCESS(
Status)) {
01086 ExAcquireSpinLock(&ExTimer->Lock, &OldIrql1);
01087
01088
if (ExTimer->ApcAssociated) {
01089 ExThread = CONTAINING_RECORD(ExTimer->TimerApc.Thread,
ETHREAD, Tcb);
01090 ExAcquireSpinLockAtDpcLevel(&ExThread->
ActiveTimerListLock);
01091 RemoveEntryList(&ExTimer->ActiveTimerListEntry);
01092 ExTimer->ApcAssociated =
FALSE;
01093 ExReleaseSpinLockFromDpcLevel(&ExThread->
ActiveTimerListLock);
01094
KeCancelTimer(&ExTimer->KeTimer);
01095
KeRemoveQueueDpc(&ExTimer->TimerDpc);
01096
KeRemoveQueueApc(&ExTimer->TimerApc);
01097 Dereference =
TRUE;
01098
01099 }
else {
01100
KeCancelTimer(&ExTimer->KeTimer);
01101 Dereference =
FALSE;
01102 }
01103
01104
01105
01106
01107
01108 State =
KeReadStateTimer(&ExTimer->KeTimer);
01109
01110
01111
01112
01113
01114 ExTimer->WakeTimer = WakeTimer;
01115 ExAcquireSpinLockAtDpcLevel(&
ExpWakeTimerListLock);
01116
if (WakeTimer) {
01117
if (!ExTimer->WakeTimerListEntry.Flink) {
01118 InsertTailList(&
ExpWakeTimerList, &ExTimer->WakeTimerListEntry);
01119 }
01120 }
else {
01121
if (ExTimer->WakeTimerListEntry.Flink) {
01122 RemoveEntryList(&ExTimer->WakeTimerListEntry);
01123 ExTimer->WakeTimerListEntry.Flink =
NULL;
01124 }
01125 }
01126 ExReleaseSpinLockFromDpcLevel(&
ExpWakeTimerListLock);
01127
01128
01129
01130
01131
01132
01133
01134
01135
01136 ExTimer->Period = Period;
01137
if (ARGUMENT_PRESENT(
TimerApcRoutine)) {
01138 ExThread =
PsGetCurrentThread();
01139
KeInitializeApc(&ExTimer->TimerApc,
01140 &ExThread->
Tcb,
01141
CurrentApcEnvironment,
01142
ExpTimerApcRoutine,
01143 (
PKRUNDOWN_ROUTINE)
NULL,
01144 (
PKNORMAL_ROUTINE)
TimerApcRoutine,
01145 PreviousMode,
01146 TimerContext);
01147
01148 ExAcquireSpinLockAtDpcLevel(&ExThread->
ActiveTimerListLock);
01149 InsertTailList(&ExThread->
ActiveTimerListHead,
01150 &ExTimer->ActiveTimerListEntry);
01151
01152 ExTimer->ApcAssociated =
TRUE;
01153 ExReleaseSpinLockFromDpcLevel(&ExThread->
ActiveTimerListLock);
01154
KeSetTimerEx(&ExTimer->KeTimer,
01155 ExpirationTime,
01156 Period,
01157 &ExTimer->TimerDpc);
01158
01159 AssociatedApc =
TRUE;
01160
01161 }
else {
01162
KeSetTimerEx(&ExTimer->KeTimer,
01163 ExpirationTime,
01164 Period,
01165
NULL);
01166
01167 AssociatedApc =
FALSE;
01168 }
01169
01170 ExReleaseSpinLock(&ExTimer->Lock, OldIrql1);
01171
01172
01173
01174
01175
01176
if (Dereference) {
01177
ObDereferenceObject((PVOID)ExTimer);
01178 }
01179
01180
if (AssociatedApc ==
FALSE) {
01181
ObDereferenceObject((PVOID)ExTimer);
01182 }
01183
01184
if (ARGUMENT_PRESENT(PreviousState)) {
01185
try {
01186 *PreviousState = State;
01187
01188 } except(
ExSystemExceptionFilter()) {
01189 }
01190 }
01191 }
01192
01193
01194
01195
01196
01197
01198
01199 } except(
ExSystemExceptionFilter()) {
01200
return GetExceptionCode();
01201 }
01202
01203
01204
01205
01206
01207
return Status;
01208 }
01209
01210
01211
VOID
01212 ExGetNextWakeTime (
01213 OUT PULONGLONG DueTime,
01214 OUT PTIME_FIELDS TimeFields,
01215 OUT PVOID *TimerObject
01216 )
01217 {
01218 PLIST_ENTRY Link;
01219
PETIMER ExTimer;
01220
PETIMER BestTimer;
01221 KIRQL OldIrql;
01222 ULONGLONG TimerDueTime;
01223 ULONGLONG BestDueTime;
01224 ULONGLONG InterruptTime;
01225 LARGE_INTEGER SystemTime;
01226 LARGE_INTEGER CmosTime;
01227
01228 ExAcquireSpinLock(&
ExpWakeTimerListLock, &OldIrql);
01229 BestDueTime = 0;
01230 BestTimer =
NULL;
01231 Link =
ExpWakeTimerList.Flink;
01232
while (Link != &
ExpWakeTimerList) {
01233 ExTimer = CONTAINING_RECORD(Link,
ETIMER, WakeTimerListEntry);
01234 Link = Link->Flink;
01235
01236
if (ExTimer->
WakeTimer) {
01237
01238 TimerDueTime =
KeQueryTimerDueTime(&ExTimer->
KeTimer);
01239 TimerDueTime = 0 - TimerDueTime;
01240
01241
01242
01243
01244
01245
if (TimerDueTime > BestDueTime) {
01246 BestDueTime = TimerDueTime;
01247 BestTimer = ExTimer;
01248 }
01249
01250 }
else {
01251
01252
01253
01254
01255
01256 RemoveEntryList(&ExTimer->
WakeTimerListEntry);
01257 ExTimer->
WakeTimerListEntry.Flink =
NULL;
01258 }
01259 }
01260
01261 ExReleaseSpinLock(&
ExpWakeTimerListLock, OldIrql);
01262
01263
if (BestDueTime) {
01264
01265
01266
01267
01268
KeQuerySystemTime (&SystemTime);
01269 InterruptTime =
KeQueryInterruptTime ();
01270 BestDueTime = 0 - BestDueTime;
01271
01272 SystemTime.QuadPart += BestDueTime - InterruptTime;
01273
01274
01275
01276
01277
01278
01279
01280 SystemTime.QuadPart += 10000000;
01281
01282
ExSystemTimeToLocalTime(&SystemTime,&CmosTime);
01283
RtlTimeToTimeFields(&CmosTime,
TimeFields);
01284 }
01285
01286 *DueTime = BestDueTime;
01287 *TimerObject = BestTimer;
01288 }