⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 psc_ac97.c

📁 基于AMD Au1200(MIPS32处理器)的AC97驱动源码
💻 C
字号:
/*++
Copyright (c) 2004  BSQUARE Corporation.  All rights reserved.

Module Name:

    psc_ac97.c

Module Description:

    This module contains the platform dependent code
    of AC97 audio driver for the Au1550

Author:

    Ian Rae  January 2004

Revision History:

--*/
#include <windows.h>
#include <bceddk.h>
#include <gpio.h>
#include "ac97.h"
#include "au1kac97.h"
#include "ac97aud.h"

#include <wavedbg.h>

#define ZONE_HACK 0

#define ZONE_AC97_ERROR ZONE_HACK
#define ZONE_AC97_FUNC  ZONE_HACK
#define ZONE_AC97_DMA   ZONE_HACK
#define ZONE_AC97_INTR  ZONE_HACK

ULONG gIntrAudio = SYSINTR_NOP;

extern HANDLE hAudioInterrupt;
extern PWAVE_DEVICE_INSTANCE WaveDevice;

BOOL
AC97ReadReg(
    IN PSC_AC97 *pAC97,
    IN ULONG    InReg,
    OUT ULONG   *Result
    )
{
    ULONG Data;
    ULONG Timeout=TIMEOUT*10;

	Data  = PSC_AC97_CDC_INDX_N(InReg);
	Data |= PSC_AC97_CDC_RD;

    WRITE_REGISTER_ULONG((PULONG)&pAC97->cdc, Data);

    while(Timeout) {
        if(READ_REGISTER_ULONG((PULONG)&pAC97->evnt)&PSC_AC97_EVNT_CD) break;
        Timeout--;
    }
	// Clear Command Done flag
	WRITE_REGISTER_ULONG((PULONG)&pAC97->evnt,PSC_AC97_EVNT_CD);


    if(Timeout) {
        Data = READ_REGISTER_ULONG((PULONG)&pAC97->cdc);
        Data &= PSC_AC97_CDC_DATA;
        *Result = Data;
        DEBUGMSG(ZONE_AC97_FUNC,(L"-Ac97ReadReg 0x%X\r\n",Data));
        return TRUE;
    }

    DEBUGMSG(ZONE_AC97_FUNC,(L"-Ac97ReadReg timeout\r\n"));
    return FALSE;
}    

BOOL
AC97WriteReg(
    IN PSC_AC97 *pAC97,
    IN ULONG    Reg,
    IN ULONG    Value
    )
{
    ULONG Data;
    ULONG Timeout=TIMEOUT*10;

    Data  = PSC_AC97_CDC_DATA_N(Value);
	Data |= PSC_AC97_CDC_INDX_N(Reg);
    
    WRITE_REGISTER_ULONG((PULONG)&pAC97->cdc, Data);

    while(Timeout) {
        if(READ_REGISTER_ULONG((PULONG)&pAC97->evnt)&PSC_AC97_EVNT_CD) break;
        Timeout--;
    }

	// Clear Command Done flag
	WRITE_REGISTER_ULONG((PULONG)&pAC97->evnt,PSC_AC97_EVNT_CD);

    if(Timeout) {
        return TRUE;
    }

    return FALSE;
}    

VOID
ShutdownDma(
    IN PWAVE_RESOURCE WaveResource
    )
{
    DEBUGMSG(ZONE_AC97_DMA, (TEXT("+ShutdownDma\r\n")));

	HalStopDMA(WaveResource->DMAChannel);

   	EnterCriticalSection(&WaveResource->CriticalSection);

    if(WaveResource->WaveDirection==WAPI_OUT) {
        WRITE_REGISTER_ULONG((PULONG)&WaveResource->AC97Ctrl->pcr, PSC_AC97_PCR_TP);
    } else {
        WRITE_REGISTER_ULONG((PULONG)&WaveResource->AC97Ctrl->pcr, PSC_AC97_PCR_RP);
    }

	WaveResource->DmaRunning = FALSE;
    
	LeaveCriticalSection(&WaveResource->CriticalSection);

	DEBUGMSG(ZONE_AC97_DMA, (TEXT("-ShutdownDma\r\n")));

}

VOID
StartDma(
    IN PWAVE_RESOURCE WaveResource
    )
{
    DEBUGMSG(ZONE_AC97_DMA, (TEXT("+StartDma\r\n")));

    EnterCriticalSection(&WaveResource->CriticalSection);
    
    if(WaveResource->WaveDirection==WAPI_OUT) {
        WRITE_REGISTER_ULONG((PULONG)&WaveResource->AC97Ctrl->pcr, PSC_AC97_PCR_TS);
    } else {
        WRITE_REGISTER_ULONG((PULONG)&WaveResource->AC97Ctrl->pcr, PSC_AC97_PCR_RS);
    }
    
	HalStartDMA(WaveResource->DMAChannel);
    
	WaveResource->DmaRunning = TRUE;
    LeaveCriticalSection(&WaveResource->CriticalSection);

    DEBUGMSG(ZONE_AC97_DMA, (TEXT("-StartDma \r\n")));

}

DWORD
FindMicVolume(    
	IN PWAVE_DEVICE_INSTANCE WaveDevice
)
{
    HKEY    hkey;
    LONG Status;
    DWORD Value = AC97_DEFAULT_MICVOLUME;    
    DWORD Size = sizeof(Value);

	// Examine Mic Volume in registry
    //
    Status = RegOpenKeyEx(
        HKEY_LOCAL_MACHINE,
        AUDIO_REGKEY,
        0,
        0,
        &hkey);
    
    
    if (ERROR_SUCCESS == Status) {
        Status = RegQueryValueEx(hkey,
                                (TEXT("MicVolume")),
                                NULL,
                                NULL,
                                (PUCHAR)&Value,
                                &Size
                                );
    }

    if (hkey) {
        RegCloseKey(hkey);
    }

	return Value;
}


BOOL
InitializeInterrupt(
    IN PWAVE_DEVICE_INSTANCE WaveDevice
    )
{
	ULONG HwIntr;

    DEBUGMSG(ZONE_AC97_FUNC,(L"+Initialize Interrupt\r\n"));

    HwIntr = HalGetDMAHwIntr(WaveDevice->DMAChannelOutput);
	HwIntr |= HalGetDMAHwIntr(WaveDevice->DMAChannelInput) << 8;

	RETAILMSG(1,(TEXT("Hooking HWINTR %08X\r\n"),HwIntr));

	gIntrAudio = InterruptConnect(Internal, 0, HwIntr, 0);

    if (gIntrAudio == SYSINTR_NOP)
    {
        RETAILMSG(1,(L"Can not allocate SYSINTR\r\n"));
        goto ErrorReturn;
    }

    DEBUGMSG(ZONE_AC97_FUNC,(L"-Initialize Interrupt\r\n"));
    return TRUE;
  ErrorReturn:
    DEBUGMSG(ZONE_AC97_FUNC,(L"-Initialize Interrupt failed\r\n"));
    return FALSE;
}

BOOL InitializeRegisters(PSC_AC97 *pAC)
{
	int timeout;
	BOOL status = TRUE;
	ULONG tmp;

	// Handle any platform specific initialization
#if defined(PLATFORM_AC97_PSC_INIT_CODE)
	PLATFORM_AC97_PSC_INIT_CODE
#endif

	// We seem to need this to be able to resume more than once
	// still working on explaining exactly why that is...
	tmp = READ_REGISTER_ULONG((PULONG)&pAC->cfg);

	// Reset and configure the PSC for AC97
	WRITE_REGISTER_ULONG((PULONG)&pAC->psc.sel, PSC_SEL_PS_AC97 | PSC_SEL_CLK_SERIAL);
	
	// Reset external codec
	WRITE_REGISTER_ULONG((PULONG)&pAC->rst, PSC_AC97_RST_RST);
	HalStallExecution(10000);
	WRITE_REGISTER_ULONG((PULONG)&pAC->rst, 0);

	// Enable the PSC
	WRITE_REGISTER_ULONG((PULONG)&pAC->psc.ctl, PSC_CTL_CE |PSC_CTL_EN);

	// wait for clock detect
	timeout = 1000;
	while (timeout && !((tmp=READ_REGISTER_ULONG((PULONG)&pAC->sts)) & PSC_AC97_STS_SR)) {
		HalStallExecution(1000);
		timeout--;
	}

	if (!timeout) {
		status = FALSE;
		goto ErrorReturn;
	}

	// configure the AC97 controller
	tmp = PSC_AC97_CFG_TRD_N(3) | 
	      PSC_AC97_CFG_RRD_N(3) | 
          PSC_AC97_CFG_LEN_N(7) | 
		  PSC_AC97_CFG_TXSLOT_N(3) | 
		  PSC_AC97_CFG_RXSLOT_N(3) | 
		  PSC_AC97_CFG_DE;

	WRITE_REGISTER_ULONG((PULONG)&pAC->cfg, tmp );

	// wait until the device is ready
	timeout = 1000;
	while (timeout && !((tmp=READ_REGISTER_ULONG((PULONG)&pAC->sts)) & PSC_AC97_STS_DR)) {
		HalStallExecution(1000);
		timeout--;
	}

	if (!timeout) {
		status = FALSE;
		goto ErrorReturn;
	}

	// mask all interrupts
	WRITE_REGISTER_ULONG((PULONG)&pAC->msk,0xFFFFFFFF);

ErrorReturn:
	return status;
}

BOOL
InitializePlatform(
    IN PWAVE_DEVICE_INSTANCE WaveDevice
    )
{
    PSC_AC97         *pAC;
	PHYSICAL_ADDRESS PhyAdd;
	BOOL             status = TRUE;
	HANDLE           hGPIO = NULL;

#if (PLATFORM_AC97_PSC==0)
	PhyAdd.QuadPart = PSC0_PHYS_ADDR;
#elif (PLATFORM_AC97_PSC==1)
	PhyAdd.QuadPart = PSC1_PHYS_ADDR;
#elif (PLATFORM_AC97_PSC==2)
	PhyAdd.QuadPart = PSC2_PHYS_ADDR;
#elif (PLATFORM_AC97_PSC==3)
	PhyAdd.QuadPart = PSC3_PHYS_ADDR;
#else
#error PLATFORM_AC97_PSC must be defined in the range 0 to 3
#endif

	RETAILMSG(1,(L"Initializing PSC%d for AC97 operation\r\n",PLATFORM_AC97_PSC));

	WaveDevice->AC97Ctrl = MmMapIoSpace(PhyAdd,
								        sizeof(WaveDevice->AC97Ctrl),
								        FALSE);

	if(WaveDevice->AC97Ctrl==NULL) {
		DEBUGMSG(ZONE_AC97_ERROR,(L"Can not map AC97 Control registers\r\n"));
		goto ErrorReturn;
	}

    hGPIO = GPIO_Init();
    if(hGPIO==INVALID_HANDLE_VALUE) {
        DEBUGMSG(ZONE_AC97_ERROR,(L"AC97: Can not open GPIO device\r\n"));
        goto ErrorReturn;
    }

	// CPU dependent GPIO pin functionality
#if defined(CPU_AU1550)
    if(!GPIO_SetPinFunction(hGPIO,SYS_PINFUNC_S1)) {
        DEBUGMSG(ZONE_ERROR,(L"AC97: Can not set pinfunc for AC97\r\n"));
        goto ErrorReturn;
    }
#elif defined(CPU_AU1200) && (PLATFORM_AC97_PSC==0)
    if(!GPIO_ClearPinFunction(hGPIO,SYS_PINFUNC_P0B | (1<<18))) {
        DEBUGMSG(ZONE_ERROR,(L"SPI: Can not set pinfunc for AC97 \r\n"));
        goto ErrorReturn;
    }
    if(!GPIO_SetPinFunction(hGPIO,(1<<17))) {
        DEBUGMSG(ZONE_ERROR,(L"SPI: Can not set pinfunc for AC97 \r\n"));
        goto ErrorReturn;
    }
#elif defined(CPU_AU1200) && (PLATFORM_AC97_PSC==1)
    if(!GPIO_ClearPinFunction(hGPIO,SYS_PINFUNC_P1A)) {
        DEBUGMSG(ZONE_ERROR,(L"AC97: Can not set pinfunc for AC97\r\n"));
        goto ErrorReturn;
    }
    if(!GPIO_SetPinFunction(hGPIO,SYS_PINFUNC_P1C)) {
        DEBUGMSG(ZONE_AC97_ERROR,(L"AC97: Can not set pinfunc for AC97\r\n"));
        goto ErrorReturn;
    }
#endif

	// we are done with GPIO now
	CloseHandle(hGPIO);

	// find a Mic Volume
	WaveDevice->MicVolume = FindMicVolume(WaveDevice);

	pAC = WaveDevice->AC97Ctrl;
	
	status = InitializeRegisters(pAC);

	DEBUGMSG(ZONE_AC97_FUNC,(L"-Initialize Platform\r\n"));
    return status;

ErrorReturn:
	if(WaveDevice->AC97Ctrl) {
		MmUnmapIoSpace((PVOID)WaveDevice->AC97Ctrl,
		               sizeof(*WaveDevice->AC97Ctrl));
	}

	if(hGPIO) {
		CloseHandle(hGPIO);
	}

	DEBUGMSG(ZONE_AC97_FUNC,(L"-Initialize Platform failed\r\n"));
	return FALSE;
}
  
BOOLEAN
InitializeCodec(
    IN OUT PWAVE_DEVICE_INSTANCE WaveDevice
    )
{
    ULONG i;
    
    //
    //  Mute all channels 
    //
    for (i=AC97_ALT_LINE_LEVEL_OUT_VOLUME;i<=AC97_RECORD_GAIN;i+=2) {
        AC97WriteReg(WaveDevice->AC97Ctrl,i,VOLUME_MUTE);
    }

    //
    // Set True Stereo Line Out as 0dB
    //
    AC97WriteReg(WaveDevice->AC97Ctrl,AC97_ALT_LINE_LEVEL_OUT_VOLUME,0);
    AC97WriteReg(WaveDevice->AC97Ctrl,AC97_RECORD_GAIN,0x0f0f);

	// Set Mic Volume
    //
    AC97WriteReg(WaveDevice->AC97Ctrl,AC97_MIC_VOLUME,WaveDevice->MicVolume);

    //  Set PCM channel to default
    //
    AC97SetPCMOutVolume(WaveDevice->AC97Ctrl,  DEFAULT_PCM_VOLUME,   DEFAULT_PCM_VOLUME);

    return TRUE;
}


BOOL
InitializeDMA(
    IN PWAVE_DEVICE_INSTANCE WaveDevice
    )
{
    DEBUGMSG(ZONE_AC97_FUNC,(L"+InitializeDMA\r\n"));

    WaveDevice->DMAChannelOutput = HalAllocateDMAChannel();

    if(WaveDevice->DMAChannelOutput==NULL) {
        DEBUGMSG(ZONE_AC97_ERROR,(L"Can not allocate output DMA Channel\r\n"));
        goto ErrorReturn;
    }
    else {
        DEBUGMSG(ZONE_AC97_DMA,(L"Using DMA channel for AC97 output\r\n"));
    }
    
    HalInitDmaChannel(WaveDevice->DMAChannelOutput,
	                  DMA_AC97_TX,
	                  WaveDevice->DmaBufferSize,
					  TRUE);

    WaveDevice->DMAChannelInput = HalAllocateDMAChannel();

    if(WaveDevice->DMAChannelInput==NULL) {
        DEBUGMSG(ZONE_AC97_ERROR,(L"Can not allocate input DMA Channel\r\n"));
        goto ErrorReturn;
    }
    else {
        DEBUGMSG(ZONE_AC97_DMA,(L"Using DMA channel for AC97 input\r\n"));
    }

    HalInitDmaChannel(WaveDevice->DMAChannelInput,
	                  DMA_AC97_RX,
	                  WaveDevice->DmaBufferSize,
					  TRUE);

    DEBUGMSG(ZONE_AC97_FUNC,(L"-InitializeDMA\r\n"));    
    return TRUE;

ErrorReturn:

    DEBUGMSG(ZONE_AC97_FUNC,(L"-InitializeDMA failed\r\n"));    
    return FALSE;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -