mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
334 lines
9.7 KiB
C
334 lines
9.7 KiB
C
/*
|
|
* Cancel-Safe Queue Library
|
|
* Created in 2004 by Vizzini (vizzini@plasmic.com)
|
|
*
|
|
* THIS SOFTWARE IS NOT COPYRIGHTED
|
|
*
|
|
* This source code is offered for use in the public domain. You may
|
|
* use, modify or distribute it freely.
|
|
*
|
|
* This code is distributed in the hope that it will be useful but
|
|
* WITHOUT ANY WARRANTY. ALL WARRANTIES, EXPRESS OR IMPLIED ARE HEREBY
|
|
* DISCLAIMED. This includes but is not limited to warranties of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
*
|
|
*
|
|
* This header defines the interface to the ReactOS Cancel-Safe Queue library.
|
|
* This interface is based on and is similar to the Microsoft Cancel-Safe
|
|
* Queue interface.
|
|
*
|
|
* BACKGROUND
|
|
*
|
|
* IRP queuing is a royal pain in the butt, due to the fact that there are
|
|
* tons of built-in race conditions. IRP handling is difficult in general,
|
|
* but the cancel logic has been particularly complicated due to some subtle
|
|
* races, coupled with the fact that the system interfaces have changed over
|
|
* time.
|
|
*
|
|
* Walter Oney (2nd. Ed. of Programming the Windows Driver Model) states a
|
|
* common opinion among driver developers when he says that it is foolish
|
|
* to try to roll your own cancel logic. There are only a very few people
|
|
* who have gotten it right in the past. He suggests, instead, that you
|
|
* either use his own well-tested code, or use the code in the Microsoft
|
|
* Cancel-Safe Queue Library.
|
|
*
|
|
* We cannot do either, of course, due to copyright issues. I have therefore
|
|
* created this clone of the Microsoft library in order to concentrate all
|
|
* of the IRP-queuing bugs in one place. I'm quite sure there are problems
|
|
* here, so if you are a driver writer, I'd be glad to hear your feedback.
|
|
*
|
|
* Apart from that, please try to use these routines, rather than building
|
|
* your own. If you think you have found a bug, please bring it up with me
|
|
* or on-list, as this is complicated and non-obvious stuff. Don't just
|
|
* change this and hope for the best!
|
|
*
|
|
* USAGE
|
|
*
|
|
* This library follows exactly the same interface as the Microsoft Cancel-Safe
|
|
* Queue routines (IoCsqXxx()). As such, the authoritative reference is the
|
|
* current DDK. There is also a DDK sample called "cancel" that has an
|
|
* example of how to use this code. I have also provided a sample driver
|
|
* that makes use of this queue. Finally, please do read the header and the
|
|
* source if you're curious about the inner workings of these routines.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#define _CSQ_H_
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/*
|
|
* Prevent including the CSQ definitions twice. They're present in NTDDK
|
|
* now too, except the *_EX versions.
|
|
*/
|
|
#ifndef IO_TYPE_CSQ_IRP_CONTEXT
|
|
|
|
typedef struct _IO_CSQ IO_CSQ, *PIO_CSQ;
|
|
|
|
/*
|
|
* STRUCTURES
|
|
*
|
|
* NOTE: Please do not use these directly. You will make incompatible code
|
|
* if you do. Always only use the documented IoCsqXxx() interfaces and you
|
|
* will amass much Good Karma.
|
|
*/
|
|
#define IO_TYPE_CSQ_IRP_CONTEXT 1
|
|
#define IO_TYPE_CSQ 2
|
|
|
|
/*
|
|
* IO_CSQ_IRP_CONTEXT - Context used to track an IRP in the CSQ
|
|
*/
|
|
typedef struct _IO_CSQ_IRP_CONTEXT {
|
|
ULONG Type;
|
|
PIRP Irp;
|
|
PIO_CSQ Csq;
|
|
} IO_CSQ_IRP_CONTEXT, *PIO_CSQ_IRP_CONTEXT;
|
|
|
|
/*
|
|
* CSQ Callbacks
|
|
*
|
|
* The cancel-safe queue is implemented as a set of IoCsqXxx() OS routines
|
|
* copuled with a set of driver callbacks to handle the basic operations of
|
|
* the queue. You need to supply one of each of these functions in your own
|
|
* driver. These routines are also documented in the DDK under CsqXxx().
|
|
* That is the authoritative documentation.
|
|
*/
|
|
|
|
/*
|
|
* Function to insert an IRP in the queue. No need to worry about locking;
|
|
* just tack it onto your list or something.
|
|
*
|
|
* Sample implementation:
|
|
*
|
|
VOID NTAPI CsqInsertIrp(PIO_CSQ Csq, PIRP Irp)
|
|
{
|
|
KdPrint(("Inserting IRP 0x%x into CSQ\n", Irp));
|
|
InsertTailList(&IrpQueue, &Irp->Tail.Overlay.ListEntry);
|
|
}
|
|
*
|
|
*/
|
|
typedef VOID
|
|
(NTAPI IO_CSQ_INSERT_IRP)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_In_ PIRP Irp);
|
|
typedef IO_CSQ_INSERT_IRP *PIO_CSQ_INSERT_IRP;
|
|
|
|
/*
|
|
* Function to remove an IRP from the queue.
|
|
*
|
|
* Sample:
|
|
*
|
|
VOID NTAPI CsqRemoveIrp(PIO_CSQ Csq, PIRP Irp)
|
|
{
|
|
KdPrint(("Removing IRP 0x%x from CSQ\n", Irp));
|
|
RemoveEntryList(&Irp->Tail.Overlay.ListEntry);
|
|
}
|
|
*
|
|
*/
|
|
typedef VOID
|
|
(NTAPI IO_CSQ_REMOVE_IRP)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_In_ PIRP Irp);
|
|
typedef IO_CSQ_REMOVE_IRP *PIO_CSQ_REMOVE_IRP;
|
|
|
|
/*
|
|
* Function to look for an IRP in the queue
|
|
*
|
|
* Sample:
|
|
*
|
|
PIRP NTAPI CsqPeekNextIrp(PIO_CSQ Csq, PIRP Irp, PVOID PeekContext)
|
|
{
|
|
KdPrint(("Peeking for next IRP\n"));
|
|
|
|
if(Irp)
|
|
return CONTAINING_RECORD(&Irp->Tail.Overlay.ListEntry.Flink, IRP, Tail.Overlay.ListEntry);
|
|
|
|
if(IsListEmpty(&IrpQueue))
|
|
return NULL;
|
|
|
|
return CONTAINING_RECORD(IrpQueue.Flink, IRP, Tail.Overlay.ListEntry);
|
|
}
|
|
*
|
|
*/
|
|
typedef PIRP
|
|
(NTAPI IO_CSQ_PEEK_NEXT_IRP)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_In_opt_ PIRP Irp,
|
|
_In_opt_ PVOID PeekContext);
|
|
typedef IO_CSQ_PEEK_NEXT_IRP *PIO_CSQ_PEEK_NEXT_IRP;
|
|
|
|
/*
|
|
* Lock the queue. This can be a spinlock, a mutex, or whatever
|
|
* else floats your boat.
|
|
*
|
|
* Sample:
|
|
*
|
|
VOID NTAPI CsqAcquireLock(PIO_CSQ Csq, PKIRQL Irql)
|
|
{
|
|
KdPrint(("Acquiring spin lock\n"));
|
|
KeAcquireSpinLock(&IrpQueueLock, Irql);
|
|
}
|
|
*
|
|
*/
|
|
typedef VOID
|
|
(NTAPI IO_CSQ_ACQUIRE_LOCK)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_Out_ PKIRQL Irql);
|
|
typedef IO_CSQ_ACQUIRE_LOCK *PIO_CSQ_ACQUIRE_LOCK;
|
|
|
|
/*
|
|
* Unlock the queue:
|
|
*
|
|
VOID NTAPI CsqReleaseLock(PIO_CSQ Csq, KIRQL Irql)
|
|
{
|
|
KdPrint(("Releasing spin lock\n"));
|
|
KeReleaseSpinLock(&IrpQueueLock, Irql);
|
|
}
|
|
*
|
|
*/
|
|
typedef VOID
|
|
(NTAPI IO_CSQ_RELEASE_LOCK)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_In_ KIRQL Irql);
|
|
typedef IO_CSQ_RELEASE_LOCK *PIO_CSQ_RELEASE_LOCK;
|
|
|
|
/*
|
|
* Finally, this is called by the queue library when it wants to complete
|
|
* a canceled IRP.
|
|
*
|
|
* Sample:
|
|
*
|
|
VOID NTAPI CsqCompleteCancelledIrp(PIO_CSQ Csq, PIRP Irp)
|
|
{
|
|
KdPrint(("cancelling irp 0x%x\n", Irp));
|
|
Irp->IoStatus.Status = STATUS_CANCELLED;
|
|
Irp->IoStatus.Information = 0;
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
|
}
|
|
*
|
|
*/
|
|
typedef VOID
|
|
(NTAPI IO_CSQ_COMPLETE_CANCELED_IRP)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_In_ PIRP Irp);
|
|
typedef IO_CSQ_COMPLETE_CANCELED_IRP *PIO_CSQ_COMPLETE_CANCELED_IRP;
|
|
|
|
/*
|
|
* IO_CSQ - Queue control structure
|
|
*/
|
|
typedef struct _IO_CSQ {
|
|
ULONG Type;
|
|
PIO_CSQ_INSERT_IRP CsqInsertIrp;
|
|
PIO_CSQ_REMOVE_IRP CsqRemoveIrp;
|
|
PIO_CSQ_PEEK_NEXT_IRP CsqPeekNextIrp;
|
|
PIO_CSQ_ACQUIRE_LOCK CsqAcquireLock;
|
|
PIO_CSQ_RELEASE_LOCK CsqReleaseLock;
|
|
PIO_CSQ_COMPLETE_CANCELED_IRP CsqCompleteCanceledIrp;
|
|
PVOID ReservePointer; /* must be NULL */
|
|
} IO_CSQ, *PIO_CSQ;
|
|
|
|
#endif /* IO_TYPE_CSQ_IRP_CONTEXT */
|
|
|
|
#ifndef IO_TYPE_CSQ_EX
|
|
|
|
/* See IO_TYPE_CSQ_* above */
|
|
#define IO_TYPE_CSQ_EX 3
|
|
|
|
/*
|
|
* Function to insert an IRP into the queue with extended context information.
|
|
* This is useful if you need to be able to de-queue particular IRPs more
|
|
* easily in some cases.
|
|
*
|
|
* Same deal as above; sample implementation:
|
|
*
|
|
NTSTATUS NTAPI CsqInsertIrpEx(PIO_CSQ Csq, PIRP Irp, PVOID InsertContext)
|
|
{
|
|
CsqInsertIrp(Csq, Irp);
|
|
return STATUS_PENDING;
|
|
}
|
|
*
|
|
*/
|
|
typedef NTSTATUS
|
|
(NTAPI IO_CSQ_INSERT_IRP_EX)(
|
|
_In_ struct _IO_CSQ *Csq,
|
|
_In_ PIRP Irp,
|
|
_In_ PVOID InsertContext);
|
|
typedef IO_CSQ_INSERT_IRP_EX *PIO_CSQ_INSERT_IRP_EX;
|
|
|
|
#endif /* IO_TYPE_CSQ_EX */
|
|
|
|
/*
|
|
* CANCEL-SAFE QUEUE DDIs
|
|
*
|
|
* These device driver interfaces are called to make use of the queue. Again,
|
|
* authoritative documentation for these functions is in the DDK. The csqtest
|
|
* driver also makes use of some of them.
|
|
*/
|
|
|
|
|
|
/*
|
|
* Call this in DriverEntry or similar in order to set up the Csq structure.
|
|
* As long as the Csq struct and the functions you pass in are resident,
|
|
* there are no IRQL restrictions.
|
|
*/
|
|
NTKERNELAPI
|
|
NTSTATUS NTAPI IoCsqInitialize(_Out_ PIO_CSQ Csq,
|
|
_In_ PIO_CSQ_INSERT_IRP CsqInsertIrp,
|
|
_In_ PIO_CSQ_REMOVE_IRP CsqRemoveIrp,
|
|
_In_ PIO_CSQ_PEEK_NEXT_IRP CsqPeekNextIrp,
|
|
_In_ PIO_CSQ_ACQUIRE_LOCK CsqAcquireLock,
|
|
_In_ PIO_CSQ_RELEASE_LOCK CsqReleaseLock,
|
|
_In_ PIO_CSQ_COMPLETE_CANCELED_IRP CsqCompleteCanceledIrp);
|
|
|
|
/*
|
|
* Same as above, except you provide a CsqInsertIrpEx routine instead of
|
|
* CsqInsertIrp. This eventually allows you to supply extra tracking
|
|
* information for use with the queue.
|
|
*/
|
|
NTKERNELAPI
|
|
NTSTATUS NTAPI IoCsqInitializeEx(_Out_ PIO_CSQ Csq,
|
|
_In_ PIO_CSQ_INSERT_IRP_EX CsqInsertIrpEx,
|
|
_In_ PIO_CSQ_REMOVE_IRP CsqRemoveIrp,
|
|
_In_ PIO_CSQ_PEEK_NEXT_IRP CsqPeekNextIrp,
|
|
_In_ PIO_CSQ_ACQUIRE_LOCK CsqAcquireLock,
|
|
_In_ PIO_CSQ_RELEASE_LOCK CsqReleaseLock,
|
|
_In_ PIO_CSQ_COMPLETE_CANCELED_IRP CsqCompleteCanceledIrp);
|
|
|
|
/*
|
|
* Insert an IRP into the queue
|
|
*/
|
|
NTKERNELAPI
|
|
VOID NTAPI IoCsqInsertIrp(_Inout_ PIO_CSQ Csq,
|
|
_Inout_ PIRP Irp,
|
|
_Out_opt_ PIO_CSQ_IRP_CONTEXT Context);
|
|
|
|
/*
|
|
* Insert an IRP into the queue, with special context maintained that
|
|
* makes it easy to find IRPs in the queue
|
|
*/
|
|
NTKERNELAPI
|
|
NTSTATUS NTAPI IoCsqInsertIrpEx(_Inout_ PIO_CSQ Csq,
|
|
_Inout_ PIRP Irp,
|
|
_Out_opt_ PIO_CSQ_IRP_CONTEXT Context,
|
|
_In_opt_ PVOID InsertContext);
|
|
|
|
/*
|
|
* Remove a particular IRP from the queue
|
|
*/
|
|
NTKERNELAPI
|
|
PIRP NTAPI IoCsqRemoveIrp(_Inout_ PIO_CSQ Csq,
|
|
_Inout_ PIO_CSQ_IRP_CONTEXT Context);
|
|
|
|
/*
|
|
* Remove the next IRP from the queue
|
|
*/
|
|
NTKERNELAPI
|
|
PIRP NTAPI IoCsqRemoveNextIrp(_Inout_ PIO_CSQ Csq,
|
|
_In_opt_ PVOID PeekContext);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|