2015-04-20 17:31:14 -04:00
/*
* Copyright 2014 Advanced Micro Devices , Inc .
*
* 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 COPYRIGHT HOLDER ( S ) OR AUTHOR ( S ) 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 .
*
* Authors : Christian König < christian . koenig @ amd . com >
*/
# include <linux/firmware.h>
# include <drm/drmP.h>
# include "amdgpu.h"
# include "amdgpu_uvd.h"
# include "vid.h"
# include "uvd/uvd_5_0_d.h"
# include "uvd/uvd_5_0_sh_mask.h"
# include "oss/oss_2_0_d.h"
# include "oss/oss_2_0_sh_mask.h"
2016-06-22 14:25:54 +02:00
# include "bif/bif_5_0_d.h"
2016-03-23 13:14:31 -04:00
# include "vi.h"
2016-10-26 17:04:33 +08:00
# include "smu/smu_7_1_2_d.h"
# include "smu/smu_7_1_2_sh_mask.h"
2015-04-20 17:31:14 -04:00
static void uvd_v5_0_set_ring_funcs ( struct amdgpu_device * adev ) ;
static void uvd_v5_0_set_irq_funcs ( struct amdgpu_device * adev ) ;
static int uvd_v5_0_start ( struct amdgpu_device * adev ) ;
static void uvd_v5_0_stop ( struct amdgpu_device * adev ) ;
2016-11-08 20:43:50 +08:00
static int uvd_v5_0_set_clockgating_state ( void * handle ,
enum amd_clockgating_state state ) ;
static void uvd_v5_0_enable_mgcg ( struct amdgpu_device * adev ,
bool enable ) ;
2015-04-20 17:31:14 -04:00
/**
* uvd_v5_0_ring_get_rptr - get read pointer
*
* @ ring : amdgpu_ring pointer
*
* Returns the current hardware read pointer
*/
2016-03-12 09:32:30 +08:00
static uint64_t uvd_v5_0_ring_get_rptr ( struct amdgpu_ring * ring )
2015-04-20 17:31:14 -04:00
{
struct amdgpu_device * adev = ring - > adev ;
return RREG32 ( mmUVD_RBC_RB_RPTR ) ;
}
/**
* uvd_v5_0_ring_get_wptr - get write pointer
*
* @ ring : amdgpu_ring pointer
*
* Returns the current hardware write pointer
*/
2016-03-12 09:32:30 +08:00
static uint64_t uvd_v5_0_ring_get_wptr ( struct amdgpu_ring * ring )
2015-04-20 17:31:14 -04:00
{
struct amdgpu_device * adev = ring - > adev ;
return RREG32 ( mmUVD_RBC_RB_WPTR ) ;
}
/**
* uvd_v5_0_ring_set_wptr - set write pointer
*
* @ ring : amdgpu_ring pointer
*
* Commits the write pointer to the hardware
*/
static void uvd_v5_0_ring_set_wptr ( struct amdgpu_ring * ring )
{
struct amdgpu_device * adev = ring - > adev ;
2016-03-12 09:32:30 +08:00
WREG32 ( mmUVD_RBC_RB_WPTR , lower_32_bits ( ring - > wptr ) ) ;
2015-04-20 17:31:14 -04:00
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_early_init ( void * handle )
2015-04-20 17:31:14 -04:00
{
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2018-05-15 14:25:46 -05:00
adev - > uvd . num_uvd_inst = 1 ;
2015-05-22 14:39:35 -04:00
2015-04-20 17:31:14 -04:00
uvd_v5_0_set_ring_funcs ( adev ) ;
uvd_v5_0_set_irq_funcs ( adev ) ;
return 0 ;
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_sw_init ( void * handle )
2015-04-20 17:31:14 -04:00
{
struct amdgpu_ring * ring ;
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
int r ;
/* UVD TRAP */
2018-05-15 14:25:46 -05:00
r = amdgpu_irq_add_id ( adev , AMDGPU_IH_CLIENTID_LEGACY , 124 , & adev - > uvd . inst - > irq ) ;
2015-04-20 17:31:14 -04:00
if ( r )
return r ;
r = amdgpu_uvd_sw_init ( adev ) ;
if ( r )
return r ;
r = amdgpu_uvd_resume ( adev ) ;
if ( r )
return r ;
2018-05-15 14:25:46 -05:00
ring = & adev - > uvd . inst - > ring ;
2015-04-20 17:31:14 -04:00
sprintf ( ring - > name , " uvd " ) ;
2018-05-15 14:25:46 -05:00
r = amdgpu_ring_init ( adev , ring , 512 , & adev - > uvd . inst - > irq , 0 ) ;
2015-04-20 17:31:14 -04:00
return r ;
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_sw_fini ( void * handle )
2015-04-20 17:31:14 -04:00
{
int r ;
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
r = amdgpu_uvd_suspend ( adev ) ;
if ( r )
return r ;
2017-03-30 15:45:52 +08:00
return amdgpu_uvd_sw_fini ( adev ) ;
2015-04-20 17:31:14 -04:00
}
/**
* uvd_v5_0_hw_init - start and test UVD block
*
* @ adev : amdgpu_device pointer
*
* Initialize the hardware , boot up the VCPU and do some testing
*/
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_hw_init ( void * handle )
2015-04-20 17:31:14 -04:00
{
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2018-05-15 14:25:46 -05:00
struct amdgpu_ring * ring = & adev - > uvd . inst - > ring ;
2015-04-20 17:31:14 -04:00
uint32_t tmp ;
int r ;
2017-01-20 17:46:34 +08:00
amdgpu_asic_set_uvd_clocks ( adev , 10000 , 10000 ) ;
uvd_v5_0_set_clockgating_state ( adev , AMD_CG_STATE_UNGATE ) ;
uvd_v5_0_enable_mgcg ( adev , true ) ;
2015-04-20 17:31:14 -04:00
ring - > ready = true ;
r = amdgpu_ring_test_ring ( ring ) ;
if ( r ) {
ring - > ready = false ;
goto done ;
}
2016-01-21 11:28:53 +01:00
r = amdgpu_ring_alloc ( ring , 10 ) ;
2015-04-20 17:31:14 -04:00
if ( r ) {
DRM_ERROR ( " amdgpu: ring failed to lock UVD ring (%d). \n " , r ) ;
goto done ;
}
tmp = PACKET0 ( mmUVD_SEMA_WAIT_FAULT_TIMEOUT_CNTL , 0 ) ;
amdgpu_ring_write ( ring , tmp ) ;
amdgpu_ring_write ( ring , 0xFFFFF ) ;
tmp = PACKET0 ( mmUVD_SEMA_WAIT_INCOMPLETE_TIMEOUT_CNTL , 0 ) ;
amdgpu_ring_write ( ring , tmp ) ;
amdgpu_ring_write ( ring , 0xFFFFF ) ;
tmp = PACKET0 ( mmUVD_SEMA_SIGNAL_INCOMPLETE_TIMEOUT_CNTL , 0 ) ;
amdgpu_ring_write ( ring , tmp ) ;
amdgpu_ring_write ( ring , 0xFFFFF ) ;
/* Clear timeout status bits */
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_SEMA_TIMEOUT_STATUS , 0 ) ) ;
amdgpu_ring_write ( ring , 0x8 ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_SEMA_CNTL , 0 ) ) ;
amdgpu_ring_write ( ring , 3 ) ;
2016-01-21 11:28:53 +01:00
amdgpu_ring_commit ( ring ) ;
2017-01-20 17:46:34 +08:00
2015-04-20 17:31:14 -04:00
done :
if ( ! r )
DRM_INFO ( " UVD initialized successfully. \n " ) ;
return r ;
2017-01-20 17:46:34 +08:00
2015-04-20 17:31:14 -04:00
}
/**
* uvd_v5_0_hw_fini - stop the hardware block
*
* @ adev : amdgpu_device pointer
*
* Stop the UVD block , mark ring as not ready any more
*/
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_hw_fini ( void * handle )
2015-04-20 17:31:14 -04:00
{
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2018-05-15 14:25:46 -05:00
struct amdgpu_ring * ring = & adev - > uvd . inst - > ring ;
2015-04-20 17:31:14 -04:00
2017-01-20 17:46:34 +08:00
if ( RREG32 ( mmUVD_STATUS ) ! = 0 )
uvd_v5_0_stop ( adev ) ;
2015-04-20 17:31:14 -04:00
ring - > ready = false ;
return 0 ;
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_suspend ( void * handle )
2015-04-20 17:31:14 -04:00
{
int r ;
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
2016-04-01 10:36:06 -04:00
r = uvd_v5_0_hw_fini ( adev ) ;
2015-04-20 17:31:14 -04:00
if ( r )
return r ;
2016-11-08 20:43:50 +08:00
uvd_v5_0_set_clockgating_state ( adev , AMD_CG_STATE_GATE ) ;
2015-04-20 17:31:14 -04:00
2017-03-30 15:45:52 +08:00
return amdgpu_uvd_suspend ( adev ) ;
2015-04-20 17:31:14 -04:00
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_resume ( void * handle )
2015-04-20 17:31:14 -04:00
{
int r ;
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
r = amdgpu_uvd_resume ( adev ) ;
if ( r )
return r ;
2017-03-30 15:45:52 +08:00
return uvd_v5_0_hw_init ( adev ) ;
2015-04-20 17:31:14 -04:00
}
/**
* uvd_v5_0_mc_resume - memory controller programming
*
* @ adev : amdgpu_device pointer
*
* Let the UVD memory controller know it ' s offsets
*/
static void uvd_v5_0_mc_resume ( struct amdgpu_device * adev )
{
uint64_t offset ;
uint32_t size ;
/* programm memory controller bits 0-27 */
WREG32 ( mmUVD_LMI_VCPU_CACHE_64BIT_BAR_LOW ,
2018-05-15 14:25:46 -05:00
lower_32_bits ( adev - > uvd . inst - > gpu_addr ) ) ;
2015-04-20 17:31:14 -04:00
WREG32 ( mmUVD_LMI_VCPU_CACHE_64BIT_BAR_HIGH ,
2018-05-15 14:25:46 -05:00
upper_32_bits ( adev - > uvd . inst - > gpu_addr ) ) ;
2015-04-20 17:31:14 -04:00
offset = AMDGPU_UVD_FIRMWARE_OFFSET ;
2017-11-10 19:28:01 +01:00
size = AMDGPU_UVD_FIRMWARE_SIZE ( adev ) ;
2015-04-20 17:31:14 -04:00
WREG32 ( mmUVD_VCPU_CACHE_OFFSET0 , offset > > 3 ) ;
WREG32 ( mmUVD_VCPU_CACHE_SIZE0 , size ) ;
offset + = size ;
2016-04-12 13:46:15 +02:00
size = AMDGPU_UVD_HEAP_SIZE ;
2015-04-20 17:31:14 -04:00
WREG32 ( mmUVD_VCPU_CACHE_OFFSET1 , offset > > 3 ) ;
WREG32 ( mmUVD_VCPU_CACHE_SIZE1 , size ) ;
offset + = size ;
2016-04-12 13:46:15 +02:00
size = AMDGPU_UVD_STACK_SIZE +
( AMDGPU_UVD_SESSION_SIZE * adev - > uvd . max_handles ) ;
2015-04-20 17:31:14 -04:00
WREG32 ( mmUVD_VCPU_CACHE_OFFSET2 , offset > > 3 ) ;
WREG32 ( mmUVD_VCPU_CACHE_SIZE2 , size ) ;
2016-02-12 03:22:34 -05:00
WREG32 ( mmUVD_UDEC_ADDR_CONFIG , adev - > gfx . config . gb_addr_config ) ;
WREG32 ( mmUVD_UDEC_DB_ADDR_CONFIG , adev - > gfx . config . gb_addr_config ) ;
WREG32 ( mmUVD_UDEC_DBW_ADDR_CONFIG , adev - > gfx . config . gb_addr_config ) ;
2015-04-20 17:31:14 -04:00
}
/**
* uvd_v5_0_start - start UVD block
*
* @ adev : amdgpu_device pointer
*
* Setup and start the UVD block
*/
static int uvd_v5_0_start ( struct amdgpu_device * adev )
{
2018-05-15 14:25:46 -05:00
struct amdgpu_ring * ring = & adev - > uvd . inst - > ring ;
2015-04-20 17:31:14 -04:00
uint32_t rb_bufsz , tmp ;
uint32_t lmi_swap_cntl ;
uint32_t mp_swap_cntl ;
int i , j , r ;
/*disable DPG */
WREG32_P ( mmUVD_POWER_STATUS , 0 , ~ ( 1 < < 2 ) ) ;
/* disable byte swapping */
lmi_swap_cntl = 0 ;
mp_swap_cntl = 0 ;
uvd_v5_0_mc_resume ( adev ) ;
/* disable interupt */
WREG32_P ( mmUVD_MASTINT_EN , 0 , ~ ( 1 < < 1 ) ) ;
/* stall UMC and register bus before resetting VCPU */
WREG32_P ( mmUVD_LMI_CTRL2 , 1 < < 8 , ~ ( 1 < < 8 ) ) ;
mdelay ( 1 ) ;
/* put LMI, VCPU, RBC etc... into reset */
WREG32 ( mmUVD_SOFT_RESET , UVD_SOFT_RESET__LMI_SOFT_RESET_MASK |
UVD_SOFT_RESET__VCPU_SOFT_RESET_MASK | UVD_SOFT_RESET__LBSI_SOFT_RESET_MASK |
UVD_SOFT_RESET__RBC_SOFT_RESET_MASK | UVD_SOFT_RESET__CSM_SOFT_RESET_MASK |
UVD_SOFT_RESET__CXW_SOFT_RESET_MASK | UVD_SOFT_RESET__TAP_SOFT_RESET_MASK |
UVD_SOFT_RESET__LMI_UMC_SOFT_RESET_MASK ) ;
mdelay ( 5 ) ;
/* take UVD block out of reset */
WREG32_P ( mmSRBM_SOFT_RESET , 0 , ~ SRBM_SOFT_RESET__SOFT_RESET_UVD_MASK ) ;
mdelay ( 5 ) ;
/* initialize UVD memory controller */
WREG32 ( mmUVD_LMI_CTRL , 0x40 | ( 1 < < 8 ) | ( 1 < < 13 ) |
( 1 < < 21 ) | ( 1 < < 9 ) | ( 1 < < 20 ) ) ;
# ifdef __BIG_ENDIAN
/* swap (8 in 32) RB and IB */
lmi_swap_cntl = 0xa ;
mp_swap_cntl = 0 ;
# endif
WREG32 ( mmUVD_LMI_SWAP_CNTL , lmi_swap_cntl ) ;
WREG32 ( mmUVD_MP_SWAP_CNTL , mp_swap_cntl ) ;
WREG32 ( mmUVD_MPC_SET_MUXA0 , 0x40c2040 ) ;
WREG32 ( mmUVD_MPC_SET_MUXA1 , 0x0 ) ;
WREG32 ( mmUVD_MPC_SET_MUXB0 , 0x40c2040 ) ;
WREG32 ( mmUVD_MPC_SET_MUXB1 , 0x0 ) ;
WREG32 ( mmUVD_MPC_SET_ALU , 0 ) ;
WREG32 ( mmUVD_MPC_SET_MUX , 0x88 ) ;
/* take all subblocks out of reset, except VCPU */
WREG32 ( mmUVD_SOFT_RESET , UVD_SOFT_RESET__VCPU_SOFT_RESET_MASK ) ;
mdelay ( 5 ) ;
/* enable VCPU clock */
WREG32 ( mmUVD_VCPU_CNTL , 1 < < 9 ) ;
/* enable UMC */
WREG32_P ( mmUVD_LMI_CTRL2 , 0 , ~ ( 1 < < 8 ) ) ;
/* boot up the VCPU */
WREG32 ( mmUVD_SOFT_RESET , 0 ) ;
mdelay ( 10 ) ;
for ( i = 0 ; i < 10 ; + + i ) {
uint32_t status ;
for ( j = 0 ; j < 100 ; + + j ) {
status = RREG32 ( mmUVD_STATUS ) ;
if ( status & 2 )
break ;
mdelay ( 10 ) ;
}
r = 0 ;
if ( status & 2 )
break ;
DRM_ERROR ( " UVD not responding, trying to reset the VCPU!!! \n " ) ;
WREG32_P ( mmUVD_SOFT_RESET , UVD_SOFT_RESET__VCPU_SOFT_RESET_MASK ,
~ UVD_SOFT_RESET__VCPU_SOFT_RESET_MASK ) ;
mdelay ( 10 ) ;
WREG32_P ( mmUVD_SOFT_RESET , 0 , ~ UVD_SOFT_RESET__VCPU_SOFT_RESET_MASK ) ;
mdelay ( 10 ) ;
r = - 1 ;
}
if ( r ) {
DRM_ERROR ( " UVD not responding, giving up!!! \n " ) ;
return r ;
}
/* enable master interrupt */
WREG32_P ( mmUVD_MASTINT_EN , 3 < < 1 , ~ ( 3 < < 1 ) ) ;
/* clear the bit 4 of UVD_STATUS */
WREG32_P ( mmUVD_STATUS , 0 , ~ ( 2 < < 1 ) ) ;
rb_bufsz = order_base_2 ( ring - > ring_size ) ;
tmp = 0 ;
tmp = REG_SET_FIELD ( tmp , UVD_RBC_RB_CNTL , RB_BUFSZ , rb_bufsz ) ;
tmp = REG_SET_FIELD ( tmp , UVD_RBC_RB_CNTL , RB_BLKSZ , 1 ) ;
tmp = REG_SET_FIELD ( tmp , UVD_RBC_RB_CNTL , RB_NO_FETCH , 1 ) ;
tmp = REG_SET_FIELD ( tmp , UVD_RBC_RB_CNTL , RB_WPTR_POLL_EN , 0 ) ;
tmp = REG_SET_FIELD ( tmp , UVD_RBC_RB_CNTL , RB_NO_UPDATE , 1 ) ;
tmp = REG_SET_FIELD ( tmp , UVD_RBC_RB_CNTL , RB_RPTR_WR_EN , 1 ) ;
/* force RBC into idle state */
WREG32 ( mmUVD_RBC_RB_CNTL , tmp ) ;
/* set the write pointer delay */
WREG32 ( mmUVD_RBC_RB_WPTR_CNTL , 0 ) ;
/* set the wb address */
WREG32 ( mmUVD_RBC_RB_RPTR_ADDR , ( upper_32_bits ( ring - > gpu_addr ) > > 2 ) ) ;
/* programm the RB_BASE for ring buffer */
WREG32 ( mmUVD_LMI_RBC_RB_64BIT_BAR_LOW ,
lower_32_bits ( ring - > gpu_addr ) ) ;
WREG32 ( mmUVD_LMI_RBC_RB_64BIT_BAR_HIGH ,
upper_32_bits ( ring - > gpu_addr ) ) ;
/* Initialize the ring buffer's read and write pointers */
WREG32 ( mmUVD_RBC_RB_RPTR , 0 ) ;
ring - > wptr = RREG32 ( mmUVD_RBC_RB_RPTR ) ;
2016-03-12 09:32:30 +08:00
WREG32 ( mmUVD_RBC_RB_WPTR , lower_32_bits ( ring - > wptr ) ) ;
2015-04-20 17:31:14 -04:00
WREG32_P ( mmUVD_RBC_RB_CNTL , 0 , ~ UVD_RBC_RB_CNTL__RB_NO_FETCH_MASK ) ;
return 0 ;
}
/**
* uvd_v5_0_stop - stop UVD block
*
* @ adev : amdgpu_device pointer
*
* stop the UVD block
*/
static void uvd_v5_0_stop ( struct amdgpu_device * adev )
{
/* force RBC into idle state */
WREG32 ( mmUVD_RBC_RB_CNTL , 0x11010101 ) ;
/* Stall UMC and register bus before resetting VCPU */
WREG32_P ( mmUVD_LMI_CTRL2 , 1 < < 8 , ~ ( 1 < < 8 ) ) ;
mdelay ( 1 ) ;
/* put VCPU into reset */
WREG32 ( mmUVD_SOFT_RESET , UVD_SOFT_RESET__VCPU_SOFT_RESET_MASK ) ;
mdelay ( 5 ) ;
/* disable VCPU clock */
WREG32 ( mmUVD_VCPU_CNTL , 0x0 ) ;
/* Unstall UMC and register bus */
WREG32_P ( mmUVD_LMI_CTRL2 , 0 , ~ ( 1 < < 8 ) ) ;
2017-01-20 17:46:34 +08:00
WREG32 ( mmUVD_STATUS , 0 ) ;
2015-04-20 17:31:14 -04:00
}
/**
* uvd_v5_0_ring_emit_fence - emit an fence & trap command
*
* @ ring : amdgpu_ring pointer
* @ fence : fence to emit
*
* Write a fence and a trap command to the ring .
*/
static void uvd_v5_0_ring_emit_fence ( struct amdgpu_ring * ring , u64 addr , u64 seq ,
2015-06-01 14:35:03 +08:00
unsigned flags )
2015-04-20 17:31:14 -04:00
{
2015-06-01 14:35:03 +08:00
WARN_ON ( flags & AMDGPU_FENCE_FLAG_64BIT ) ;
2015-04-20 17:31:14 -04:00
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_CONTEXT_ID , 0 ) ) ;
amdgpu_ring_write ( ring , seq ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_GPCOM_VCPU_DATA0 , 0 ) ) ;
amdgpu_ring_write ( ring , addr & 0xffffffff ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_GPCOM_VCPU_DATA1 , 0 ) ) ;
amdgpu_ring_write ( ring , upper_32_bits ( addr ) & 0xff ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_GPCOM_VCPU_CMD , 0 ) ) ;
amdgpu_ring_write ( ring , 0 ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_GPCOM_VCPU_DATA0 , 0 ) ) ;
amdgpu_ring_write ( ring , 0 ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_GPCOM_VCPU_DATA1 , 0 ) ) ;
amdgpu_ring_write ( ring , 0 ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_GPCOM_VCPU_CMD , 0 ) ) ;
amdgpu_ring_write ( ring , 2 ) ;
}
/**
* uvd_v5_0_ring_test_ring - register write test
*
* @ ring : amdgpu_ring pointer
*
* Test if we can successfully write to the context register
*/
static int uvd_v5_0_ring_test_ring ( struct amdgpu_ring * ring )
{
struct amdgpu_device * adev = ring - > adev ;
uint32_t tmp = 0 ;
unsigned i ;
int r ;
WREG32 ( mmUVD_CONTEXT_ID , 0xCAFEDEAD ) ;
2016-01-21 11:28:53 +01:00
r = amdgpu_ring_alloc ( ring , 3 ) ;
2015-04-20 17:31:14 -04:00
if ( r ) {
DRM_ERROR ( " amdgpu: cp failed to lock ring %d (%d). \n " ,
ring - > idx , r ) ;
return r ;
}
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_CONTEXT_ID , 0 ) ) ;
amdgpu_ring_write ( ring , 0xDEADBEEF ) ;
2016-01-21 11:28:53 +01:00
amdgpu_ring_commit ( ring ) ;
2015-04-20 17:31:14 -04:00
for ( i = 0 ; i < adev - > usec_timeout ; i + + ) {
tmp = RREG32 ( mmUVD_CONTEXT_ID ) ;
if ( tmp = = 0xDEADBEEF )
break ;
DRM_UDELAY ( 1 ) ;
}
if ( i < adev - > usec_timeout ) {
2017-10-26 09:30:38 +08:00
DRM_DEBUG ( " ring test on %d succeeded in %d usecs \n " ,
2015-04-20 17:31:14 -04:00
ring - > idx , i ) ;
} else {
DRM_ERROR ( " amdgpu: ring %d test failed (0x%08X) \n " ,
ring - > idx , tmp ) ;
r = - EINVAL ;
}
return r ;
}
/**
* uvd_v5_0_ring_emit_ib - execute indirect buffer
*
* @ ring : amdgpu_ring pointer
* @ ib : indirect buffer to execute
*
* Write ring commands to execute the indirect buffer
*/
static void uvd_v5_0_ring_emit_ib ( struct amdgpu_ring * ring ,
2016-05-06 17:50:03 +02:00
struct amdgpu_ib * ib ,
2017-12-18 17:08:25 +01:00
unsigned vmid , bool ctx_switch )
2015-04-20 17:31:14 -04:00
{
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_LMI_RBC_IB_64BIT_BAR_LOW , 0 ) ) ;
amdgpu_ring_write ( ring , lower_32_bits ( ib - > gpu_addr ) ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_LMI_RBC_IB_64BIT_BAR_HIGH , 0 ) ) ;
amdgpu_ring_write ( ring , upper_32_bits ( ib - > gpu_addr ) ) ;
amdgpu_ring_write ( ring , PACKET0 ( mmUVD_RBC_IB_SIZE , 0 ) ) ;
amdgpu_ring_write ( ring , ib - > length_dw ) ;
}
2015-05-22 14:39:35 -04:00
static bool uvd_v5_0_is_idle ( void * handle )
2015-04-20 17:31:14 -04:00
{
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
return ! ( RREG32 ( mmSRBM_STATUS ) & SRBM_STATUS__UVD_BUSY_MASK ) ;
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_wait_for_idle ( void * handle )
2015-04-20 17:31:14 -04:00
{
unsigned i ;
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
for ( i = 0 ; i < adev - > usec_timeout ; i + + ) {
if ( ! ( RREG32 ( mmSRBM_STATUS ) & SRBM_STATUS__UVD_BUSY_MASK ) )
return 0 ;
}
return - ETIMEDOUT ;
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_soft_reset ( void * handle )
2015-04-20 17:31:14 -04:00
{
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2015-04-20 17:31:14 -04:00
uvd_v5_0_stop ( adev ) ;
WREG32_P ( mmSRBM_SOFT_RESET , SRBM_SOFT_RESET__SOFT_RESET_UVD_MASK ,
~ SRBM_SOFT_RESET__SOFT_RESET_UVD_MASK ) ;
mdelay ( 5 ) ;
return uvd_v5_0_start ( adev ) ;
}
static int uvd_v5_0_set_interrupt_state ( struct amdgpu_device * adev ,
struct amdgpu_irq_src * source ,
unsigned type ,
enum amdgpu_interrupt_state state )
{
// TODO
return 0 ;
}
static int uvd_v5_0_process_interrupt ( struct amdgpu_device * adev ,
struct amdgpu_irq_src * source ,
struct amdgpu_iv_entry * entry )
{
DRM_DEBUG ( " IH: UVD TRAP \n " ) ;
2018-05-15 14:25:46 -05:00
amdgpu_fence_process ( & adev - > uvd . inst - > ring ) ;
2015-04-20 17:31:14 -04:00
return 0 ;
}
2016-11-08 20:43:50 +08:00
static void uvd_v5_0_enable_clock_gating ( struct amdgpu_device * adev , bool enable )
2016-03-23 13:14:31 -04:00
{
2016-11-08 20:43:50 +08:00
uint32_t data1 , data3 , suvd_flags ;
2016-03-23 13:14:31 -04:00
data1 = RREG32 ( mmUVD_SUVD_CGC_GATE ) ;
2016-11-08 20:43:50 +08:00
data3 = RREG32 ( mmUVD_CGC_GATE ) ;
2016-03-23 13:14:31 -04:00
suvd_flags = UVD_SUVD_CGC_GATE__SRE_MASK |
UVD_SUVD_CGC_GATE__SIT_MASK |
UVD_SUVD_CGC_GATE__SMP_MASK |
UVD_SUVD_CGC_GATE__SCM_MASK |
UVD_SUVD_CGC_GATE__SDB_MASK ;
2016-11-08 20:43:50 +08:00
if ( enable ) {
2016-11-23 18:09:22 +08:00
data3 | = ( UVD_CGC_GATE__SYS_MASK |
2016-11-08 20:43:50 +08:00
UVD_CGC_GATE__UDEC_MASK |
UVD_CGC_GATE__MPEG2_MASK |
UVD_CGC_GATE__RBC_MASK |
UVD_CGC_GATE__LMI_MC_MASK |
UVD_CGC_GATE__IDCT_MASK |
UVD_CGC_GATE__MPRD_MASK |
UVD_CGC_GATE__MPC_MASK |
UVD_CGC_GATE__LBSI_MASK |
UVD_CGC_GATE__LRBBM_MASK |
UVD_CGC_GATE__UDEC_RE_MASK |
UVD_CGC_GATE__UDEC_CM_MASK |
UVD_CGC_GATE__UDEC_IT_MASK |
UVD_CGC_GATE__UDEC_DB_MASK |
UVD_CGC_GATE__UDEC_MP_MASK |
UVD_CGC_GATE__WCB_MASK |
UVD_CGC_GATE__JPEG_MASK |
UVD_CGC_GATE__SCPU_MASK ) ;
2016-11-23 18:09:22 +08:00
/* only in pg enabled, we can gate clock to vcpu*/
if ( adev - > pg_flags & AMD_PG_SUPPORT_UVD )
data3 | = UVD_CGC_GATE__VCPU_MASK ;
2016-11-08 20:43:50 +08:00
data3 & = ~ UVD_CGC_GATE__REGS_MASK ;
data1 | = suvd_flags ;
} else {
data3 = 0 ;
data1 = 0 ;
}
WREG32 ( mmUVD_SUVD_CGC_GATE , data1 ) ;
WREG32 ( mmUVD_CGC_GATE , data3 ) ;
}
static void uvd_v5_0_set_sw_clock_gating ( struct amdgpu_device * adev )
{
uint32_t data , data2 ;
data = RREG32 ( mmUVD_CGC_CTRL ) ;
data2 = RREG32 ( mmUVD_SUVD_CGC_CTRL ) ;
data & = ~ ( UVD_CGC_CTRL__CLK_OFF_DELAY_MASK |
UVD_CGC_CTRL__CLK_GATE_DLY_TIMER_MASK ) ;
2016-03-23 13:14:31 -04:00
data | = UVD_CGC_CTRL__DYN_CLOCK_MODE_MASK |
( 1 < < REG_FIELD_SHIFT ( UVD_CGC_CTRL , CLK_GATE_DLY_TIMER ) ) |
( 4 < < REG_FIELD_SHIFT ( UVD_CGC_CTRL , CLK_OFF_DELAY ) ) ;
data & = ~ ( UVD_CGC_CTRL__UDEC_RE_MODE_MASK |
UVD_CGC_CTRL__UDEC_CM_MODE_MASK |
UVD_CGC_CTRL__UDEC_IT_MODE_MASK |
UVD_CGC_CTRL__UDEC_DB_MODE_MASK |
UVD_CGC_CTRL__UDEC_MP_MODE_MASK |
UVD_CGC_CTRL__SYS_MODE_MASK |
UVD_CGC_CTRL__UDEC_MODE_MASK |
UVD_CGC_CTRL__MPEG2_MODE_MASK |
UVD_CGC_CTRL__REGS_MODE_MASK |
UVD_CGC_CTRL__RBC_MODE_MASK |
UVD_CGC_CTRL__LMI_MC_MODE_MASK |
UVD_CGC_CTRL__LMI_UMC_MODE_MASK |
UVD_CGC_CTRL__IDCT_MODE_MASK |
UVD_CGC_CTRL__MPRD_MODE_MASK |
UVD_CGC_CTRL__MPC_MODE_MASK |
UVD_CGC_CTRL__LBSI_MODE_MASK |
UVD_CGC_CTRL__LRBBM_MODE_MASK |
UVD_CGC_CTRL__WCB_MODE_MASK |
UVD_CGC_CTRL__VCPU_MODE_MASK |
UVD_CGC_CTRL__JPEG_MODE_MASK |
UVD_CGC_CTRL__SCPU_MODE_MASK ) ;
data2 & = ~ ( UVD_SUVD_CGC_CTRL__SRE_MODE_MASK |
UVD_SUVD_CGC_CTRL__SIT_MODE_MASK |
UVD_SUVD_CGC_CTRL__SMP_MODE_MASK |
UVD_SUVD_CGC_CTRL__SCM_MODE_MASK |
UVD_SUVD_CGC_CTRL__SDB_MODE_MASK ) ;
WREG32 ( mmUVD_CGC_CTRL , data ) ;
WREG32 ( mmUVD_SUVD_CGC_CTRL , data2 ) ;
}
#if 0
static void uvd_v5_0_set_hw_clock_gating ( struct amdgpu_device * adev )
{
uint32_t data , data1 , cgc_flags , suvd_flags ;
data = RREG32 ( mmUVD_CGC_GATE ) ;
data1 = RREG32 ( mmUVD_SUVD_CGC_GATE ) ;
cgc_flags = UVD_CGC_GATE__SYS_MASK |
UVD_CGC_GATE__UDEC_MASK |
UVD_CGC_GATE__MPEG2_MASK |
UVD_CGC_GATE__RBC_MASK |
UVD_CGC_GATE__LMI_MC_MASK |
UVD_CGC_GATE__IDCT_MASK |
UVD_CGC_GATE__MPRD_MASK |
UVD_CGC_GATE__MPC_MASK |
UVD_CGC_GATE__LBSI_MASK |
UVD_CGC_GATE__LRBBM_MASK |
UVD_CGC_GATE__UDEC_RE_MASK |
UVD_CGC_GATE__UDEC_CM_MASK |
UVD_CGC_GATE__UDEC_IT_MASK |
UVD_CGC_GATE__UDEC_DB_MASK |
UVD_CGC_GATE__UDEC_MP_MASK |
UVD_CGC_GATE__WCB_MASK |
UVD_CGC_GATE__VCPU_MASK |
UVD_CGC_GATE__SCPU_MASK ;
suvd_flags = UVD_SUVD_CGC_GATE__SRE_MASK |
UVD_SUVD_CGC_GATE__SIT_MASK |
UVD_SUVD_CGC_GATE__SMP_MASK |
UVD_SUVD_CGC_GATE__SCM_MASK |
UVD_SUVD_CGC_GATE__SDB_MASK ;
data | = cgc_flags ;
data1 | = suvd_flags ;
WREG32 ( mmUVD_CGC_GATE , data ) ;
WREG32 ( mmUVD_SUVD_CGC_GATE , data1 ) ;
}
# endif
2016-11-08 20:43:50 +08:00
static void uvd_v5_0_enable_mgcg ( struct amdgpu_device * adev ,
bool enable )
{
u32 orig , data ;
if ( enable & & ( adev - > cg_flags & AMD_CG_SUPPORT_UVD_MGCG ) ) {
data = RREG32_UVD_CTX ( ixUVD_CGC_MEM_CTRL ) ;
data | = 0xfff ;
WREG32_UVD_CTX ( ixUVD_CGC_MEM_CTRL , data ) ;
orig = data = RREG32 ( mmUVD_CGC_CTRL ) ;
data | = UVD_CGC_CTRL__DYN_CLOCK_MODE_MASK ;
if ( orig ! = data )
WREG32 ( mmUVD_CGC_CTRL , data ) ;
} else {
data = RREG32_UVD_CTX ( ixUVD_CGC_MEM_CTRL ) ;
data & = ~ 0xfff ;
WREG32_UVD_CTX ( ixUVD_CGC_MEM_CTRL , data ) ;
orig = data = RREG32 ( mmUVD_CGC_CTRL ) ;
data & = ~ UVD_CGC_CTRL__DYN_CLOCK_MODE_MASK ;
if ( orig ! = data )
WREG32 ( mmUVD_CGC_CTRL , data ) ;
}
}
2016-10-26 17:04:33 +08:00
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_set_clockgating_state ( void * handle ,
enum amd_clockgating_state state )
2015-04-20 17:31:14 -04:00
{
2016-02-04 23:33:56 -05:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2016-03-23 13:14:31 -04:00
bool enable = ( state = = AMD_CG_STATE_GATE ) ? true : false ;
2016-02-04 23:33:56 -05:00
2016-03-23 13:14:31 -04:00
if ( enable ) {
/* wait for STATUS to clear */
if ( uvd_v5_0_wait_for_idle ( handle ) )
return - EBUSY ;
2016-11-08 20:43:50 +08:00
uvd_v5_0_enable_clock_gating ( adev , true ) ;
2016-03-23 13:14:31 -04:00
/* enable HW gates because UVD is idle */
/* uvd_v5_0_set_hw_clock_gating(adev); */
2016-11-08 20:43:50 +08:00
} else {
uvd_v5_0_enable_clock_gating ( adev , false ) ;
2016-03-23 13:14:31 -04:00
}
2016-11-08 20:43:50 +08:00
uvd_v5_0_set_sw_clock_gating ( adev ) ;
2015-04-20 17:31:14 -04:00
return 0 ;
}
2015-05-22 14:39:35 -04:00
static int uvd_v5_0_set_powergating_state ( void * handle ,
enum amd_powergating_state state )
2015-04-20 17:31:14 -04:00
{
/* This doesn't actually powergate the UVD block.
* That ' s done in the dpm code via the SMC . This
* just re - inits the block as necessary . The actual
* gating still happens in the dpm code . We should
* revisit this when there is a cleaner line between
* the smc and the hw blocks
*/
2015-05-22 14:39:35 -04:00
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
2017-01-05 21:07:02 +08:00
int ret = 0 ;
2015-05-22 14:39:35 -04:00
if ( state = = AMD_PG_STATE_GATE ) {
2015-04-20 17:31:14 -04:00
uvd_v5_0_stop ( adev ) ;
} else {
2017-01-05 21:07:02 +08:00
ret = uvd_v5_0_start ( adev ) ;
if ( ret )
goto out ;
}
out :
return ret ;
}
static void uvd_v5_0_get_clockgating_state ( void * handle , u32 * flags )
{
struct amdgpu_device * adev = ( struct amdgpu_device * ) handle ;
int data ;
mutex_lock ( & adev - > pm . mutex ) ;
2017-02-08 17:17:55 +08:00
if ( RREG32_SMC ( ixCURRENT_PG_STATUS ) &
CURRENT_PG_STATUS__UVD_PG_STATUS_MASK ) {
2017-01-05 21:07:02 +08:00
DRM_INFO ( " Cannot get clockgating state when UVD is powergated. \n " ) ;
goto out ;
2015-04-20 17:31:14 -04:00
}
2017-01-05 21:07:02 +08:00
/* AMD_CG_SUPPORT_UVD_MGCG */
data = RREG32 ( mmUVD_CGC_CTRL ) ;
if ( data & UVD_CGC_CTRL__DYN_CLOCK_MODE_MASK )
* flags | = AMD_CG_SUPPORT_UVD_MGCG ;
out :
mutex_unlock ( & adev - > pm . mutex ) ;
2015-04-20 17:31:14 -04:00
}
2016-10-13 17:41:13 -04:00
static const struct amd_ip_funcs uvd_v5_0_ip_funcs = {
2016-05-04 14:28:35 -04:00
. name = " uvd_v5_0 " ,
2015-04-20 17:31:14 -04:00
. early_init = uvd_v5_0_early_init ,
. late_init = NULL ,
. sw_init = uvd_v5_0_sw_init ,
. sw_fini = uvd_v5_0_sw_fini ,
. hw_init = uvd_v5_0_hw_init ,
. hw_fini = uvd_v5_0_hw_fini ,
. suspend = uvd_v5_0_suspend ,
. resume = uvd_v5_0_resume ,
. is_idle = uvd_v5_0_is_idle ,
. wait_for_idle = uvd_v5_0_wait_for_idle ,
. soft_reset = uvd_v5_0_soft_reset ,
. set_clockgating_state = uvd_v5_0_set_clockgating_state ,
. set_powergating_state = uvd_v5_0_set_powergating_state ,
2017-01-05 21:07:02 +08:00
. get_clockgating_state = uvd_v5_0_get_clockgating_state ,
2015-04-20 17:31:14 -04:00
} ;
static const struct amdgpu_ring_funcs uvd_v5_0_ring_funcs = {
2016-10-05 15:36:39 +02:00
. type = AMDGPU_RING_TYPE_UVD ,
2016-10-05 16:09:32 +02:00
. align_mask = 0xf ,
. nop = PACKET0 ( mmUVD_NO_OP , 0 ) ,
2016-03-12 09:32:30 +08:00
. support_64bit_ptrs = false ,
2015-04-20 17:31:14 -04:00
. get_rptr = uvd_v5_0_ring_get_rptr ,
. get_wptr = uvd_v5_0_ring_get_wptr ,
. set_wptr = uvd_v5_0_ring_set_wptr ,
. parse_cs = amdgpu_uvd_ring_parse_cs ,
2016-10-05 14:29:38 +02:00
. emit_frame_size =
14 , /* uvd_v5_0_ring_emit_fence x1 no user fence */
. emit_ib_size = 6 , /* uvd_v5_0_ring_emit_ib */
2015-04-20 17:31:14 -04:00
. emit_ib = uvd_v5_0_ring_emit_ib ,
. emit_fence = uvd_v5_0_ring_emit_fence ,
. test_ring = uvd_v5_0_ring_test_ring ,
2016-07-05 16:47:54 +02:00
. test_ib = amdgpu_uvd_ring_test_ib ,
2015-09-01 13:04:08 +08:00
. insert_nop = amdgpu_ring_insert_nop ,
2016-01-31 12:20:55 +01:00
. pad_ib = amdgpu_ring_generic_pad_ib ,
2016-07-20 14:11:26 +02:00
. begin_use = amdgpu_uvd_ring_begin_use ,
. end_use = amdgpu_uvd_ring_end_use ,
2015-04-20 17:31:14 -04:00
} ;
static void uvd_v5_0_set_ring_funcs ( struct amdgpu_device * adev )
{
2018-05-15 14:25:46 -05:00
adev - > uvd . inst - > ring . funcs = & uvd_v5_0_ring_funcs ;
2015-04-20 17:31:14 -04:00
}
static const struct amdgpu_irq_src_funcs uvd_v5_0_irq_funcs = {
. set = uvd_v5_0_set_interrupt_state ,
. process = uvd_v5_0_process_interrupt ,
} ;
static void uvd_v5_0_set_irq_funcs ( struct amdgpu_device * adev )
{
2018-05-15 14:25:46 -05:00
adev - > uvd . inst - > irq . num_types = 1 ;
adev - > uvd . inst - > irq . funcs = & uvd_v5_0_irq_funcs ;
2015-04-20 17:31:14 -04:00
}
2016-10-13 17:41:13 -04:00
const struct amdgpu_ip_block_version uvd_v5_0_ip_block =
{
. type = AMD_IP_BLOCK_TYPE_UVD ,
. major = 5 ,
. minor = 0 ,
. rev = 0 ,
. funcs = & uvd_v5_0_ip_funcs ,
} ;