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
00027
#include "ki.h"
00028
00029
00030
00031
00032
00033
00034 #define ASSERT_DPC(E) { \
00035
ASSERT((E)->Type == DpcObject); \
00036
}
00037
00038
VOID
00039 KeInitializeDpc (
00040 IN
PRKDPC Dpc,
00041 IN
PKDEFERRED_ROUTINE DeferredRoutine,
00042 IN PVOID DeferredContext
00043 )
00044
00045
00046
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068 {
00069
00070
00071
00072
00073
00074 Dpc->Type =
DpcObject;
00075 Dpc->Number = 0;
00076 Dpc->Importance =
MediumImportance;
00077
00078
00079
00080
00081
00082 Dpc->DeferredRoutine = DeferredRoutine;
00083 Dpc->DeferredContext = DeferredContext;
00084 Dpc->Lock =
NULL;
00085
return;
00086 }
00087
00088 BOOLEAN
00089 KeInsertQueueDpc (
00090 IN
PRKDPC Dpc,
00091 IN PVOID SystemArgument1,
00092 IN PVOID SystemArgument2
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
00118 {
00119
00120 ULONG
Index;
00121 PKSPIN_LOCK
Lock;
00122 KIRQL OldIrql;
00123 PKPRCB Prcb;
00124 ULONG Processor;
00125
00126
ASSERT_DPC(Dpc);
00127
00128
00129
00130
00131
00132
KeRaiseIrql(
HIGH_LEVEL, &OldIrql);
00133
00134
00135
00136
00137
00138
#if !defined(NT_UP)
00139
00140
if (Dpc->Number >=
MAXIMUM_PROCESSORS) {
00141 Processor = Dpc->Number -
MAXIMUM_PROCESSORS;
00142 Prcb =
KiProcessorBlock[Processor];
00143
00144 }
else {
00145 Prcb =
KeGetCurrentPrcb();
00146 }
00147
00148 KiAcquireSpinLock(&Prcb->DpcLock);
00149
00150
#else
00151
00152 Prcb =
KeGetCurrentPrcb();
00153
00154
#endif
00155
00156
00157
00158
00159
00160
00161
00162
00163
00164
if ((
Lock = InterlockedCompareExchangePointer(&Dpc->Lock, &Prcb->DpcLock,
NULL)) ==
NULL) {
00165 Prcb->DpcCount += 1;
00166 Prcb->DpcQueueDepth += 1;
00167 Dpc->SystemArgument1 = SystemArgument1;
00168 Dpc->SystemArgument2 = SystemArgument2;
00169
00170
00171
00172
00173
00174
00175
00176
if (Dpc->Importance ==
HighImportance) {
00177 InsertHeadList(&Prcb->DpcListHead, &Dpc->DpcListEntry);
00178
00179 }
else {
00180 InsertTailList(&Prcb->DpcListHead, &Dpc->DpcListEntry);
00181 }
00182
00183
00184
00185
00186
00187
00188
#if defined(_ALPHA_) && !defined(NT_UP)
00189
00190 __MB();
00191
00192
#endif
00193
00194
00195
00196
00197
00198
00199
if ((Prcb->DpcRoutineActive ==
FALSE) &&
00200 (Prcb->DpcInterruptRequested ==
FALSE)) {
00201
00202
00203
00204
00205
00206
00207
00208
00209
#if defined(NT_UP)
00210
00211
if ((Dpc->Importance !=
LowImportance) ||
00212 (Prcb->DpcQueueDepth >= Prcb->MaximumDpcQueueDepth) ||
00213 (Prcb->DpcRequestRate < Prcb->MinimumDpcRate)) {
00214 Prcb->DpcInterruptRequested =
TRUE;
00215
KiRequestSoftwareInterrupt(
DISPATCH_LEVEL);
00216 }
00217
00218
00219
00220
00221
00222
00223
00224
00225
#else
00226
00227
if (Prcb !=
KeGetCurrentPrcb()) {
00228
if (((Dpc->Importance ==
HighImportance) ||
00229 (Prcb->DpcQueueDepth >= Prcb->MaximumDpcQueueDepth))) {
00230 Prcb->DpcInterruptRequested =
TRUE;
00231
KiIpiSend((KAFFINITY)(1 << Processor),
IPI_DPC);
00232 }
00233
00234 }
else {
00235
00236
00237
00238
00239
00240
00241
00242
00243
if ((Dpc->Importance !=
LowImportance) ||
00244 (Prcb->DpcQueueDepth >= Prcb->MaximumDpcQueueDepth) ||
00245 (Prcb->DpcRequestRate < Prcb->MinimumDpcRate)) {
00246 Prcb->DpcInterruptRequested =
TRUE;
00247
KiRequestSoftwareInterrupt(
DISPATCH_LEVEL);
00248 }
00249 }
00250
00251
#endif
00252
00253 }
00254 }
00255
00256
00257
00258
00259
00260
00261
#if !defined(NT_UP)
00262
00263 KiReleaseSpinLock(&Prcb->DpcLock);
00264
00265
#endif
00266
00267
KeLowerIrql(OldIrql);
00268
return (
Lock ==
NULL);
00269 }
00270
00271 BOOLEAN
00272 KeRemoveQueueDpc (
00273 IN
PRKDPC Dpc
00274 )
00275
00276
00277
00278
00279
00280
00281
00282
00283
00284
00285
00286
00287
00288
00289
00290
00291
00292
00293
00294
00295
00296 {
00297
00298 PKSPIN_LOCK
Lock;
00299 PKPRCB Prcb;
00300
00301
ASSERT_DPC(Dpc);
00302
00303
00304
00305
00306
00307
00308 _disable();
00309
Lock = Dpc->Lock;
00310
if (
Lock !=
NULL) {
00311
00312
00313
00314
00315
00316
#if !defined(NT_UP)
00317
00318 KiAcquireSpinLock(
Lock);
00319
00320
#endif
00321
00322
00323
00324
00325
00326
00327
00328
00329
00330
00331
if (
Lock == Dpc->Lock) {
00332 Prcb = CONTAINING_RECORD(
Lock, KPRCB, DpcLock);
00333 Prcb->DpcQueueDepth -= 1;
00334 RemoveEntryList(&Dpc->DpcListEntry);
00335
00336
#if defined(_ALPHA_) && !defined(NT_UP)
00337
00338 __MB();
00339
00340
#endif
00341
00342 Dpc->Lock =
NULL;
00343 }
00344
00345
00346
00347
00348
00349
#if !defined(NT_UP)
00350
00351 KiReleaseSpinLock(
Lock);
00352
00353
#endif
00354
00355 }
00356
00357
00358
00359
00360
00361
00362 _enable();
00363
return (
Lock !=
NULL);
00364 }
00365
00366
VOID
00367 KeSetImportanceDpc (
00368 IN
PRKDPC Dpc,
00369 IN
KDPC_IMPORTANCE Importance
00370 )
00371
00372
00373
00374
00375
00376
00377
00378
00379
00380
00381
00382
00383
00384
00385
00386
00387
00388
00389
00390 {
00391
00392
00393
00394
00395
00396 Dpc->Importance = (UCHAR)Importance;
00397
return;
00398 }
00399
00400
VOID
00401 KeSetTargetProcessorDpc (
00402 IN
PRKDPC Dpc,
00403 IN CCHAR Number
00404 )
00405
00406
00407
00408
00409
00410
00411
00412
00413
00414
00415
00416
00417
00418
00419
00420
00421
00422
00423
00424 {
00425
00426
00427
00428
00429
00430
00431
00432
00433 Dpc->Number =
MAXIMUM_PROCESSORS + Number;
00434
return;
00435 }