2017-03-07 00:41:22 +07:00
|
|
|
/*
|
|
|
|
* Copyright 2016 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __SOC15_COMMON_H__
|
|
|
|
#define __SOC15_COMMON_H__
|
|
|
|
|
2017-04-07 18:53:31 +07:00
|
|
|
/* Register Access Macros */
|
2017-11-30 01:51:32 +07:00
|
|
|
#define SOC15_REG_OFFSET(ip, inst, reg) (adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg)
|
2017-03-07 00:41:22 +07:00
|
|
|
|
2017-04-07 18:53:31 +07:00
|
|
|
#define WREG32_FIELD15(ip, idx, reg, field, val) \
|
2017-11-28 01:20:38 +07:00
|
|
|
WREG32(adev->reg_offset[ip##_HWIP][idx][mm##reg##_BASE_IDX] + mm##reg, \
|
|
|
|
(RREG32(adev->reg_offset[ip##_HWIP][idx][mm##reg##_BASE_IDX] + mm##reg) \
|
|
|
|
& ~REG_FIELD_MASK(reg, field)) | (val) << REG_FIELD_SHIFT(reg, field))
|
2017-04-07 18:53:31 +07:00
|
|
|
|
|
|
|
#define RREG32_SOC15(ip, inst, reg) \
|
2017-11-28 01:20:38 +07:00
|
|
|
RREG32(adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg)
|
2017-04-07 18:53:31 +07:00
|
|
|
|
2017-06-12 23:05:42 +07:00
|
|
|
#define RREG32_SOC15_OFFSET(ip, inst, reg, offset) \
|
2017-11-28 01:20:38 +07:00
|
|
|
RREG32((adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg) + offset)
|
2017-06-12 23:05:42 +07:00
|
|
|
|
2017-04-07 18:53:31 +07:00
|
|
|
#define WREG32_SOC15(ip, inst, reg, value) \
|
2017-11-28 01:20:38 +07:00
|
|
|
WREG32((adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg), value)
|
2017-04-07 18:53:31 +07:00
|
|
|
|
2017-07-05 21:53:55 +07:00
|
|
|
#define WREG32_SOC15_NO_KIQ(ip, inst, reg, value) \
|
2017-11-28 01:20:38 +07:00
|
|
|
WREG32_NO_KIQ((adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg), value)
|
2017-07-05 21:53:55 +07:00
|
|
|
|
2017-06-12 23:05:42 +07:00
|
|
|
#define WREG32_SOC15_OFFSET(ip, inst, reg, offset, value) \
|
2017-11-28 01:20:38 +07:00
|
|
|
WREG32((adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg) + offset, value)
|
2017-06-12 23:05:42 +07:00
|
|
|
|
2018-05-17 14:58:53 +07:00
|
|
|
#define SOC15_WAIT_ON_RREG(ip, inst, reg, expected_value, mask, ret) \
|
|
|
|
do { \
|
2018-12-17 20:35:05 +07:00
|
|
|
uint32_t old_ = 0; \
|
2018-05-17 14:58:53 +07:00
|
|
|
uint32_t tmp_ = RREG32(adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg); \
|
|
|
|
uint32_t loop = adev->usec_timeout; \
|
drm/amdgpu: Ensure ret is always initialized when using SOC15_WAIT_ON_RREG
Commit b0f3cd3191cd ("drm/amdgpu: remove unnecessary JPEG2.0 code from
VCN2.0") introduced a new clang warning in the vcn_v2_0_stop function:
../drivers/gpu/drm/amd/amdgpu/vcn_v2_0.c:1082:2: warning: variable 'r'
is used uninitialized whenever 'while' loop exits because its condition
is false [-Wsometimes-uninitialized]
SOC15_WAIT_ON_RREG(VCN, 0, mmUVD_STATUS, UVD_STATUS__IDLE, 0x7, r);
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
../drivers/gpu/drm/amd/amdgpu/../amdgpu/soc15_common.h:55:10: note:
expanded from macro 'SOC15_WAIT_ON_RREG'
while ((tmp_ & (mask)) != (expected_value)) { \
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
../drivers/gpu/drm/amd/amdgpu/vcn_v2_0.c:1083:6: note: uninitialized use
occurs here
if (r)
^
../drivers/gpu/drm/amd/amdgpu/vcn_v2_0.c:1082:2: note: remove the
condition if it is always true
SOC15_WAIT_ON_RREG(VCN, 0, mmUVD_STATUS, UVD_STATUS__IDLE, 0x7, r);
^
../drivers/gpu/drm/amd/amdgpu/../amdgpu/soc15_common.h:55:10: note:
expanded from macro 'SOC15_WAIT_ON_RREG'
while ((tmp_ & (mask)) != (expected_value)) { \
^
../drivers/gpu/drm/amd/amdgpu/vcn_v2_0.c:1072:7: note: initialize the
variable 'r' to silence this warning
int r;
^
= 0
1 warning generated.
To prevent warnings like this from happening in the future, make the
SOC15_WAIT_ON_RREG macro initialize its ret variable before the while
loop that can time out. This macro's return value is always checked so
it should set ret in both the success and fail path.
Link: https://github.com/ClangBuiltLinux/linux/issues/776
Signed-off-by: Nathan Chancellor <natechancellor@gmail.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
2019-11-24 02:23:36 +07:00
|
|
|
ret = 0; \
|
2018-05-17 14:58:53 +07:00
|
|
|
while ((tmp_ & (mask)) != (expected_value)) { \
|
2018-12-17 20:35:05 +07:00
|
|
|
if (old_ != tmp_) { \
|
|
|
|
loop = adev->usec_timeout; \
|
|
|
|
old_ = tmp_; \
|
|
|
|
} else \
|
|
|
|
udelay(1); \
|
2018-05-17 14:58:53 +07:00
|
|
|
tmp_ = RREG32(adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg); \
|
|
|
|
loop--; \
|
|
|
|
if (!loop) { \
|
2018-12-17 20:35:05 +07:00
|
|
|
DRM_WARN("Register(%d) [%s] failed to reach value 0x%08x != 0x%08x\n", \
|
2018-09-26 23:18:47 +07:00
|
|
|
inst, #reg, (unsigned)expected_value, (unsigned)(tmp_ & (mask))); \
|
2018-05-17 14:58:53 +07:00
|
|
|
ret = -ETIMEDOUT; \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2019-07-30 16:21:19 +07:00
|
|
|
#define AMDGPU_VIRT_SUPPORT_RLC_PRG_REG(a) (amdgpu_sriov_vf((a)) && !amdgpu_sriov_runtime((a)))
|
drm/amdgpu: add basic func for RLC program reg
New feature for RLC, some registers can be programmed by
RLC interface under SR-IOV VF:
WREG32_SOC15_RLC_SHADOW:
1, for GRBM_GFX_CNTL, firstly the new register value should be be
programmed to SCRATCH_REG2
1, for GRBM_GFX_INDEX, firstly the new register value should be be
programmed to SCRATCH_REG3
WREG32_RLC:
for registers supported to be programmed by RLC interface, the
following sequence should be used:
1, write the value to SCRATCH_REG0
2, write reg | 0x80000000 to SCRATCH_REG1
3, write 0x1 to RLC_SPARE_INT to notify RLC
4, polling SCRATCH_REG1 to check if finished
Signed-off-by: Trigger Huang <Trigger.Huang@amd.com>
Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
2019-03-01 10:56:20 +07:00
|
|
|
#define WREG32_RLC(reg, value) \
|
|
|
|
do { \
|
2019-07-30 16:21:19 +07:00
|
|
|
if (AMDGPU_VIRT_SUPPORT_RLC_PRG_REG(adev)) { \
|
drm/amdgpu: add basic func for RLC program reg
New feature for RLC, some registers can be programmed by
RLC interface under SR-IOV VF:
WREG32_SOC15_RLC_SHADOW:
1, for GRBM_GFX_CNTL, firstly the new register value should be be
programmed to SCRATCH_REG2
1, for GRBM_GFX_INDEX, firstly the new register value should be be
programmed to SCRATCH_REG3
WREG32_RLC:
for registers supported to be programmed by RLC interface, the
following sequence should be used:
1, write the value to SCRATCH_REG0
2, write reg | 0x80000000 to SCRATCH_REG1
3, write 0x1 to RLC_SPARE_INT to notify RLC
4, polling SCRATCH_REG1 to check if finished
Signed-off-by: Trigger Huang <Trigger.Huang@amd.com>
Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
2019-03-01 10:56:20 +07:00
|
|
|
uint32_t i = 0; \
|
|
|
|
uint32_t retries = 50000; \
|
|
|
|
uint32_t r0 = adev->reg_offset[GC_HWIP][0][mmSCRATCH_REG0_BASE_IDX] + mmSCRATCH_REG0; \
|
|
|
|
uint32_t r1 = adev->reg_offset[GC_HWIP][0][mmSCRATCH_REG1_BASE_IDX] + mmSCRATCH_REG1; \
|
|
|
|
uint32_t spare_int = adev->reg_offset[GC_HWIP][0][mmRLC_SPARE_INT_BASE_IDX] + mmRLC_SPARE_INT; \
|
|
|
|
WREG32(r0, value); \
|
|
|
|
WREG32(r1, (reg | 0x80000000)); \
|
|
|
|
WREG32(spare_int, 0x1); \
|
|
|
|
for (i = 0; i < retries; i++) { \
|
|
|
|
u32 tmp = RREG32(r1); \
|
|
|
|
if (!(tmp & 0x80000000)) \
|
|
|
|
break; \
|
|
|
|
udelay(10); \
|
|
|
|
} \
|
|
|
|
if (i >= retries) \
|
|
|
|
pr_err("timeout: rlcg program reg:0x%05x failed !\n", reg); \
|
|
|
|
} else { \
|
|
|
|
WREG32(reg, value); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define WREG32_SOC15_RLC_SHADOW(ip, inst, reg, value) \
|
|
|
|
do { \
|
|
|
|
uint32_t target_reg = adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg;\
|
2019-07-30 16:21:19 +07:00
|
|
|
if (AMDGPU_VIRT_SUPPORT_RLC_PRG_REG(adev)) { \
|
drm/amdgpu: add basic func for RLC program reg
New feature for RLC, some registers can be programmed by
RLC interface under SR-IOV VF:
WREG32_SOC15_RLC_SHADOW:
1, for GRBM_GFX_CNTL, firstly the new register value should be be
programmed to SCRATCH_REG2
1, for GRBM_GFX_INDEX, firstly the new register value should be be
programmed to SCRATCH_REG3
WREG32_RLC:
for registers supported to be programmed by RLC interface, the
following sequence should be used:
1, write the value to SCRATCH_REG0
2, write reg | 0x80000000 to SCRATCH_REG1
3, write 0x1 to RLC_SPARE_INT to notify RLC
4, polling SCRATCH_REG1 to check if finished
Signed-off-by: Trigger Huang <Trigger.Huang@amd.com>
Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
2019-03-01 10:56:20 +07:00
|
|
|
uint32_t r2 = adev->reg_offset[GC_HWIP][0][mmSCRATCH_REG1_BASE_IDX] + mmSCRATCH_REG2; \
|
|
|
|
uint32_t r3 = adev->reg_offset[GC_HWIP][0][mmSCRATCH_REG1_BASE_IDX] + mmSCRATCH_REG3; \
|
|
|
|
uint32_t grbm_cntl = adev->reg_offset[GC_HWIP][0][mmGRBM_GFX_CNTL_BASE_IDX] + mmGRBM_GFX_CNTL; \
|
|
|
|
uint32_t grbm_idx = adev->reg_offset[GC_HWIP][0][mmGRBM_GFX_INDEX_BASE_IDX] + mmGRBM_GFX_INDEX; \
|
|
|
|
if (target_reg == grbm_cntl) \
|
|
|
|
WREG32(r2, value); \
|
|
|
|
else if (target_reg == grbm_idx) \
|
|
|
|
WREG32(r3, value); \
|
|
|
|
WREG32(target_reg, value); \
|
|
|
|
} else { \
|
|
|
|
WREG32(target_reg, value); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define WREG32_SOC15_RLC(ip, inst, reg, value) \
|
|
|
|
do { \
|
|
|
|
uint32_t target_reg = adev->reg_offset[GC_HWIP][0][reg##_BASE_IDX] + reg;\
|
|
|
|
WREG32_RLC(target_reg, value); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define WREG32_FIELD15_RLC(ip, idx, reg, field, val) \
|
|
|
|
WREG32_RLC((adev->reg_offset[ip##_HWIP][idx][mm##reg##_BASE_IDX] + mm##reg), \
|
|
|
|
(RREG32(adev->reg_offset[ip##_HWIP][idx][mm##reg##_BASE_IDX] + mm##reg) \
|
|
|
|
& ~REG_FIELD_MASK(reg, field)) | (val) << REG_FIELD_SHIFT(reg, field))
|
|
|
|
|
|
|
|
#define WREG32_SOC15_OFFSET_RLC(ip, inst, reg, offset, value) \
|
|
|
|
WREG32_RLC(((adev->reg_offset[ip##_HWIP][inst][reg##_BASE_IDX] + reg) + offset), value)
|
2017-03-07 00:41:22 +07:00
|
|
|
|
drm/amdgpu: add basic func for RLC program reg
New feature for RLC, some registers can be programmed by
RLC interface under SR-IOV VF:
WREG32_SOC15_RLC_SHADOW:
1, for GRBM_GFX_CNTL, firstly the new register value should be be
programmed to SCRATCH_REG2
1, for GRBM_GFX_INDEX, firstly the new register value should be be
programmed to SCRATCH_REG3
WREG32_RLC:
for registers supported to be programmed by RLC interface, the
following sequence should be used:
1, write the value to SCRATCH_REG0
2, write reg | 0x80000000 to SCRATCH_REG1
3, write 0x1 to RLC_SPARE_INT to notify RLC
4, polling SCRATCH_REG1 to check if finished
Signed-off-by: Trigger Huang <Trigger.Huang@amd.com>
Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
2019-03-01 10:56:20 +07:00
|
|
|
#endif
|