Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

3046 lines
91 KiB

/*++
Copyright (C) Microsoft Corporation, 1999 - 2000
Module Name:
MsTpUtil.c
Abstract:
Provide utility functions for MSTAPE.
Last changed by:
Author: Yee J. Wu
Environment:
Kernel mode only
Revision History:
$Revision:: $
$Date:: $
--*/
#include "strmini.h"
#include "ksmedia.h"
#include "1394.h"
#include "61883.h"
#include "avc.h"
#include "dbg.h"
#include "MsTpFmt.h"
#include "MsTpDef.h"
#include "MsTpAvc.h"
#include "MsTpUtil.h"
#include "XPrtDefs.h"
#if 0 // Enable later
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, DVDelayExecutionThread)
#pragma alloc_text(PAGE, DVGetUnitCapabilities)
// Local variables might paged out but the called might use it in DISPATCH level!
// #pragma alloc_text(PAGE, DVGetDevModeOfOperation)
// #pragma alloc_text(PAGE, DVGetDevIsItDVCPro)
// #pragma alloc_text(PAGE, DVGetDevSignalFormat)
#pragma alloc_text(PAGE, DvAllocatePCResource)
#pragma alloc_text(PAGE, DvFreePCResource)
#pragma alloc_text(PAGE, DVGetPlugState)
#endif
#endif
extern AVCSTRM_FORMAT_INFO AVCStrmFormatInfoTable[];
VOID
DVDelayExecutionThread(
ULONG ulDelayMSec
)
/*
Device might need a "wait" in between AV/C commands.
*/
{
PAGED_CODE();
if (ulDelayMSec)
{
LARGE_INTEGER tmDelay;
TRACE(TL_PNP_TRACE,("DelayExeThrd: %d MSec\n", ulDelayMSec));
tmDelay.LowPart = (ULONG) (-1 * ulDelayMSec * 10000);
tmDelay.HighPart = -1;
KeDelayExecutionThread(KernelMode, FALSE, &tmDelay);
}
}
NTSTATUS
DVIrpSynchCR(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP pIrp,
IN PKEVENT Event
)
{
KeSetEvent(Event, 0, FALSE);
return STATUS_MORE_PROCESSING_REQUIRED;
} // DVIrpSynchCR
NTSTATUS
DVSubmitIrpSynch(
IN PDVCR_EXTENSION pDevExt,
IN PIRP pIrp,
IN PAV_61883_REQUEST pAVReq
)
{
NTSTATUS Status;
KEVENT Event;
PIO_STACK_LOCATION NextIrpStack;
Status = STATUS_SUCCESS;;
NextIrpStack = IoGetNextIrpStackLocation(pIrp);
NextIrpStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
NextIrpStack->Parameters.DeviceIoControl.IoControlCode = IOCTL_61883_CLASS;
NextIrpStack->Parameters.Others.Argument1 = pAVReq;
KeInitializeEvent(&Event, NotificationEvent, FALSE);
IoSetCompletionRoutine(
pIrp,
DVIrpSynchCR,
&Event,
TRUE,
TRUE,
TRUE
);
Status =
IoCallDriver(
pDevExt->pBusDeviceObject,
pIrp
);
if (Status == STATUS_PENDING) {
TRACE(TL_PNP_TRACE,("Irp is pending...\n"));
if(KeGetCurrentIrql() < DISPATCH_LEVEL) {
KeWaitForSingleObject(
&Event,
Executive,
KernelMode,
FALSE,
NULL
);
TRACE(TL_PNP_TRACE,("Irp has completed; IoStatus.Status %x\n", pIrp->IoStatus.Status));
Status = pIrp->IoStatus.Status; // Final status
}
else {
ASSERT(FALSE && "Pending but in DISPATCH_LEVEL!");
return Status;
}
}
return Status;
} // DVSubmitIrpSynchAV
#ifdef SUPPORT_LOCAL_PLUGS
BOOL
AVCTapeCreateLocalPlug(
IN PDVCR_EXTENSION pDevExt,
IN AV_61883_REQUEST * pAVReq,
IN CMP_PLUG_TYPE PlugType,
IN AV_PCR *pPCR,
OUT ULONG *pPlugNumber,
OUT HANDLE *pPlugHandle
)
/*
To be a compliant device, we need to have both input and output
plugs in order to do isoch streaming. These plug is belong to
the device and is part of the device extension. In theory, the
lugs belong to the unit (ei.e. avc.sys) and not this subunit
Driver; however, in this case, we create directly from 61883.sys.
*/
{
NTSTATUS Status = STATUS_SUCCESS;
PIRP pIrp;
pIrp = IoAllocateIrp(pDevExt->pBusDeviceObject->StackSize, FALSE);
if(!pIrp)
return FALSE;
// Create a local oPCR
// Need to correctly update Overhead_ID and payload fields of PC's own oPCR
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_CreatePlug);
pAVReq->CreatePlug.Context = NULL;
pAVReq->CreatePlug.pfnNotify = NULL;
pAVReq->CreatePlug.PlugType = PlugType;
if(PlugType == CMP_PlugOut)
pAVReq->CreatePlug.Pcr.oPCR = pPCR->oPCR;
else
pAVReq->CreatePlug.Pcr.iPCR = pPCR->iPCR;
Status = DVSubmitIrpSynch(pDevExt, pIrp, pAVReq);
if(!NT_SUCCESS(Status)) {
*pPlugNumber = 0xffffffff;
*pPlugHandle = 0;
TRACE(TL_61883_ERROR,("Av61883_CreatePlug (%s) Failed:%x\n",
PlugType == CMP_PlugOut ? "oPCR":"iPCR", Status));
} else {
*pPlugNumber = pAVReq->CreatePlug.PlugNum;
*pPlugHandle = pAVReq->CreatePlug.hPlug;
TRACE(TL_61883_TRACE,("Av61883_CreatePlug (%s): PlugNum:%d, hPlug:%x\n",
PlugType == CMP_PlugOut ? "oPCR":"iPCR", *pPlugNumber, *pPlugHandle));
#if DBG
if(PlugType == CMP_PlugOut) {
TRACE(TL_61883_WARNING,("Av61883_CreatePlug: oPCR DataRate:%d (%s); Payload:%d, Overhead_ID:0x%x\n",
pPCR->oPCR.DataRate,
(pPCR->oPCR.DataRate == CMP_SPEED_S100) ? "S100" :
(pPCR->oPCR.DataRate == CMP_SPEED_S200) ? "S200" :
(pPCR->oPCR.DataRate == CMP_SPEED_S400) ? "S400" : "Sxxx",
pPCR->oPCR.Payload,
pPCR->oPCR.OverheadID
));
}
#endif
}
IoFreeIrp(pIrp);
pIrp = NULL;
return NT_SUCCESS(Status);
}
BOOL
AVCTapeDeleteLocalPlug(
IN PDVCR_EXTENSION pDevExt,
IN AV_61883_REQUEST * pAVReq,
OUT ULONG *pPlugNumber,
OUT HANDLE *pPlugHandle
)
/*
Delete a local plug.
*/
{
NTSTATUS Status = STATUS_SUCCESS;
PIRP pIrp;
TRACE(TL_61883_TRACE,("Deleting hPlug[%d]:%x\n", *pPlugNumber, *pPlugHandle));
pIrp = IoAllocateIrp(pDevExt->pBusDeviceObject->StackSize, FALSE);
if(!pIrp)
return FALSE;
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_DeletePlug);
pAVReq->DeletePlug.hPlug = *pPlugHandle;
Status = DVSubmitIrpSynch(pDevExt, pIrp, pAVReq);
if(!NT_SUCCESS(Status)) {
TRACE(TL_61883_ERROR,("Av61883_DeletePlug Failed; ST:%x\n", Status));
// Do not care if this result in error.
} else {
*pPlugNumber = 0xffffffff;
*pPlugHandle = 0;
TRACE(TL_61883_TRACE,("Av61883_DeltePlug suceeded.\n"));
}
IoFreeIrp(pIrp);
pIrp = NULL;
return NT_SUCCESS(Status);
}
BOOL
AVCTapeSetLocalPlug(
IN PDVCR_EXTENSION pDevExt,
IN AV_61883_REQUEST * pAVReq,
IN HANDLE *pPlugHandle,
IN AV_PCR *pPCR
)
/*
Set the content of a local plug.
*/
{
NTSTATUS Status = STATUS_SUCCESS;
PIRP pIrp;
pIrp = IoAllocateIrp(pDevExt->pBusDeviceObject->StackSize, FALSE);
if(!pIrp)
return FALSE;
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_SetPlug);
pAVReq->SetPlug.hPlug = *pPlugHandle;
pAVReq->SetPlug.Pcr = *pPCR;
TRACE(TL_61883_TRACE,("Av61883_SetPlug hPlug:%x to %x.\n", *pPlugHandle, pPCR->ulongData));
Status = DVSubmitIrpSynch(pDevExt, pIrp, pAVReq);
#if DBG
if(!NT_SUCCESS(Status)) {
TRACE(TL_61883_ERROR,("Av61883_SetPlug to %x Failed; ST:%x\n", pPCR->ulongData, Status));
}
#endif
IoFreeIrp(pIrp);
pIrp = NULL;
return NT_SUCCESS(Status);
}
#endif // SUPPORT_LOCAL_PLUGS
//
// Get device plug and query its state
//
NTSTATUS
AVCDevGetDevPlug(
IN PDVCR_EXTENSION pDevExt,
IN CMP_PLUG_TYPE PlugType,
IN ULONG PlugNum,
OUT HANDLE *pPlugHandle
)
/*++
Routine Description:
Get the targe device's plug handle
Arguments:
Return Value:
STATUS_SUCCESS
STATUS_INSUFFICIENT_RESOURCES
status return from 61883.
--*/
{
PIRP pIrp;
PAV_61883_REQUEST pAVReq;
NTSTATUS Status = STATUS_SUCCESS;
PAGED_CODE();
if(!(pIrp = IoAllocateIrp(pDevExt->pBusDeviceObject->StackSize, FALSE)))
return STATUS_INSUFFICIENT_RESOURCES;
if(!(pAVReq = (AV_61883_REQUEST *) ExAllocatePool(NonPagedPool, sizeof(AV_61883_REQUEST)))) {
IoFreeIrp(pIrp); pIrp = NULL;
return STATUS_INSUFFICIENT_RESOURCES;
}
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_GetPlugHandle);
pAVReq->GetPlugHandle.PlugNum = PlugNum;
pAVReq->GetPlugHandle.hPlug = 0;
pAVReq->GetPlugHandle.Type = PlugType;
if(NT_SUCCESS(
Status = DVSubmitIrpSynch(
pDevExt,
pIrp,
pAVReq
))) {
*pPlugHandle = pAVReq->GetPlugHandle.hPlug;
TRACE(TL_61883_WARNING,("Created h%sPlugDV[%d]=%x\n", PlugType == CMP_PlugIn ? "I" : "O", PlugNum, *pPlugHandle));
} else {
TRACE(TL_61883_ERROR,("Created h%sPlugDV[%d] failed; Status:%x\n", PlugType == CMP_PlugIn ? "I" : "O", PlugNum, Status));
Status = STATUS_INSUFFICIENT_RESOURCES;
}
IoFreeIrp(pIrp); pIrp = NULL;
ExFreePool(pAVReq); pAVReq = NULL;
return Status;
}
NTSTATUS
AVCDevGetPlugState(
IN PDVCR_EXTENSION pDevExt,
IN HANDLE hPlug,
OUT CMP_GET_PLUG_STATE *pPlugState
)
/*++
Routine Description:
Ask 61883.sys for the plug state.
Arguments:
Return Value:
Nothing
--*/
{
PIRP pIrp;
PAV_61883_REQUEST pAVReq;
NTSTATUS Status = STATUS_SUCCESS;
PAGED_CODE();
if(!hPlug || !pPlugState)
return STATUS_INVALID_PARAMETER;
if(!(pIrp = IoAllocateIrp(pDevExt->pBusDeviceObject->StackSize, FALSE)))
return STATUS_INSUFFICIENT_RESOURCES;
if(!(pAVReq = (AV_61883_REQUEST *) ExAllocatePool(NonPagedPool, sizeof(AV_61883_REQUEST)))) {
IoFreeIrp(pIrp); pIrp = NULL;
return STATUS_INSUFFICIENT_RESOURCES;
}
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_GetPlugState);
pAVReq->GetPlugState.hPlug = hPlug;
if(NT_SUCCESS(
Status = DVSubmitIrpSynch(
pDevExt,
pIrp,
pAVReq
))) {
//
// Transfer plug state (note: these are dynamic values)
//
*pPlugState = pAVReq->GetPlugState;
TRACE(TL_61883_WARNING,("GetPlugState: ST %x; State %x; DRate %d (%s); Payld %d; BCCnt %d; PPCnt %d\n",
pAVReq->Flags ,
pAVReq->GetPlugState.State,
pAVReq->GetPlugState.DataRate,
(pAVReq->GetPlugState.DataRate == CMP_SPEED_S100) ? "S100" :
(pAVReq->GetPlugState.DataRate == CMP_SPEED_S200) ? "S200" :
(pAVReq->GetPlugState.DataRate == CMP_SPEED_S400) ? "S400" : "Sxxx",
pAVReq->GetPlugState.Payload,
pAVReq->GetPlugState.BC_Connections,
pAVReq->GetPlugState.PP_Connections
));
}
else {
TRACE(TL_61883_ERROR,("GetPlugState Failed %x\n", Status));
}
IoFreeIrp(pIrp); pIrp = NULL;
ExFreePool(pAVReq); pAVReq = NULL;
return Status;
}
#ifndef NT51_61883
NTSTATUS
AVCDevSubmitIrpSynch1394(
IN PDEVICE_OBJECT pDevObj,
IN PIRP pIrp,
IN PIRB pIrb
)
{
NTSTATUS Status;
KEVENT Event;
PIO_STACK_LOCATION NextIrpStack;
Status = STATUS_SUCCESS;;
NextIrpStack = IoGetNextIrpStackLocation(pIrp);
NextIrpStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
NextIrpStack->Parameters.DeviceIoControl.IoControlCode = IOCTL_1394_CLASS;
NextIrpStack->Parameters.Others.Argument1 = pIrb;
KeInitializeEvent(&Event, NotificationEvent, FALSE);
IoSetCompletionRoutine(
pIrp,
DVIrpSynchCR,
&Event,
TRUE,
TRUE,
TRUE
);
Status =
IoCallDriver(
pDevObj,
pIrp
);
if (Status == STATUS_PENDING) {
TRACE(TL_PNP_TRACE,("Irp is pending...\n"));
if(KeGetCurrentIrql() < DISPATCH_LEVEL) {
KeWaitForSingleObject(
&Event,
Executive,
KernelMode,
FALSE,
NULL
);
TRACE(TL_PNP_TRACE,("Irp has completed; IoStatus.Status %x\n", pIrp->IoStatus.Status));
Status = pIrp->IoStatus.Status; // Final status
}
else {
ASSERT(FALSE && "Pending but in DISPATCH_LEVEL!");
return Status;
}
}
return Status;
} // AVCDevSubmitIrpSynch1394
NTSTATUS
Av1394_GetGenerationCount(
IN PDVCR_EXTENSION pDevExt,
OUT PULONG pGenerationCount
)
{
NTSTATUS ntStatus = STATUS_SUCCESS;
PIRP pIrp = NULL;
PIRB p1394Irb = NULL;
CCHAR StackSize;
PAGED_CODE();
StackSize = pDevExt->pBusDeviceObject->StackSize;
pIrp = IoAllocateIrp(StackSize, FALSE);
p1394Irb = ExAllocatePool(NonPagedPool, sizeof(IRB));
if ((pIrp == NULL) || (p1394Irb == NULL)) {
TRACE(TL_PNP_ERROR, ("Failed to allocate pIrp (%x) or p1394Irb (%x)", pIrp, p1394Irb));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto Exit_GetGenerationCount;
}
//
// Get the current generation count first
//
p1394Irb->FunctionNumber = REQUEST_GET_GENERATION_COUNT;
p1394Irb->Flags = 0;
ntStatus = AVCDevSubmitIrpSynch1394(pDevExt->pBusDeviceObject, pIrp, p1394Irb);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("REQUEST_GET_GENERATION_COUNT Failed %x", ntStatus));
goto Exit_GetGenerationCount;
}
*pGenerationCount = p1394Irb->u.GetGenerationCount.GenerationCount;
Exit_GetGenerationCount:
if(pIrp) {
IoFreeIrp(pIrp); pIrp = NULL;
}
if(p1394Irb) {
ExFreePool(p1394Irb); p1394Irb = NULL;
}
return(ntStatus);
} // Av1394_GetGenerationCount
#define RETRY_COUNT 4
//
// IEEE 1212 Directory definition
//
typedef struct _DIRECTORY_INFO {
union {
USHORT DI_CRC;
USHORT DI_Saved_Length;
} u;
USHORT DI_Length;
} DIRECTORY_INFO, *PDIRECTORY_INFO;
//
// IEEE 1212 Immediate entry definition
//
typedef struct _IMMEDIATE_ENTRY {
ULONG IE_Value:24;
ULONG IE_Key:8;
} IMMEDIATE_ENTRY, *PIMMEDIATE_ENTRY;
NTSTATUS
Av1394_QuadletRead(
IN PDVCR_EXTENSION pDevExt,
IN OUT PULONG pData,
IN ULONG Address
)
{
NTSTATUS ntStatus = STATUS_SUCCESS;
PIRP pIrp;
PIRB p1394Irb;
PMDL Mdl = NULL;
ULONG Retries = RETRY_COUNT;
CCHAR StackSize;
PAGED_CODE();
StackSize = pDevExt->pBusDeviceObject->StackSize;
pIrp = IoAllocateIrp(StackSize, FALSE);
p1394Irb = ExAllocatePool(NonPagedPool, sizeof(IRB));
if ((pIrp == NULL) || (p1394Irb == NULL)) {
TRACE(TL_PNP_ERROR, ("Failed to allocate Irp (0x%x) or Irb (0x%x)", pIrp, p1394Irb));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto Exit_Av1394_QuadletRead;
}
Mdl = IoAllocateMdl(pData, sizeof(ULONG), FALSE, FALSE, NULL);
if (!Mdl) {
TRACE(TL_PNP_ERROR, ("Failed to allocate Mdl!"));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto Exit_Av1394_QuadletRead;
}
MmBuildMdlForNonPagedPool(Mdl);
do {
p1394Irb->FunctionNumber = REQUEST_ASYNC_READ;
p1394Irb->Flags = 0;
p1394Irb->u.AsyncRead.DestinationAddress.IA_Destination_Offset.Off_High = (USHORT)0xffff;
p1394Irb->u.AsyncRead.DestinationAddress.IA_Destination_Offset.Off_Low = Address;
p1394Irb->u.AsyncRead.nNumberOfBytesToRead = 4;
p1394Irb->u.AsyncRead.nBlockSize = 0;
p1394Irb->u.AsyncRead.fulFlags = 0;
p1394Irb->u.AsyncRead.Mdl = Mdl;
p1394Irb->u.AsyncRead.ulGeneration = pDevExt->GenerationCount;
p1394Irb->u.AsyncRead.chPriority = 0;
p1394Irb->u.AsyncRead.nSpeed = 0;
p1394Irb->u.AsyncRead.tCode = 0;
p1394Irb->u.AsyncRead.Reserved = 0;
ntStatus = AVCDevSubmitIrpSynch1394(pDevExt->pBusDeviceObject, pIrp, p1394Irb);
if (ntStatus == STATUS_INVALID_GENERATION) {
TRACE(TL_PNP_WARNING, ("QuadletRead: Invalid GenerationCount = %d", pDevExt->GenerationCount));
Av1394_GetGenerationCount(pDevExt, &pDevExt->GenerationCount);
}
else if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("Av1394_QuadletRead Failed = 0x%x Address = 0x%x", ntStatus, Address));
}
else {
goto Exit_Av1394_QuadletRead;
}
} while ((ntStatus == STATUS_INVALID_GENERATION) || (Retries--));
Exit_Av1394_QuadletRead:
if(pIrp) {
IoFreeIrp(pIrp); pIrp = NULL;
}
if(p1394Irb) {
ExFreePool(p1394Irb); p1394Irb = NULL;
}
if(Mdl) {
IoFreeMdl(Mdl); Mdl = NULL;
}
return(ntStatus);
} // Av1394_QuadletRead
#define KEY_ModuleVendorId (0x03)
#define KEY_ModuleHwVersion (0x04)
#define KEY_UnitSwVersion (0x13)
#define KEY_ModelId (0x17)
#define DEVICE_NAME_MAX_CHARS 100*sizeof(WCHAR)
NTSTATUS
Av1394_ReadTextualDescriptor(
IN PDVCR_EXTENSION pDevExt,
IN OUT PUNICODE_STRING uniString,
IN ULONG Address
)
{
NTSTATUS ntStatus = STATUS_SUCCESS;
PULONG pData = NULL;
ULONG DataLength, i, n;
ULONG ulUnicode;
ULONG ulQuadlet;
union {
ULONG asUlong;
UCHAR asUchar[4];
DIRECTORY_INFO DirectoryHeader;
} u;
PAGED_CODE();
TRACE(TL_PNP_TRACE, ("Address = 0x%x", Address));
// read the first quadlet of leaf, this is the header
ntStatus = Av1394_QuadletRead(pDevExt, &ulQuadlet, Address);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("GetUnitInfo: QuadletRead Error = 0x%x", ntStatus));
goto Exit_Av1394_ReadTextualDescriptor;
}
// number of entries
u.asUlong = bswap(ulQuadlet);
DataLength = u.DirectoryHeader.DI_Length-2; // one extra for the header
// read the second quadlet of leaf to determine unicode
Address += 4;
ntStatus = Av1394_QuadletRead(pDevExt, &ulQuadlet, Address);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("GetUnitInfo: QuadletRead Error = 0x%x", ntStatus));
goto Exit_Av1394_ReadTextualDescriptor;
}
// save spec type
ulUnicode = bswap(ulQuadlet);
pData = ExAllocatePool(NonPagedPool, DataLength*sizeof(ULONG)+2);
if (pData == NULL) {
TRACE(TL_PNP_ERROR, ("Failed to allocate pData"));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto Exit_Av1394_ReadTextualDescriptor;
}
RtlZeroMemory(pData, DataLength*sizeof(ULONG)+2);
// lets read in each quad
Address += 8;
for (i=0; i<DataLength; i++) {
ntStatus = Av1394_QuadletRead(pDevExt, &u.asUlong, Address+(sizeof(ULONG)*i));
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("GetUnitInfo: QuadletRead Error = 0x%x", ntStatus));
goto Exit_Av1394_ReadTextualDescriptor;
}
// need to make sure we have valid characters...
for (n=0; n<4; n++) {
// we should be done if the char equals 0x00
if (u.asUchar[n] == 0x00)
break;
if ((u.asUchar[n] == 0x2C) || (u.asUchar[n] < 0x20) || (u.asUchar[n] > 0x7F)) {
TRACE(TL_PNP_WARNING, ("Invalid Character = 0x%x", u.asUchar[n]));
// set it to space
u.asUchar[n] = 0x20;
}
if (ulUnicode & 0x80000000)
n++;
}
RtlCopyMemory((PULONG)pData+i, &u.asUlong, sizeof(ULONG));
}
// if there's a vendor leaf, then convert it to unicode
{
ANSI_STRING ansiString;
uniString->Length = 0;
uniString->MaximumLength = DEVICE_NAME_MAX_CHARS;
uniString->Buffer = ExAllocatePool(NonPagedPool, uniString->MaximumLength);
if (!uniString->Buffer) {
TRACE(TL_PNP_ERROR, ("Failed to allocate uniString.Buffer!"));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto Exit_Av1394_ReadTextualDescriptor;
}
RtlZeroMemory(uniString->Buffer, uniString->MaximumLength);
// unicode??
if (ulUnicode & 0x80000000) {
RtlAppendUnicodeToString(uniString, ((PWSTR)pData));
}
else {
RtlInitAnsiString(&ansiString, (PUCHAR)pData);
RtlAnsiStringToUnicodeString(uniString, &ansiString, FALSE);
}
}
Exit_Av1394_ReadTextualDescriptor:
if (pData)
ExFreePool(pData);
return(ntStatus);
} // ReadTextualLeaf
NTSTATUS
AVCDevGetModelText(
IN PDVCR_EXTENSION pDevExt,
PUNICODE_STRING pUniRootModelString,
PUNICODE_STRING pUniUnitModelString
)
{
CCHAR StackSize;
PIRP pIrp = NULL;
PIRB p1394Irb = NULL;
NTSTATUS ntStatus = STATUS_SUCCESS;
CONFIG_ROM ConfigRom;
ULONG ulQuadlet = 0;
ULONG CurrAddress;
PULONG UnitDir = NULL, UnitDirToFree = NULL;
ULONG i;
ULONG LastKey;
union {
ULONG asUlong;
DIRECTORY_INFO DirInfo;
IMMEDIATE_ENTRY Entry;
} u, u2; //, u3;
PAGED_CODE();
StackSize = pDevExt->pBusDeviceObject->StackSize;
pIrp = IoAllocateIrp(StackSize, FALSE);
p1394Irb = ExAllocatePool(NonPagedPool, sizeof(IRB));
if ((pIrp == NULL) || (p1394Irb == NULL)) {
TRACE(TL_PNP_ERROR, ("Failed to allocate pIrp (%x) or p1394Irb (%x)", pIrp, p1394Irb));
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Get the current generation count (used to read config rom)
//
Av1394_GetGenerationCount(pDevExt, &pDevExt->GenerationCount);
//
// Get Model Text from the Root directory
//
CurrAddress = 0xF0000414;
// root directory
ntStatus = Av1394_QuadletRead(pDevExt, &ulQuadlet, CurrAddress);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("GetUnitInfo: QuadletRead Error = 0x%x", ntStatus));
goto Exit_GetUnitInfo;
}
u.asUlong = bswap(ulQuadlet);
TRACE(TL_PNP_TRACE, ("RootDir: Length = %d", u.DirInfo.DI_Length));
// process the root directory
for (i=0; i<u.DirInfo.DI_Length; i++) {
CurrAddress += sizeof(ULONG);
ntStatus = Av1394_QuadletRead(pDevExt, &ulQuadlet, CurrAddress);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("GetUnitInfo: QuadletRead Error = 0x%x", ntStatus));
goto Exit_GetUnitInfo;
}
u2.asUlong = bswap(ulQuadlet);
TRACE(TL_PNP_TRACE, ("CurrAddress = 0x%x Key = 0x%x Value = 0x%x",
CurrAddress, u2.Entry.IE_Key, u2.Entry.IE_Value));
// ModelId Textual Descriptor
if ((u2.Entry.IE_Key == 0x81) && (LastKey == KEY_ModelId)) {
// get the first entry of the textual descriptor
Av1394_ReadTextualDescriptor( pDevExt,
pUniRootModelString,
CurrAddress+(u2.Entry.IE_Value*sizeof(ULONG))
);
}
#if 0
// ModelId Textual Descriptor Layer
if ((u2.Entry.IE_Key == 0xC1) && (LastKey == KEY_ModelId)) {
ULONG DescAddress;
DescAddress = CurrAddress+(u2.Entry.IE_Value*sizeof(ULONG));
Av1394_QuadletRead(pDevExt, &ulQuadlet, DescAddress);
u3.asUlong = bswap(ulQuadlet);
// get the first entry of the textual descriptor
Av1394_ReadTextualDescriptor( pDevExt,
pUniRootModelString,
DescAddress+(u3.Entry.IE_Value*sizeof(ULONG))
);
}
#endif
LastKey = u2.Entry.IE_Key;
}
//
// Get Configuration Info
//
p1394Irb->FunctionNumber = REQUEST_GET_CONFIGURATION_INFO;
p1394Irb->Flags = 0;
p1394Irb->u.GetConfigurationInformation.ConfigRom = NULL;
p1394Irb->u.GetConfigurationInformation.UnitDirectoryBufferSize = 0;
p1394Irb->u.GetConfigurationInformation.UnitDirectory = NULL;
p1394Irb->u.GetConfigurationInformation.UnitDependentDirectoryBufferSize = 0;
p1394Irb->u.GetConfigurationInformation.UnitDependentDirectory = NULL;
p1394Irb->u.GetConfigurationInformation.VendorLeafBufferSize = 0;
p1394Irb->u.GetConfigurationInformation.VendorLeaf = NULL;
p1394Irb->u.GetConfigurationInformation.ModelLeafBufferSize = 0;
p1394Irb->u.GetConfigurationInformation.ModelLeaf = NULL;
ntStatus = AVCDevSubmitIrpSynch1394(pDevExt->pBusDeviceObject, pIrp, p1394Irb);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("REQUEST_GET_CONFIGURATION_INFO Failed %x", ntStatus));
goto Exit_GetUnitInfo;
}
//
// Allocate buffer in order retrieve unit directory of a config rom
//
if (p1394Irb->u.GetConfigurationInformation.UnitDirectoryBufferSize) {
UnitDir = UnitDirToFree =
p1394Irb->u.GetConfigurationInformation.UnitDirectory =
ExAllocatePool(NonPagedPool, p1394Irb->u.GetConfigurationInformation.UnitDirectoryBufferSize);
if (!p1394Irb->u.GetConfigurationInformation.UnitDirectory) {
TRACE(TL_PNP_ERROR, ("Couldn't allocate memory for the UnitDirectory"));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto Exit_GetUnitInfo;
}
}
else {
TRACE(TL_PNP_ERROR, ("No Unit directory. Bad Device."));
ntStatus = STATUS_BAD_DEVICE_TYPE;
goto Exit_GetUnitInfo;
}
p1394Irb->u.GetConfigurationInformation.ConfigRom = &ConfigRom;
p1394Irb->u.GetConfigurationInformation.UnitDependentDirectoryBufferSize = 0;
p1394Irb->u.GetConfigurationInformation.VendorLeafBufferSize = 0;
p1394Irb->u.GetConfigurationInformation.ModelLeafBufferSize = 0;
ntStatus = AVCDevSubmitIrpSynch1394(pDevExt->pBusDeviceObject, pIrp, p1394Irb);
if (!NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_ERROR, ("2nd REQUEST_GET_CONFIGURATION_INFO Failed = 0x%x", ntStatus));
goto Exit_GetUnitInfo;
}
//
// Process unit directory; see this doc for detail:
// 1394TA specification: Configuration ROM for AV/C device 1.0 (AVWG)
//
u.asUlong = bswap(*UnitDir++); // Get length, and dkip first quadlet
TRACE(TL_PNP_TRACE, ("UnitDir: Length = %d", u.DirInfo.DI_Length));
CurrAddress = p1394Irb->u.GetConfigurationInformation.UnitDirectoryLocation.IA_Destination_Offset.Off_Low;
for (i=0; i<u.DirInfo.DI_Length; i++) {
TRACE(TL_PNP_TRACE, ("i = %d UnitDir = 0x%x *UnitDir = 0x%x", i, UnitDir, *UnitDir));
u2.asUlong = bswap(*UnitDir++);
CurrAddress += sizeof(ULONG);
TRACE(TL_PNP_TRACE, ("UnitDir Quadlet = 0x%x", u2.asUlong));
//
// ModelId Textual Descriptor
//
if ((u2.Entry.IE_Key == 0x81) && (LastKey == KEY_ModelId)) {
// get the first entry of the textual descriptor
Av1394_ReadTextualDescriptor(
pDevExt,
pUniUnitModelString,
CurrAddress+(u2.Entry.IE_Value*sizeof(ULONG))
);
}
#if 0
//
// UnitModelId Textual Descriptor Layer
//
if ((u2.Entry.IE_Key == 0xC1) && (LastKey == KEY_ModelId)) {
ULONG DescAddress;
DescAddress = CurrAddress+(u2.Entry.IE_Value*sizeof(ULONG));
Av1394_QuadletRead(pDevExt, &ulQuadlet, DescAddress);
u3.asUlong = bswap(ulQuadlet);
// get the first entry of the textual descriptor
Av1394_ReadTextualDescriptor(
pDevExt,
pUniUnitModelString,
DescAddress+(u3.Entry.IE_Value*sizeof(ULONG))
);
}
#endif
LastKey = u2.Entry.IE_Key;
}
Exit_GetUnitInfo:
if (UnitDirToFree) {
ExFreePool(UnitDirToFree); UnitDirToFree = NULL;
}
if(pIrp) {
IoFreeIrp(pIrp); pIrp = NULL;
}
if(p1394Irb) {
ExFreePool(p1394Irb); p1394Irb = NULL;
}
return ntStatus;
}
#endif
NTSTATUS
DVGetUnitCapabilities(
IN PDVCR_EXTENSION pDevExt,
IN PIRP pIrp,
IN PAV_61883_REQUEST pAVReq
)
{
NTSTATUS Status;
GET_UNIT_IDS * pUnitIds;
GET_UNIT_CAPABILITIES * pUnitCaps;
PAGED_CODE();
//
// Query device's capability
//
pUnitCaps = (GET_UNIT_CAPABILITIES *) ExAllocatePool(NonPagedPool, sizeof(GET_UNIT_CAPABILITIES));
if(!pUnitCaps) {
TRACE(TL_61883_ERROR,("DVGetUnitCapabilities: Allocate pUnitCaps (%d bytes) failed\n", sizeof(GET_UNIT_CAPABILITIES)));
return STATUS_INSUFFICIENT_RESOURCES;
}
// UnitIDS is cached in DevExt.
pUnitIds = &pDevExt->UnitIDs;
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_GetUnitInfo);
pAVReq->GetUnitInfo.nLevel = GET_UNIT_INFO_IDS;
RtlZeroMemory(pUnitIds, sizeof(GET_UNIT_IDS)); // Initialize pointers.
pAVReq->GetUnitInfo.Information = (PVOID) pUnitIds;
Status =
DVSubmitIrpSynch(
pDevExt,
pIrp,
pAVReq
);
if(!NT_SUCCESS(Status)) {
TRACE(TL_61883_ERROR,("Av61883_GetUnitCapabilities Failed = 0x%x\n", Status));
pDevExt->UniqueID.QuadPart = 0;
pDevExt->ulVendorID = 0;
pDevExt->ulModelID = 0;
}
else {
pDevExt->UniqueID = pUnitIds->UniqueID;
pDevExt->ulVendorID = pUnitIds->VendorID;
pDevExt->ulModelID = pUnitIds->ModelID;
TRACE(TL_61883_TRACE,("UniqueId:(Low)%x:(High)%x; VendorID:%x; ModelID:%x\n",
pDevExt->UniqueID.LowPart, pDevExt->UniqueID.HighPart, pDevExt->ulVendorID, pDevExt->ulModelID));
//
// Allocate memory needed for the text string for VendorText,
// ModelText and UntiModelText.
//
if(pUnitIds->ulVendorLength) {
pUnitIds->VendorText = (PWSTR) ExAllocatePool(NonPagedPool, pUnitIds->ulVendorLength);
if(!pUnitIds->VendorText)
goto AbortGetUnitCapabilities;
}
if(pUnitIds->ulModelLength) {
pUnitIds->ModelText = (PWSTR) ExAllocatePool(NonPagedPool, pUnitIds->ulModelLength);
if(!pUnitIds->ModelText)
goto AbortGetUnitCapabilities;
}
#ifdef NT51_61883
if(pUnitIds->ulUnitModelLength) {
pUnitIds->UnitModelText = (PWSTR) ExAllocatePool(NonPagedPool, pUnitIds->ulUnitModelLength);
if(!pUnitIds->UnitModelText)
goto AbortGetUnitCapabilities;
}
#else
//
// 1st version of 61883.sys does not retrieve Root and Unit model text
// the same way as in WinXP; so we retieve them directly using 1394 API
//
if(!NT_SUCCESS(AVCDevGetModelText(
pDevExt,
&pDevExt->UniRootModelString,
&pDevExt->UniUnitModelString
))) {
goto AbortGetUnitCapabilities;
}
#endif
Status =
DVSubmitIrpSynch(
pDevExt,
pIrp,
pAVReq
);
}
RtlZeroMemory(pAVReq, sizeof(AV_61883_REQUEST));
INIT_61883_HEADER(pAVReq, Av61883_GetUnitInfo);
pAVReq->GetUnitInfo.nLevel = GET_UNIT_INFO_CAPABILITIES;
RtlZeroMemory(pUnitCaps, sizeof(GET_UNIT_CAPABILITIES)); // Initialize pointers.
pAVReq->GetUnitInfo.Information = (PVOID) pUnitCaps;
Status =
DVSubmitIrpSynch(
pDevExt,
pIrp,
pAVReq
);
if(!NT_SUCCESS(Status)) {
TRACE(TL_61883_ERROR,("Av61883_GetUnitCapabilities Failed = 0x%x\n", Status));
pDevExt->pDevOutPlugs->MaxDataRate = 0;
pDevExt->pDevOutPlugs->NumPlugs = 0;
pDevExt->pDevInPlugs->MaxDataRate = 0;
pDevExt->pDevInPlugs->NumPlugs = 0;
}
else {
//
// There can never be more than MAX_NUM_PCR (= 31) of plugs
//
ASSERT(pUnitCaps->NumOutputPlugs <= MAX_NUM_PCR);
ASSERT(pUnitCaps->NumInputPlugs <= MAX_NUM_PCR);
pDevExt->pDevOutPlugs->MaxDataRate = pUnitCaps->MaxDataRate;
pDevExt->pDevOutPlugs->NumPlugs = (pUnitCaps->NumOutputPlugs > MAX_NUM_PCR ? MAX_NUM_PCR : pUnitCaps->NumOutputPlugs);
pDevExt->pDevInPlugs->MaxDataRate = pUnitCaps->MaxDataRate;
pDevExt->pDevInPlugs->NumPlugs = (pUnitCaps->NumInputPlugs > MAX_NUM_PCR ? MAX_NUM_PCR : pUnitCaps->NumInputPlugs);
}
TRACE(TL_61883_TRACE,("** UnitCaps: OutP:%d; InP:%d; MDRate:%s; CtsF:%x; HwF:%x; VID:%x; MID:%x\n",
pUnitCaps->NumOutputPlugs,
pUnitCaps->NumInputPlugs,
pUnitCaps->MaxDataRate == 0 ? "S100": pUnitCaps->MaxDataRate == 1? "S200" : "S400 or +",
pUnitCaps->CTSFlags,
pUnitCaps->HardwareFlags,
pUnitIds->VendorID,
pUnitIds->ModelID
));
AbortGetUnitCapabilities:
if(pUnitIds->ulVendorLength && pUnitIds->VendorText) {
TRACE(TL_61883_TRACE,("Vendor: Len:%d; \"%S\"\n", pUnitIds->ulVendorLength, pUnitIds->VendorText));
if(!NT_SUCCESS(Status)) {
ExFreePool(pUnitIds->VendorText); pUnitIds->VendorText = NULL;
}
}
if(pUnitIds->ulModelLength && pUnitIds->ModelText) {
TRACE(TL_61883_TRACE,("Model: Len:%d; \"%S\"\n", pUnitIds->ulModelLength, pUnitIds->ModelText));
if(!NT_SUCCESS(Status)) {
ExFreePool(pUnitIds->ModelText); pUnitIds->ModelText = NULL;
}
}
#ifdef NT51_61883
if(pUnitIds->ulUnitModelLength && pUnitIds->UnitModelText) {
TRACE(TL_61883_TRACE,("UnitModel (61883): Len:%d; \"%S\"\n", pUnitIds->ulUnitModelLength, pUnitIds->UnitModelText));
if(!NT_SUCCESS(Status)) {
ExFreePool(pUnitIds->UnitModelText); pUnitIds->UnitModelText = NULL;
}
}
#else
if(pDevExt->UniRootModelString.Length && pDevExt->UniRootModelString.Buffer) {
TRACE(TL_61883_TRACE,("RootModel (MSTape): Len:%d; \"%S\"\n", pDevExt->UniRootModelString.Length, pDevExt->UniRootModelString.Buffer));
if(!NT_SUCCESS(Status)) {
ExFreePool(pDevExt->UniRootModelString.Buffer); pDevExt->UniRootModelString.Buffer = NULL;
}
}
if(pDevExt->UniUnitModelString.Length && pDevExt->UniUnitModelString.Buffer) {
TRACE(TL_61883_TRACE,("UnitModel (MSTape): Len:%d; \"%S\"\n", pDevExt->UniUnitModelString.Length, pDevExt->UniUnitModelString.Buffer));
if(!NT_SUCCESS(Status)) {
ExFreePool(pDevExt->UniUnitModelString.Buffer); pDevExt->UniUnitModelString.Buffer = NULL;
}
}
#endif
ExFreePool(pUnitCaps); pUnitCaps = NULL;
return Status;
}
#ifdef SUPPORT_NEW_AVC_CMD
BOOL
InitializeAVCCommand (
PAVC_CMD pAVCCmd,
AvcCommandType CmdType,
AvcSubunitType SubunitType,
UCHAR SubunitID,
AVC_COMMAND_OP_CODE Opcode
)
{
switch(Opcode) {
case OPC_UNIT_CONNECT_AV_20:
pAVCCmd->DataLen = 8;
pAVCCmd->ConnectAV.AudSrc = 3;
pAVCCmd->ConnectAV.VidSrc = 3;
pAVCCmd->ConnectAV.AudDst = 0; // subunit
pAVCCmd->ConnectAV.VidDst = 0; // subunit
pAVCCmd->ConnectAV.VidSrc = 0xff;
pAVCCmd->ConnectAV.AudSrc = 0xff;
pAVCCmd->ConnectAV.VidDst = 0x20;
pAVCCmd->ConnectAV.AudDst = 0x20;
break;
case OPC_TAPE_PLAY_C3:
pAVCCmd->DataLen = 4;
// pAVCCmd->TapePlay.PlaybackMode =
break;
default:
return FALSE;
}
pAVCCmd->CmdFrame.CmdHeader.CTS = 0;
pAVCCmd->CmdFrame.CmdHeader.CmdType = CmdType;
pAVCCmd->CmdFrame.CmdHeader.SubunitTypeID.SubunitType = SubunitType;
pAVCCmd->CmdFrame.CmdHeader.SubunitTypeID.SubunitID = SubunitID;
pAVCCmd->CmdFrame.CmdHeader.Opcode = Opcode;
return TRUE;
}
#endif // SUPPORT_NEW_AVC_CMD
BOOL
DVGetDevModeOfOperation(
IN PDVCR_EXTENSION pDevExt
)
{
NTSTATUS Status;
BYTE bAvcBuf[MAX_FCP_PAYLOAD_SIZE];
#ifdef SUPPORT_NEW_AVC_CMD
AVC_CMD AVCCmd;
#endif
PAGED_CODE();
#ifdef SUPPORT_NEW_AVC_CMD
InitializeAVCCommand(&AVCCmd, AVC_CTYPE_STATUS, AVC_SUBUNITTYPE_UNIT, 0, OPC_UNIT_CONNECT_AV_20);
InitializeAVCCommand(&AVCCmd, AVC_CTYPE_CONTROL, AVC_SUBUNITTYPE_TAPE_PLAYER, 0, OPC_TAPE_PLAY_C3);
AVCCmd.TapePlay.PlaybackMode = NEXT_FRAME; // Testing...
#endif
//
// Use ConnectAV STATUS cmd to determine mode of operation,
// except for some Canon DVs that it requires its vendor specific command
//
Status = DVIssueAVCCommand(pDevExt, AVC_CTYPE_STATUS, DV_CONNECT_AV_MODE, (PVOID) bAvcBuf);
TRACE(TL_61883_TRACE,("GetDevModeOfOperation(DV_CONNECT_AV_MODE): Status %x, %x %x %x %x : %x %x %x %x\n",
Status, bAvcBuf[0], bAvcBuf[1], bAvcBuf[2], bAvcBuf[3], bAvcBuf[4], bAvcBuf[5], bAvcBuf[6], bAvcBuf[7]));
if(Status == STATUS_SUCCESS) {
if(bAvcBuf[0] == 0x0c) {
if(bAvcBuf[1] == 0x00 &&
bAvcBuf[2] == 0x38 &&
bAvcBuf[3] == 0x38) {
pDevExt->ulDevType = ED_DEVTYPE_CAMERA;
} else
if(bAvcBuf[1] == 0xa0 &&
bAvcBuf[2] == 0x00 &&
bAvcBuf[3] == 0x00) {
pDevExt->ulDevType = ED_DEVTYPE_VCR;
}
}
} else if(pDevExt->ulVendorID == VENDORID_CANON) {
// If this is a Canon, we can try this:
Status = DVIssueAVCCommand(pDevExt, AVC_CTYPE_STATUS, DV_VEN_DEP_CANON_MODE, (PVOID) bAvcBuf);
TRACE(TL_61883_TRACE,("GetDevModeOfOperation(DV_VEN_DEP_CANON_MODE): Status %x, %x %x %x %x : %x %x %x %x\n",
Status, bAvcBuf[0], bAvcBuf[1], bAvcBuf[2], bAvcBuf[3], bAvcBuf[4], bAvcBuf[5], bAvcBuf[6], bAvcBuf[7]));
if(Status == STATUS_SUCCESS) {
if(bAvcBuf[0] == 0x0c) {
if(bAvcBuf[7] == 0x38) {
pDevExt->ulDevType = ED_DEVTYPE_CAMERA;
} else
if(bAvcBuf[7] == 0x20) {
pDevExt->ulDevType = ED_DEVTYPE_VCR;
}
}
}
}
//
// Connect AV is an optional command, a device may not support it.
// If this device support a tape subunit, we will assume we are in that device type.
//
if(Status != STATUS_SUCCESS) {
// We are the subunit driver so if any of the device type is a
// tape subunit, we are in that device type.
if( pDevExt->Subunit_Type[0] == AVC_DEVICE_TAPE_REC
|| pDevExt->Subunit_Type[1] == AVC_DEVICE_TAPE_REC
|| pDevExt->Subunit_Type[2] == AVC_DEVICE_TAPE_REC
|| pDevExt->Subunit_Type[3] == AVC_DEVICE_TAPE_REC) {
pDevExt->ulDevType = ED_DEVTYPE_VCR;
} else {
pDevExt->ulDevType = ED_DEVTYPE_UNKNOWN; // Such as MediaConverter box.
}
TRACE(TL_PNP_ERROR|TL_FCP_ERROR,("GetDevModeOfOperation: failed but we choose DevType:%x\n", pDevExt->ulDevType));
}
TRACE(TL_61883_TRACE,("** Mode of operation: %s (%x); NumOPlg:%d; NumIPlg:%d\n",
pDevExt->ulDevType == ED_DEVTYPE_CAMERA ? "Camera" : pDevExt->ulDevType == ED_DEVTYPE_VCR ? "Tape" : "Unknown",
pDevExt->ulDevType, pDevExt->pDevOutPlugs->NumPlugs, pDevExt->pDevInPlugs->NumPlugs));
return TRUE;
}
BOOL
DVGetDevIsItDVCPro(
IN PDVCR_EXTENSION pDevExt
)
{
NTSTATUS Status;
BYTE bAvcBuf[MAX_FCP_PAYLOAD_SIZE];
PAGED_CODE();
//
// Use Panasnoic's vendor dependent command to determine if the system support DVCPro
//
Status = DVIssueAVCCommand(pDevExt, AVC_CTYPE_STATUS, DV_VEN_DEP_DVCPRO, (PVOID) bAvcBuf);
pDevExt->bDVCPro = (Status == STATUS_SUCCESS);
TRACE(TL_61883_TRACE,("GetDevIsItDVCPro? %s; Status %x, %x %x %x %x : %x %x %x %x\n",
pDevExt->bDVCPro ? "Yes":"No",
Status, bAvcBuf[0], bAvcBuf[1], bAvcBuf[2], bAvcBuf[3], bAvcBuf[4], bAvcBuf[5], bAvcBuf[6], bAvcBuf[7]));
return pDevExt->bDVCPro;
}
#define GET_MEDIA_FMT_MAX_RETRIES 10 // AVC.sys will retry so we may rey just once.
BOOL
DVGetDevSignalFormat(
IN PDVCR_EXTENSION pDevExt,
IN KSPIN_DATAFLOW DataFlow,
IN PSTREAMEX pStrmExt
)
{
NTSTATUS Status;
BYTE bAvcBuf[MAX_FCP_PAYLOAD_SIZE];
LONG lRetries = GET_MEDIA_FMT_MAX_RETRIES;
PAGED_CODE();
//
// Respone of Input/output signal mode is used to determine plug signal format:
//
// FMT:
// DVCR 10:00 0000 = 0x80; Canon returns 00:100000 (0x20)
// 50/60: 0:NTSC/60; 1:PAL/50
// STYPE:
// SD: 00000 (DVCPRO:11110)
// HD: 00010
// SDL:00001
// 00:
// SYT:
// MPEG 10:10 0000 = 0xa0
// TSF:0:NotTimeShifted; 1:Time shifted
// 000 0000 0000 0000 0000 0000
//
// If this command failed, we can use Input/Output Signal Mode subunit command
// to determine signal format.
//
do {
RtlZeroMemory(bAvcBuf, sizeof(bAvcBuf));
Status =
DVIssueAVCCommand(
pDevExt,
AVC_CTYPE_STATUS,
pStrmExt == NULL ? DV_OUT_PLUG_SIGNAL_FMT : pStrmExt->pStrmInfo->DataFlow == KSPIN_DATAFLOW_OUT ? DV_OUT_PLUG_SIGNAL_FMT : DV_IN_PLUG_SIGNAL_FMT,
(PVOID) bAvcBuf
);
--lRetries;
//
// Camcorders that has problem with this command:
//
// Panasonic's DVCPRO: if power on while connected to PC, it will
// reject this command with (STATUS_REQUEST_NOT_ACCEPTED)
// so we will retry up to 10 time with .5 second wait between tries.
//
// JVC: returns STATUS_NOT_SUPPORTED.
//
// SONY DV Decoder Box: return STATUS_TIMEOUT
//
if(Status == STATUS_SUCCESS ||
Status == STATUS_NOT_SUPPORTED ||
Status == STATUS_TIMEOUT) {
break; // No need to retry
} else
if(Status == STATUS_REQUEST_NOT_ACCEPTED) {
if(lRetries >= 0)
DVDelayExecutionThread(DV_AVC_CMD_DELAY_DVCPRO);
}
// else retry.
} while (lRetries >= 0);
if(NT_SUCCESS(Status)) {
switch(bAvcBuf[0]) {
case FMT_DVCR:
case FMT_DVCR_CANON: // Workaround for buggy Canon Camcorders
switch(bAvcBuf[1] & FDF0_STYPE_MASK) {
case FDF0_STYPE_SD_DVCR:
case FDF0_STYPE_SD_DVCPRO:
pDevExt->VideoFormatIndex = ((bAvcBuf[1] & FDF0_50_60_MASK) ? AVCSTRM_FORMAT_SDDV_PAL : AVCSTRM_FORMAT_SDDV_NTSC);
break;
case FDF0_STYPE_HD_DVCR:
pDevExt->VideoFormatIndex = ((bAvcBuf[1] & FDF0_50_60_MASK) ? AVCSTRM_FORMAT_HDDV_PAL : AVCSTRM_FORMAT_HDDV_NTSC);
break;
case FDF0_STYPE_SDL_DVCR:
pDevExt->VideoFormatIndex = ((bAvcBuf[1] & FDF0_50_60_MASK) ? AVCSTRM_FORMAT_SDLDV_PAL : AVCSTRM_FORMAT_SDLDV_NTSC);
break;
default: // Unknown format
Status = STATUS_UNSUCCESSFUL;
break;
}
break;
case FMT_MPEG:
pDevExt->VideoFormatIndex = AVCSTRM_FORMAT_MPEG2TS;
break;
default:
Status = STATUS_UNSUCCESSFUL;
}
if(NT_SUCCESS(Status)) {
TRACE(TL_PNP_ERROR|TL_FCP_ERROR,("ST:%x; PlugSignal:FMT[%x %x %x %x]; VideoFormatIndex;%d\n", Status, bAvcBuf[0], bAvcBuf[1], bAvcBuf[2] , bAvcBuf[3], pDevExt->VideoFormatIndex));
return TRUE; // Success
}
}
TRACE(TL_FCP_TRACE,("ST:%x; PlugSignal:FMT[%x %x %x %x]\n", Status, bAvcBuf[0], bAvcBuf[1], bAvcBuf[2] , bAvcBuf[3], pDevExt->VideoFormatIndex));
//
// If "recommended" unit input/output plug signal status command fails,
// try "manadatory" input/output signal mode status command.
// This command may failed some device if its tape is not playing for
// output signal mode command.
//
RtlZeroMemory(bAvcBuf, sizeof(bAvcBuf));
Status =
DVIssueAVCCommand(
pDevExt,
AVC_CTYPE_STATUS,
DataFlow == KSPIN_DATAFLOW_OUT ? VCR_OUTPUT_SIGNAL_MODE : VCR_INPUT_SIGNAL_MODE,
(PVOID) bAvcBuf
);
if(STATUS_SUCCESS == Status) {
PKSPROPERTY_EXTXPORT_S pXPrtProperty;
pXPrtProperty = (PKSPROPERTY_EXTXPORT_S) bAvcBuf;
TRACE(TL_STRM_TRACE|TL_FCP_TRACE,("** MediaFormat: Retry %d mSec; ST:%x; SignalMode:%dL\n",
(GET_MEDIA_FMT_MAX_RETRIES - lRetries) * DV_AVC_CMD_DELAY_DVCPRO, Status, pXPrtProperty->u.SignalMode - ED_BASE));
switch(pXPrtProperty->u.SignalMode) {
case ED_TRANSBASIC_SIGNAL_525_60_SD:
pDevExt->VideoFormatIndex = AVCSTRM_FORMAT_SDDV_NTSC;
if(pStrmExt) {
pStrmExt->cipQuad2[0] = FMT_DVCR; // 0x80
if(pDevExt->bDVCPro)
pStrmExt->cipQuad2[1] = FDF0_50_60_NTSC | FDF0_STYPE_SD_DVCPRO; // 0x78 = NTSC(0):STYPE(11110):RSV(00)
else
pStrmExt->cipQuad2[1] = FDF0_50_60_NTSC | FDF0_STYPE_SD_DVCR; // 0x00 = NTSC(0):STYPE(00000):RSV(00)
}
break;
case ED_TRANSBASIC_SIGNAL_625_50_SD:
pDevExt->VideoFormatIndex = AVCSTRM_FORMAT_SDDV_PAL;
if(pStrmExt) {
pStrmExt->cipQuad2[0] = FMT_DVCR; // 0x80
if(pDevExt->bDVCPro)
pStrmExt->cipQuad2[1] = FDF0_50_60_PAL | FDF0_STYPE_SD_DVCPRO; // 0xf8 = PAL(1):STYPE(11110):RSV(00)
else
pStrmExt->cipQuad2[1] = FDF0_50_60_PAL | FDF0_STYPE_SD_DVCR; // 0x80 = PAL(1):STYPE(00000):RSV(00)
}
break;
case ED_TRANSBASIC_SIGNAL_MPEG2TS:
pDevExt->VideoFormatIndex = AVCSTRM_FORMAT_MPEG2TS;
break;
default:
TRACE(TL_PNP_ERROR|TL_FCP_ERROR,("Unsupported SignalMode:%dL", pXPrtProperty->u.SignalMode - ED_BASE));
ASSERT(FALSE && "Unsupported IoSignal! Refuse to load.");
return FALSE;
break;
}
}
// WORKITEM Sony HW CODEC does not response to any AVC command.
// We are making an exception here to load it.
if(Status == STATUS_TIMEOUT) {
Status = STATUS_SUCCESS;
}
// We must know the signal format!! If this failed, the driver will either:
// fail to load, or fail to open an stream
ASSERT(Status == STATUS_SUCCESS && "Failed to get media signal format!\n");
#if DBG
if(pStrmExt) {
// Note: bAvcBuf[0] is operand[1] == 10:fmt
TRACE(TL_STRM_TRACE|TL_CIP_TRACE,("** MediaFormat: St:%x; idx:%d; CIP:[FMT:%.2x(%s); FDF:[%.2x(%s,%s):SYT]\n",
Status,
pDevExt->VideoFormatIndex,
pStrmExt->cipQuad2[0],
pStrmExt->cipQuad2[0] == FMT_DVCR ? "DVCR" : pStrmExt->cipQuad2[0] == FMT_MPEG ? "MPEG" : "Fmt:???",
pStrmExt->cipQuad2[1],
(pStrmExt->cipQuad2[1] & FDF0_50_60_MASK) == FDF0_50_60_PAL ? "PAL" : "NTSC",
(pStrmExt->cipQuad2[1] & FDF0_STYPE_MASK) == FDF0_STYPE_SD_DVCR ? "SD" : \
(pStrmExt->cipQuad2[1] & FDF0_STYPE_MASK) == FDF0_STYPE_SDL_DVCR ? "SDL" : \
(pStrmExt->cipQuad2[1] & FDF0_STYPE_MASK) == FDF0_STYPE_HD_DVCR ? "HD" : \
(pStrmExt->cipQuad2[1] & FDF0_STYPE_MASK) == FDF0_STYPE_SD_DVCPRO ? "DVCPRO" : "DV:????"
));
} else
TRACE(TL_STRM_TRACE|TL_CIP_TRACE,("** MediaFormat: St:%x; use idx:%d\n", Status, pDevExt->VideoFormatIndex));
#endif
return STATUS_SUCCESS == Status;
}
BOOL
DVCmpGUIDsAndFormatSize(
IN PKSDATARANGE pDataRange1,
IN PKSDATARANGE pDataRange2,
IN BOOL fCompareFormatSize
)
/*++
Routine Description:
Checks for a match on the three GUIDs and FormatSize
Arguments:
IN pDataRange1
IN pDataRange2
Return Value:
TRUE if all elements match
FALSE if any are different
--*/
{
return (
IsEqualGUID (
&pDataRange1->MajorFormat,
&pDataRange2->MajorFormat) &&
IsEqualGUID (
&pDataRange1->SubFormat,
&pDataRange2->SubFormat) &&
IsEqualGUID (
&pDataRange1->Specifier,
&pDataRange2->Specifier) &&
(fCompareFormatSize ?
(pDataRange1->FormatSize == pDataRange2->FormatSize) : TRUE ));
}
//
// GetSystemTime in 100 nS units
//
ULONGLONG GetSystemTime()
{
LARGE_INTEGER rate, ticks;
ticks = KeQueryPerformanceCounter(&rate);
return (KSCONVERT_PERFORMANCE_TIME(rate.QuadPart, ticks));
}
VOID
DvFreeTextualString(
PDVCR_EXTENSION pDevExt,
GET_UNIT_IDS * pUnitIds
)
{
if(pUnitIds->ulVendorLength && pUnitIds->VendorText) {
ExFreePool(pUnitIds->VendorText); pUnitIds->VendorText = NULL;
}
if(pUnitIds->ulModelLength && pUnitIds->ModelText) {
ExFreePool(pUnitIds->ModelText); pUnitIds->ModelText = NULL;
}
#ifdef NT51_61883
if(pUnitIds->ulUnitModelLength && pUnitIds->UnitModelText) {
ExFreePool(pUnitIds->UnitModelText); pUnitIds->UnitModelText = NULL;
}
#else
if(pDevExt->UniRootModelString.Length && pDevExt->UniRootModelString.Buffer) {
ExFreePool(pDevExt->UniRootModelString.Buffer); pDevExt->UniRootModelString.Buffer = NULL;
}
if(pDevExt->UniUnitModelString.Length && pDevExt->UniUnitModelString.Buffer) {
ExFreePool(pDevExt->UniUnitModelString.Buffer); pDevExt->UniUnitModelString.Buffer = NULL;
}
#endif
}
#define DIFBLK_SIZE 12000
#define PACK_NO_INFO 0xff
// Subcode header identifier
#define SC_HDR_TIMECODE 0x13
#define SC_HDR_BINARYGROUP 0x14
// header identifier
#define AAUX_HDR_SOURCE 0x50
#define AAUX_HDR_SOURCE_CONTROL 0x51
#define AAUX_HDR_REC_DATE 0x52
#define AAUX_HDR_REC_TIME 0x53
#define AAUX_HDR_BINARY_GROUP 0x54
#define AAUX_HDR_CC 0x55
#define AAUX_HDR_TR 0x56
#define VAUX_HDR_SOURCE 0x60
#define VAUX_HDR_SOURCE_CONTROL 0x61
#define VAUX_HDR_REC_DATE 0x62
#define VAUX_HDR_REC_TIME 0x63
#define VAUX_HDR_BINARY_GROUP 0x64
#define VAUX_HDR_CC 0x65
#define VAUX_HDR_TR 0x66
// Determine section type (MS 3 bits); Fig.66; Table 36.
#define ID0_SCT_MASK 0xe0
#define ID0_SCT_HEADER 0x00
#define ID0_SCT_SUBCODE 0x20
#define ID0_SCT_VAUX 0x40
#define ID0_SCT_AUDIO 0x60
#define ID0_SCT_VIDEO 0x80
// A pack is consisted of one byte of header identifier and 4 bytes of data; Part2, annex D.
typedef struct _DV_PACK {
UCHAR Header;
UCHAR Data[4];
} DV_PACK, *PDV_PACK;
typedef struct _DV_H0 {
UCHAR ID0;
UCHAR ID1;
UCHAR ID2;
UCHAR DSF;
UCHAR DFTIA;
UCHAR TF1;
UCHAR TF2;
UCHAR TF3;
UCHAR Reserved[72];
} DV_H0, *PDV_H0;
typedef struct _DV_SC {
UCHAR ID0;
UCHAR ID1;
UCHAR ID2;
struct {
UCHAR SID0;
UCHAR SID1;
UCHAR Reserved;
DV_PACK Pack;
} SSyb0;
struct {
UCHAR SID0;
UCHAR SID1;
UCHAR Reserved;
DV_PACK Pack;
} SSyb1;
struct {
UCHAR SID0;
UCHAR SID1;
UCHAR Reserved;
DV_PACK Pack;
} SSyb2;
struct {
UCHAR SID0;
UCHAR SID1;
UCHAR Reserved;
DV_PACK Pack;
} SSyb3;
struct {
UCHAR SID0;
UCHAR SID1;
UCHAR Reserved;
DV_PACK Pack;
} SSyb4;
struct {
UCHAR SID0;
UCHAR SID1;
UCHAR Reserved;
DV_PACK Pack;
} SSyb5;
UCHAR Reserved[29];
} DV_SC, *PDV_SC;
#define MAX_VAUX_PACK 15
typedef struct _DV_VAUX {
UCHAR ID0;
UCHAR ID1;
UCHAR ID2;
DV_PACK Pack[MAX_VAUX_PACK];
UCHAR Reserved[2];
} DV_VAUX, *PDV_VAUX;
typedef struct _DV_A {
UCHAR ID0;
UCHAR ID1;
UCHAR ID2;
DV_PACK Pack;
UCHAR Data[72];
} DV_A, *PDV_A;
typedef struct _DV_V {
UCHAR ID0;
UCHAR ID1;
UCHAR ID2;
UCHAR Data[77]; // 3..79
} DV_V, *PDV_V;
// Two source packets
#define V_BLOCKS 15
typedef struct _DV_AV {
DV_A A;
DV_V V[V_BLOCKS];
} DV_AV, *PDV_AV;
#define SC_SECTIONS 2
#define VAUX_SECTIONS 3
#define AV_SECTIONS 9
typedef struct _DV_DIF_SEQ {
DV_H0 H0;
DV_SC SC[SC_SECTIONS];
DV_VAUX VAux[VAUX_SECTIONS];
DV_AV AV[AV_SECTIONS];
} DV_DIF_SEQ, *PDV_DIF_SEQ;
typedef struct _DV_FRAME_NTSC {
DV_DIF_SEQ DifSeq[10];
} DV_FRAME_NTSC, *PDV_FRAME_NTSC;
typedef struct _DV_FRAME_PAL {
DV_DIF_SEQ DifSeq[12];
} DV_FRAME_PAL, *PDV_FRAME_PAL;
// By setting REC MODE to 111b (invalid recording) can
// cause DV to mute the audio
#define AAUX_REC_MODE_INVALID_MASK 0x38 // xx11:1xxx
#define AAUX_REC_MODE_ORIGINAL 0x08 // xx00:1xxx
#ifdef MSDV_SUPPORT_MUTE_AUDIO
BOOL
DVMuteDVFrame(
IN PDVCR_EXTENSION pDevExt,
IN OUT PUCHAR pFrameBuffer,
IN BOOL bMuteAudio
)
{
PDV_DIF_SEQ pDifSeq;
#if 0
PDV_VAUX pVAux;
ULONG k;
#endif
ULONG i, j;
#if 0
BOOL bFound1 = FALSE;
#endif
BOOL bFound2 = FALSE;
pDifSeq = (PDV_DIF_SEQ) pFrameBuffer;
// find the VVAX Source pack
for (i=0; i < AVCStrmFormatInfoTable[pDevExt->VideoFormatIndex].FrameSize/DIFBLK_SIZE; i++) {
// #define SHOW_ONE_FIELD_TWICE
#ifdef SHOW_ONE_FIELD_TWICE // Advise by Adobe that we may want to show bothj field but mute audio
// Make the field2 output twice, FrameChange to 0 (same as previous frame)
for (j=0; j < VAUX_SECTIONS; j++) {
pVAux = &pDifSeq->VAux[j];
if((pVAux->ID0 & ID0_SCT_MASK) != ID0_SCT_VAUX) {
TRACE(TL_CIP_TRACE,("Invalid ID0:%.2x for pVAUX:%x (Dif:%d;V%d;S%d)\n", pVAux->ID0, pVAux, i, j, k));
continue;
}
for (k=0; k< MAX_VAUX_PACK; k++) {
if(pVAux->Pack[k].Header == VAUX_HDR_SOURCE_CONTROL) {
if(bMuteAudio) {
TRACE(TL_CIP_TRACE,("Mute Audio; pDifSeq:%x; pVAux:%x; (Dif:%d,V%d,S%d); %.2x,[%.2x,%.2x,%.2x,%.2x]; pack[2]->%.2x\n", \
pDifSeq, pVAux, i, j, k, \
pVAux->Pack[k].Header, pVAux->Pack[k].Data[0], pVAux->Pack[k].Data[1], pVAux->Pack[k].Data[2], pVAux->Pack[k].Data[3], \
(pVAux->Pack[k].Data[2] & 0x1F) ));
pVAux->Pack[k].Data[2] &= 0x1f; // 0x1F; // set FF, FS and FC to 0
TRACE(TL_CIP_INFO,("pVAux->Pack[k].Data[2] = %.2x\n", pVAux->Pack[k].Data[2]));
} else {
TRACE(TL_CIP_INFO,("un-Mute Audio; pack[2]: %.2x ->%.2x\n", pVAux->Pack[k].Data[2], (pVAux->Pack[k].Data[2] | 0xc0) ));
pVAux->Pack[k].Data[2] |= 0xe0; // set FF, FS and FCto 1; Show both fields in field 1,2 order
}
bFound1 = TRUE;
break; // Set only the 1st occurrence
}
}
}
#endif
for (j=0; j < AV_SECTIONS; j++) {
if(pDifSeq->AV[j].A.Pack.Header == AAUX_HDR_SOURCE_CONTROL) {
TRACE(TL_CIP_INFO,("A0Aux %.2x,[%.2x,%.2x,%.2x,%.2x] %.2x->%.2x\n", \
pDifSeq->AV[j].A.Pack.Header, pDifSeq->AV[j].A.Pack.Data[0], \
pDifSeq->AV[j].A.Pack.Data[1], pDifSeq->AV[j].A.Pack.Data[2], pDifSeq->AV[j].A.Pack.Data[3], \
pDifSeq->AV[j].A.Pack.Data[1], pDifSeq->AV[j].A.Pack.Data[1] | AAUX_REC_MODE_INVALID_MASK
));
if(bMuteAudio)
pDifSeq->AV[j].A.Pack.Data[1] |= AAUX_REC_MODE_INVALID_MASK; // Cause DV to mute this.
else
pDifSeq->AV[j].A.Pack.Data[1] = \
(pDifSeq->AV[j].A.Pack.Data[1] & ~AAUX_REC_MODE_INVALID_MASK) | AAUX_REC_MODE_ORIGINAL;
bFound2 = TRUE;
break; // Set only the 1st occurrence
}
}
// Must do the 1st occurance of all Dif sequences;
pDifSeq++; // Next DIF sequence
}
#if 0
return (bFound1 && bFound2);
#else
return bFound2;
#endif
}
#endif
#ifdef MSDV_SUPPORT_EXTRACT_SUBCODE_DATA
VOID
DVCRExtractTimecodeFromFrame(
IN PDVCR_EXTENSION pDevExt,
IN PSTREAMEX pStrmExt,
IN PUCHAR pFrameBuffer
)
{
PUCHAR pDIFBlk;
PUCHAR pS0, pS1, pSID0;
ULONG i, j;
BYTE LastTimecode[4], Timecode[4]; // hh:mm:ss,ff
DWORD LastAbsTrackNumber, AbsTrackNumber;
PUCHAR pSID1;
BYTE Timecode2[4]; // hh:mm:ss,ff
DWORD AbsTrackNumber2;
BOOL bGetAbsT = TRUE, bGetTimecode = TRUE;
// Can be called at DISPATCH_LEVEL
pDIFBlk = (PUCHAR) pFrameBuffer;
// Save the last timecode so we will now if it has
LastTimecode[0] = pStrmExt->Timecode[0];
LastTimecode[1] = pStrmExt->Timecode[1];
LastTimecode[2] = pStrmExt->Timecode[2];
LastTimecode[3] = pStrmExt->Timecode[3];
LastAbsTrackNumber = pStrmExt->AbsTrackNumber;
//
// Traverse thru every DIF BLOCK looking for VA0,1 and 2
for(i=0; i < AVCStrmFormatInfoTable[pDevExt->VideoFormatIndex].ulNumOfDIFSequences; i++) {
pS0 = pDIFBlk + 80;
pS1 = pS0 + 80;
//
// Is this Subcode source packet? See Table 36 (P.111) of the Blue Book
//
if ((pS0[0] & 0xe0) == 0x20 && (pS1[0] & 0xe0) == 0x20) {
if(bGetAbsT) {
//
// See Figure 42 (p. 94) of the Blue book
// SID0(Low nibble),1 (high nibble) of every three subcode sync block can form the ATN
//
pSID0 = &pS0[3];
AbsTrackNumber = 0;
for (j = 0 ; j < 3; j++) {
AbsTrackNumber = (( ( (pSID0[0] & 0x0f) << 4) | (pSID0[1] >> 4) ) << (j * 8)) | AbsTrackNumber;
pSID0 += 8;
bGetAbsT = FALSE;
}
pSID1 = &pS1[3];
AbsTrackNumber2 = 0;
for (j = 0 ; j < 3; j++) {
AbsTrackNumber2 = (( ( (pSID1[0] & 0x0f) << 4) | (pSID1[1] >> 4) ) << (j * 8)) | AbsTrackNumber2;
pSID1 += 8;
}
// Verify that the track number is the same!
if(AbsTrackNumber == AbsTrackNumber2) {
bGetAbsT = FALSE;
} else {
bGetAbsT = TRUE;
TRACE(TL_CIP_TRACE,("%d Sequence; AbsT (%d,%d) != AbsT2 (%d,%d)\n",
i,
AbsTrackNumber / 2, AbsTrackNumber & 0x01,
AbsTrackNumber2 / 2, AbsTrackNumber2 & 0x01
));
}
}
if(bGetTimecode) {
// See Figure 68 (p. 114) of the Blue Book
// Subcode sync block number 3, 4 and 5
for(j = 3; j <= 5; j++) {
// 3 bytes of IDs and follow by sequence of 8 bytes SyncBlock (3:5);
// 0x13 == TIMECODE
if(pS0[3+3+j*8] == 0x13
&& pS0[3+3+j*8+4] != 0xff
&& pS0[3+3+j*8+3] != 0xff
&& pS0[3+3+j*8+2] != 0xff
&& pS0[3+3+j*8+1] != 0xff) {
Timecode[0] = pS0[3+3+j*8+4]&0x3f; // hh
Timecode[1] = pS0[3+3+j*8+3]&0x7f; // mm
Timecode[2] = pS0[3+3+j*8+2]&0x7f; // ss
Timecode[3] = pS0[3+3+j*8+1]&0x3f; // ff
bGetTimecode = FALSE;
break;
}
}
// Subcode sync block number 9, 10 and 11
for(j = 3; j <= 5; j++) {
// 3 bytes of IDs and follow by sequence of 8 bytes SyncBlock (3:5);
// 0x13 == TIMECODE
if(pS1[3+3+j*8] == 0x13
&& pS1[3+3+j*8+4] != 0xff
&& pS1[3+3+j*8+3] != 0xff
&& pS1[3+3+j*8+2] != 0xff
&& pS1[3+3+j*8+1] != 0xff) {
Timecode2[0] = pS1[3+3+j*8+4]&0x3f; // hh
Timecode2[1] = pS1[3+3+j*8+3]&0x7f; // mm
Timecode2[2] = pS1[3+3+j*8+2]&0x7f; // ss
Timecode2[3] = pS1[3+3+j*8+1]&0x3f; // ff
bGetTimecode = FALSE;
break;
}
}
//
// Verify
//
if(!bGetTimecode) {
if( Timecode[0] == Timecode2[0]
&& Timecode[1] == Timecode2[1]
&& Timecode[2] == Timecode2[2]
&& Timecode[3] == Timecode2[3]) {
} else {
bGetTimecode = TRUE;
TRACE(TL_CIP_TRACE,("%d Sequence; %.2x:%.2x:%.2x,%.2x != %.2x:%.2x:%.2x,%.2x\n",
i,
Timecode[0], Timecode[1], Timecode[2], Timecode[3],
Timecode2[0], Timecode2[1], Timecode2[2], Timecode2[3]
));
}
}
}
}
if(!bGetAbsT && !bGetTimecode)
break;
pDIFBlk += DIFBLK_SIZE; // Get to next block
}
if(!bGetAbsT && pStrmExt->AbsTrackNumber != AbsTrackNumber) {
pStrmExt->AbsTrackNumber = AbsTrackNumber; // BF is the LSB
pStrmExt->bATNUpdated = TRUE;
TRACE(TL_CIP_INFO,("Extracted TrackNum:%d; DicontBit:%d\n", AbsTrackNumber / 2, AbsTrackNumber & 0x01));
}
if(!bGetTimecode &&
(
Timecode[0] != LastTimecode[0] ||
Timecode[1] != LastTimecode[1] ||
Timecode[2] != LastTimecode[2] ||
Timecode[3] != LastTimecode[3]
)
) {
pStrmExt->Timecode[0] = Timecode[0]; // hh
pStrmExt->Timecode[1] = Timecode[1]; // mm
pStrmExt->Timecode[2] = Timecode[2]; // mm
pStrmExt->Timecode[3] = Timecode[3]; // ff
pStrmExt->bTimecodeUpdated = TRUE;
TRACE(TL_CIP_INFO,("Extracted Timecode %.2x:%.2x:%.2x,%.2x\n", Timecode[0], Timecode[1], Timecode[2], Timecode[3]));
}
}
#endif // MSDV_SUPPORT_EXTRACT_SUBCODE_DATA
#ifdef MSDV_SUPPORT_EXTRACT_DV_DATE_TIME
VOID
DVCRExtractRecDateAndTimeFromFrame(
IN PDVCR_EXTENSION pDevExt,
IN PSTREAMEX pStrmExt,
IN PUCHAR pFrameBuffer
)
{
PUCHAR pDIFBlk;
PUCHAR pS0, pS1;
ULONG i, j;
BOOL bGetRecDate = TRUE, bGetRecTime = TRUE;
// Can be called at DISPATCH_LEVEL
pDIFBlk = (PUCHAR) pFrameBuffer + DIFBLK_SIZE * AVCStrmFormatInfoTable[pDevExt->VideoFormatIndex].ulNumOfDIFSequences/2;
//
// REC Data (VRD) and Time (VRT) on in the 2nd half oa a video frame
//
for(i=AVCStrmFormatInfoTable[pDevExt->VideoFormatIndex].ulNumOfDIFSequences/2; i < AVCStrmFormatInfoTable[pDevExt->VideoFormatIndex].ulNumOfDIFSequences; i++) {
pS0 = pDIFBlk + 80;
pS1 = pS0 + 80;
//
// Find SC0 and SC1. See Table 36 (P.111) of the Blue Book
//
// SC0/1: ID(0,1,2), Data (3,50), Reserved(51-79)
// SC0:Data: SSYB0(3..10), SSYB1(11..18), SSYB2(19..26), SSYB3(27..34), SSYB4(35..42), SSYB5(43..50)
// SC1:Data: SSYB6(3..10), SSYB7(11..18), SSYB8(19..26), SSYB9(27..34), SSYB10(35..42), SSYB11(43..50)
// SSYBx(SubCodeId0, SubcodeID1, Reserved, Pack(3,4,5,6,7))
//
// TTC are in the 1st half: SSYB0..11 (every)
// TTC are in the 2nd half: SSYB0,3,6,9
// VRD are in the 2nd half of a video frame, SSYB1,4,7,10
// VRT are in the 2nd half of a video frame, SSYB2,5,8,11
//
// Subcode data ?
if ((pS0[0] & 0xe0) == 0x20 && (pS1[0] & 0xe0) == 0x20) {
//
// RecDate: VRD
//
if(bGetRecDate) {
// go thru 6 sync blocks (8 bytes per block) per Subcode; idx 1(SSYB1),4(SSYB4) for SC0
for(j=0; j <= 5 ; j++) {
if(j == 1 || j == 4) {
// 0x62== RecDate
if(pS0[3+3+j*8] == 0x62) {
pStrmExt->RecDate[0] = pS0[3+3+j*8+4]; // Year
pStrmExt->RecDate[1] = pS0[3+3+j*8+3]&0x1f; // Month
pStrmExt->RecDate[2] = pS0[3+3+j*8+2]&0x3f; // Day
pStrmExt->RecDate[3] = pS0[3+3+j*8+1]&0x3f; // TimeZone
bGetRecDate = FALSE;
break;
}
}
}
}
if(bGetRecDate) {
// go thru 6 sync blocks (8 bytes per block) per Subcode; idx 1 (SSYB7),4(SSYB10) for SC1
for(j=0; j <= 5; j++) {
if(j == 1 || j == 4) {
// 0x62== RecDate
if(pS1[3+3+j*8] == 0x62) {
pStrmExt->RecDate[0] = pS1[3+3+j*8+4]; // Year
pStrmExt->RecDate[1] = pS1[3+3+j*8+3]&0x1f; // Month
pStrmExt->RecDate[2] = pS1[3+3+j*8+2]&0x3f; // Day
pStrmExt->RecDate[3] = pS1[3+3+j*8+1]&0x3f; // TimeZone
bGetRecDate = FALSE;
break;
}
}
}
}
//
// RecTime: VRT
//
if(bGetRecTime) {
// go thru 6 sync blocks (8 bytes per block) per Subcode; idx 2(SSYB2),5(SSYB5) for SC0
for(j=0; j <= 5 ; j++) {
if(j == 2 || j == 5) {
// 0x63== RecTime
if(pS0[3+3+j*8] == 0x63) {
pStrmExt->RecTime[0] = pS0[3+3+j*8+4]&0x3f;
pStrmExt->RecTime[1] = pS0[3+3+j*8+3]&0x7f;
pStrmExt->RecTime[2] = pS0[3+3+j*8+2]&0x7f;
pStrmExt->RecTime[3] = pS0[3+3+j*8+1]&0x3f;
bGetRecTime = FALSE;
break;
}
}
}
}
if(bGetRecTime) {
// go thru 6 sync blocks (8 bytes per block) per Subcode; idx 2 (SSYB8),5(SSYB11) for SC1
for(j=0; j <= 5; j++) {
if(j == 2 || j == 5) {
// 0x63== RecTime
if(pS1[3+3+j*8] == 0x63) {
pStrmExt->RecTime[0] = pS1[3+3+j*8+4]&0x3f;
pStrmExt->RecTime[1] = pS1[3+3+j*8+3]&0x7f;
pStrmExt->RecTime[2] = pS1[3+3+j*8+2]&0x7f;
pStrmExt->RecTime[3] = pS1[3+3+j*8+1]&0x3f;
bGetRecTime = FALSE;
break;
}
}
}
}
}
if(!bGetRecDate && !bGetRecTime)
break;
pDIFBlk += DIFBLK_SIZE; // Next sequence
}
TRACE(TL_PNP_TRACE,("Frame# %.5d, Date %s %x-%.2x-%.2x, Time %s %.2x:%.2x:%.2x,%.2x\n",
(ULONG) pStrmExt->FramesProcessed,
bGetRecDate ? "NF:" : "Found:", pStrmExt->RecDate[0], pStrmExt->RecDate[1] & 0x1f, pStrmExt->RecDate[2] & 0x3f,
bGetRecTime ? "NF:" : "Found:",pStrmExt->RecTime[0], pStrmExt->RecTime[1], pStrmExt->RecTime[2], pStrmExt->RecTime[3]
));
}
#endif // MSDV_SUPPORT_EXTRACT_DV_DATE_TIME
#ifdef READ_CUTOMIZE_REG_VALUES
NTSTATUS
CreateRegistryKeySingle(
IN HANDLE hKey,
IN ACCESS_MASK desiredAccess,
PWCHAR pwszSection,
OUT PHANDLE phKeySection
)
{
NTSTATUS status;
UNICODE_STRING ustr;
OBJECT_ATTRIBUTES objectAttributes;
RtlInitUnicodeString(&ustr, pwszSection);
InitializeObjectAttributes(
&objectAttributes,
&ustr,
OBJ_CASE_INSENSITIVE,
hKey,
NULL
);
status =
ZwCreateKey(
phKeySection,
desiredAccess,
&objectAttributes,
0,
NULL, /* optional*/
REG_OPTION_NON_VOLATILE,
NULL
);
return status;
}
NTSTATUS
CreateRegistrySubKey(
IN HANDLE hKey,
IN ACCESS_MASK desiredAccess,
PWCHAR pwszSection,
OUT PHANDLE phKeySection
)
{
UNICODE_STRING ustr;
USHORT usPos = 1; // Skip first backslash
static WCHAR wSep = '\\';
NTSTATUS status = STATUS_SUCCESS;
RtlInitUnicodeString(&ustr, pwszSection);
while(usPos < ustr.Length) {
if(ustr.Buffer[usPos] == wSep) {
// NULL terminate our partial string
ustr.Buffer[usPos] = UNICODE_NULL;
status =
CreateRegistryKeySingle(
hKey,
desiredAccess,
ustr.Buffer,
phKeySection
);
ustr.Buffer[usPos] = wSep;
if(NT_SUCCESS(status)) {
ZwClose(*phKeySection);
} else {
break;
}
}
usPos++;
}
// Create the full key
if(NT_SUCCESS(status)) {
status =
CreateRegistryKeySingle(
hKey,
desiredAccess,
ustr.Buffer,
phKeySection
);
}
return status;
}
NTSTATUS
GetRegistryKeyValue (
IN HANDLE Handle,
IN PWCHAR KeyNameString,
IN ULONG KeyNameStringLength,
IN PVOID Data,
IN PULONG DataLength
)
/*++
Routine Description:
This routine gets the specified value out of the registry
Arguments:
Handle - Handle to location in registry
KeyNameString - registry key we're looking for
KeyNameStringLength - length of registry key we're looking for
Data - where to return the data
DataLength - how big the data is
Return Value:
status is returned from ZwQueryValueKey
--*/
{
NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
UNICODE_STRING keyName;
ULONG length;
PKEY_VALUE_FULL_INFORMATION fullInfo;
RtlInitUnicodeString(&keyName, KeyNameString);
length = sizeof(KEY_VALUE_FULL_INFORMATION) +
KeyNameStringLength + *DataLength;
fullInfo = ExAllocatePool(PagedPool, length);
if (fullInfo) {
status = ZwQueryValueKey(
Handle,
&keyName,
KeyValueFullInformation,
fullInfo,
length,
&length
);
if (NT_SUCCESS(status)){
ASSERT(fullInfo->DataLength <= *DataLength);
RtlCopyMemory(
Data,
((PUCHAR) fullInfo) + fullInfo->DataOffset,
fullInfo->DataLength
);
}
*DataLength = fullInfo->DataLength;
ExFreePool(fullInfo);
}
return (status);
}
NTSTATUS
SetRegistryKeyValue(
HANDLE hKey,
PWCHAR pwszEntry,
LONG nValue
)
{
NTSTATUS status;
UNICODE_STRING ustr;
RtlInitUnicodeString(&ustr, pwszEntry);
status =
ZwSetValueKey(
hKey,
&ustr,
0, /* optional */
REG_DWORD,
&nValue,
sizeof(nValue)
);
return status;
}
//
// Registry subky and values wide character strings.
//
WCHAR wszSettings[] = L"Settings";
WCHAR wszATNSearch[] = L"bSupportATNSearch";
WCHAR wszSyncRecording[] = L"bSyncRecording";
WCHAR wszMaxDataSync[] = L"tmMaxDataSync";
WCHAR wszPlayPs2RecPs[] = L"fmPlayPause2RecPause";
WCHAR wszStop2RecPs[] = L"fmStop2RecPause";
WCHAR wszRecPs2Rec[] = L"tmRecPause2Rec";
BOOL
DVGetPropertyValuesFromRegistry(
IN PDVCR_EXTENSION pDevExt
)
{
NTSTATUS Status;
HANDLE hPDOKey, hKeySettings;
ULONG ulLength;
//
// Registry key:
// Windows 2000:
// HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Class\
// {6BDD1FC6-810F-11D0-BEC7-08002BE2092F\000x
//
// Win98:
// HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\Class\Image\000x
//
Status =
IoOpenDeviceRegistryKey(
pDevExt->pPhysicalDeviceObject,
PLUGPLAY_REGKEY_DRIVER,
STANDARD_RIGHTS_READ,
&hPDOKey
);
// PDO might be deleted when it was removed.
if(! pDevExt->bDevRemoved) {
ASSERT(Status == STATUS_SUCCESS);
}
//
// loop through our table of strings,
// reading the registry for each.
//
if(NT_SUCCESS(Status)) {
// Create or open the settings key
Status =
CreateRegistrySubKey(
hPDOKey,
KEY_ALL_ACCESS,
wszSettings,
&hKeySettings
);
if(NT_SUCCESS(Status)) {
// Note: we can be more selective by checking
// pDevExt->ulDevType
// ATNSearch
ulLength = sizeof(LONG);
Status = GetRegistryKeyValue(
hKeySettings,
wszATNSearch,
sizeof(wszATNSearch),
(PVOID) &pDevExt->bATNSearch,
&ulLength);
TRACE(TL_PNP_TRACE,("GetRegVal: St:%x, Len:%d, bATNSearch:%d (1:Yes)\n", Status, ulLength, pDevExt->bATNSearch));
if(!NT_SUCCESS(Status)) pDevExt->bATNSearch = FALSE;
// bSyncRecording
ulLength = sizeof(LONG);
Status = GetRegistryKeyValue(
hKeySettings,
wszSyncRecording,
sizeof(wszSyncRecording),
(PVOID) &pDevExt->bSyncRecording,
&ulLength);
TRACE(TL_PNP_TRACE,("GetRegVal: St:%x, Len:%d, bSyncRecording:%d (1:Yes)\n", Status, ulLength, pDevExt->bSyncRecording));
if(!NT_SUCCESS(Status)) pDevExt->bSyncRecording = FALSE;
// tmMaxDataSync
ulLength = sizeof(LONG);
Status = GetRegistryKeyValue(
hKeySettings,
wszMaxDataSync,
sizeof(wszMaxDataSync),
(PVOID) &pDevExt->tmMaxDataSync,
&ulLength);
TRACE(TL_PNP_TRACE,("GetRegVal: St:%x, Len:%d, tmMaxDataSync:%d (msec)\n", Status, ulLength, pDevExt->tmMaxDataSync));
// fmPlayPs2RecPs
ulLength = sizeof(LONG);
Status = GetRegistryKeyValue(
hKeySettings,
wszPlayPs2RecPs,
sizeof(wszPlayPs2RecPs),
(PVOID) &pDevExt->fmPlayPs2RecPs,
&ulLength);
TRACE(TL_PNP_TRACE,("GetRegVal: St:%x, Len:%d, fmPlayPs2RecPs:%d (frames)\n", Status, ulLength, pDevExt->fmPlayPs2RecPs));
// fmStop2RecPs
ulLength = sizeof(LONG);
Status = GetRegistryKeyValue(
hKeySettings,
wszStop2RecPs,
sizeof(wszStop2RecPs),
(PVOID) &pDevExt->fmStop2RecPs,
&ulLength);
TRACE(TL_PNP_TRACE,("GetRegVal: St:%x, Len:%d, fmStop2RecPs:%d (frames)\n", Status, ulLength, pDevExt->fmStop2RecPs));
// tmRecPs2Rec
ulLength = sizeof(LONG);
Status = GetRegistryKeyValue(
hKeySettings,
wszRecPs2Rec,
sizeof(wszRecPs2Rec),
(PVOID) &pDevExt->tmRecPs2Rec,
&ulLength);
TRACE(TL_PNP_TRACE,("GetRegVal: St:%x, Len:%d, tmRecPs2Rec:%d (msec)\n", Status, ulLength, pDevExt->tmRecPs2Rec));
ZwClose(hKeySettings);
ZwClose(hPDOKey);
return TRUE;
} else {
TRACE(TL_PNP_ERROR,("GetPropertyValuesFromRegistry: CreateRegistrySubKey failed with Status=%x\n", Status));
}
ZwClose(hPDOKey);
} else {
TRACE(TL_PNP_ERROR,("GetPropertyValuesFromRegistry: IoOpenDeviceRegistryKey failed with Status=%x\n", Status));
}
// Not implemented so always return FALSE to use the defaults.
return FALSE;
}
BOOL
DVSetPropertyValuesToRegistry(
PDVCR_EXTENSION pDevExt
)
{
// Set the default to :
// HLM\Software\DeviceExtension->pchVendorName\1394DCam
NTSTATUS Status;
HANDLE hPDOKey, hKeySettings;
TRACE(TL_PNP_TRACE,("SetPropertyValuesToRegistry: pDevExt=%x; pDevExt->pBusDeviceObject=%x\n", pDevExt, pDevExt->pBusDeviceObject));
//
// Registry key:
// HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Class\
// {6BDD1FC6-810F-11D0-BEC7-08002BE2092F\000x
//
Status =
IoOpenDeviceRegistryKey(
pDevExt->pPhysicalDeviceObject,
PLUGPLAY_REGKEY_DRIVER,
STANDARD_RIGHTS_WRITE,
&hPDOKey);
// PDO might be deleted when it was removed.
if(! pDevExt->bDevRemoved) {
ASSERT(Status == STATUS_SUCCESS);
}
//
// loop through our table of strings,
// reading the registry for each.
//
if(NT_SUCCESS(Status)) {
// Create or open the settings key
Status =
CreateRegistrySubKey(
hPDOKey,
KEY_ALL_ACCESS,
wszSettings,
&hKeySettings
);
if(NT_SUCCESS(Status)) {
#if 0 // Note used, just an example:
// Brightness
Status = SetRegistryKeyValue(
hKeySettings,
wszBrightness,
pDevExt->XXXX);
TRACE(TL_PNP_TRACE,("SetPropertyValuesToRegistry: Status %x, Brightness %d\n", Status, pDevExt->Brightness));
#endif
ZwClose(hKeySettings);
ZwClose(hPDOKey);
return TRUE;
} else {
TRACE(TL_PNP_ERROR,("GetPropertyValuesToRegistry: CreateRegistrySubKey failed with Status=%x\n", Status));
}
ZwClose(hPDOKey);
} else {
TRACE(TL_PNP_TRACE,("GetPropertyValuesToRegistry: IoOpenDeviceRegistryKey failed with Status=%x\n", Status));
}
return FALSE;
}
#endif // READ_CUTOMIZE_REG_VALUES
#ifdef SUPPORT_ACCESS_DEVICE_INTERFACE
#if DBG
NTSTATUS
DVGetRegistryValue(
IN HANDLE Handle,
IN PWCHAR KeyNameString,
IN ULONG KeyNameStringLength,
IN PVOID Data,
IN ULONG DataLength
)
/*++
Routine Description:
Reads the specified registry value
Arguments:
Handle - handle to the registry key
KeyNameString - value to read
KeyNameStringLength - length of string
Data - buffer to read data into
DataLength - length of data buffer
Return Value:
NTSTATUS returned as appropriate
--*/
{
NTSTATUS Status = STATUS_INSUFFICIENT_RESOURCES;
UNICODE_STRING KeyName;
ULONG Length;
PKEY_VALUE_FULL_INFORMATION FullInfo;
PAGED_CODE();
RtlInitUnicodeString(&KeyName, KeyNameString);
Length = sizeof(KEY_VALUE_FULL_INFORMATION) +
KeyNameStringLength + DataLength;
FullInfo = ExAllocatePool(PagedPool, Length);
if (FullInfo) {
Status = ZwQueryValueKey(Handle,
&KeyName,
KeyValueFullInformation,
FullInfo,
Length,
&Length);
if (NT_SUCCESS(Status)) {
if (DataLength >= FullInfo->DataLength) {
RtlCopyMemory(Data, ((PUCHAR) FullInfo) + FullInfo->DataOffset, FullInfo->DataLength);
} else {
Status = STATUS_BUFFER_TOO_SMALL;
} // buffer right length
} // if success
else {
TRACE(TL_PNP_ERROR,("ReadRegValue failed; ST:%x\n", Status));
}
ExFreePool(FullInfo);
} // if fullinfo
return Status;
}
#endif
#ifdef NT51_61883
static const WCHAR DeviceTypeName[] = L"GLOBAL";
#endif
static const WCHAR UniqueID_Low[] = L"UniqueID_Low";
static const WCHAR UniqueID_High[] = L"UniqueID_High";
static const WCHAR VendorID[] = L"VendorID";
static const WCHAR ModelID[] = L"ModelID";
static const WCHAR VendorText[] = L"VendorText";
static const WCHAR ModelText[] = L"ModelText";
static const WCHAR UnitModelText[] = L"UnitModelText";
static const WCHAR DeviceOPcr0Payload[] = L"DeviceOPcr0Payload";
static const WCHAR DeviceOPcr0DataRate[] = L"DeviceOPcr0DataRate";
#if DBG
static const WCHAR FriendlyName[] = L"FriendlyName";
#endif
BOOL
DVAccessDeviceInterface(
IN PDVCR_EXTENSION pDevExt,
IN const ULONG ulNumCategories,
IN GUID DVCategories[]
)
/*++
Routine Description:
Access to the device's interface section and update the VendorText,
ModelText and UnitModelText.
--*/
{
NTSTATUS ntStatus = STATUS_SUCCESS;
HANDLE hDevIntfKey;
UNICODE_STRING *aSymbolicLinkNames,
#ifdef NT51_61883
RefString,
#endif
TempUnicodeString;
ULONG i;
#ifdef NT51_61883
#if DBG
WCHAR DataBuffer[MAX_PATH];
#endif
#endif
//
// allocate space for the array of catagory names
//
if (!(aSymbolicLinkNames = ExAllocatePool(PagedPool,
sizeof(UNICODE_STRING) * ulNumCategories))) {
return FALSE;
}
//
// zero the array in case we're unable to fill it in below. the Destroy
// routine below will then correctly handle this case.
//
RtlZeroMemory(aSymbolicLinkNames, sizeof(UNICODE_STRING) * ulNumCategories);
#ifdef NT51_61883
//
// loop through each of the catagory GUID's for each of the pins,
// creating a symbolic link for each one.
//
RtlInitUnicodeString(&RefString, DeviceTypeName);
#endif
for (i = 0; i < ulNumCategories; i++) {
// Register our Device Interface
ntStatus = IoRegisterDeviceInterface(
pDevExt->pPhysicalDeviceObject,
&DVCategories[i],
#ifdef NT51_61883
&RefString,
#else
NULL,
#endif
&aSymbolicLinkNames[i]
);
if(!NT_SUCCESS(ntStatus)) {
//
// Can't register device interface
//
TRACE(TL_PNP_WARNING,("Cannot register device interface! ST:%x\n", ntStatus));
goto Exit;
}
TRACE(TL_PNP_TRACE,("AccessDeviceInterface:%d) Cateogory (Len:%d; MaxLen:%d); Name:\n%S\n", i,
aSymbolicLinkNames[i].Length, aSymbolicLinkNames[i].MaximumLength, aSymbolicLinkNames[i].Buffer));
//
// Get deice interface
//
hDevIntfKey = 0;
ntStatus = IoOpenDeviceInterfaceRegistryKey(&aSymbolicLinkNames[i],
STANDARD_RIGHTS_ALL,
&hDevIntfKey);
if (NT_SUCCESS(ntStatus)) {
#ifdef NT51_61883
#if DBG
// Get DeviceInstance
ntStatus = DVGetRegistryValue(hDevIntfKey,
(PWCHAR) FriendlyName,
sizeof(FriendlyName),
&DataBuffer,
MAX_PATH);
if(NT_SUCCESS(ntStatus)) {
TRACE(TL_PNP_TRACE,("AccessDeviceInterface:%S:%S\n", FriendlyName, DataBuffer));
} else {
TRACE(TL_PNP_WARNING,("Get %S failed; ST:%x\n", FriendlyName, ntStatus));
}
#endif
#endif
//
// Update ConfigROM info read from an AV/C device:
// UniqueID, VendorID, ModelID,
// VendorText, ModelText and UnitModelText
//
if(pDevExt->UniqueID.LowPart || pDevExt->UniqueID.HighPart) {
RtlInitUnicodeString(&TempUnicodeString, UniqueID_High);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_DWORD,
&pDevExt->UniqueID.HighPart,
sizeof(pDevExt->UniqueID.HighPart));
RtlInitUnicodeString(&TempUnicodeString, UniqueID_Low);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_DWORD,
&pDevExt->UniqueID.LowPart,
sizeof(pDevExt->UniqueID.LowPart));
TRACE(TL_PNP_TRACE,("Reg: %S = (Low)%x:(High)%x\n", TempUnicodeString.Buffer, pDevExt->UniqueID.LowPart, pDevExt->UniqueID.HighPart));
}
if(pDevExt->ulVendorID) {
RtlInitUnicodeString(&TempUnicodeString, VendorID);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_DWORD,
&pDevExt->ulVendorID,
sizeof(pDevExt->ulVendorID));
TRACE(TL_PNP_TRACE,("Reg: %S = %x\n", TempUnicodeString.Buffer, pDevExt->ulVendorID));
}
if(pDevExt->ulModelID) {
RtlInitUnicodeString(&TempUnicodeString, ModelID);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_DWORD,
&pDevExt->ulModelID,
sizeof(pDevExt->ulModelID));
TRACE(TL_PNP_TRACE,("Reg: %S = %x\n", TempUnicodeString.Buffer, pDevExt->ulModelID));
}
if(pDevExt->UnitIDs.ulVendorLength && pDevExt->UnitIDs.VendorText) {
RtlInitUnicodeString(&TempUnicodeString, VendorText);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_SZ,
pDevExt->UnitIDs.VendorText,
pDevExt->UnitIDs.ulVendorLength);
TRACE(TL_PNP_TRACE,("Reg: %S = %S\n", TempUnicodeString.Buffer, pDevExt->UnitIDs.VendorText));
}
#ifdef NT51_61883
if(pDevExt->UnitIDs.ulModelLength && pDevExt->UnitIDs.ModelText) {
RtlInitUnicodeString(&TempUnicodeString, ModelText);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_SZ,
pDevExt->UnitIDs.ModelText,
pDevExt->UnitIDs.ulModelLength);
TRACE(TL_PNP_WARNING,("Reg: %S = %S\n", TempUnicodeString.Buffer, pDevExt->UnitIDs.ModelText));
}
#else
if(pDevExt->UniRootModelString.Length && pDevExt->UniRootModelString.Buffer) {
RtlInitUnicodeString(&TempUnicodeString, ModelText);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_SZ,
pDevExt->UniRootModelString.Buffer,
pDevExt->UniRootModelString.Length);
TRACE(TL_PNP_WARNING,("Reg: %S = %S\n", TempUnicodeString.Buffer, pDevExt->UniRootModelString.Buffer));
}
#endif
#ifdef NT51_61883
if(pDevExt->UnitIDs.ulUnitModelLength && pDevExt->UnitIDs.UnitModelText) {
RtlInitUnicodeString(&TempUnicodeString, UnitModelText);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_SZ,
pDevExt->UnitIDs.UnitModelText,
pDevExt->UnitIDs.ulUnitModelLength);
TRACE(TL_PNP_WARNING,("Reg: %S = %S\n", TempUnicodeString.Buffer, pDevExt->UnitIDs.UnitModelText));
}
#else
if(pDevExt->UniUnitModelString.Length && pDevExt->UniUnitModelString.Buffer) {
RtlInitUnicodeString(&TempUnicodeString, UnitModelText);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_SZ,
pDevExt->UniUnitModelString.Buffer,
pDevExt->UniUnitModelString.Length);
TRACE(TL_PNP_WARNING,("Reg: %S = %S\n", TempUnicodeString.Buffer, pDevExt->UniUnitModelString.Buffer));
}
#endif
//
// Cache cache device's payload field of the oPCR[0];
// The valid range is 0 to 1023 where 0 is 1024.
// This value is needed for an application figure out
// max data rate. However, if this payload is
// dynamically changing, it will not be accurate.
//
if(pDevExt->pDevOutPlugs->NumPlugs) {
//
// 98 and 146 are two known valid payloads
//
ASSERT(pDevExt->pDevOutPlugs->DevPlug[0].PlugState.Payload <= MAX_PAYLOAD-1); // 0..MAX_PAYLOAD-1 is the valid range; "0" is MAX_PAYLOAD (1024) quadlets.
RtlInitUnicodeString(&TempUnicodeString, DeviceOPcr0Payload);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_DWORD,
&pDevExt->pDevOutPlugs->DevPlug[0].PlugState.Payload,
sizeof(pDevExt->pDevOutPlugs->DevPlug[0].PlugState.Payload));
TRACE(TL_PNP_TRACE,("Reg: %S = %d quadlets\n", TempUnicodeString.Buffer, pDevExt->pDevOutPlugs->DevPlug[0].PlugState.Payload));
//
// 0 (S100), 1(S200) and 2(S400) are the only three valid data rates.
//
ASSERT(pDevExt->pDevOutPlugs->DevPlug[0].PlugState.DataRate <= CMP_SPEED_S400);
RtlInitUnicodeString(&TempUnicodeString, DeviceOPcr0DataRate);
ZwSetValueKey(hDevIntfKey,
&TempUnicodeString,
0,
REG_DWORD,
&pDevExt->pDevOutPlugs->DevPlug[0].PlugState.DataRate,
sizeof(pDevExt->pDevOutPlugs->DevPlug[0].PlugState.DataRate));
TRACE(TL_PNP_TRACE,("Reg: %S = %d (0:S100,1:S200,2:S400)\n", TempUnicodeString.Buffer,
pDevExt->pDevOutPlugs->DevPlug[0].PlugState.DataRate));
}
ZwClose(hDevIntfKey);
} else {
TRACE(TL_PNP_ERROR,("IoOpenDeviceInterfaceRegistryKey failed; ST:%x\n", ntStatus));
}
RtlFreeUnicodeString(&aSymbolicLinkNames[i]);
}
Exit:
ExFreePool(aSymbolicLinkNames); aSymbolicLinkNames = NULL;
return ntStatus == STATUS_SUCCESS;
}
#endif