blob: bf7a832aaa0b2ed1e1070f791cbd802359dcfeac [file] [log] [blame]
/*************************************************************************/ /*!
@File power.c
@Title Power management functions
@Copyright Copyright (c) Imagination Technologies Ltd. All Rights Reserved
@Description Main APIs for power management functions
@License Dual MIT/GPLv2
The contents of this file are subject to the MIT license as set out below.
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.
Alternatively, the contents of this file may be used under the terms of
the GNU General Public License Version 2 ("GPL") in which case the provisions
of GPL are applicable instead of those above.
If you wish to allow use of your version of this file only under the terms of
GPL, and not to allow others to use your version of this file under the terms
of the MIT license, indicate your decision by deleting the provisions above
and replace them with the notice and other provisions required by GPL as set
out in the file called "GPL-COPYING" included in this distribution. If you do
not delete the provisions above, a recipient may use your version of this file
under the terms of either the MIT license or GPL.
This License is also included in this distribution in the file called
"MIT-COPYING".
EXCEPT AS OTHERWISE STATED IN A NEGOTIATED AGREEMENT: (A) 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; AND (B) 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 "pdump_km.h"
#include "allocmem.h"
#include "osfunc.h"
#include "pvrsrv.h"
#include "pvr_debug.h"
#include "process_stats.h"
struct _PVRSRV_POWER_DEV_TAG_
{
PFN_PRE_POWER pfnDevicePrePower;
PFN_POST_POWER pfnDevicePostPower;
PFN_SYS_DEV_PRE_POWER pfnSystemPrePower;
PFN_SYS_DEV_POST_POWER pfnSystemPostPower;
PFN_PRE_CLOCKSPEED_CHANGE pfnPreClockSpeedChange;
PFN_POST_CLOCKSPEED_CHANGE pfnPostClockSpeedChange;
PFN_FORCED_IDLE_REQUEST pfnForcedIdleRequest;
PFN_FORCED_IDLE_CANCEL_REQUEST pfnForcedIdleCancelRequest;
PFN_DUST_COUNT_REQUEST pfnDustCountRequest;
IMG_HANDLE hSysData;
IMG_HANDLE hDevCookie;
PVRSRV_DEV_POWER_STATE eDefaultPowerState;
PVRSRV_DEV_POWER_STATE eCurrentPowerState;
};
/*!
Typedef for a pointer to a function that will be called for re-acquiring
device powerlock after releasing it temporarily for some timeout period
in function PVRSRVDeviceIdleRequestKM
*/
typedef PVRSRV_ERROR (*PFN_POWER_LOCK_ACQUIRE) (PCPVRSRV_DEVICE_NODE psDevNode);
static inline IMG_UINT64 PVRSRVProcessStatsGetTimeNs(void)
{
#if defined(PVRSRV_ENABLE_PROCESS_STATS)
return OSClockns64();
#else
return 0;
#endif
}
static inline IMG_UINT64 PVRSRVProcessStatsGetTimeUs(void)
{
#if defined(PVRSRV_ENABLE_PROCESS_STATS)
return OSClockus();
#else
return 0;
#endif
}
/*!
******************************************************************************
@Function _IsSystemStatePowered
@Description Tests whether a given system state represents powered-up.
@Input eSystemPowerState : a system power state
@Return IMG_BOOL
******************************************************************************/
static IMG_BOOL _IsSystemStatePowered(PVRSRV_SYS_POWER_STATE eSystemPowerState)
{
return (eSystemPowerState == PVRSRV_SYS_POWER_STATE_ON);
}
/*!
******************************************************************************
@Function PVRSRVPowerLock
@Description Obtain the mutex for power transitions. Only allowed when
system power is on.
@Return PVRSRV_ERROR_SYSTEM_STATE_POWERED_OFF or PVRSRV_OK
******************************************************************************/
PVRSRV_ERROR PVRSRVPowerLock(PCPVRSRV_DEVICE_NODE psDeviceNode)
{
OSLockAcquire(psDeviceNode->hPowerLock);
/* Only allow to take powerlock when the system power is on */
if (_IsSystemStatePowered(psDeviceNode->eCurrentSysPowerState))
{
return PVRSRV_OK;
}
OSLockRelease(psDeviceNode->hPowerLock);
return PVRSRV_ERROR_SYSTEM_STATE_POWERED_OFF;
}
/*!
******************************************************************************
@Function PVRSRVPowerTryLock
@Description Try to obtain the mutex for power transitions. Only allowed when
system power is on.
@Return PVRSRV_ERROR_RETRY or PVRSRV_ERROR_SYSTEM_STATE_POWERED_OFF or
PVRSRV_OK
******************************************************************************/
PVRSRV_ERROR PVRSRVPowerTryLock(PCPVRSRV_DEVICE_NODE psDeviceNode)
{
if (!(OSTryLockAcquire(psDeviceNode->hPowerLock)))
{
return PVRSRV_ERROR_RETRY;
}
/* Only allow to take powerlock when the system power is on */
if (_IsSystemStatePowered(psDeviceNode->eCurrentSysPowerState))
{
/* System is powered ON, return OK */
return PVRSRV_OK;
}
else
{
/* System is powered OFF, release the lock and return error */
OSLockRelease(psDeviceNode->hPowerLock);
return PVRSRV_ERROR_SYSTEM_STATE_POWERED_OFF;
}
}
/*!
******************************************************************************
@Function _PVRSRVForcedPowerLock
@Description Obtain the mutex for power transitions regardless of system
power state
@Return Always returns PVRSRV_OK. Function prototype required same as
PFN_POWER_LOCK_ACQUIRE
******************************************************************************/
static PVRSRV_ERROR _PVRSRVForcedPowerLock(PCPVRSRV_DEVICE_NODE psDeviceNode)
{
OSLockAcquire(psDeviceNode->hPowerLock);
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVPowerUnlock
@Description Release the mutex for power transitions
@Return PVRSRV_ERROR
******************************************************************************/
void PVRSRVPowerUnlock(PCPVRSRV_DEVICE_NODE psDeviceNode)
{
OSLockRelease(psDeviceNode->hPowerLock);
}
IMG_BOOL PVRSRVDeviceIsDefaultStateOFF(PVRSRV_POWER_DEV *psPowerDevice)
{
return (psPowerDevice->eDefaultPowerState == PVRSRV_DEV_POWER_STATE_OFF);
}
/*!
******************************************************************************
@Function PVRSRVSetDeviceDefaultPowerState
@Description Set the default device power state to eNewPowerState
@Input psDeviceNode : Device node
@Input eNewPowerState : New power state
@Return PVRSRV_ERROR
******************************************************************************/
PVRSRV_ERROR PVRSRVSetDeviceDefaultPowerState(PCPVRSRV_DEVICE_NODE psDeviceNode,
PVRSRV_DEV_POWER_STATE eNewPowerState)
{
PVRSRV_POWER_DEV *psPowerDevice;
psPowerDevice = psDeviceNode->psPowerDev;
if (psPowerDevice == NULL)
{
return PVRSRV_ERROR_INVALID_DEVICE;
}
psPowerDevice->eDefaultPowerState = eNewPowerState;
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function _PVRSRVDeviceIdleRequestKM
@Description Perform device-specific processing required to force the device
idle. The device power-lock might be temporarily released (and
again re-acquired) during the course of this call, hence to
maintain lock-ordering power-lock should be the last acquired
lock before calling this function
@Input psDeviceNode : Device node
@Input pfnIsDefaultStateOff : When specified, the idle request is only
processed if this function passes.
@Input bDeviceOffPermitted : IMG_TRUE if the transition should not fail
if device off
IMG_FALSE if the transition should fail if
device off
@Input pfnPowerLockAcquire : Function to re-acquire power-lock in-case
it was necessary to release it.
@Return PVRSRV_ERROR_PWLOCK_RELEASED_REACQ_FAILED
When re-acquisition of power-lock failed.
This error NEEDS EXPLICIT HANDLING at call
site as it signifies the caller needs to
AVOID calling PVRSRVPowerUnlock, since
power-lock is no longer "possessed" by
this context.
PVRSRV_OK When idle request succeeded.
PVRSRV_ERROR Other system errors.
******************************************************************************/
static PVRSRV_ERROR _PVRSRVDeviceIdleRequestKM(PPVRSRV_DEVICE_NODE psDeviceNode,
PFN_SYS_DEV_IS_DEFAULT_STATE_OFF pfnIsDefaultStateOff,
IMG_BOOL bDeviceOffPermitted,
PFN_POWER_LOCK_ACQUIRE pfnPowerLockAcquire)
{
PVRSRV_POWER_DEV *psPowerDev = psDeviceNode->psPowerDev;
PVRSRV_ERROR eError;
if ((psPowerDev && psPowerDev->pfnForcedIdleRequest) &&
(!pfnIsDefaultStateOff || pfnIsDefaultStateOff(psPowerDev)))
{
LOOP_UNTIL_TIMEOUT(MAX_HW_TIME_US)
{
eError = psPowerDev->pfnForcedIdleRequest(psPowerDev->hDevCookie,
bDeviceOffPermitted);
if (eError == PVRSRV_OK)
{
/* Idle request was successful */
break;
}
else if (eError == PVRSRV_ERROR_DEVICE_IDLE_REQUEST_DENIED)
{
PVRSRV_ERROR eErrPwrLockAcq;
/* FW denied idle request */
PVRSRVPowerUnlock(psDeviceNode);
OSWaitus(MAX_HW_TIME_US/WAIT_TRY_COUNT);
eErrPwrLockAcq = pfnPowerLockAcquire(psDeviceNode);
if (eErrPwrLockAcq != PVRSRV_OK)
{
/* We only understand PVRSRV_ERROR_RETRY, so assert on others.
* Moreover, we've ended-up releasing the power-lock which was
* originally "held" by caller before calling this function -
* since this needs vigilant handling at call-site, we pass
* back an explicit error, for caller(s) to "avoid" calling
* PVRSRVPowerUnlock */
PVR_ASSERT(eErrPwrLockAcq == PVRSRV_ERROR_RETRY);
PVR_DPF((PVR_DBG_ERROR, "%s: Failed to re-acquire power-lock "
"(%s) after releasing it for a time-out",
__func__, PVRSRVGetErrorString(eErrPwrLockAcq)));
return PVRSRV_ERROR_PWLOCK_RELEASED_REACQ_FAILED;
}
}
else
{
/* some other error occurred, return failure */
break;
}
} END_LOOP_UNTIL_TIMEOUT();
}
else
{
return PVRSRV_OK;
}
return eError;
}
/*
* Wrapper function helps limiting calling complexity of supplying additional
* PFN_POWER_LOCK_ACQUIRE argument (required by _PVRSRVDeviceIdleRequestKM)
*/
inline PVRSRV_ERROR PVRSRVDeviceIdleRequestKM(PPVRSRV_DEVICE_NODE psDeviceNode,
PFN_SYS_DEV_IS_DEFAULT_STATE_OFF pfnIsDefaultStateOff,
IMG_BOOL bDeviceOffPermitted)
{
return _PVRSRVDeviceIdleRequestKM(psDeviceNode,
pfnIsDefaultStateOff,
bDeviceOffPermitted,
PVRSRVPowerLock);
}
/*!
******************************************************************************
@Function PVRSRVDeviceIdleCancelRequestKM
@Description
Perform device-specific processing required to cancel the forced idle state on the device, returning to normal operation.
@Input psDeviceNode : Device node
@Return PVRSRV_ERROR
******************************************************************************/
PVRSRV_ERROR PVRSRVDeviceIdleCancelRequestKM(PPVRSRV_DEVICE_NODE psDeviceNode)
{
PVRSRV_POWER_DEV *psPowerDev = psDeviceNode->psPowerDev;
if (psPowerDev && psPowerDev->pfnForcedIdleCancelRequest)
{
return psPowerDev->pfnForcedIdleCancelRequest(psPowerDev->hDevCookie);
}
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVDevicePrePowerStateKM
@Description
Perform device-specific processing required before a power transition
@Input psPowerDevice : Power device
@Input eNewPowerState : New power state
@Input bForced : TRUE if the transition should not fail (e.g. OS request)
@Return PVRSRV_ERROR
******************************************************************************/
static
PVRSRV_ERROR PVRSRVDevicePrePowerStateKM(PVRSRV_POWER_DEV *psPowerDevice,
PVRSRV_DEV_POWER_STATE eNewPowerState,
IMG_BOOL bForced)
{
IMG_UINT64 ui64SysTimer1 = 0;
IMG_UINT64 ui64SysTimer2 = 0;
IMG_UINT64 ui64DevTimer1 = 0;
IMG_UINT64 ui64DevTimer2 = 0;
PVRSRV_ERROR eError;
PVR_ASSERT(eNewPowerState != PVRSRV_DEV_POWER_STATE_DEFAULT);
if (psPowerDevice->pfnDevicePrePower != NULL)
{
ui64DevTimer1 = PVRSRVProcessStatsGetTimeNs();
/* Call the device's power callback. */
eError = psPowerDevice->pfnDevicePrePower(psPowerDevice->hDevCookie,
eNewPowerState,
psPowerDevice->eCurrentPowerState,
bForced);
ui64DevTimer2 = PVRSRVProcessStatsGetTimeNs();
if (eError != PVRSRV_OK)
{
return eError;
}
}
/* Do any required system-layer processing. */
if (psPowerDevice->pfnSystemPrePower != NULL)
{
ui64SysTimer1 = PVRSRVProcessStatsGetTimeNs();
eError = psPowerDevice->pfnSystemPrePower(psPowerDevice->hSysData,
eNewPowerState,
psPowerDevice->eCurrentPowerState,
bForced);
ui64SysTimer2 = PVRSRVProcessStatsGetTimeNs();
if (eError != PVRSRV_OK)
{
return eError;
}
}
InsertPowerTimeStatistic(ui64SysTimer1, ui64SysTimer2,
ui64DevTimer1, ui64DevTimer2,
bForced,
eNewPowerState == PVRSRV_DEV_POWER_STATE_ON,
IMG_TRUE);
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVDevicePostPowerStateKM
@Description
Perform device-specific processing required after a power transition
@Input psPowerDevice : Power device
@Input eNewPowerState : New power state
@Input bForced : TRUE if the transition should not fail (e.g. OS request)
@Return PVRSRV_ERROR
******************************************************************************/
static
PVRSRV_ERROR PVRSRVDevicePostPowerStateKM(PVRSRV_POWER_DEV *psPowerDevice,
PVRSRV_DEV_POWER_STATE eNewPowerState,
IMG_BOOL bForced)
{
IMG_UINT64 ui64SysTimer1 = 0;
IMG_UINT64 ui64SysTimer2 = 0;
IMG_UINT64 ui64DevTimer1 = 0;
IMG_UINT64 ui64DevTimer2 = 0;
PVRSRV_ERROR eError;
PVR_ASSERT(eNewPowerState != PVRSRV_DEV_POWER_STATE_DEFAULT);
/* Do any required system-layer processing. */
if (psPowerDevice->pfnSystemPostPower != NULL)
{
ui64SysTimer1 = PVRSRVProcessStatsGetTimeNs();
eError = psPowerDevice->pfnSystemPostPower(psPowerDevice->hSysData,
eNewPowerState,
psPowerDevice->eCurrentPowerState,
bForced);
ui64SysTimer2 = PVRSRVProcessStatsGetTimeNs();
if (eError != PVRSRV_OK)
{
return eError;
}
}
if (psPowerDevice->pfnDevicePostPower != NULL)
{
ui64DevTimer1 = PVRSRVProcessStatsGetTimeNs();
/* Call the device's power callback. */
eError = psPowerDevice->pfnDevicePostPower(psPowerDevice->hDevCookie,
eNewPowerState,
psPowerDevice->eCurrentPowerState,
bForced);
ui64DevTimer2 = PVRSRVProcessStatsGetTimeNs();
if (eError != PVRSRV_OK)
{
return eError;
}
}
InsertPowerTimeStatistic(ui64SysTimer1, ui64SysTimer2,
ui64DevTimer1, ui64DevTimer2,
bForced,
eNewPowerState == PVRSRV_DEV_POWER_STATE_ON,
IMG_FALSE);
psPowerDevice->eCurrentPowerState = eNewPowerState;
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVSetDevicePowerStateKM
@Description Set the Device into a new state
@Input psDeviceNode : Device node
@Input eNewPowerState : New power state
@Input bForced : TRUE if the transition should not fail (e.g. OS request)
@Return PVRSRV_ERROR
******************************************************************************/
PVRSRV_ERROR PVRSRVSetDevicePowerStateKM(PPVRSRV_DEVICE_NODE psDeviceNode,
PVRSRV_DEV_POWER_STATE eNewPowerState,
IMG_BOOL bForced)
{
PVRSRV_ERROR eError;
PVRSRV_DATA* psPVRSRVData = PVRSRVGetPVRSRVData();
PVRSRV_POWER_DEV *psPowerDevice;
psPowerDevice = psDeviceNode->psPowerDev;
if (!psPowerDevice)
{
return PVRSRV_OK;
}
if (eNewPowerState == PVRSRV_DEV_POWER_STATE_DEFAULT)
{
eNewPowerState = psPowerDevice->eDefaultPowerState;
}
if (psPowerDevice->eCurrentPowerState != eNewPowerState)
{
eError = PVRSRVDevicePrePowerStateKM(psPowerDevice,
eNewPowerState,
bForced);
if (eError != PVRSRV_OK)
{
goto ErrorExit;
}
eError = PVRSRVDevicePostPowerStateKM(psPowerDevice,
eNewPowerState,
bForced);
if (eError != PVRSRV_OK)
{
goto ErrorExit;
}
/* Signal Device Watchdog Thread about power mode change. */
if (eNewPowerState == PVRSRV_DEV_POWER_STATE_ON)
{
psPVRSRVData->ui32DevicesWatchdogPwrTrans++;
#if !defined(PVRSRV_SERVER_THREADS_INDEFINITE_SLEEP)
if (psPVRSRVData->ui32DevicesWatchdogTimeout == DEVICES_WATCHDOG_POWER_OFF_SLEEP_TIMEOUT)
#endif
{
eError = OSEventObjectSignal(psPVRSRVData->hDevicesWatchdogEvObj);
PVR_LOG_IF_ERROR(eError, "OSEventObjectSignal");
}
}
#if defined(PVRSRV_SERVER_THREADS_INDEFINITE_SLEEP)
else if (eNewPowerState == PVRSRV_DEV_POWER_STATE_OFF)
{
/* signal watchdog thread and give it a chance to switch to
* longer / infinite wait time */
eError = OSEventObjectSignal(psPVRSRVData->hDevicesWatchdogEvObj);
PVR_LOG_IF_ERROR(eError, "OSEventObjectSignal");
}
#endif /* defined(PVRSRV_SERVER_THREADS_INDEFINITE_SLEEP) */
}
return PVRSRV_OK;
ErrorExit:
if (eError == PVRSRV_ERROR_DEVICE_POWER_CHANGE_DENIED)
{
PVR_DPF((PVR_DBG_MESSAGE,
"%s: Transition to %d was denied, Forced=%d",
__func__, eNewPowerState, bForced));
}
else if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_WARNING,
"%s: Transition to %d FAILED (%s)",
__func__, eNewPowerState, PVRSRVGetErrorString(eError)));
}
return eError;
}
/*************************************************************************/ /*!
@Function PVRSRVSetDeviceSystemPowerState
@Description Set the device into a new power state based on the systems power
state
@Input psDeviceNode Device node
@Input eNewSysPowerState New system power state
@Return PVRSRV_ERROR PVRSRV_OK on success or an error otherwise
*/ /**************************************************************************/
PVRSRV_ERROR PVRSRVSetDeviceSystemPowerState(PPVRSRV_DEVICE_NODE psDeviceNode,
PVRSRV_SYS_POWER_STATE eNewSysPowerState)
{
PVRSRV_ERROR eError;
IMG_UINT uiStage = 0;
PVRSRV_DEV_POWER_STATE eNewDevicePowerState =
_IsSystemStatePowered(eNewSysPowerState)? PVRSRV_DEV_POWER_STATE_DEFAULT : PVRSRV_DEV_POWER_STATE_OFF;
/* If setting devices to default state, force idle all devices whose default state is off */
PFN_SYS_DEV_IS_DEFAULT_STATE_OFF pfnIsDefaultStateOff =
(eNewDevicePowerState == PVRSRV_DEV_POWER_STATE_DEFAULT) ? PVRSRVDeviceIsDefaultStateOFF : NULL;
/* require a proper power state */
if (eNewSysPowerState == PVRSRV_SYS_POWER_STATE_Unspecified)
{
return PVRSRV_ERROR_INVALID_PARAMS;
}
/* Prevent simultaneous SetPowerStateKM calls */
_PVRSRVForcedPowerLock(psDeviceNode);
/* no power transition requested, so do nothing */
if (eNewSysPowerState == psDeviceNode->eCurrentSysPowerState)
{
PVRSRVPowerUnlock(psDeviceNode);
return PVRSRV_OK;
}
eError = _PVRSRVDeviceIdleRequestKM(psDeviceNode, pfnIsDefaultStateOff,
IMG_TRUE, _PVRSRVForcedPowerLock);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR, "%s: Forced idle request failure (%s)",
__func__, PVRSRVGetErrorString(eError)));
uiStage++;
goto ErrorExit;
}
eError = PVRSRVSetDevicePowerStateKM(psDeviceNode, eNewDevicePowerState,
IMG_TRUE);
if (eError != PVRSRV_OK)
{
uiStage++;
goto ErrorExit;
}
psDeviceNode->eCurrentSysPowerState = eNewSysPowerState;
PVRSRVPowerUnlock(psDeviceNode);
return PVRSRV_OK;
ErrorExit:
PVRSRVPowerUnlock(psDeviceNode);
PVR_DPF((PVR_DBG_ERROR,
"%s: Transition from %d to %d FAILED (%s) at stage %u. Dumping debug info.",
__func__, psDeviceNode->eCurrentSysPowerState, eNewSysPowerState,
PVRSRVGetErrorString(eError), uiStage));
PVRSRVDebugRequest(psDeviceNode, DEBUG_REQUEST_VERBOSITY_MAX, NULL, NULL);
return eError;
}
PVRSRV_ERROR PVRSRVRegisterPowerDevice(PPVRSRV_DEVICE_NODE psDeviceNode,
PFN_PRE_POWER pfnDevicePrePower,
PFN_POST_POWER pfnDevicePostPower,
PFN_SYS_DEV_PRE_POWER pfnSystemPrePower,
PFN_SYS_DEV_POST_POWER pfnSystemPostPower,
PFN_PRE_CLOCKSPEED_CHANGE pfnPreClockSpeedChange,
PFN_POST_CLOCKSPEED_CHANGE pfnPostClockSpeedChange,
PFN_FORCED_IDLE_REQUEST pfnForcedIdleRequest,
PFN_FORCED_IDLE_CANCEL_REQUEST pfnForcedIdleCancelRequest,
PFN_DUST_COUNT_REQUEST pfnDustCountRequest,
IMG_HANDLE hDevCookie,
PVRSRV_DEV_POWER_STATE eCurrentPowerState,
PVRSRV_DEV_POWER_STATE eDefaultPowerState)
{
PVRSRV_POWER_DEV *psPowerDevice;
PVR_ASSERT(!psDeviceNode->psPowerDev);
PVR_ASSERT(eCurrentPowerState != PVRSRV_DEV_POWER_STATE_DEFAULT);
PVR_ASSERT(eDefaultPowerState != PVRSRV_DEV_POWER_STATE_DEFAULT);
psPowerDevice = OSAllocMem(sizeof(PVRSRV_POWER_DEV));
if (psPowerDevice == NULL)
{
PVR_DPF((PVR_DBG_ERROR,
"%s: Failed to alloc PVRSRV_POWER_DEV", __func__));
return PVRSRV_ERROR_OUT_OF_MEMORY;
}
/* setup device for power manager */
psPowerDevice->pfnDevicePrePower = pfnDevicePrePower;
psPowerDevice->pfnDevicePostPower = pfnDevicePostPower;
psPowerDevice->pfnSystemPrePower = pfnSystemPrePower;
psPowerDevice->pfnSystemPostPower = pfnSystemPostPower;
psPowerDevice->pfnPreClockSpeedChange = pfnPreClockSpeedChange;
psPowerDevice->pfnPostClockSpeedChange = pfnPostClockSpeedChange;
psPowerDevice->pfnForcedIdleRequest = pfnForcedIdleRequest;
psPowerDevice->pfnForcedIdleCancelRequest = pfnForcedIdleCancelRequest;
psPowerDevice->pfnDustCountRequest = pfnDustCountRequest;
psPowerDevice->hSysData = psDeviceNode->psDevConfig->hSysData;
psPowerDevice->hDevCookie = hDevCookie;
psPowerDevice->eCurrentPowerState = eCurrentPowerState;
psPowerDevice->eDefaultPowerState = eDefaultPowerState;
psDeviceNode->psPowerDev = psPowerDevice;
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVRemovePowerDevice
@Description
Removes device from power management register. Device is located by Device Index
@Input psDeviceNode : Device node
@Return PVRSRV_ERROR
******************************************************************************/
PVRSRV_ERROR PVRSRVRemovePowerDevice(PPVRSRV_DEVICE_NODE psDeviceNode)
{
if (psDeviceNode->psPowerDev)
{
OSFreeMem(psDeviceNode->psPowerDev);
psDeviceNode->psPowerDev = NULL;
}
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVGetDevicePowerState
@Description
Return the device power state
@Input psDeviceNode : Device node
@Output psPowerState : Current power state
@Return PVRSRV_ERROR_UNKNOWN_POWER_STATE if device could not be found. PVRSRV_OK otherwise.
******************************************************************************/
PVRSRV_ERROR PVRSRVGetDevicePowerState(PCPVRSRV_DEVICE_NODE psDeviceNode,
PPVRSRV_DEV_POWER_STATE pePowerState)
{
PVRSRV_POWER_DEV *psPowerDevice;
psPowerDevice = psDeviceNode->psPowerDev;
if (psPowerDevice == NULL)
{
return PVRSRV_ERROR_UNKNOWN_POWER_STATE;
}
*pePowerState = psPowerDevice->eCurrentPowerState;
return PVRSRV_OK;
}
/*!
******************************************************************************
@Function PVRSRVIsDevicePowered
@Description
Whether the device is powered, for the purposes of lockup detection.
@Input psDeviceNode : Device node
@Return IMG_BOOL
******************************************************************************/
IMG_BOOL PVRSRVIsDevicePowered(PPVRSRV_DEVICE_NODE psDeviceNode)
{
PVRSRV_DEV_POWER_STATE ePowerState;
if (OSLockIsLocked(psDeviceNode->hPowerLock))
{
return IMG_FALSE;
}
if (PVRSRVGetDevicePowerState(psDeviceNode, &ePowerState) != PVRSRV_OK)
{
return IMG_FALSE;
}
return (ePowerState == PVRSRV_DEV_POWER_STATE_ON);
}
/**************************************************************************/ /*!
@Function PVRSRVDevicePreClockSpeedChange
@Description This function is called before a voltage/frequency change is
made to the GPU HW. It informs the host driver of the intention
to make a DVFS change. If allows the host driver to idle
the GPU and begin a hold off period from starting new work
on the GPU.
When this call succeeds the caller *must* call
PVRSRVDevicePostClockSpeedChange() to end the hold off period
to allow new work to be submitted to the GPU.
Called form system layer or OS layer implementation that
is responsible for triggering a GPU DVFS transition.
@Input psDeviceNode pointer to the device affected by DVFS transition.
@Input bIdleDevice when True, the driver will wait for the GPU to
reach an idle state before the call returns.
@Input pvInfo unused
@Return PVRSRV_OK on success, power lock acquired and held on exit,
GPU idle.
PVRSRV_ERROR on failure, power lock not held on exit, do not
call PVRSRVDevicePostClockSpeedChange().
*/ /**************************************************************************/
PVRSRV_ERROR
PVRSRVDevicePreClockSpeedChange(PPVRSRV_DEVICE_NODE psDeviceNode,
IMG_BOOL bIdleDevice,
void* pvInfo)
{
PVRSRV_ERROR eError = PVRSRV_OK;
PVRSRV_POWER_DEV *psPowerDevice;
IMG_UINT64 ui64StartTimer, ui64StopTimer;
PVR_UNREFERENCED_PARAMETER(pvInfo);
ui64StartTimer = PVRSRVProcessStatsGetTimeUs();
/* This mutex is released in PVRSRVDevicePostClockSpeedChange. */
eError = PVRSRVPowerLock(psDeviceNode);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR,
"%s: failed to acquire lock (%s)",
__func__, PVRSRVGetErrorString(eError)));
return eError;
}
psPowerDevice = psDeviceNode->psPowerDev;
if (psPowerDevice)
{
if ((psPowerDevice->eCurrentPowerState == PVRSRV_DEV_POWER_STATE_ON) && bIdleDevice)
{
/* We can change the clock speed if the device is either IDLE or OFF */
eError = PVRSRVDeviceIdleRequestKM(psDeviceNode, NULL, IMG_TRUE);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR, "%s: Forced idle request failed (%s)",
__func__, PVRSRVGetErrorString(eError)));
if (eError != PVRSRV_ERROR_PWLOCK_RELEASED_REACQ_FAILED)
{
PVRSRVPowerUnlock(psDeviceNode);
}
return eError;
}
}
eError = psPowerDevice->pfnPreClockSpeedChange(psPowerDevice->hDevCookie,
psPowerDevice->eCurrentPowerState);
}
ui64StopTimer = PVRSRVProcessStatsGetTimeUs();
InsertPowerTimeStatisticExtraPre(ui64StartTimer, ui64StopTimer);
return eError;
}
/**************************************************************************/ /*!
@Function PVRSRVDevicePostClockSpeedChange
@Description This function is called after a voltage/frequency change has
been made to the GPU HW following a call to
PVRSRVDevicePreClockSpeedChange().
Before calling this function the caller must ensure the system
data RGX_DATA->RGX_TIMING_INFORMATION->ui32CoreClockSpeed has
been updated with the new frequency set, measured in Hz.
The function informs the host driver that the DVFS change has
completed. The driver will end the work hold off period, cancel
the device idle period and update its time data records.
When this call returns work submissions are unblocked and
are submitted to the GPU as normal.
This function *must* not be called if the preceding call to
PVRSRVDevicePreClockSpeedChange() failed.
Called form system layer or OS layer implementation that
is responsible for triggering a GPU DVFS transition.
@Input psDeviceNode pointer to the device affected by DVFS transition.
@Input bIdleDevice when True, the driver will cancel the GPU
device idle state before the call returns. Value
given must match that used in the call to
PVRSRVDevicePreClockSpeedChange() otherwise
undefined behaviour will result.
@Input pvInfo unused
@Return void power lock released, no longer held on exit.
*/ /**************************************************************************/
void
PVRSRVDevicePostClockSpeedChange(PPVRSRV_DEVICE_NODE psDeviceNode,
IMG_BOOL bIdleDevice,
void* pvInfo)
{
PVRSRV_ERROR eError;
PVRSRV_POWER_DEV *psPowerDevice;
IMG_UINT64 ui64StartTimer, ui64StopTimer;
PVR_UNREFERENCED_PARAMETER(pvInfo);
ui64StartTimer = PVRSRVProcessStatsGetTimeUs();
psPowerDevice = psDeviceNode->psPowerDev;
if (psPowerDevice)
{
eError = psPowerDevice->pfnPostClockSpeedChange(psPowerDevice->hDevCookie,
psPowerDevice->eCurrentPowerState);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR, "%s: Device %p failed (%s)",
__func__, psDeviceNode, PVRSRVGetErrorString(eError)));
}
if ((psPowerDevice->eCurrentPowerState == PVRSRV_DEV_POWER_STATE_ON) && bIdleDevice)
{
eError = PVRSRVDeviceIdleCancelRequestKM(psDeviceNode);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR,
"%s: Failed to cancel forced IDLE.", __func__));
}
}
}
/* This mutex was acquired in PVRSRVDevicePreClockSpeedChange. */
PVRSRVPowerUnlock(psDeviceNode);
OSAtomicIncrement(&psDeviceNode->iNumClockSpeedChanges);
ui64StopTimer = PVRSRVProcessStatsGetTimeUs();
InsertPowerTimeStatisticExtraPost(ui64StartTimer, ui64StopTimer);
}
/*!
******************************************************************************
@Function PVRSRVDeviceDustCountChange
@Description
Request from system layer that a dust count change is requested.
@Input psDeviceNode : Device node
@Input ui32DustCount : dust count to be set
@Return PVRSRV_ERROR
******************************************************************************/
PVRSRV_ERROR PVRSRVDeviceDustCountChange(PPVRSRV_DEVICE_NODE psDeviceNode,
IMG_UINT32 ui32DustCount)
{
PVRSRV_ERROR eError = PVRSRV_OK;
PVRSRV_POWER_DEV *psPowerDevice;
psPowerDevice = psDeviceNode->psPowerDev;
if (psPowerDevice)
{
PVRSRV_DEV_POWER_STATE eDevicePowerState;
eError = PVRSRVPowerLock(psDeviceNode);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR, "%s: failed to acquire lock (%s)",
__func__, PVRSRVGetErrorString(eError)));
return eError;
}
eDevicePowerState = psPowerDevice->eCurrentPowerState;
if (eDevicePowerState == PVRSRV_DEV_POWER_STATE_ON)
{
/* Device must be idle to change dust count */
eError = PVRSRVDeviceIdleRequestKM(psDeviceNode, NULL, IMG_FALSE);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR, "%s: Forced idle request failure (%s)",
__func__, PVRSRVGetErrorString(eError)));
if (eError == PVRSRV_ERROR_PWLOCK_RELEASED_REACQ_FAILED)
{
goto ErrorExit;
}
goto ErrorUnlockAndExit;
}
}
if (psPowerDevice->pfnDustCountRequest != NULL)
{
PVRSRV_ERROR eError2 = psPowerDevice->pfnDustCountRequest(psPowerDevice->hDevCookie, ui32DustCount);
if (eError2 != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR, "%s: Device %p failed (%s)",
__func__, psDeviceNode,
PVRSRVGetErrorString(eError)));
}
}
if (eDevicePowerState == PVRSRV_DEV_POWER_STATE_ON)
{
eError = PVRSRVDeviceIdleCancelRequestKM(psDeviceNode);
if (eError != PVRSRV_OK)
{
PVR_DPF((PVR_DBG_ERROR,
"%s: Failed to cancel forced IDLE.", __func__));
goto ErrorUnlockAndExit;
}
}
PVRSRVPowerUnlock(psDeviceNode);
}
return eError;
ErrorUnlockAndExit:
PVRSRVPowerUnlock(psDeviceNode);
ErrorExit:
return eError;
}
/******************************************************************************
End of file (power.c)
******************************************************************************/