mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2025-01-05 18:28:46 +01:00
7c0aa811ec
Signed-off-by: Sergey Isakov <isakov-sl@bk.ru>
868 lines
33 KiB
C
868 lines
33 KiB
C
/*
|
|
* File: HdaController.c
|
|
*
|
|
* Copyright (c) 2018 John Davis
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
* copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*/
|
|
|
|
#include "HdaController.h"
|
|
#include "HdaControllerComponentName.h"
|
|
|
|
VOID
|
|
EFIAPI
|
|
HdaControllerStreamPollTimerHandler(
|
|
IN EFI_EVENT Event,
|
|
IN VOID *Context)
|
|
{
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
HDA_STREAM *HdaStream = (HDA_STREAM*)Context;
|
|
EFI_PCI_IO_PROTOCOL *PciIo = HdaStream->HdaControllerDev->PciIo;
|
|
UINT8 HdaStreamSts = 0;
|
|
UINT32 HdaStreamDmaPos;
|
|
UINTN HdaSourceLength;
|
|
UINTN HdaCurrentBlock;
|
|
UINTN HdaNextBlock;
|
|
|
|
// Get stream status.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthFifoUint8, PCI_HDA_BAR, HDA_REG_SDNSTS(HdaStream->Index), 1, &HdaStreamSts);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// If there was a FIFO error or DESC error, halt.
|
|
ASSERT ((HdaStreamSts & (HDA_REG_SDNSTS_FIFOE | HDA_REG_SDNSTS_DESE)) == 0);
|
|
|
|
// Has the completion bit been set?
|
|
if (HdaStreamSts & HDA_REG_SDNSTS_BCIS) {
|
|
// Are we done playing the stream? If so we can stop now.
|
|
if (HdaStream->BufferSourceDone) {
|
|
// Stop stream.
|
|
Status = HdaControllerSetStream(HdaStream, FALSE);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// Stop timer.
|
|
Status = gBS->SetTimer(HdaStream->PollTimer, TimerCancel, 0);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// Trigger callback.
|
|
if (HdaStream->Callback)
|
|
HdaStream->Callback(HdaStream->Output ? EfiHdaIoTypeOutput : EfiHdaIoTypeInput,
|
|
HdaStream->CallbackContext1, HdaStream->CallbackContext2, HdaStream->CallbackContext3);
|
|
goto CLEAR_BIT;
|
|
}
|
|
|
|
// Get stream DMA position.
|
|
HdaStreamDmaPos = HdaStream->HdaControllerDev->DmaPositions[HdaStream->Index].Position;
|
|
HdaCurrentBlock = HdaStreamDmaPos / HDA_BDL_BLOCKSIZE;
|
|
HdaNextBlock = HdaCurrentBlock + 1;
|
|
HdaNextBlock %= HDA_BDL_ENTRY_COUNT;
|
|
|
|
// Have we reached the end of the source buffer? If so the stream will stop on the next block.
|
|
if (HdaStream->BufferSourcePosition >= HdaStream->BufferSourceLength) {
|
|
// Zero out next block.
|
|
gBS->SetMem(HdaStream->BufferData + (HdaNextBlock * HDA_BDL_BLOCKSIZE), HDA_BDL_BLOCKSIZE, 0);
|
|
|
|
// Set flag to stop stream on the next block.
|
|
HdaStream->BufferSourceDone = TRUE;
|
|
// DEBUG((DEBUG_INFO, "Block %u of %u is the last! (current position 0x%X, buffer 0x%X)\n",
|
|
// HdaStreamDmaPos / HDA_BDL_BLOCKSIZE, HDA_BDL_ENTRY_COUNT, HdaStreamDmaPos, HdaStream->BufferSourcePosition));
|
|
goto CLEAR_BIT;
|
|
}
|
|
|
|
// Determine number of bytes to pull from or push to source data.
|
|
HdaSourceLength = HDA_BDL_BLOCKSIZE;
|
|
if ((HdaStream->BufferSourcePosition + HdaSourceLength) > HdaStream->BufferSourceLength)
|
|
HdaSourceLength = HdaStream->BufferSourceLength - HdaStream->BufferSourcePosition;
|
|
|
|
// Is this an output stream (copy data to)?
|
|
if (HdaStream->Output) {
|
|
// Copy data to DMA buffer.
|
|
if (HdaSourceLength < HDA_BDL_BLOCKSIZE)
|
|
gBS->SetMem(HdaStream->BufferData + (HdaNextBlock * HDA_BDL_BLOCKSIZE), HDA_BDL_BLOCKSIZE, 0);
|
|
gBS->CopyMem(HdaStream->BufferData + (HdaNextBlock * HDA_BDL_BLOCKSIZE), HdaStream->BufferSource + HdaStream->BufferSourcePosition, HdaSourceLength);
|
|
} else { // Input stream (copy data from).
|
|
// Copy data from DMA buffer.
|
|
gBS->CopyMem(HdaStream->BufferSource + HdaStream->BufferSourcePosition, HdaStream->BufferData + (HdaNextBlock * HDA_BDL_BLOCKSIZE), HdaSourceLength);
|
|
}
|
|
|
|
// Increase source position.
|
|
HdaStream->BufferSourcePosition += HdaSourceLength;
|
|
// DEBUG((DEBUG_INFO, "Block %u of %u filled! (current position 0x%X, buffer 0x%X)\n",
|
|
// HdaStreamDmaPos / HDA_BDL_BLOCKSIZE, HDA_BDL_ENTRY_COUNT, HdaStreamDmaPos, HdaStream->BufferSourcePosition));
|
|
|
|
CLEAR_BIT:
|
|
// Reset completion bit.
|
|
HdaStreamSts = HDA_REG_SDNSTS_BCIS;
|
|
Status = PciIo->Mem.Write(PciIo, EfiPciIoWidthUint8, PCI_HDA_BAR, HDA_REG_SDNSTS(HdaStream->Index), 1, &HdaStreamSts);
|
|
ASSERT_EFI_ERROR(Status);
|
|
}
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerInitPciHw(
|
|
IN HDA_CONTROLLER_DEV *HdaControllerDev)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerInitPciHw(): start\n"));
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo = HdaControllerDev->PciIo;
|
|
UINT64 PciSupports = 0;
|
|
UINT8 HdaTcSel;
|
|
UINT16 HdaDevC;
|
|
|
|
// Get original PCI I/O attributes.
|
|
Status = PciIo->Attributes(PciIo, EfiPciIoAttributeOperationGet, 0, &HdaControllerDev->OriginalPciAttributes);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
HdaControllerDev->OriginalPciAttributesSaved = TRUE;
|
|
|
|
// Get currently supported PCI I/O attributes.
|
|
Status = PciIo->Attributes(PciIo, EfiPciIoAttributeOperationSupported, 0, &PciSupports);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Enable the PCI device.
|
|
PciSupports &= EFI_PCI_DEVICE_ENABLE;
|
|
Status = PciIo->Attributes(PciIo, EfiPciIoAttributeOperationEnable, PciSupports, NULL);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Get vendor and device IDs of PCI device.
|
|
Status = PciIo->Pci.Read(PciIo, EfiPciIoWidthUint32, PCI_VENDOR_ID_OFFSET, 1, &HdaControllerDev->VendorId);
|
|
if (EFI_ERROR (Status))
|
|
return Status;
|
|
|
|
// DEBUG((DEBUG_INFO, "HdaControllerInitPciHw(): controller %4X:%4X\n",
|
|
// GET_PCI_VENDOR_ID(HdaControllerDev->VendorId), GET_PCI_DEVICE_ID(HdaControllerDev->VendorId)));
|
|
|
|
// Is this an Intel controller?
|
|
if (GET_PCI_VENDOR_ID(HdaControllerDev->VendorId) == VEN_INTEL_ID) {
|
|
// Set TC0 in TCSEL register.
|
|
Status = PciIo->Pci.Read(PciIo, EfiPciIoWidthUint8, PCI_HDA_TCSEL_OFFSET, 1, &HdaTcSel);
|
|
if (EFI_ERROR (Status))
|
|
return Status;
|
|
HdaTcSel &= PCI_HDA_TCSEL_TC0_MASK;
|
|
Status = PciIo->Pci.Write(PciIo, EfiPciIoWidthUint8, PCI_HDA_TCSEL_OFFSET, 1, &HdaTcSel);
|
|
if (EFI_ERROR (Status))
|
|
return Status;
|
|
}
|
|
|
|
// Get device control PCI register.
|
|
Status = PciIo->Pci.Read(PciIo, EfiPciIoWidthUint16, PCI_HDA_DEVC_OFFSET, 1, &HdaDevC);
|
|
if (EFI_ERROR (Status))
|
|
return Status;
|
|
|
|
// If No Snoop is currently enabled, disable it.
|
|
if (HdaDevC & PCI_HDA_DEVC_NOSNOOPEN) {
|
|
HdaDevC &= ~PCI_HDA_DEVC_NOSNOOPEN;
|
|
Status = PciIo->Pci.Write(PciIo, EfiPciIoWidthUint16, PCI_HDA_DEVC_OFFSET, 1, &HdaDevC);
|
|
if (EFI_ERROR (Status))
|
|
return Status;
|
|
}
|
|
|
|
// Get major/minor version.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint8, PCI_HDA_BAR, HDA_REG_VMAJ, 1, &HdaControllerDev->MajorVersion);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint8, PCI_HDA_BAR, HDA_REG_VMIN, 1, &HdaControllerDev->MinorVersion);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Validate version. If invalid abort.
|
|
// DEBUG((DEBUG_INFO, "HdaControllerInitPciHw(): controller version %u.%u\n",
|
|
// HdaControllerDev->MajorVersion, HdaControllerDev->MinorVersion));
|
|
if (HdaControllerDev->MajorVersion < HDA_VERSION_MIN_MAJOR) {
|
|
Status = EFI_UNSUPPORTED;
|
|
return Status;
|
|
}
|
|
|
|
// Get capabilities.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_GCAP, 1, &HdaControllerDev->Capabilities);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
// DEBUG((DEBUG_INFO, "HdaControllerInitPciHw(): capabilities:\n 64-bit: %s Serial Data Out Signals: %u\n",
|
|
// HdaControllerDev->Capabilities & HDA_REG_GCAP_64OK ? L"Yes" : L"No",
|
|
// HDA_REG_GCAP_NSDO(HdaControllerDev->Capabilities)));
|
|
// DEBUG((DEBUG_INFO, " Bidir streams: %u Input streams: %u Output streams: %u\n",
|
|
// HDA_REG_GCAP_BSS(HdaControllerDev->Capabilities), HDA_REG_GCAP_ISS(HdaControllerDev->Capabilities),
|
|
// HDA_REG_GCAP_OSS(HdaControllerDev->Capabilities)));
|
|
|
|
// Success.
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
VOID
|
|
EFIAPI
|
|
HdaControllerGetName(
|
|
IN HDA_CONTROLLER_DEV *HdaControllerDev)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerGetName(): start\n"));
|
|
|
|
// Try to match controller name.
|
|
HdaControllerDev->Name = NULL;
|
|
UINTN ControllerIndex = 0;
|
|
while (gHdaControllerList[ControllerIndex].Id != 0) {
|
|
// Check ID and revision against array element.
|
|
if (gHdaControllerList[ControllerIndex].Id == HdaControllerDev->VendorId)
|
|
HdaControllerDev->Name = gHdaControllerList[ControllerIndex].Name;
|
|
ControllerIndex++;
|
|
}
|
|
|
|
// If match wasn't found, try again with a generic device ID.
|
|
if (HdaControllerDev->Name == NULL) {
|
|
ControllerIndex = 0;
|
|
while (gHdaControllerList[ControllerIndex].Id != 0) {
|
|
// Check ID and revision against array element.
|
|
if (gHdaControllerList[ControllerIndex].Id == GET_PCI_GENERIC_ID(HdaControllerDev->VendorId))
|
|
HdaControllerDev->Name = gHdaControllerList[ControllerIndex].Name;
|
|
ControllerIndex++;
|
|
}
|
|
}
|
|
|
|
// If match still wasn't found, controller is unknown.
|
|
if (HdaControllerDev->Name == NULL)
|
|
HdaControllerDev->Name = HDA_CONTROLLER_MODEL_GENERIC;
|
|
// DEBUG((DEBUG_INFO, "HdaControllerGetName(): controller is %s\n", HdaControllerDev->Name));
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerReset(
|
|
IN HDA_CONTROLLER_DEV *HdaControllerDev)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerReset(): start\n"));
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo = HdaControllerDev->PciIo;
|
|
UINT32 HdaGCtl = 0;
|
|
UINT64 Tmp = 0;
|
|
|
|
// Get value of CRST bit.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint32, PCI_HDA_BAR, HDA_REG_GCTL, 1, &HdaGCtl);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Check if the controller is already in reset. If not, clear bit.
|
|
if (!(HdaGCtl & HDA_REG_GCTL_CRST)) {
|
|
HdaGCtl &= ~HDA_REG_GCTL_CRST;
|
|
Status = PciIo->Mem.Write(PciIo, EfiPciIoWidthUint32, PCI_HDA_BAR, HDA_REG_GCTL, 1, &HdaGCtl);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
}
|
|
|
|
// Set CRST bit to begin the process of coming out of reset.
|
|
HdaGCtl |= HDA_REG_GCTL_CRST;
|
|
Status = PciIo->Mem.Write(PciIo, EfiPciIoWidthUint32, PCI_HDA_BAR, HDA_REG_GCTL, 1, &HdaGCtl);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Wait for bit to be set. Once bit is set, the controller is ready.
|
|
Status = PciIo->PollMem(PciIo, EfiPciIoWidthUint32, PCI_HDA_BAR, HDA_REG_GCTL,
|
|
HDA_REG_GCTL_CRST, HDA_REG_GCTL_CRST, MS_TO_NANOSECOND(100), &Tmp);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Wait 100ms to ensure all codecs have also reset.
|
|
gBS->Stall(MS_TO_MICROSECOND(100));
|
|
|
|
// Controller is reset.
|
|
// DEBUG((DEBUG_INFO, "HdaControllerReset(): done\n"));
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerScanCodecs(
|
|
IN HDA_CONTROLLER_DEV *HdaControllerDev)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerScanCodecs(): start\n"));
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
|
UINT16 HdaStatests = 0;
|
|
EFI_HDA_IO_VERB_LIST HdaCodecVerbList;
|
|
UINT32 VendorVerb;
|
|
UINT32 VendorResponse;
|
|
UINT8 i;
|
|
|
|
// Streams.
|
|
UINTN CurrentOutputStreamIndex = 0;
|
|
UINTN CurrentInputStreamIndex = 0;
|
|
|
|
// Protocols.
|
|
HDA_IO_PRIVATE_DATA *HdaIoPrivateData;
|
|
VOID *TmpProtocol;
|
|
|
|
if (!HdaControllerDev) {
|
|
return EFI_INVALID_PARAMETER;
|
|
}
|
|
|
|
PciIo = HdaControllerDev->PciIo;
|
|
|
|
// Get STATESTS register.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_STATESTS, 1, &HdaStatests);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Create verb list with single item.
|
|
VendorVerb = HDA_CODEC_VERB(HDA_VERB_GET_PARAMETER, HDA_PARAMETER_VENDOR_ID);
|
|
gBS->SetMem(&HdaCodecVerbList, sizeof(EFI_HDA_IO_VERB_LIST), 0);
|
|
HdaCodecVerbList.Count = 1;
|
|
HdaCodecVerbList.Verbs = &VendorVerb;
|
|
HdaCodecVerbList.Responses = &VendorResponse;
|
|
|
|
// Iterate through register looking for active codecs.
|
|
for (i = 0; i < HDA_MAX_CODECS; i++) {
|
|
// Do we have a codec at this address?
|
|
if ((HdaStatests & (1 << i))) {
|
|
// DEBUG((DEBUG_INFO, "HdaControllerScanCodecs(): found codec @ 0x%X\n", i));
|
|
|
|
// Try to get the vendor ID. If this fails, ignore the codec.
|
|
VendorResponse = 0;
|
|
Status = HdaControllerSendCommands(HdaControllerDev, i, HDA_NID_ROOT, &HdaCodecVerbList);
|
|
if ((EFI_ERROR(Status)) || (VendorResponse == 0))
|
|
continue;
|
|
|
|
// Create HDA I/O protocol private data structure.
|
|
HdaIoPrivateData = AllocateZeroPool(sizeof(HDA_IO_PRIVATE_DATA));
|
|
if (HdaIoPrivateData == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto FREE_CODECS;
|
|
}
|
|
|
|
// Fill HDA I/O protocol private data structure.
|
|
HdaIoPrivateData->Signature = HDA_CONTROLLER_PRIVATE_DATA_SIGNATURE;
|
|
HdaIoPrivateData->HdaCodecAddress = i;
|
|
HdaIoPrivateData->HdaControllerDev = HdaControllerDev;
|
|
HdaIoPrivateData->HdaIo.GetAddress = HdaControllerHdaIoGetAddress;
|
|
HdaIoPrivateData->HdaIo.SendCommand = HdaControllerHdaIoSendCommand;
|
|
HdaIoPrivateData->HdaIo.SetupStream = HdaControllerHdaIoSetupStream;
|
|
HdaIoPrivateData->HdaIo.CloseStream = HdaControllerHdaIoCloseStream;
|
|
HdaIoPrivateData->HdaIo.GetStream = HdaControllerHdaIoGetStream;
|
|
HdaIoPrivateData->HdaIo.StartStream = HdaControllerHdaIoStartStream;
|
|
HdaIoPrivateData->HdaIo.StopStream = HdaControllerHdaIoStopStream;
|
|
|
|
// Assign output stream.
|
|
if (CurrentOutputStreamIndex < HdaControllerDev->OutputStreamsCount) {
|
|
// DEBUG((DEBUG_INFO, "Assigning output stream %u to codec\n", CurrentOutputStreamIndex));
|
|
HdaIoPrivateData->HdaOutputStream = HdaControllerDev->OutputStreams + CurrentOutputStreamIndex;
|
|
CurrentOutputStreamIndex++;
|
|
}
|
|
|
|
// Assign input stream.
|
|
if (CurrentInputStreamIndex < HdaControllerDev->InputStreamsCount) {
|
|
// DEBUG((DEBUG_INFO, "Assigning input stream %u to codec\n", CurrentInputStreamIndex));
|
|
HdaIoPrivateData->HdaInputStream = HdaControllerDev->InputStreams + CurrentInputStreamIndex;
|
|
CurrentInputStreamIndex++;
|
|
}
|
|
|
|
// Add to array.
|
|
HdaControllerDev->HdaIoChildren[i].PrivateData = HdaIoPrivateData;
|
|
}
|
|
}
|
|
|
|
// Clear STATESTS register.
|
|
HdaStatests = HDA_REG_STATESTS_CLEAR;
|
|
Status = PciIo->Mem.Write(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_STATESTS, 1, &HdaStatests);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Install protocols on each codec.
|
|
|
|
for (i = 0; i < HDA_MAX_CODECS; i++) {
|
|
// Do we have a codec at this address?
|
|
if (HdaControllerDev->HdaIoChildren[i].PrivateData != NULL) {
|
|
// Create Device Path for codec.
|
|
EFI_HDA_IO_DEVICE_PATH HdaIoDevicePathNode; //EFI_HDA_IO_DEVICE_PATH_TEMPLATE;
|
|
HdaIoDevicePathNode.Header.Type = MESSAGING_DEVICE_PATH;
|
|
HdaIoDevicePathNode.Header.SubType = MSG_VENDOR_DP;
|
|
HdaIoDevicePathNode.Header.Length[0] = (UINT8)(sizeof(EFI_HDA_IO_DEVICE_PATH));
|
|
HdaIoDevicePathNode.Header.Length[1] = (UINT8)((sizeof(EFI_HDA_IO_DEVICE_PATH)) >> 8);
|
|
HdaIoDevicePathNode.Guid = gEfiHdaIoDevicePathGuid;
|
|
// gBS->CopyMem((VOID*)&HdaIoDevicePathNode.Guid, (VOID*)&gEfiHdaIoDevicePathGuid, sizeof(EFI_GUID));
|
|
HdaIoDevicePathNode.Address = i;
|
|
HdaControllerDev->HdaIoChildren[i].DevicePath = AppendDevicePathNode(HdaControllerDev->DevicePath, (EFI_DEVICE_PATH_PROTOCOL*)&HdaIoDevicePathNode);
|
|
if (HdaControllerDev->HdaIoChildren[i].DevicePath == NULL) {
|
|
Status = EFI_INVALID_PARAMETER;
|
|
goto FREE_CODECS;
|
|
}
|
|
|
|
// Install protocols for the codec. The codec driver will later bind to this.
|
|
HdaControllerDev->HdaIoChildren[i].Handle = NULL;
|
|
Status = gBS->InstallMultipleProtocolInterfaces(&HdaControllerDev->HdaIoChildren[i].Handle,
|
|
&gEfiDevicePathProtocolGuid, HdaControllerDev->HdaIoChildren[i].DevicePath,
|
|
&gEfiHdaIoProtocolGuid, &HdaControllerDev->HdaIoChildren[i].PrivateData->HdaIo, NULL);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CODECS;
|
|
|
|
// Connect child to parent.
|
|
Status = gBS->OpenProtocol(HdaControllerDev->ControllerHandle, &gEfiPciIoProtocolGuid, &TmpProtocol,
|
|
HdaControllerDev->DriverBinding->DriverBindingHandle, HdaControllerDev->HdaIoChildren[i].Handle, EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CODECS;
|
|
}
|
|
}
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
FREE_CODECS:
|
|
//DEBUG((DEBUG_INFO, "HdaControllerScanCodecs(): failed to load driver for codec @ 0x%X\n", i));
|
|
|
|
return Status;
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerSendCommands(
|
|
IN HDA_CONTROLLER_DEV *HdaDev,
|
|
IN UINT8 CodecAddress,
|
|
IN UINT8 Node,
|
|
IN EFI_HDA_IO_VERB_LIST *Verbs)
|
|
{
|
|
//DEBUG((DEBUG_INFO, "HdaControllerSendCommands(): start\n"));
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
|
UINT32 *HdaCorb;
|
|
UINT64 *HdaRirb;
|
|
UINT32 RemainingVerbs;
|
|
UINT32 RemainingResponses;
|
|
UINT16 HdaCorbReadPointer = 0;
|
|
UINT16 HdaRirbWritePointer = 0;
|
|
BOOLEAN ResponseReceived;
|
|
UINT8 ResponseTimeout;
|
|
UINT64 RirbResponse;
|
|
UINT32 VerbCommand;
|
|
BOOLEAN Retry = FALSE;
|
|
|
|
// Ensure parameters are valid.
|
|
if (!HdaDev || (CodecAddress >= HDA_MAX_CODECS) || !Verbs || (Verbs->Count < 1))
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
// Get pointers to CORB and RIRB.
|
|
HdaCorb = HdaDev->CorbBuffer;
|
|
HdaRirb = HdaDev->RirbBuffer;
|
|
|
|
PciIo = HdaDev->PciIo;
|
|
|
|
// Lock.
|
|
AcquireSpinLock(&HdaDev->SpinLock);
|
|
|
|
START:
|
|
RemainingVerbs = Verbs->Count;
|
|
RemainingResponses = Verbs->Count;
|
|
do {
|
|
// Keep sending verbs until they are all sent.
|
|
if (RemainingVerbs) {
|
|
// Get current CORB read pointer.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_CORBRP, 1, &HdaCorbReadPointer);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
//DEBUG((DEBUG_INFO, "old RP: 0x%X\n", HdaCorbReadPointer));
|
|
|
|
// Add verbs to CORB until all of them are added or the CORB becomes full.
|
|
while (RemainingVerbs && ((HdaDev->CorbWritePointer + 1 % HdaDev->CorbEntryCount) != HdaCorbReadPointer)) {
|
|
// Move write pointer and write verb to CORB.
|
|
HdaDev->CorbWritePointer++;
|
|
HdaDev->CorbWritePointer %= HdaDev->CorbEntryCount;
|
|
VerbCommand = HDA_CORB_VERB(CodecAddress, Node, Verbs->Verbs[Verbs->Count - RemainingVerbs]);
|
|
HdaCorb[HdaDev->CorbWritePointer] = VerbCommand;
|
|
|
|
// Move to next verb.
|
|
RemainingVerbs--;
|
|
}
|
|
|
|
// Set CORB write pointer.
|
|
Status = PciIo->Mem.Write(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_CORBWP, 1, &HdaDev->CorbWritePointer);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
}
|
|
|
|
// Get responses from RIRB.
|
|
ResponseReceived = FALSE;
|
|
ResponseTimeout = 10;
|
|
while (!ResponseReceived) {
|
|
// Get current RIRB write pointer.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_RIRBWP, 1, &HdaRirbWritePointer);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
|
|
// If the read and write pointers differ, there are responses waiting.
|
|
while (HdaDev->RirbReadPointer != HdaRirbWritePointer) {
|
|
// Increment RIRB read pointer.
|
|
HdaDev->RirbReadPointer++;
|
|
HdaDev->RirbReadPointer %= HdaDev->RirbEntryCount;
|
|
|
|
// Get response and ensure it belongs to the current codec.
|
|
RirbResponse = HdaRirb[HdaDev->RirbReadPointer];
|
|
if (HDA_RIRB_CAD(RirbResponse) != CodecAddress || HDA_RIRB_UNSOL(RirbResponse)) {
|
|
DEBUG((DEBUG_INFO, "Unknown response!\n"));
|
|
continue;
|
|
}
|
|
|
|
// Add response to list.
|
|
Verbs->Responses[Verbs->Count - RemainingResponses] = HDA_RIRB_RESP(RirbResponse);
|
|
RemainingResponses--;
|
|
ResponseReceived = TRUE;
|
|
}
|
|
|
|
// If no response still, wait a bit.
|
|
if (!ResponseReceived) {
|
|
// If timeout reached, fail.
|
|
if (!ResponseTimeout) {
|
|
// DEBUG((DEBUG_INFO, "Command: 0x%X\n", VerbCommand));
|
|
Status = EFI_TIMEOUT;
|
|
goto TIMEOUT;
|
|
}
|
|
|
|
ResponseTimeout--;
|
|
gBS->Stall(MS_TO_MICROSECOND(5));
|
|
if (ResponseTimeout < 5) {
|
|
DEBUG((DEBUG_INFO, "%u timeouts reached while waiting for response!\n", ResponseTimeout));
|
|
}
|
|
}
|
|
}
|
|
|
|
} while (RemainingVerbs || RemainingResponses);
|
|
Status = EFI_SUCCESS;
|
|
goto DONE;
|
|
|
|
TIMEOUT:
|
|
// DEBUG((DEBUG_INFO, "Timeout!\n"));
|
|
if (!Retry) {
|
|
// DEBUG((DEBUG_INFO, "Stall detected, restarting CORB and RIRB!\n"));
|
|
Status = HdaControllerSetCorb(HdaDev, FALSE);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
Status = HdaControllerSetRirb(HdaDev, FALSE);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
Status = HdaControllerSetCorb(HdaDev, TRUE);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
Status = HdaControllerSetRirb(HdaDev, TRUE);
|
|
if (EFI_ERROR(Status))
|
|
goto DONE;
|
|
|
|
// Try again.
|
|
Retry = TRUE;
|
|
goto START;
|
|
}
|
|
|
|
DONE:
|
|
ReleaseSpinLock(&HdaDev->SpinLock);
|
|
return Status;
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerInstallProtocols(
|
|
IN HDA_CONTROLLER_DEV *HdaControllerDev)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerInstallProtocols(): start\n"));
|
|
|
|
// Create variables.
|
|
HDA_CONTROLLER_INFO_PRIVATE_DATA *HdaControllerInfoData;
|
|
|
|
// Allocate space for info protocol data.
|
|
HdaControllerInfoData = AllocateZeroPool(sizeof(HDA_CONTROLLER_INFO_PRIVATE_DATA));
|
|
if (HdaControllerInfoData == NULL)
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
// Populate data.
|
|
HdaControllerInfoData->Signature = HDA_CONTROLLER_PRIVATE_DATA_SIGNATURE;
|
|
HdaControllerInfoData->HdaControllerDev = HdaControllerDev;
|
|
HdaControllerInfoData->HdaControllerInfo.GetName = HdaControllerInfoGetName;
|
|
|
|
// Install protocols.
|
|
HdaControllerDev->HdaControllerInfoData = HdaControllerInfoData;
|
|
return gBS->InstallMultipleProtocolInterfaces(&HdaControllerDev->ControllerHandle,
|
|
&gEfiHdaControllerInfoProtocolGuid, &HdaControllerInfoData->HdaControllerInfo,
|
|
&gEfiCallerIdGuid, HdaControllerDev, NULL);
|
|
}
|
|
|
|
VOID
|
|
EFIAPI
|
|
HdaControllerCleanup(
|
|
IN HDA_CONTROLLER_DEV *HdaControllerDev)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerCleanup(): start\n"));
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
|
UINT32 HdaGCtl = 0;
|
|
UINT8 i;
|
|
|
|
// If controller device is already free, we are done.
|
|
if (HdaControllerDev == NULL)
|
|
return;
|
|
|
|
PciIo = HdaControllerDev->PciIo;
|
|
|
|
// Clean HDA Controller info protocol.
|
|
if (HdaControllerDev->HdaControllerInfoData != NULL) {
|
|
// Uninstall protocol.
|
|
// DEBUG((DEBUG_INFO, "HdaControllerCleanup(): clean HDA Controller Info\n"));
|
|
Status = gBS->UninstallProtocolInterface(HdaControllerDev->ControllerHandle,
|
|
&gEfiHdaControllerInfoProtocolGuid, &HdaControllerDev->HdaControllerInfoData->HdaControllerInfo);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// Free data.
|
|
FreePool(HdaControllerDev->HdaControllerInfoData);
|
|
}
|
|
|
|
// Clean HDA I/O children.
|
|
for (i = 0; i < HDA_MAX_CODECS; i++) {
|
|
// Clean Device Path protocol.
|
|
if (HdaControllerDev->HdaIoChildren[i].DevicePath != NULL) {
|
|
// Uninstall protocol.
|
|
// DEBUG((DEBUG_INFO, "HdaControllerCleanup(): clean Device Path index %u\n", i));
|
|
Status = gBS->UninstallProtocolInterface(HdaControllerDev->HdaIoChildren[i].Handle,
|
|
&gEfiDevicePathProtocolGuid, HdaControllerDev->HdaIoChildren[i].DevicePath);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// Free Device Path.
|
|
FreePool(HdaControllerDev->HdaIoChildren[i].DevicePath);
|
|
}
|
|
|
|
// Clean HDA I/O protocol.
|
|
if (HdaControllerDev->HdaIoChildren[i].PrivateData != NULL) {
|
|
// Uninstall protocol.
|
|
// DEBUG((DEBUG_INFO, "HdaControllerCleanup(): clean HDA I/O index %u\n", i));
|
|
Status = gBS->UninstallProtocolInterface(HdaControllerDev->HdaIoChildren[i].Handle,
|
|
&gEfiHdaIoProtocolGuid, &HdaControllerDev->HdaIoChildren[i].PrivateData->HdaIo);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// Free private data.
|
|
FreePool(HdaControllerDev->HdaIoChildren[i].PrivateData);
|
|
}
|
|
}
|
|
|
|
// Cleanup streams.
|
|
HdaControllerCleanupStreams(HdaControllerDev);
|
|
|
|
// Stop and cleanup CORB and RIRB.
|
|
HdaControllerCleanupCorb(HdaControllerDev);
|
|
HdaControllerCleanupRirb(HdaControllerDev);
|
|
|
|
// Get value of CRST bit.
|
|
Status = PciIo->Mem.Read(PciIo, EfiPciIoWidthUint32, PCI_HDA_BAR, HDA_REG_GCTL, 1, &HdaGCtl);
|
|
|
|
// Place controller into a reset state to stop it.
|
|
if (!(EFI_ERROR(Status))) {
|
|
HdaGCtl &= ~HDA_REG_GCTL_CRST;
|
|
Status = PciIo->Mem.Write(PciIo, EfiPciIoWidthUint32, PCI_HDA_BAR, HDA_REG_GCTL, 1, &HdaGCtl);
|
|
}
|
|
|
|
// Free controller device.
|
|
gBS->UninstallProtocolInterface(HdaControllerDev->ControllerHandle,
|
|
&gEfiCallerIdGuid, HdaControllerDev);
|
|
FreePool(HdaControllerDev);
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerDriverBindingSupported(
|
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
|
IN EFI_HANDLE ControllerHandle,
|
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL)
|
|
{
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo = 0;
|
|
HDA_PCI_CLASSREG HdaClassReg;
|
|
|
|
// Open PCI I/O protocol. If this fails, it's not a PCI device.
|
|
Status = gBS->OpenProtocol(ControllerHandle, &gEfiPciIoProtocolGuid, (VOID**)&PciIo,
|
|
This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_BY_DRIVER);
|
|
if (EFI_ERROR(Status))
|
|
return Status;
|
|
|
|
// Read class code from PCI.
|
|
Status = PciIo->Pci.Read(PciIo, EfiPciIoWidthUint8, PCI_CLASSCODE_OFFSET,
|
|
sizeof(HDA_PCI_CLASSREG) / sizeof(UINT8), &HdaClassReg);
|
|
if (EFI_ERROR(Status))
|
|
goto CLOSE_PCIIO;
|
|
|
|
// Check class code. If not an HDA controller, we cannot support it.
|
|
//may check also PCI_CLASS_MEDIA_AUDIO
|
|
if ((HdaClassReg.Class != PCI_CLASS_MEDIA) || (HdaClassReg.SubClass != PCI_CLASS_MEDIA_HDA)) {
|
|
Status = EFI_UNSUPPORTED;
|
|
goto CLOSE_PCIIO;
|
|
}
|
|
Status = EFI_SUCCESS;
|
|
|
|
CLOSE_PCIIO:
|
|
// Close PCI I/O protocol and return status.
|
|
gBS->CloseProtocol(ControllerHandle, &gEfiPciIoProtocolGuid, This->DriverBindingHandle, ControllerHandle);
|
|
return Status;
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerDriverBindingStart(
|
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
|
IN EFI_HANDLE ControllerHandle,
|
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerDriverBindingStart(): start\n"));
|
|
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
|
EFI_DEVICE_PATH_PROTOCOL *HdaControllerDevicePath;
|
|
HDA_CONTROLLER_DEV *HdaControllerDev;
|
|
|
|
// Open PCI I/O protocol.
|
|
Status = gBS->OpenProtocol(ControllerHandle, &gEfiPciIoProtocolGuid, (VOID**)&PciIo,
|
|
This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_BY_DRIVER);
|
|
if (EFI_ERROR (Status))
|
|
return Status;
|
|
|
|
// Open Device Path protocol.
|
|
Status = gBS->OpenProtocol(ControllerHandle, &gEfiDevicePathProtocolGuid, (VOID**)&HdaControllerDevicePath,
|
|
This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_BY_DRIVER);
|
|
if (EFI_ERROR (Status))
|
|
goto CLOSE_PCIIO;
|
|
|
|
// Allocate controller device.
|
|
HdaControllerDev = AllocateZeroPool(sizeof(HDA_CONTROLLER_DEV));
|
|
if (HdaControllerDev == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto CLOSE_PCIIO;
|
|
}
|
|
|
|
// Fill controller device data.
|
|
HdaControllerDev->Signature = HDA_CONTROLLER_PRIVATE_DATA_SIGNATURE;
|
|
HdaControllerDev->PciIo = PciIo;
|
|
HdaControllerDev->DevicePath = HdaControllerDevicePath;
|
|
HdaControllerDev->DriverBinding = This;
|
|
HdaControllerDev->ControllerHandle = ControllerHandle;
|
|
InitializeSpinLock(&HdaControllerDev->SpinLock);
|
|
|
|
// Setup PCI hardware.
|
|
Status = HdaControllerInitPciHw(HdaControllerDev);
|
|
if (EFI_ERROR (Status))
|
|
goto FREE_CONTROLLER;
|
|
|
|
// Get controller name.
|
|
HdaControllerGetName(HdaControllerDev);
|
|
|
|
// Reset controller.
|
|
Status = HdaControllerReset(HdaControllerDev);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
|
|
// Install info protocol.
|
|
Status = HdaControllerInstallProtocols(HdaControllerDev);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
|
|
// Initialize CORB and RIRB.
|
|
Status = HdaControllerInitCorb(HdaControllerDev);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
Status = HdaControllerInitRirb(HdaControllerDev);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
|
|
// needed for QEMU.
|
|
#ifdef QEMU
|
|
UINT16 dd = 0xFF;
|
|
PciIo->Mem.Write(PciIo, EfiPciIoWidthUint16, PCI_HDA_BAR, HDA_REG_RINTCNT, 1, &dd);
|
|
#endif
|
|
|
|
// Start CORB and RIRB
|
|
Status = HdaControllerSetCorb(HdaControllerDev, TRUE);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
Status = HdaControllerSetRirb(HdaControllerDev, TRUE);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
|
|
// Init streams.
|
|
Status = HdaControllerInitStreams(HdaControllerDev);
|
|
if (EFI_ERROR(Status))
|
|
goto FREE_CONTROLLER;
|
|
|
|
// Scan for codecs.
|
|
Status = HdaControllerScanCodecs(HdaControllerDev);
|
|
ASSERT_EFI_ERROR(Status);
|
|
|
|
// DEBUG((DEBUG_INFO, "HdaControllerDriverBindingStart(): done\n"));
|
|
return Status;
|
|
|
|
FREE_CONTROLLER:
|
|
// Restore PCI attributes if needed.
|
|
if (HdaControllerDev->OriginalPciAttributesSaved)
|
|
PciIo->Attributes(PciIo, EfiPciIoAttributeOperationSet, HdaControllerDev->OriginalPciAttributes, NULL);
|
|
|
|
// Free controller device.
|
|
HdaControllerCleanup(HdaControllerDev);
|
|
|
|
CLOSE_PCIIO:
|
|
// Close protocols.
|
|
gBS->CloseProtocol(ControllerHandle, &gEfiDevicePathProtocolGuid, This->DriverBindingHandle, ControllerHandle);
|
|
gBS->CloseProtocol(ControllerHandle, &gEfiPciIoProtocolGuid, This->DriverBindingHandle, ControllerHandle);
|
|
return Status;
|
|
}
|
|
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HdaControllerDriverBindingStop(
|
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
|
IN EFI_HANDLE ControllerHandle,
|
|
IN UINTN NumberOfChildren,
|
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL)
|
|
{
|
|
// DEBUG((DEBUG_INFO, "HdaControllerDriverBindingStop(): start\n"));
|
|
// Create variables.
|
|
EFI_STATUS Status;
|
|
HDA_CONTROLLER_DEV *HdaControllerDev = NULL;
|
|
|
|
// Get codec device.
|
|
Status = gBS->OpenProtocol(ControllerHandle, &gEfiCallerIdGuid, (VOID**)&HdaControllerDev,
|
|
This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_GET_PROTOCOL);
|
|
if (!(EFI_ERROR(Status))) {
|
|
// Ensure controller device is valid.
|
|
if (HdaControllerDev->Signature != HDA_CONTROLLER_PRIVATE_DATA_SIGNATURE)
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
// Restore PCI attributes if needed.
|
|
if (HdaControllerDev->OriginalPciAttributesSaved)
|
|
HdaControllerDev->PciIo->Attributes(HdaControllerDev->PciIo, EfiPciIoAttributeOperationSet,
|
|
HdaControllerDev->OriginalPciAttributes, NULL);
|
|
|
|
// Cleanup controller.
|
|
HdaControllerCleanup(HdaControllerDev);
|
|
}
|
|
|
|
// Close protocols.
|
|
gBS->CloseProtocol(ControllerHandle, &gEfiDevicePathProtocolGuid, This->DriverBindingHandle, ControllerHandle);
|
|
gBS->CloseProtocol(ControllerHandle, &gEfiPciIoProtocolGuid, This->DriverBindingHandle, ControllerHandle);
|
|
return EFI_SUCCESS;
|
|
}
|