mirror of
https://github.com/reactos/reactos.git
synced 2025-01-04 05:20:54 +00:00
9c79a7982b
The source code is licensed under MIT license, taken from "MSDN Code Gallery Microsoft Samples" repository (https://github.com/microsoftarchive/msdn-code-gallery-microsoft) The original license was MS-PL, but the driver was later relicensed as MIT. Adopted to ReactOS code base by Michael Stamper. Co-authored-by: Michael Stamper <michaelstamper1@gmail.com>
2309 lines
74 KiB
C++
2309 lines
74 KiB
C++
/********************************************************************************
|
|
** Copyright (c) 1998-2000 Microsoft Corporation. All Rights Reserved.
|
|
**
|
|
** Portions Copyright (c) 1998-1999 Intel Corporation
|
|
**
|
|
********************************************************************************/
|
|
|
|
/* The file common.cpp was reviewed by LCA in June 2011 and is acceptable for use by Microsoft. */
|
|
|
|
// Every debug output has "Modulname text"
|
|
#define STR_MODULENAME "AC97 Common: "
|
|
|
|
#include "common.h"
|
|
|
|
|
|
/*****************************************************************************
|
|
* Static Members
|
|
*****************************************************************************
|
|
*/
|
|
|
|
//
|
|
// This is the register cache including registry names and default values. The
|
|
// first WORD contains the register value and the second WORD contains a flag.
|
|
// Currently, we only set SHREG_INVALID if we have to read the register at
|
|
// startup (that's true when there is no constant default value for the
|
|
// register). Note that we cache the registers only to prevent read access to
|
|
// the AC97 CoDec during runtime, because this is slow (40us).
|
|
// We only set SHREG_INIT if we want to set the register to default at driver
|
|
// startup. If needed, the third field contains the registry name and the
|
|
// forth field contains a default value that is used when there is no registry
|
|
// entry.
|
|
// The flag SHREG_NOCACHE is used when we don't want to cache the register
|
|
// at all. This is neccessary for status registers and sample rate registers.
|
|
//
|
|
tAC97Registers CAC97AdapterCommon::m_stAC97Registers[] =
|
|
{
|
|
{0x0000, SHREG_INVALID, NULL, 0}, // AC97REG_RESET
|
|
{0x8000, SHREG_INIT, L"MasterVolume", 0x0000}, // AC97REG_MASTER_VOLUME
|
|
{0x8000, SHREG_INIT, L"HeadphoneVolume", 0x0000}, // AC97REG_HPHONE_VOLUME
|
|
{0x8000, SHREG_INIT, L"MonooutVolume", 0x0000}, // AC97REG_MMONO_VOLUME
|
|
{0x0F0F, SHREG_INIT, L"ToneControls", 0x0F0F}, // AC97REG_MASTER_TONE
|
|
{0x0000, SHREG_INVALID |
|
|
SHREG_INIT, L"BeepVolume", 0x0000}, // AC97REG_BEEP_VOLUME
|
|
{0x8008, SHREG_INIT, L"PhoneVolume", 0x8008}, // AC97REG_PHONE_VOLUME
|
|
{0x8008, SHREG_INIT, L"MicVolume", 0x8008}, // AC97REG_MIC_VOLUME
|
|
{0x8808, SHREG_INIT, L"LineInVolume", 0x0808}, // AC97REG_LINE_IN_VOLUME
|
|
{0x8808, SHREG_INIT, L"CDVolume", 0x0808}, // AC97REG_CD_VOLUME
|
|
{0x8808, SHREG_INIT, L"VideoVolume", 0x0808}, // AC97REG_VIDEO_VOLUME
|
|
{0x8808, SHREG_INIT, L"AUXVolume", 0x0808}, // AC97REG_AUX_VOLUME
|
|
{0x8808, SHREG_INIT, L"WaveOutVolume", 0x0808}, // AC97REG_PCM_OUT_VOLUME
|
|
{0x0000, SHREG_INIT, L"RecordSelect", 0x0404}, // AC97REG_RECORD_SELECT
|
|
{0x8000, SHREG_INIT, L"RecordGain", 0x0000}, // AC97REG_RECORD_GAIN
|
|
{0x8000, SHREG_INIT, L"RecordGainMic", 0x0000}, // AC97REG_RECORD_GAIN_MIC
|
|
{0x0000, SHREG_INIT, L"GeneralPurpose", 0x0000}, // AC97REG_GENERAL
|
|
{0x0000, SHREG_INIT, L"3DControl", 0x0000}, // AC97REG_3D_CONTROL
|
|
{0x0000, SHREG_NOCACHE, NULL, 0}, // AC97REG_RESERVED
|
|
{0x0000, SHREG_NOCACHE |
|
|
SHREG_INIT, L"PowerDown", 0}, // AC97REG_POWERDOWN
|
|
// AC97-2.0 registers
|
|
{0x0000, SHREG_INVALID, NULL, 0}, // AC97REG_EXT_AUDIO_ID
|
|
{0x0000, SHREG_NOCACHE |
|
|
SHREG_INIT, L"ExtAudioCtrl", 0x4001}, // AC97REG_EXT_AUDIO_CTRL
|
|
{0xBB80, SHREG_NOCACHE, NULL, 0}, // AC97REG_FRONT_SAMPLERATE
|
|
{0xBB80, SHREG_NOCACHE, NULL, 0}, // AC97REG_SURROUND_SAMPLERATE
|
|
{0xBB80, SHREG_NOCACHE, NULL, 0}, // AC97REG_LFE_SAMPLERATE
|
|
{0xBB80, SHREG_NOCACHE, NULL, 0}, // AC97REG_RECORD_SAMPLERATE
|
|
{0xBB80, SHREG_NOCACHE, NULL, 0}, // AC97REG_MIC_SAMPLERATE
|
|
{0x8080, SHREG_INIT, L"CenterLFEVolume", 0x0000}, // AC97REG_CENTER_LFE_VOLUME
|
|
{0x8080, SHREG_INIT, L"SurroundVolume", 0x0000}, // AC97REG_SURROUND_VOLUME
|
|
{0x0000, SHREG_NOCACHE, NULL, 0} // AC97REG_RESERVED2
|
|
|
|
// We leave the other values blank. There would be a huge gap with 31
|
|
// elements that are currently unused, and then there would be 2 other
|
|
// (used) values, the vendor IDs. We just force a read from the vendor
|
|
// IDs in the end of ProbeHWConfig to fill the cache.
|
|
};
|
|
|
|
|
|
//
|
|
// This is the hardware configuration information. The first struct is for
|
|
// nodes, which we default to FALSE. The second struct is for Pins, which
|
|
// contains the configuration (FALSE) and the registry string which is the
|
|
// reason for making a static struct so we can just fill in the name.
|
|
//
|
|
tHardwareConfig CAC97AdapterCommon::m_stHardwareConfig =
|
|
{
|
|
// Nodes
|
|
{{FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE},
|
|
{FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE}, {FALSE},
|
|
{FALSE}},
|
|
// Pins
|
|
{{FALSE, L"DisablePCBeep"}, // PINC_PCBEEP_PRESENT
|
|
{FALSE, L"DisablePhone"}, // PINC_PHONE_PRESENT
|
|
{FALSE, L"DisableMic2"}, // PINC_MIC2_PRESENT
|
|
{FALSE, L"DisableVideo"}, // PINC_VIDEO_PRESENT
|
|
{FALSE, L"DisableAUX"}, // PINC_AUX_PRESENT
|
|
{FALSE, L"DisableHeadphone"}, // PINC_HPOUT_PRESENT
|
|
{FALSE, L"DisableMonoOut"}, // PINC_MONOOUT_PRESENT
|
|
{FALSE, L"DisableMicIn"}, // PINC_MICIN_PRESENT
|
|
{FALSE, L"DisableMic"}, // PINC_MIC_PRESENT
|
|
{FALSE, L"DisableLineIn"}, // PINC_LINEIN_PRESENT
|
|
{FALSE, L"DisableCD"}, // PINC_CD_PRESENT
|
|
{FALSE, L"DisableSurround"}, // PINC_SURROUND_PRESENT
|
|
{FALSE, L"DisableCenterLFE"}} // PINC_CENTER_LFE_PRESENT
|
|
};
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma code_seg("PAGE")
|
|
#endif
|
|
/*****************************************************************************
|
|
* NewAdapterCommon
|
|
*****************************************************************************
|
|
* Create a new adapter common object.
|
|
*/
|
|
NTSTATUS NewAdapterCommon
|
|
(
|
|
OUT PUNKNOWN *Unknown,
|
|
IN REFCLSID,
|
|
IN PUNKNOWN UnknownOuter OPTIONAL,
|
|
_When_((PoolType & NonPagedPoolMustSucceed) != 0,
|
|
__drv_reportError("Must succeed pool allocations are forbidden. "
|
|
"Allocation failures cause a system crash"))
|
|
IN POOL_TYPE PoolType
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
ASSERT (Unknown);
|
|
|
|
DOUT (DBG_PRINT, ("[NewAdapterCommon]"));
|
|
|
|
STD_CREATE_BODY_WITH_TAG_(CAC97AdapterCommon,Unknown,UnknownOuter,PoolType,
|
|
PoolTag, PADAPTERCOMMON);
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::Init
|
|
*****************************************************************************
|
|
* Initialize the adapter common object -> initialize and probe HW.
|
|
* Pass only checked resources.
|
|
*/
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::Init
|
|
(
|
|
IN PRESOURCELIST ResourceList,
|
|
IN PDEVICE_OBJECT DeviceObject
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
ASSERT (ResourceList);
|
|
ASSERT (DeviceObject);
|
|
|
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::Init]"));
|
|
|
|
//
|
|
// Set the topology pointer to NULL.
|
|
//
|
|
m_Topology = NULL;
|
|
|
|
//
|
|
// Save the device object
|
|
//
|
|
m_pDeviceObject = DeviceObject;
|
|
|
|
//
|
|
// Get the base address for the AC97 codec and bus master.
|
|
//
|
|
ASSERT (ResourceList->FindTranslatedPort (0));
|
|
m_pCodecBase = (PUSHORT)ResourceList->FindTranslatedPort (0)->
|
|
u.Port.Start.QuadPart;
|
|
|
|
ASSERT (ResourceList->FindTranslatedPort (1));
|
|
m_pBusMasterBase = (PUCHAR)ResourceList->FindTranslatedPort (1)->
|
|
u.Port.Start.QuadPart;
|
|
|
|
DOUT (DBG_SYSINFO, ("Configuration:\n"
|
|
" Bus Master = 0x%p\n"
|
|
" Codec = 0x%p",
|
|
m_pBusMasterBase, m_pCodecBase));
|
|
|
|
//
|
|
// Set m_bDirectRead to TRUE so that all AC97 register read and
|
|
// writes are going directly to the HW
|
|
//
|
|
m_bDirectRead = TRUE;
|
|
|
|
//
|
|
// Initialize the hardware.
|
|
//
|
|
ntStatus = InitAC97 ();
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Probe hardware configuration
|
|
//
|
|
ntStatus = ProbeHWConfig ();
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("Probing of hardware configuration failed!"));
|
|
return ntStatus;
|
|
}
|
|
|
|
//
|
|
// Now, every AC97 read access goes to the cache.
|
|
//
|
|
m_bDirectRead = FALSE;
|
|
|
|
//
|
|
// Restore the AC97 registers now.
|
|
//
|
|
#if (DBG)
|
|
DumpConfig ();
|
|
#endif
|
|
ntStatus = SetAC97Default ();
|
|
|
|
//
|
|
// Initialize the device state.
|
|
//
|
|
m_PowerState = PowerDeviceD0;
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::~CAC97AdapterCommon
|
|
*****************************************************************************
|
|
* Destructor.
|
|
*/
|
|
CAC97AdapterCommon::~CAC97AdapterCommon ()
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::~CAC97AdapterCommon]"));
|
|
}
|
|
|
|
|
|
#if (DBG)
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::DumpConfig
|
|
*****************************************************************************
|
|
* Dumps the HW configuration for the AC97 codec.
|
|
*/
|
|
void CAC97AdapterCommon::DumpConfig (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::DumpConfig]"));
|
|
|
|
//
|
|
// Print debug output for MICIN.
|
|
//
|
|
if (GetPinConfig (PINC_MICIN_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("MICIN found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No MICIN found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for tone controls.
|
|
//
|
|
if (GetNodeConfig (NODEC_TONE_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("Tone controls found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No tone controls found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for mono out.
|
|
//
|
|
if (!GetPinConfig (PINC_MONOOUT_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("No mono out found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for headphones.
|
|
//
|
|
if (!GetPinConfig (PINC_HPOUT_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("No headphone out found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for loudness.
|
|
//
|
|
if (GetNodeConfig (NODEC_LOUDNESS_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("Loudness found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No Loudness found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for 3D.
|
|
//
|
|
if (GetNodeConfig (NODEC_3D_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("3D controls found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No 3D controls found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for pc beep.
|
|
//
|
|
if (GetPinConfig (PINC_PCBEEP_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("PC beep found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No PC beep found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for phone line (or mono line input).
|
|
//
|
|
if (GetPinConfig (PINC_PHONE_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("Phone found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No Phone found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for video.
|
|
//
|
|
if (GetPinConfig (PINC_VIDEO_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("Video in found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No Video in found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for AUX.
|
|
//
|
|
if (GetPinConfig (PINC_AUX_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("AUX in found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No AUX in found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for second miorophone.
|
|
//
|
|
if (GetPinConfig (PINC_MIC2_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("MIC2 found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("No MIC2 found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for 3D stuff.
|
|
//
|
|
if (GetNodeConfig (NODEC_3D_PRESENT))
|
|
{
|
|
if (GetNodeConfig (NODEC_3D_CENTER_ADJUSTABLE))
|
|
{
|
|
DOUT (DBG_PROBE, ("Adjustable 3D center control found"));
|
|
}
|
|
if (GetNodeConfig (NODEC_3D_DEPTH_ADJUSTABLE))
|
|
{
|
|
DOUT (DBG_PROBE, ("Nonadjustable 3D depth control found"));
|
|
}
|
|
}
|
|
|
|
//
|
|
// Print debug output for quality of master volume.
|
|
//
|
|
if (GetNodeConfig (NODEC_6BIT_MASTER_VOLUME))
|
|
{
|
|
DOUT (DBG_PROBE, ("6bit master out found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("5bit master out found"));
|
|
}
|
|
|
|
//
|
|
// Print debug output for quality of headphones volume.
|
|
//
|
|
if (GetPinConfig (PINC_HPOUT_PRESENT))
|
|
{
|
|
if (GetNodeConfig (NODEC_6BIT_HPOUT_VOLUME))
|
|
{
|
|
DOUT (DBG_PROBE, ("6bit headphone out found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("5bit headphone out found"));
|
|
}
|
|
}
|
|
|
|
//
|
|
// Print debug output for quality of mono out volume.
|
|
//
|
|
if (GetPinConfig (PINC_MONOOUT_PRESENT))
|
|
{
|
|
if (GetNodeConfig (NODEC_6BIT_MONOOUT_VOLUME))
|
|
{
|
|
DOUT (DBG_PROBE, ("6bit mono out found"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("5bit mono out found"));
|
|
}
|
|
}
|
|
|
|
//
|
|
// Print sample rate information.
|
|
//
|
|
if (GetNodeConfig (NODEC_PCM_VARIABLERATE_SUPPORTED))
|
|
{
|
|
DOUT (DBG_PROBE, ("PCM variable sample rate supported"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("only 48KHz PCM supported"));
|
|
}
|
|
|
|
//
|
|
// Print double rate information.
|
|
//
|
|
if (GetNodeConfig (NODEC_PCM_DOUBLERATE_SUPPORTED))
|
|
{
|
|
DOUT (DBG_PROBE, ("PCM double sample rate supported"));
|
|
}
|
|
|
|
//
|
|
// Print mic rate information.
|
|
//
|
|
if (GetNodeConfig (NODEC_MIC_VARIABLERATE_SUPPORTED))
|
|
{
|
|
DOUT (DBG_PROBE, ("MIC variable sample rate supported"));
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_PROBE, ("only 48KHz MIC supported"));
|
|
}
|
|
|
|
// print DAC information
|
|
if (GetNodeConfig (NODEC_CENTER_DAC_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("center DAC found"));
|
|
}
|
|
if (GetNodeConfig (NODEC_SURROUND_DAC_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("surround DAC found"));
|
|
}
|
|
if (GetNodeConfig (NODEC_LFE_DAC_PRESENT))
|
|
{
|
|
DOUT (DBG_PROBE, ("LFE DAC found"));
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::NonDelegatingQueryInterface
|
|
*****************************************************************************
|
|
* Obtains an interface. This function works just like a COM QueryInterface
|
|
* call and is used if the object is not being aggregated.
|
|
* We basically just check any GUID we know and return this object in case we
|
|
* know it.
|
|
*/
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::NonDelegatingQueryInterface
|
|
(
|
|
_In_ REFIID Interface,
|
|
_COM_Outptr_ PVOID * Object
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
ASSERT (Object);
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::NonDelegatingQueryInterface]"));
|
|
|
|
// Is it IID_IUnknown?
|
|
if (IsEqualGUIDAligned (Interface, IID_IUnknown))
|
|
{
|
|
*Object = (PVOID)(PUNKNOWN)(PADAPTERCOMMON)this;
|
|
}
|
|
else
|
|
// or IID_IAC97AdapterCommon ...
|
|
if (IsEqualGUIDAligned (Interface, IID_IAC97AdapterCommon))
|
|
{
|
|
*Object = (PVOID)(PADAPTERCOMMON)this;
|
|
}
|
|
else
|
|
// or IID_IAdapterPowerManagement ...
|
|
if (IsEqualGUIDAligned (Interface, IID_IAdapterPowerManagement))
|
|
{
|
|
*Object = (PVOID)(PADAPTERPOWERMANAGEMENT)this;
|
|
}
|
|
else
|
|
{
|
|
// nothing found, must be an unknown interface.
|
|
*Object = NULL;
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// We reference the interface for the caller.
|
|
//
|
|
((PUNKNOWN)*Object)->AddRef ();
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::InitAC97
|
|
*****************************************************************************
|
|
* Initialize the AC97 (without hosing the modem if it got installed first).
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::InitAC97 (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::InitAC97]"));
|
|
|
|
//
|
|
// First check if there is an AC link to the primary CoDec.
|
|
//
|
|
NTSTATUS ntStatus = PrimaryCodecReady ();
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
//
|
|
// Second, reset this primary CoDec; If this is a AMC97 CoDec, only
|
|
// the audio registers are reset. If this is a MC97 CoDec, the CoDec
|
|
// should ignore the reset (according to the spec).
|
|
//
|
|
WriteCodecRegister (AC97REG_RESET, 0x00, 0xFFFF);
|
|
|
|
ntStatus = PowerUpCodec ();
|
|
}
|
|
else
|
|
{
|
|
DOUT (DBG_ERROR, ("Initialization of AC97 CoDec failed."));
|
|
}
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::Check6thBitSupport
|
|
*****************************************************************************
|
|
* Probes for 6th bit volume control support.
|
|
* The passed parameters are the AC97 register that has the volume control and
|
|
* the node config that should be set in this case.
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::Check6thBitSupport
|
|
(
|
|
IN AC97Register AC97Reg,
|
|
IN TopoNodeConfig Config
|
|
)
|
|
{
|
|
PAGED_CODE();
|
|
|
|
NTSTATUS ntStatus;
|
|
WORD wCodecReg;
|
|
WORD wOriginal;
|
|
|
|
// Read the current value.
|
|
ntStatus = ReadCodecRegister (AC97Reg, &wOriginal);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Write the 6th bit; for mono controls we write 0x20, for stereo
|
|
// controls 0x2020.
|
|
ntStatus = WriteCodecRegister (AC97Reg,
|
|
(AC97Reg == AC97REG_MMONO_VOLUME) ? 0x0020 : 0x2020, 0xFFFF);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// And read back.
|
|
ntStatus = ReadCodecRegister (AC97Reg, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Check return. For mono 0x20 and for stereo 0x2020.
|
|
if (((wCodecReg & 0x0020) && (AC97Reg == AC97REG_MMONO_VOLUME)) ||
|
|
(wCodecReg & 0x2020))
|
|
{
|
|
SetNodeConfig (Config, TRUE);
|
|
}
|
|
else
|
|
{
|
|
SetNodeConfig (Config, FALSE);
|
|
}
|
|
|
|
// Restore original value.
|
|
WriteCodecRegister (AC97Reg, wOriginal, 0xFFFF);
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ProbeHWConfig
|
|
*****************************************************************************
|
|
* Probes the hardware configuration.
|
|
* If this function returns with an error, then the configuration is not
|
|
* complete! Probing the registers is done by reading them (and comparing with
|
|
* the HW default value) or when the default is unknown, writing to them and
|
|
* reading back + restoring.
|
|
* Additionally, we read the registry so that a HW vendor can overwrite (means
|
|
* disable) found registers in case the adapter (e.g. video) is not visible to
|
|
* the user (he can't plug in a video audio there).
|
|
*
|
|
* This is a very long function with all of the error checking!
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::ProbeHWConfig (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
|
DWORD dwGlobalStatus;
|
|
WORD wCodecID;
|
|
WORD wCodecReg;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ProbeHWConfig]"));
|
|
|
|
//
|
|
// Wait for the whatever 97 to complete reset and establish a link.
|
|
//
|
|
ntStatus = PrimaryCodecReady ();
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Master volume is one of the supported registers on an AC97
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_MASTER_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Default is x8000.
|
|
if (wCodecReg != 0x8000)
|
|
return STATUS_NO_SUCH_DEVICE;
|
|
|
|
//
|
|
// This gives us information about the AC97 CoDec
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_RESET, &wCodecID);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Fill out the configuration stuff.
|
|
//
|
|
|
|
SetPinConfig (PINC_MICIN_PRESENT, wCodecID & 0x0001);
|
|
|
|
// Check if OEM wants to disable MIC record line.
|
|
if (DisableAC97Pin (PINC_MICIN_PRESENT))
|
|
SetPinConfig (PINC_MICIN_PRESENT, FALSE);
|
|
|
|
// If we still have MIC record line, enable the DAC in ext. audio register.
|
|
if (GetPinConfig (PINC_MICIN_PRESENT))
|
|
// Enable ADC MIC.
|
|
WriteCodecRegister (AC97REG_EXT_AUDIO_CTRL, 0, 0x4000);
|
|
else
|
|
// Disable ADC MIC.
|
|
WriteCodecRegister (AC97REG_EXT_AUDIO_CTRL, 0x4000, 0x4000);
|
|
|
|
//
|
|
// Continue setting configuration information.
|
|
//
|
|
|
|
SetNodeConfig (NODEC_TONE_PRESENT, wCodecID & 0x0004);
|
|
SetNodeConfig (NODEC_SIMUL_STEREO_PRESENT, wCodecID & 0x0008);
|
|
SetPinConfig (PINC_HPOUT_PRESENT, wCodecID & 0x0010);
|
|
|
|
// Check if OEM wants to disable headphone output.
|
|
if (DisableAC97Pin (PINC_HPOUT_PRESENT))
|
|
SetPinConfig (PINC_HPOUT_PRESENT, FALSE);
|
|
|
|
SetNodeConfig (NODEC_LOUDNESS_PRESENT, wCodecID & 0x0020);
|
|
SetNodeConfig (NODEC_3D_PRESENT, wCodecID & 0x7C00);
|
|
|
|
//
|
|
// Test for the input pins that are always there but could be disabled
|
|
// by the HW vender
|
|
//
|
|
|
|
// Check if OEM wants to disable mic input.
|
|
SetPinConfig (PINC_MIC_PRESENT, !DisableAC97Pin (PINC_MIC_PRESENT));
|
|
|
|
// Check if OEM wants to disable line input.
|
|
SetPinConfig (PINC_LINEIN_PRESENT, !DisableAC97Pin (PINC_LINEIN_PRESENT));
|
|
|
|
// Check if OEM wants to disable CD input.
|
|
SetPinConfig (PINC_CD_PRESENT, !DisableAC97Pin (PINC_CD_PRESENT));
|
|
|
|
|
|
//
|
|
// For the rest, we have to probe the registers.
|
|
//
|
|
|
|
//
|
|
// Test for Mono out.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_MMONO_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Default is x8000.
|
|
SetPinConfig (PINC_MONOOUT_PRESENT, (wCodecReg == 0x8000));
|
|
|
|
// Check if OEM wants to disable mono output.
|
|
if (DisableAC97Pin (PINC_MONOOUT_PRESENT))
|
|
SetPinConfig (PINC_MONOOUT_PRESENT, FALSE);
|
|
|
|
//
|
|
// Test for PC beeper support.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_BEEP_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// default is x0 or x8000. If it's 0x8000 then we know for sure that the
|
|
// CoDec has a PcBeep, otherwise we have to check the register
|
|
if (wCodecReg == 0x8000)
|
|
SetPinConfig (PINC_PCBEEP_PRESENT, TRUE);
|
|
else if (!wCodecReg)
|
|
{
|
|
// mute the pc beeper.
|
|
ntStatus = WriteCodecRegister (AC97REG_BEEP_VOLUME, 0x8000, 0xFFFF);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// read back
|
|
ntStatus = ReadCodecRegister (AC97REG_BEEP_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
if (wCodecReg == 0x8000)
|
|
{
|
|
// yep, we have support.
|
|
SetPinConfig (PINC_PCBEEP_PRESENT, TRUE);
|
|
// reset to default value.
|
|
WriteCodecRegister (AC97REG_BEEP_VOLUME, 0x0, 0xFFFF);
|
|
}
|
|
else
|
|
// nope, not present
|
|
SetPinConfig (PINC_PCBEEP_PRESENT, FALSE);
|
|
}
|
|
else
|
|
// any other value then 0x0 and 0x8000.
|
|
SetPinConfig (PINC_PCBEEP_PRESENT, FALSE);
|
|
|
|
// Check if OEM wants to disable beeper support.
|
|
if (DisableAC97Pin (PINC_PCBEEP_PRESENT))
|
|
SetPinConfig (PINC_PCBEEP_PRESENT, FALSE);
|
|
|
|
//
|
|
// Test for phone support.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_PHONE_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Default is x8008.
|
|
SetPinConfig (PINC_PHONE_PRESENT, (wCodecReg == 0x8008));
|
|
|
|
// Check if OEM wants to disable phone input.
|
|
if (DisableAC97Pin (PINC_PHONE_PRESENT))
|
|
SetPinConfig (PINC_PHONE_PRESENT, FALSE);
|
|
|
|
//
|
|
// Test for video support.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_VIDEO_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Default is x8808.
|
|
SetPinConfig (PINC_VIDEO_PRESENT, (wCodecReg == 0x8808));
|
|
|
|
// Check if OEM wants to disable video input.
|
|
if (DisableAC97Pin (PINC_VIDEO_PRESENT))
|
|
SetPinConfig (PINC_VIDEO_PRESENT, FALSE);
|
|
|
|
//
|
|
// Test for Aux support.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_AUX_VOLUME, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Default is 0x8808.
|
|
SetPinConfig (PINC_AUX_PRESENT, (wCodecReg == 0x8808));
|
|
|
|
// Check if OEM wants to disable aux input.
|
|
if (DisableAC97Pin (PINC_AUX_PRESENT))
|
|
SetPinConfig (PINC_AUX_PRESENT, FALSE);
|
|
|
|
//
|
|
// Test for Mic2 source.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_GENERAL, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Test for Mic2 select bit.
|
|
if (wCodecReg & 0x0100)
|
|
SetPinConfig (PINC_MIC2_PRESENT, TRUE);
|
|
else
|
|
{
|
|
// Select Mic2 as source.
|
|
ntStatus = WriteCodecRegister (AC97REG_GENERAL, 0x0100, 0x0100);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Read back.
|
|
ntStatus = ReadCodecRegister (AC97REG_GENERAL, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
if (wCodecReg & 0x0100)
|
|
{
|
|
// Yep, we have support so set it to the default value.
|
|
SetPinConfig (PINC_MIC2_PRESENT, TRUE);
|
|
// reset to default value.
|
|
WriteCodecRegister (AC97REG_GENERAL, 0, 0x0100);
|
|
}
|
|
else
|
|
SetPinConfig (PINC_MIC2_PRESENT, FALSE);
|
|
}
|
|
|
|
// Check if OEM wants to disable mic2 input.
|
|
if (DisableAC97Pin (PINC_MIC2_PRESENT))
|
|
SetPinConfig (PINC_MIC2_PRESENT, FALSE);
|
|
|
|
//
|
|
// Test the 3D controls.
|
|
//
|
|
if (GetNodeConfig (NODEC_3D_PRESENT))
|
|
{
|
|
//
|
|
// First test for fixed 3D controls. Write default value ...
|
|
//
|
|
ntStatus = WriteCodecRegister (AC97REG_3D_CONTROL, 0, 0xFFFF);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
// Read 3D register. Default is 0 when adjustable, otherwise it is
|
|
// a fixed value.
|
|
ntStatus = ReadCodecRegister (AC97REG_3D_CONTROL, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Check center and depth separately.
|
|
//
|
|
|
|
// For center
|
|
SetNodeConfig (NODEC_3D_CENTER_ADJUSTABLE, !(wCodecReg & 0x0F00));
|
|
|
|
// For depth
|
|
SetNodeConfig (NODEC_3D_DEPTH_ADJUSTABLE, !(wCodecReg & 0x000F));
|
|
|
|
//
|
|
// Test for adjustable controls.
|
|
//
|
|
WriteCodecRegister (AC97REG_3D_CONTROL, 0x0A0A, 0xFFFF);
|
|
|
|
// Read 3D register. Now it should be 0x0A0A for adjustable controls,
|
|
// otherwise it is a fixed control or simply not there.
|
|
ReadCodecRegister (AC97REG_3D_CONTROL, &wCodecReg);
|
|
|
|
// Restore the default value
|
|
WriteCodecRegister (AC97REG_3D_CONTROL, 0, 0xFFFF);
|
|
|
|
// Check the center control for beeing adjustable
|
|
if (GetNodeConfig (NODEC_3D_CENTER_ADJUSTABLE) &&
|
|
(wCodecReg & 0x0F00) != 0x0A00)
|
|
{
|
|
SetNodeConfig (NODEC_3D_CENTER_ADJUSTABLE, FALSE);
|
|
}
|
|
|
|
// Check the depth control for beeing adjustable
|
|
if (GetNodeConfig (NODEC_3D_DEPTH_ADJUSTABLE) &&
|
|
(wCodecReg & 0x000F) != 0x000A)
|
|
{
|
|
SetNodeConfig (NODEC_3D_DEPTH_ADJUSTABLE, FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check for 6th bit support in volume controls. To check the 6th bit,
|
|
// we first have to write a value (with 6th bit set) and then read it
|
|
// back. After that, we should restore the register to its default value.
|
|
//
|
|
|
|
//
|
|
// Start with the master volume.
|
|
//
|
|
Check6thBitSupport (AC97REG_MASTER_VOLUME, NODEC_6BIT_MASTER_VOLUME);
|
|
|
|
//
|
|
// Check for a headphone volume control.
|
|
//
|
|
if (GetPinConfig (PINC_HPOUT_PRESENT))
|
|
{
|
|
Check6thBitSupport (AC97REG_HPHONE_VOLUME, NODEC_6BIT_HPOUT_VOLUME);
|
|
}
|
|
|
|
//
|
|
// Mono out there?
|
|
//
|
|
if (GetPinConfig (PINC_MONOOUT_PRESENT))
|
|
{
|
|
Check6thBitSupport (AC97REG_MMONO_VOLUME, NODEC_6BIT_MONOOUT_VOLUME);
|
|
}
|
|
|
|
//
|
|
// Get extended AC97 V2.0 information
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_EXT_AUDIO_ID, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Store the information
|
|
//
|
|
SetNodeConfig (NODEC_PCM_VARIABLERATE_SUPPORTED, wCodecReg & 0x0001);
|
|
SetNodeConfig (NODEC_PCM_DOUBLERATE_SUPPORTED, wCodecReg & 0x0002);
|
|
SetNodeConfig (NODEC_MIC_VARIABLERATE_SUPPORTED, wCodecReg & 0x0008);
|
|
SetNodeConfig (NODEC_CENTER_DAC_PRESENT, wCodecReg & 0x0040);
|
|
SetNodeConfig (NODEC_SURROUND_DAC_PRESENT, wCodecReg & 0x0080);
|
|
SetNodeConfig (NODEC_LFE_DAC_PRESENT, wCodecReg & 0x0100);
|
|
|
|
//
|
|
// In case we have some features get some more information and program
|
|
// the codec.
|
|
//
|
|
if (wCodecReg)
|
|
{
|
|
//
|
|
// Enable variable sample rate in the control register and disable
|
|
// double rate. Also enable all DACs.
|
|
//
|
|
WriteCodecRegister (AC97REG_EXT_AUDIO_CTRL, wCodecReg & 0x0009, 0x380B);
|
|
|
|
//
|
|
// Check for codecs that have only one sample rate converter. These
|
|
// codecs will stick registers AC97REG_FRONT_SAMPLERATE and
|
|
// AC97REG_RECORD_SAMPLERATE together.
|
|
//
|
|
if (GetNodeConfig (NODEC_PCM_VARIABLERATE_SUPPORTED))
|
|
{
|
|
// The default of the sample rate registers should be 0xBB80.
|
|
WriteCodecRegister (AC97REG_FRONT_SAMPLERATE, 0xBB80, 0xFFFF);
|
|
|
|
// Write 44.1KHz into record VSR, then check playback again.
|
|
WriteCodecRegister (AC97REG_RECORD_SAMPLERATE, 0xAC44, 0xFFFF);
|
|
ntStatus = ReadCodecRegister (AC97REG_FRONT_SAMPLERATE, &wCodecReg);
|
|
WriteCodecRegister (AC97REG_RECORD_SAMPLERATE, 0xBB80, 0xFFFF);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Set the flag accordingly
|
|
//
|
|
SetNodeConfig (NODEC_PCM_VSR_INDEPENDENT_RATES, (wCodecReg == 0xBB80));
|
|
}
|
|
|
|
//
|
|
// Check multichanel support on the AC97.
|
|
//
|
|
if (GetNodeConfig (NODEC_SURROUND_DAC_PRESENT))
|
|
{
|
|
dwGlobalStatus = ReadBMControlRegister32 (GLOB_STA);
|
|
|
|
//
|
|
// Codec supports >2 chanel, does AC97 too?
|
|
//
|
|
if ((GetNodeConfig (NODEC_CENTER_DAC_PRESENT) ||
|
|
GetNodeConfig (NODEC_LFE_DAC_PRESENT)) &&
|
|
(dwGlobalStatus & GLOB_STA_MC6))
|
|
{
|
|
SetPinConfig (PINC_CENTER_LFE_PRESENT, TRUE);
|
|
}
|
|
else
|
|
{
|
|
SetPinConfig (PINC_CENTER_LFE_PRESENT, FALSE);
|
|
}
|
|
|
|
//
|
|
// Do we support at least 4 channels?
|
|
//
|
|
SetPinConfig (PINC_SURROUND_PRESENT, (dwGlobalStatus & GLOB_STA_MC4));
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Only 2 channel (stereo) support.
|
|
//
|
|
SetPinConfig (PINC_CENTER_LFE_PRESENT, FALSE);
|
|
SetPinConfig (PINC_SURROUND_PRESENT, FALSE);
|
|
}
|
|
}
|
|
|
|
// Check if OEM wants to disable surround output.
|
|
if (DisableAC97Pin (PINC_SURROUND_PRESENT))
|
|
SetPinConfig (PINC_SURROUND_PRESENT, FALSE);
|
|
|
|
// Check if OEM wants to disable center and LFE output.
|
|
if (DisableAC97Pin (PINC_CENTER_LFE_PRESENT))
|
|
SetPinConfig (PINC_CENTER_LFE_PRESENT, FALSE);
|
|
|
|
//
|
|
// Check the 6th bit support for the additional channels.
|
|
//
|
|
if (GetPinConfig (PINC_SURROUND_PRESENT))
|
|
Check6thBitSupport (AC97REG_SURROUND_VOLUME, NODEC_6BIT_SURROUND_VOLUME);
|
|
|
|
if (GetPinConfig (PINC_CENTER_LFE_PRESENT))
|
|
Check6thBitSupport (AC97REG_CENTER_LFE_VOLUME, NODEC_6BIT_CENTER_LFE_VOLUME);
|
|
|
|
//
|
|
// We read these registers because they are dependent on the codec.
|
|
//
|
|
ReadCodecRegister (AC97REG_VENDOR_ID1, &wCodecReg);
|
|
ReadCodecRegister (AC97REG_VENDOR_ID2, &wCodecReg);
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::AcquireCodecSemiphore
|
|
*****************************************************************************
|
|
* Acquires the AC97 semiphore. This can not be called at dispatch level
|
|
* because it can timeout if a lower IRQL thread has the semaphore.
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::AcquireCodecSemiphore ()
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::AcquireCodecSemiphore]"));
|
|
|
|
ULONG ulCount = 0;
|
|
while (READ_PORT_UCHAR (m_pBusMasterBase + CAS) & CAS_CAS)
|
|
{
|
|
//
|
|
// Do we want to give up??
|
|
//
|
|
if (ulCount++ > 100)
|
|
{
|
|
DOUT (DBG_ERROR, ("Cannot acquire semaphore."));
|
|
return STATUS_IO_TIMEOUT;
|
|
}
|
|
|
|
//
|
|
// Let's wait a little, 40us and then try again.
|
|
//
|
|
KeStallExecutionProcessor (40L);
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ReadCodecRegister
|
|
*****************************************************************************
|
|
* Reads a AC97 register. Don't call at PASSIVE_LEVEL.
|
|
*/
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::ReadCodecRegister
|
|
(
|
|
_In_range_(0, AC97REG_INVALID) AC97Register reg,
|
|
_Out_ PWORD wData
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
ASSERT (wData);
|
|
ASSERT (reg < AC97REG_INVALID); // audio can only be in the primary codec
|
|
_Analysis_assume_(reg < AC97REG_INVALID);
|
|
|
|
NTSTATUS ntStatus;
|
|
ULONG Status;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ReadCodecRegister]"));
|
|
|
|
//
|
|
// Check if we have to access the HW directly.
|
|
//
|
|
if (m_bDirectRead || (m_stAC97Registers[reg].wFlags & SHREG_INVALID) ||
|
|
(m_stAC97Registers[reg].wFlags & SHREG_NOCACHE))
|
|
{
|
|
//
|
|
// Grab the codec access semiphore.
|
|
//
|
|
ntStatus = AcquireCodecSemiphore ();
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("ReadCodecRegister couldn't acquire the semiphore"
|
|
" for reg. %s", reg <= AC97REG_RESERVED2 ? RegStrings[reg] :
|
|
reg == AC97REG_VENDOR_ID1 ? "REG_VENDOR_ID1" :
|
|
reg == AC97REG_VENDOR_ID2 ? "REG_VENDOR_ID2" :
|
|
"REG_INVALID"));
|
|
return ntStatus;
|
|
}
|
|
|
|
//
|
|
// Read the data.
|
|
//
|
|
*wData = READ_PORT_USHORT (m_pCodecBase + reg);
|
|
|
|
//
|
|
// Check to see if the read was successful.
|
|
//
|
|
Status = READ_PORT_ULONG ((PULONG)(m_pBusMasterBase + GLOB_STA));
|
|
if (Status & GLOB_STA_RCS)
|
|
{
|
|
//
|
|
// clear the timeout bit
|
|
//
|
|
WRITE_PORT_ULONG ((PULONG)(m_pBusMasterBase + GLOB_STA), Status);
|
|
*wData = 0;
|
|
DOUT (DBG_ERROR, ("ReadCodecRegister timed out for register %s",
|
|
reg <= AC97REG_RESERVED2 ? RegStrings[reg] :
|
|
reg == AC97REG_VENDOR_ID1 ? "REG_VENDOR_ID1" :
|
|
reg == AC97REG_VENDOR_ID2 ? "REG_VENDOR_ID2" :
|
|
"REG_INVALID"));
|
|
return STATUS_IO_TIMEOUT;
|
|
}
|
|
|
|
//
|
|
// Clear invalid flag
|
|
//
|
|
m_stAC97Registers[reg].wCache = *wData;
|
|
m_stAC97Registers[reg].wFlags &= ~SHREG_INVALID;
|
|
|
|
DOUT (DBG_REGS, ("AC97READ: %s = 0x%04x (HW)",
|
|
reg <= AC97REG_RESERVED2 ? RegStrings[reg] :
|
|
reg == AC97REG_VENDOR_ID1 ? "REG_VENDOR_ID1" :
|
|
reg == AC97REG_VENDOR_ID2 ? "REG_VENDOR_ID2" :
|
|
"REG_INVALID", *wData));
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Otherwise, use the value in the cache.
|
|
//
|
|
*wData = m_stAC97Registers[reg].wCache;
|
|
DOUT (DBG_REGS, ("AC97READ: %s = 0x%04x (C)",
|
|
reg <= AC97REG_RESERVED2 ? RegStrings[reg] :
|
|
reg == AC97REG_VENDOR_ID1 ? "REG_VENDOR_ID1" :
|
|
reg == AC97REG_VENDOR_ID2 ? "REG_VENDOR_ID2" :
|
|
"REG_INVALID", *wData));
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::WriteCodecRegister
|
|
*****************************************************************************
|
|
* Writes to a AC97 register. This can only be done at passive level because
|
|
* the AcquireCodecSemiphore call could fail!
|
|
*/
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::WriteCodecRegister
|
|
(
|
|
_In_range_(0, AC97REG_INVALID) AC97Register reg,
|
|
_In_ WORD wData,
|
|
_In_ WORD wMask
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
ASSERT (reg < AC97REG_INVALID); // audio can only be in the primary codec
|
|
|
|
WORD TempData = 0;
|
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::WriteCodecRegister]"));
|
|
|
|
//
|
|
// No mask? Could happen when you try to prg. left channel of a
|
|
// mono volume.
|
|
//
|
|
if (!wMask)
|
|
return STATUS_SUCCESS;
|
|
|
|
//
|
|
// Check to see if we are only writing specific bits. If so, we want
|
|
// to leave some bits in the register alone.
|
|
//
|
|
if (wMask != 0xffff)
|
|
{
|
|
//
|
|
// Read the current register contents.
|
|
//
|
|
ntStatus = ReadCodecRegister (reg, &TempData);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("WriteCodecRegiser read for mask failed"));
|
|
return ntStatus;
|
|
}
|
|
|
|
//
|
|
// Do the masking.
|
|
//
|
|
TempData &= ~wMask;
|
|
TempData |= (wMask & wData);
|
|
}
|
|
else
|
|
{
|
|
TempData = wData;
|
|
}
|
|
|
|
|
|
//
|
|
// Grab the codec access semiphore.
|
|
//
|
|
ntStatus = AcquireCodecSemiphore ();
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("WriteCodecRegister failed for register %s",
|
|
reg <= AC97REG_RESERVED2 ? RegStrings[reg] :
|
|
reg == AC97REG_VENDOR_ID1 ? "REG_VENDOR_ID1" :
|
|
reg == AC97REG_VENDOR_ID2 ? "REG_VENDOR_ID2" : "REG_INVALID"));
|
|
return ntStatus;
|
|
}
|
|
|
|
//
|
|
// Write the data.
|
|
//
|
|
WRITE_PORT_USHORT (m_pCodecBase + reg, TempData);
|
|
|
|
//
|
|
// Update cache.
|
|
//
|
|
_Analysis_assume_(reg < AC97REG_INVALID);
|
|
m_stAC97Registers[reg].wCache = TempData;
|
|
|
|
DOUT (DBG_REGS, ("AC97WRITE: %s -> 0x%04x",
|
|
reg <= AC97REG_RESERVED2 ? RegStrings[reg] :
|
|
reg == AC97REG_VENDOR_ID1 ? "REG_VENDOR_ID1" :
|
|
reg == AC97REG_VENDOR_ID2 ? "REG_VENDOR_ID2" :
|
|
"REG_INVALID", TempData));
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::PrimaryCodecReady
|
|
*****************************************************************************
|
|
* Checks whether the primary codec is present and ready. This may take
|
|
* awhile if we are bringing it up from a cold reset so give it a second
|
|
* before giving up.
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::PrimaryCodecReady (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::PrimaryCodecReady]"));
|
|
|
|
|
|
//
|
|
// Enable the AC link and raise the reset line.
|
|
//
|
|
DWORD dwRegValue = ReadBMControlRegister32 (GLOB_CNT);
|
|
|
|
// If someone enabled GPI Interrupt Enable, then he hopefully handles that
|
|
// too.
|
|
dwRegValue = (dwRegValue | GLOB_CNT_COLD) & ~(GLOB_CNT_ACLOFF | GLOB_CNT_PRIE);
|
|
WriteBMControlRegister (GLOB_CNT, dwRegValue);
|
|
|
|
//
|
|
// Wait for the Codec to be ready.
|
|
//
|
|
ULONG WaitCycles = 200;
|
|
LARGE_INTEGER WaitTime;
|
|
|
|
WaitTime.QuadPart = (-50000); // wait 5000us (5ms) relative
|
|
|
|
do
|
|
{
|
|
if (READ_PORT_ULONG ((PULONG)(m_pBusMasterBase + GLOB_STA)) &
|
|
GLOB_STA_PCR)
|
|
{
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
KeDelayExecutionThread (KernelMode, FALSE, &WaitTime);
|
|
} while (WaitCycles--);
|
|
|
|
DOUT (DBG_ERROR, ("PrimaryCodecReady timed out!"));
|
|
return STATUS_IO_TIMEOUT;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::PowerUpCodec
|
|
*****************************************************************************
|
|
* Sets the Codec to the highest power state and waits until the Codec reports
|
|
* that the power state is reached.
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::PowerUpCodec (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
WORD wCodecReg;
|
|
NTSTATUS ntStatus;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::PowerUpCodec]"));
|
|
|
|
//
|
|
// Power up the Codec.
|
|
//
|
|
WriteCodecRegister (AC97REG_POWERDOWN, 0x00, 0xFFFF);
|
|
|
|
//
|
|
// Wait for the Codec to be powered up.
|
|
//
|
|
ULONG WaitCycles = 200;
|
|
LARGE_INTEGER WaitTime;
|
|
|
|
WaitTime.QuadPart = (-50000); // wait 5000us (5ms) relative
|
|
|
|
do
|
|
{
|
|
//
|
|
// Read the power management register.
|
|
//
|
|
ntStatus = ReadCodecRegister (AC97REG_POWERDOWN, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
wCodecReg = 0; // Will cause an error.
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Check the power state. Should be ready.
|
|
//
|
|
if ((wCodecReg & 0x0f) == 0x0f)
|
|
break;
|
|
|
|
//
|
|
// Let's wait a little, 5ms and then try again.
|
|
//
|
|
KeDelayExecutionThread (KernelMode, FALSE, &WaitTime);
|
|
} while (WaitCycles--);
|
|
|
|
// Check if we timed out.
|
|
if ((wCodecReg & 0x0f) != 0x0f)
|
|
{
|
|
DOUT (DBG_ERROR, ("PowerUpCodec timed out. CoDec not powered up."));
|
|
ntStatus = STATUS_DEVICE_NOT_READY;
|
|
}
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ProgramSampleRate
|
|
*****************************************************************************
|
|
* Programs the sample rate. If the rate cannot be programmed, the routine
|
|
* restores the register and returns STATUS_UNSUCCESSFUL.
|
|
* We don't handle double rate sample rates here, because the Intel AC97 con-
|
|
* troller cannot serve CoDecs with double rate or surround sound. If you want
|
|
* to modify this driver for another AC97 controller, then you might want to
|
|
* change this function too.
|
|
*/
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::ProgramSampleRate
|
|
(
|
|
IN AC97Register Register,
|
|
IN DWORD dwSampleRate
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
WORD wOldRateReg, wCodecReg;
|
|
NTSTATUS ntStatus;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ProgramSampleRate]"));
|
|
|
|
//
|
|
// Check if we support variable sample rate.
|
|
//
|
|
switch(Register)
|
|
{
|
|
case AC97REG_MIC_SAMPLERATE:
|
|
//
|
|
// Variable sample rate supported?
|
|
//
|
|
if (GetNodeConfig (NODEC_MIC_VARIABLERATE_SUPPORTED))
|
|
{
|
|
// Range supported?
|
|
if (dwSampleRate > 48000ul)
|
|
{
|
|
// Not possible.
|
|
DOUT (DBG_VSR, ("Samplerate %d not supported", dwSampleRate));
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Only 48000KHz possible.
|
|
if (dwSampleRate != 48000ul)
|
|
{
|
|
DOUT (DBG_VSR, ("Samplerate %d not supported", dwSampleRate));
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
break;
|
|
|
|
case AC97REG_FRONT_SAMPLERATE:
|
|
case AC97REG_SURROUND_SAMPLERATE:
|
|
case AC97REG_LFE_SAMPLERATE:
|
|
case AC97REG_RECORD_SAMPLERATE:
|
|
//
|
|
// Variable sample rate supported?
|
|
//
|
|
if (GetNodeConfig (NODEC_PCM_VARIABLERATE_SUPPORTED))
|
|
{
|
|
//
|
|
// Check range supported
|
|
//
|
|
if (dwSampleRate > 48000ul)
|
|
{
|
|
DOUT (DBG_VSR, ("Samplerate %d not supported", dwSampleRate));
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Only 48KHz possible.
|
|
if (dwSampleRate != 48000ul)
|
|
{
|
|
DOUT (DBG_VSR, ("Samplerate %d not supported", dwSampleRate));
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
DOUT (DBG_ERROR, ("Invalid sample rate register!"));
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
|
|
//
|
|
// Save the old sample rate register.
|
|
//
|
|
ntStatus = ReadCodecRegister (Register, &wOldRateReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// program the rate.
|
|
//
|
|
ntStatus = WriteCodecRegister (Register, (WORD)dwSampleRate, 0xFFFF);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("Cannot program sample rate."));
|
|
return ntStatus;
|
|
}
|
|
|
|
//
|
|
// Read it back.
|
|
//
|
|
ntStatus = ReadCodecRegister (Register, &wCodecReg);
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("Cannot read sample rate."));
|
|
return ntStatus;
|
|
}
|
|
|
|
//
|
|
// Validate.
|
|
//
|
|
if (wCodecReg != dwSampleRate)
|
|
{
|
|
//
|
|
// restore sample rate and ctrl register.
|
|
//
|
|
WriteCodecRegister (Register, wOldRateReg, 0xFFFF);
|
|
|
|
DOUT (DBG_VSR, ("Samplerate %d not supported", dwSampleRate));
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
DOUT (DBG_VSR, ("Samplerate changed to %d.", dwSampleRate));
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::PowerChangeState
|
|
*****************************************************************************
|
|
* Change power state for the device. We handle the codec, PowerChangeNotify
|
|
* in the wave miniport handles the DMA registers.
|
|
*/
|
|
STDMETHODIMP_(void) CAC97AdapterCommon::PowerChangeState
|
|
(
|
|
_In_ POWER_STATE NewState
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::PowerChangeNotify]"));
|
|
|
|
//
|
|
// Check to see if this is the current power state.
|
|
//
|
|
if (NewState.DeviceState == m_PowerState)
|
|
{
|
|
DOUT (DBG_POWER, ("New device state equals old state."));
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Check the new device state.
|
|
//
|
|
if ((NewState.DeviceState < PowerDeviceD0) ||
|
|
(NewState.DeviceState > PowerDeviceD3))
|
|
{
|
|
DOUT (DBG_ERROR, ("Unknown device state: D%d.",
|
|
(ULONG)NewState.DeviceState - (ULONG)PowerDeviceD0));
|
|
return;
|
|
}
|
|
|
|
DOUT (DBG_POWER, ("Changing state to D%d.", (ULONG)NewState.DeviceState -
|
|
(ULONG)PowerDeviceD0));
|
|
|
|
//
|
|
// Switch on new state.
|
|
//
|
|
switch (NewState.DeviceState)
|
|
{
|
|
case PowerDeviceD0:
|
|
//
|
|
// If we are coming from D2 or D3 we have to restore the registers cause
|
|
// there might have been a power loss.
|
|
//
|
|
if ((m_PowerState == PowerDeviceD3) || (m_PowerState == PowerDeviceD2))
|
|
{
|
|
//
|
|
// Reset AD3 to indicate that we are now awake.
|
|
// Because the system has only one power irp at a time, we are sure
|
|
// that the modem driver doesn't get called while we are restoring
|
|
// power.
|
|
//
|
|
WriteBMControlRegister (GLOB_STA,
|
|
ReadBMControlRegister32 (GLOB_STA) & ~GLOB_STA_AD3);
|
|
|
|
//
|
|
// Restore codec registers.
|
|
//
|
|
ntStatus = RestoreCodecRegisters ();
|
|
}
|
|
else // We are coming from power state D1
|
|
{
|
|
ntStatus = PowerUpCodec ();
|
|
}
|
|
|
|
// Print error code.
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("PowerChangeState failed to restore the codec."));
|
|
}
|
|
break;
|
|
|
|
case PowerDeviceD1:
|
|
//
|
|
// This sleep state is the lowest latency sleep state with respect
|
|
// to the latency time required to return to D0. If the
|
|
// driver is not being used an inactivity timer in portcls will
|
|
// place the driver in this state after a timeout period
|
|
// controllable via the registry.
|
|
//
|
|
|
|
// Let's power down the DAC/ADC's and analog mixer.
|
|
WriteCodecRegister (AC97REG_POWERDOWN, 0x0700, 0xFFFF);
|
|
break;
|
|
|
|
case PowerDeviceD2:
|
|
case PowerDeviceD3:
|
|
//
|
|
// This is a full hibernation state and is the longest latency sleep
|
|
// state. In this modes the power could be removed or reduced that
|
|
// much that the AC97 controller looses information, so we save
|
|
// whatever we have to save.
|
|
//
|
|
|
|
//
|
|
// Powerdown ADC, DAC, Mixer, Vref, HP amp, and Exernal Amp but not
|
|
// AC-link and Clk
|
|
//
|
|
WriteCodecRegister (AC97REG_POWERDOWN, 0xCF00, 0xFFFF);
|
|
|
|
//
|
|
// Only in D3 mode we set the AD3 bit and evtl. shut off the AC link.
|
|
//
|
|
if (NewState.DeviceState == PowerDeviceD3)
|
|
{
|
|
//
|
|
// Set the AD3 bit.
|
|
//
|
|
ULONG ulReg = ReadBMControlRegister32 (GLOB_STA);
|
|
WriteBMControlRegister (GLOB_STA, ulReg | GLOB_STA_AD3);
|
|
|
|
//
|
|
// We check if the modem is sleeping. If it is, we can shut off the
|
|
// AC link also. We shut off the AC link also if the modem is not
|
|
// there.
|
|
//
|
|
if ((ulReg & GLOB_STA_MD3) || !(ulReg & GLOB_STA_SCR))
|
|
{
|
|
// Set Codec to super sleep
|
|
WriteCodecRegister (AC97REG_POWERDOWN, 0xFF00, 0xFFFF);
|
|
|
|
// Disable the AC-link signals
|
|
ulReg = ReadBMControlRegister32 (GLOB_CNT);
|
|
WriteBMControlRegister (GLOB_CNT, (ulReg | GLOB_CNT_ACLOFF) & ~GLOB_CNT_COLD);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Save the new state. This local value is used to determine when to
|
|
// cache property accesses and when to permit the driver from accessing
|
|
// the hardware.
|
|
//
|
|
m_PowerState = NewState.DeviceState;
|
|
DOUT (DBG_POWER, ("Entering D%d", (ULONG)m_PowerState -
|
|
(ULONG)PowerDeviceD0));
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::QueryPowerChangeState
|
|
*****************************************************************************
|
|
* Query to see if the device can change to this power state
|
|
*/
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::QueryPowerChangeState
|
|
(
|
|
_In_ POWER_STATE NewState
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::QueryPowerChangeState]"));
|
|
|
|
// Check here to see of a legitimate state is being requested
|
|
// based on the device state and fail the call if the device/driver
|
|
// cannot support the change requested. Otherwise, return STATUS_SUCCESS.
|
|
// Note: A QueryPowerChangeState() call is not guaranteed to always preceed
|
|
// a PowerChangeState() call.
|
|
|
|
// check the new state being requested
|
|
switch (NewState.DeviceState)
|
|
{
|
|
case PowerDeviceD0:
|
|
case PowerDeviceD1:
|
|
case PowerDeviceD2:
|
|
case PowerDeviceD3:
|
|
return STATUS_SUCCESS;
|
|
|
|
default:
|
|
DOUT (DBG_ERROR, ("Unknown device state: D%d.",
|
|
(ULONG)NewState.DeviceState - (ULONG)PowerDeviceD0));
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::QueryDeviceCapabilities
|
|
*****************************************************************************
|
|
* Called at startup to get the caps for the device. This structure provides
|
|
* the system with the mappings between system power state and device power
|
|
* state. This typically will not need modification by the driver.
|
|
* If the driver modifies these mappings then the driver is not allowed to
|
|
* change the mapping to a weaker power state (e.g. from S1->D3 to S1->D1).
|
|
*
|
|
*/
|
|
_Use_decl_annotations_
|
|
STDMETHODIMP_(NTSTATUS) CAC97AdapterCommon::QueryDeviceCapabilities
|
|
(
|
|
PDEVICE_CAPABILITIES PowerDeviceCaps
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::QueryDeviceCapabilities]"));
|
|
|
|
UNREFERENCED_PARAMETER(PowerDeviceCaps);
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::RestoreAC97Registers
|
|
*****************************************************************************
|
|
* Preset the AC97 registers with default values. The routine first checks if
|
|
* There are registry entries for the default values. If not, we have hard
|
|
* coded values too ;)
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::SetAC97Default (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
PREGISTRYKEY DriverKey;
|
|
PREGISTRYKEY SettingsKey;
|
|
UNICODE_STRING sKeyName;
|
|
ULONG ulDisposition;
|
|
ULONG ulResultLength;
|
|
PVOID KeyInfo = NULL;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::SetAC97Default]"));
|
|
|
|
// open the driver registry key
|
|
NTSTATUS ntStatus = PcNewRegistryKey (&DriverKey, // IRegistryKey
|
|
NULL, // OuterUnknown
|
|
DriverRegistryKey, // Registry key type
|
|
KEY_READ, // Access flags
|
|
m_pDeviceObject, // Device object
|
|
NULL, // Subdevice
|
|
NULL, // ObjectAttributes
|
|
0, // Create options
|
|
NULL); // Disposition
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// make a unicode string for the subkey name
|
|
RtlInitUnicodeString (&sKeyName, L"Settings");
|
|
|
|
// open the settings subkey
|
|
ntStatus = DriverKey->NewSubKey (&SettingsKey, // Subkey
|
|
NULL, // OuterUnknown
|
|
KEY_READ, // Access flags
|
|
&sKeyName, // Subkey name
|
|
REG_OPTION_NON_VOLATILE, // Create options
|
|
&ulDisposition);
|
|
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// allocate data to hold key info
|
|
KeyInfo = ExAllocatePoolWithTag (PagedPool,
|
|
sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
sizeof(WORD), PoolTag);
|
|
if (NULL != KeyInfo)
|
|
{
|
|
// loop through all mixer settings
|
|
for (AC97Register i = AC97REG_RESET; i <= AC97REG_RESERVED2;
|
|
i = (AC97Register)(i + 1))
|
|
{
|
|
if (m_stAC97Registers[i].wFlags & SHREG_INIT)
|
|
{
|
|
// init key name
|
|
RtlInitUnicodeString (&sKeyName,
|
|
m_stAC97Registers[i].sRegistryName);
|
|
|
|
// query the value key
|
|
ntStatus = SettingsKey->QueryValueKey (&sKeyName,
|
|
KeyValuePartialInformation,
|
|
KeyInfo,
|
|
sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
sizeof(WORD),
|
|
&ulResultLength);
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
PKEY_VALUE_PARTIAL_INFORMATION PartialInfo =
|
|
(PKEY_VALUE_PARTIAL_INFORMATION)KeyInfo;
|
|
|
|
if (PartialInfo->DataLength == sizeof(WORD))
|
|
{
|
|
// set mixer register to registry value
|
|
WriteCodecRegister
|
|
(i, *(PWORD)PartialInfo->Data, 0xFFFF);
|
|
}
|
|
else // write the hard coded default
|
|
{
|
|
// if key access failed, set to default
|
|
WriteCodecRegister
|
|
(i, m_stAC97Registers[i].wWantedDefault, 0xFFFF);
|
|
}
|
|
}
|
|
else // write the hard coded default
|
|
{
|
|
// if key access failed, set to default
|
|
WriteCodecRegister
|
|
(i, m_stAC97Registers[i].wWantedDefault, 0xFFFF);
|
|
}
|
|
}
|
|
}
|
|
|
|
// we want to return status success even if the last QueryValueKey
|
|
// failed.
|
|
ntStatus = STATUS_SUCCESS;
|
|
|
|
// free the key info
|
|
ExFreePoolWithTag (KeyInfo,PoolTag);
|
|
}
|
|
else
|
|
{
|
|
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
// release the settings key
|
|
SettingsKey->Release ();
|
|
}
|
|
|
|
// release the driver key
|
|
DriverKey->Release ();
|
|
}
|
|
|
|
|
|
// in case we did not query the registry (cause of lack of resources)
|
|
// restore default values and return insufficient resources.
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
// copy hard coded default settings
|
|
for (AC97Register i = AC97REG_RESET; i < AC97REG_RESERVED2;
|
|
i = (AC97Register)(i + 1))
|
|
{
|
|
if (m_stAC97Registers[i].wFlags & SHREG_INIT)
|
|
{
|
|
WriteCodecRegister (i, m_stAC97Registers[i].wWantedDefault, 0xFFFF);
|
|
}
|
|
}
|
|
}
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::DisableAC97Pin
|
|
*****************************************************************************
|
|
* Returns TRUE when the HW vendor wants to disable the pin. A disabled pin is
|
|
* not shown to the user (means it is not included in the topology). The
|
|
* reason for doing this could be that some of the input lines like Aux or
|
|
* Video are not available to the user (to plug in something) but the codec
|
|
* can handle those lines.
|
|
*/
|
|
BOOL CAC97AdapterCommon::DisableAC97Pin
|
|
(
|
|
IN TopoPinConfig pin
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
PREGISTRYKEY DriverKey;
|
|
PREGISTRYKEY SettingsKey;
|
|
UNICODE_STRING sKeyName;
|
|
ULONG ulDisposition;
|
|
ULONG ulResultLength;
|
|
PVOID KeyInfo = NULL;
|
|
BOOL bDisable = FALSE;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::DisableAC97Pin]"));
|
|
|
|
// open the driver registry key
|
|
NTSTATUS ntStatus = PcNewRegistryKey (&DriverKey, // IRegistryKey
|
|
NULL, // OuterUnknown
|
|
DriverRegistryKey, // Registry key type
|
|
KEY_READ, // Access flags
|
|
m_pDeviceObject, // Device object
|
|
NULL, // Subdevice
|
|
NULL, // ObjectAttributes
|
|
0, // Create options
|
|
NULL); // Disposition
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// make a unicode string for the subkey name
|
|
RtlInitUnicodeString (&sKeyName, L"Settings");
|
|
|
|
// open the settings subkey
|
|
ntStatus = DriverKey->NewSubKey (&SettingsKey, // Subkey
|
|
NULL, // OuterUnknown
|
|
KEY_READ, // Access flags
|
|
&sKeyName, // Subkey name
|
|
REG_OPTION_NON_VOLATILE, // Create options
|
|
&ulDisposition);
|
|
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// allocate data to hold key info
|
|
KeyInfo = ExAllocatePoolWithTag (PagedPool,
|
|
sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
sizeof(BYTE), PoolTag);
|
|
if (NULL != KeyInfo)
|
|
{
|
|
// init key name
|
|
RtlInitUnicodeString (&sKeyName, m_stHardwareConfig.
|
|
Pins[pin].sRegistryName);
|
|
|
|
// query the value key
|
|
ntStatus = SettingsKey->QueryValueKey (&sKeyName,
|
|
KeyValuePartialInformation,
|
|
KeyInfo,
|
|
sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
sizeof(BYTE),
|
|
&ulResultLength );
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
PKEY_VALUE_PARTIAL_INFORMATION PartialInfo =
|
|
(PKEY_VALUE_PARTIAL_INFORMATION)KeyInfo;
|
|
|
|
if (PartialInfo->DataLength == sizeof(BYTE))
|
|
{
|
|
// store the value
|
|
if (*(PBYTE)PartialInfo->Data)
|
|
bDisable = TRUE;
|
|
else
|
|
bDisable = FALSE;
|
|
}
|
|
}
|
|
|
|
// free the key info
|
|
ExFreePoolWithTag (KeyInfo,PoolTag);
|
|
}
|
|
|
|
// release the settings key
|
|
SettingsKey->Release ();
|
|
}
|
|
|
|
// release the driver key
|
|
DriverKey->Release ();
|
|
}
|
|
|
|
// if one of the stuff above fails we return the default, which is FALSE.
|
|
return bDisable;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::RestoreCodecRegisters
|
|
*****************************************************************************
|
|
* write back cached mixer values to codec registers
|
|
*/
|
|
NTSTATUS CAC97AdapterCommon::RestoreCodecRegisters (void)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::RestoreCodecRegisters]"));
|
|
|
|
//
|
|
// Initialize the AC97 codec.
|
|
//
|
|
NTSTATUS ntStatus = InitAC97 ();
|
|
if (!NT_SUCCESS (ntStatus))
|
|
return ntStatus;
|
|
|
|
//
|
|
// Restore all codec registers. Failure is not critical.
|
|
//
|
|
for (AC97Register i = AC97REG_MASTER_VOLUME; i < AC97REG_RESERVED2;
|
|
i = (AC97Register)(i + 1))
|
|
{
|
|
WriteCodecRegister (i, m_stAC97Registers[i].wCache, 0xFFFF);
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ReadChannelConfigDefault
|
|
*****************************************************************************
|
|
* This function reads the default channel config from the registry. The
|
|
* registry entry "ChannelConfig" is set every every time we get a
|
|
* KSPROPERTY_AUDIO_CHANNEL_CONFIG for the DAC node.
|
|
* In case the key doesn't exist we assume a channel config of stereo speakers,
|
|
* cause that is the default of DSOUND.
|
|
*/
|
|
STDMETHODIMP_(void) CAC97AdapterCommon::ReadChannelConfigDefault
|
|
(
|
|
PDWORD pdwChannelConfig,
|
|
PWORD pwChannels
|
|
)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
PREGISTRYKEY DriverKey;
|
|
PREGISTRYKEY SettingsKey;
|
|
UNICODE_STRING sKeyName;
|
|
ULONG ulDisposition;
|
|
ULONG ulResultLength;
|
|
PVOID KeyInfo = NULL;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ReadChannelConfigDefault]"));
|
|
|
|
// This is the default: 2 speakers, stereo.
|
|
*pdwChannelConfig = KSAUDIO_SPEAKER_STEREO;
|
|
*pwChannels = 2;
|
|
|
|
// open the driver registry key
|
|
NTSTATUS ntStatus = PcNewRegistryKey (&DriverKey, // IRegistryKey
|
|
NULL, // OuterUnknown
|
|
DriverRegistryKey, // Registry key type
|
|
KEY_READ, // Access flags
|
|
m_pDeviceObject, // Device object
|
|
NULL, // Subdevice
|
|
NULL, // ObjectAttributes
|
|
0, // Create options
|
|
NULL); // Disposition
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// make a unicode string for the subkey name
|
|
RtlInitUnicodeString (&sKeyName, L"Settings");
|
|
|
|
// open the settings subkey
|
|
ntStatus = DriverKey->NewSubKey (&SettingsKey, // Subkey
|
|
NULL, // OuterUnknown
|
|
KEY_READ, // Access flags
|
|
&sKeyName, // Subkey name
|
|
REG_OPTION_NON_VOLATILE, // Create options
|
|
&ulDisposition);
|
|
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// allocate data to hold key info
|
|
KeyInfo = ExAllocatePoolWithTag (PagedPool,
|
|
sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
sizeof(DWORD), PoolTag);
|
|
if (NULL != KeyInfo)
|
|
{
|
|
// init key name
|
|
RtlInitUnicodeString (&sKeyName, L"ChannelConfig");
|
|
|
|
// query the value key
|
|
ntStatus = SettingsKey->QueryValueKey (&sKeyName,
|
|
KeyValuePartialInformation,
|
|
KeyInfo,
|
|
sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
sizeof(DWORD),
|
|
&ulResultLength );
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
PKEY_VALUE_PARTIAL_INFORMATION PartialInfo =
|
|
(PKEY_VALUE_PARTIAL_INFORMATION)KeyInfo;
|
|
|
|
if (PartialInfo->DataLength == sizeof(DWORD))
|
|
{
|
|
switch (*(PLONG)PartialInfo->Data)
|
|
{
|
|
case KSAUDIO_SPEAKER_QUAD:
|
|
case KSAUDIO_SPEAKER_SURROUND:
|
|
if (GetPinConfig (PINC_SURROUND_PRESENT))
|
|
{
|
|
*pdwChannelConfig = *(PDWORD)PartialInfo->Data;
|
|
*pwChannels = 4;
|
|
}
|
|
break;
|
|
|
|
case KSAUDIO_SPEAKER_5POINT1:
|
|
if (GetPinConfig (PINC_SURROUND_PRESENT) &&
|
|
GetPinConfig (PINC_CENTER_LFE_PRESENT))
|
|
{
|
|
*pdwChannelConfig = *(PDWORD)PartialInfo->Data;
|
|
*pwChannels = 6;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// free the key info
|
|
ExFreePoolWithTag (KeyInfo,PoolTag);
|
|
}
|
|
|
|
// release the settings key
|
|
SettingsKey->Release ();
|
|
}
|
|
|
|
// release the driver key
|
|
DriverKey->Release ();
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::WriteChannelConfigDefault
|
|
*****************************************************************************
|
|
* This function writes the default channel config to the registry. The
|
|
* registry entry "ChannelConfig" is set every every time we get a
|
|
* KSPROPERTY_AUDIO_CHANNEL_CONFIG for the DAC node.
|
|
*/
|
|
STDMETHODIMP_(void) CAC97AdapterCommon::WriteChannelConfigDefault (DWORD dwChannelConfig)
|
|
{
|
|
PAGED_CODE ();
|
|
|
|
PREGISTRYKEY DriverKey;
|
|
PREGISTRYKEY SettingsKey;
|
|
UNICODE_STRING sKeyName;
|
|
ULONG ulDisposition;
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::WriteChannelConfigDefault]"));
|
|
|
|
// open the driver registry key
|
|
NTSTATUS ntStatus = PcNewRegistryKey (&DriverKey, // IRegistryKey
|
|
NULL, // OuterUnknown
|
|
DriverRegistryKey, // Registry key type
|
|
KEY_WRITE, // Access flags
|
|
m_pDeviceObject, // Device object
|
|
NULL, // Subdevice
|
|
NULL, // ObjectAttributes
|
|
0, // Create options
|
|
NULL); // Disposition
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// make a unicode string for the subkey name
|
|
RtlInitUnicodeString (&sKeyName, L"Settings");
|
|
|
|
// open the settings subkey
|
|
ntStatus = DriverKey->NewSubKey (&SettingsKey, // Subkey
|
|
NULL, // OuterUnknown
|
|
KEY_WRITE, // Access flags
|
|
&sKeyName, // Subkey name
|
|
REG_OPTION_NON_VOLATILE, // Create options
|
|
&ulDisposition);
|
|
|
|
if (NT_SUCCESS (ntStatus))
|
|
{
|
|
// init key name
|
|
RtlInitUnicodeString (&sKeyName, L"ChannelConfig");
|
|
|
|
// query the value key
|
|
ntStatus = SettingsKey->SetValueKey (&sKeyName,
|
|
REG_DWORD,
|
|
&dwChannelConfig,
|
|
sizeof (DWORD));
|
|
if (!NT_SUCCESS (ntStatus))
|
|
{
|
|
DOUT (DBG_ERROR, ("Could not write the ChannelConfig to registry."));
|
|
}
|
|
|
|
// release the settings key
|
|
SettingsKey->Release ();
|
|
}
|
|
|
|
// release the driver key
|
|
DriverKey->Release ();
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Non paged code begins here
|
|
*****************************************************************************
|
|
*/
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma code_seg()
|
|
#endif
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::WriteBMControlRegister
|
|
*****************************************************************************
|
|
* Writes a byte (UCHAR) to BusMaster Control register.
|
|
*/
|
|
STDMETHODIMP_(void) CAC97AdapterCommon::WriteBMControlRegister
|
|
(
|
|
IN ULONG ulOffset,
|
|
IN UCHAR ucValue
|
|
)
|
|
{
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::WriteBMControlRegister] (UCHAR)"));
|
|
|
|
WRITE_PORT_UCHAR ((PUCHAR)(m_pBusMasterBase + ulOffset), ucValue);
|
|
|
|
DOUT (DBG_REGS, ("WriteBMControlRegister wrote 0x%2x to 0x%4p.",
|
|
ucValue, m_pBusMasterBase + ulOffset));
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::WriteBMControlRegister
|
|
*****************************************************************************
|
|
* Writes a word (USHORT) to BusMaster Control register.
|
|
*/
|
|
STDMETHODIMP_(void) CAC97AdapterCommon::WriteBMControlRegister
|
|
(
|
|
IN ULONG ulOffset,
|
|
IN USHORT usValue
|
|
)
|
|
{
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::WriteBMControlRegister (USHORT)]"));
|
|
|
|
WRITE_PORT_USHORT ((PUSHORT)(m_pBusMasterBase + ulOffset), usValue);
|
|
|
|
DOUT (DBG_REGS, ("WriteBMControlRegister wrote 0x%4x to 0x%4p",
|
|
usValue, m_pBusMasterBase + ulOffset));
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::WriteBMControlRegister
|
|
*****************************************************************************
|
|
* Writes a DWORD (ULONG) to BusMaster Control register.
|
|
*/
|
|
STDMETHODIMP_(void) CAC97AdapterCommon::WriteBMControlRegister
|
|
(
|
|
IN ULONG ulOffset,
|
|
IN ULONG ulValue
|
|
)
|
|
{
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::WriteBMControlRegister (ULONG)]"));
|
|
|
|
WRITE_PORT_ULONG ((PULONG)(m_pBusMasterBase + ulOffset), ulValue);
|
|
|
|
DOUT (DBG_REGS, ("WriteBMControlRegister wrote 0x%8x to 0x%4p.",
|
|
ulValue, m_pBusMasterBase + ulOffset));
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ReadBMControlRegister8
|
|
*****************************************************************************
|
|
* Read a byte (UCHAR) from BusMaster Control register.
|
|
*/
|
|
STDMETHODIMP_(UCHAR) CAC97AdapterCommon::ReadBMControlRegister8
|
|
(
|
|
IN ULONG ulOffset
|
|
)
|
|
{
|
|
UCHAR ucValue = UCHAR(-1);
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ReadBMControlRegister8]"));
|
|
|
|
ucValue = READ_PORT_UCHAR ((PUCHAR)(m_pBusMasterBase + ulOffset));
|
|
|
|
DOUT (DBG_REGS, ("ReadBMControlRegister read 0x%2x from 0x%4p.", ucValue,
|
|
m_pBusMasterBase + ulOffset));
|
|
|
|
return ucValue;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ReadBMControlRegister16
|
|
*****************************************************************************
|
|
* Read a word (USHORT) from BusMaster Control register.
|
|
*/
|
|
STDMETHODIMP_(USHORT) CAC97AdapterCommon::ReadBMControlRegister16
|
|
(
|
|
IN ULONG ulOffset
|
|
)
|
|
{
|
|
USHORT usValue = USHORT(-1);
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ReadBMControlRegister16]"));
|
|
|
|
usValue = READ_PORT_USHORT ((PUSHORT)(m_pBusMasterBase + ulOffset));
|
|
|
|
DOUT (DBG_REGS, ("ReadBMControlRegister read 0x%4x = 0x%4p", usValue,
|
|
m_pBusMasterBase + ulOffset));
|
|
|
|
return usValue;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* CAC97AdapterCommon::ReadBMControlRegister32
|
|
*****************************************************************************
|
|
* Read a dword (ULONG) from BusMaster Control register.
|
|
*/
|
|
STDMETHODIMP_(ULONG) CAC97AdapterCommon::ReadBMControlRegister32
|
|
(
|
|
IN ULONG ulOffset
|
|
)
|
|
{
|
|
ULONG ulValue = ULONG(-1);
|
|
|
|
DOUT (DBG_PRINT, ("[CAC97AdapterCommon::ReadBMControlRegister32]"));
|
|
|
|
ulValue = READ_PORT_ULONG ((PULONG)(m_pBusMasterBase + ulOffset));
|
|
|
|
DOUT (DBG_REGS, ("ReadBMControlRegister read 0x%8x = 0x%4p", ulValue,
|
|
m_pBusMasterBase + ulOffset));
|
|
|
|
return ulValue;
|
|
}
|
|
|