diff options
| author | Bartlomiej Zolnierkiewicz <[email protected]> | 2018-09-26 15:54:31 +0200 | 
|---|---|---|
| committer | Bartlomiej Zolnierkiewicz <[email protected]> | 2018-09-26 15:54:31 +0200 | 
| commit | aaccf3c97418f169afdbb5855e9cbcbda34e90fd (patch) | |
| tree | 5d4207e67958bdbc23288cf30178692f5534e1a0 /drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c | |
| parent | f39684524b391c5a7ed0ac44db4fec3357af1c5d (diff) | |
| parent | 6bf4ca7fbc85d80446ac01c0d1d77db4d91a6d84 (diff) | |
Merge tag 'v4.19-rc5' of https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux into fbdev-for-next
Sync with upstream (which now contains fbdev-v4.19 changes) to
prepare a base for fbdev-v4.20 changes.
Diffstat (limited to 'drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c')
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c | 131 | 
1 files changed, 97 insertions, 34 deletions
| diff --git a/drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c b/drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c index 57d32f21b3a6..9b7f8469bc5c 100644 --- a/drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c +++ b/drivers/gpu/drm/amd/amdgpu/uvd_v7_0.c @@ -39,6 +39,13 @@  #include "hdp/hdp_4_0_offset.h"  #include "mmhub/mmhub_1_0_offset.h"  #include "mmhub/mmhub_1_0_sh_mask.h" +#include "ivsrcid/uvd/irqsrcs_uvd_7_0.h" + +#define mmUVD_PG0_CC_UVD_HARVESTING                                                                    0x00c7 +#define mmUVD_PG0_CC_UVD_HARVESTING_BASE_IDX                                                           1 +//UVD_PG0_CC_UVD_HARVESTING +#define UVD_PG0_CC_UVD_HARVESTING__UVD_DISABLE__SHIFT                                                         0x1 +#define UVD_PG0_CC_UVD_HARVESTING__UVD_DISABLE_MASK                                                           0x00000002L  #define UVD7_MAX_HW_INSTANCES_VEGA20			2 @@ -249,12 +256,10 @@ static int uvd_v7_0_enc_get_create_msg(struct amdgpu_ring *ring, uint32_t handle  	for (i = ib->length_dw; i < ib_size_dw; ++i)  		ib->ptr[i] = 0x0; -	r = amdgpu_ib_schedule(ring, 1, ib, NULL, &f); -	job->fence = dma_fence_get(f); +	r = amdgpu_job_submit_direct(job, ring, &f);  	if (r)  		goto err; -	amdgpu_job_free(job);  	if (fence)  		*fence = dma_fence_get(f);  	dma_fence_put(f); @@ -312,19 +317,13 @@ int uvd_v7_0_enc_get_destroy_msg(struct amdgpu_ring *ring, uint32_t handle,  	for (i = ib->length_dw; i < ib_size_dw; ++i)  		ib->ptr[i] = 0x0; -	if (direct) { -		r = amdgpu_ib_schedule(ring, 1, ib, NULL, &f); -		job->fence = dma_fence_get(f); -		if (r) -			goto err; - -		amdgpu_job_free(job); -	} else { -		r = amdgpu_job_submit(job, ring, &ring->adev->vce.entity, +	if (direct) +		r = amdgpu_job_submit_direct(job, ring, &f); +	else +		r = amdgpu_job_submit(job, &ring->adev->vce.entity,  				      AMDGPU_FENCE_OWNER_UNDEFINED, &f); -		if (r) -			goto err; -	} +	if (r) +		goto err;  	if (fence)  		*fence = dma_fence_get(f); @@ -377,10 +376,25 @@ error:  static int uvd_v7_0_early_init(void *handle)  {  	struct amdgpu_device *adev = (struct amdgpu_device *)handle; -	if (adev->asic_type == CHIP_VEGA20) + +	if (adev->asic_type == CHIP_VEGA20) { +		u32 harvest; +		int i; +  		adev->uvd.num_uvd_inst = UVD7_MAX_HW_INSTANCES_VEGA20; -	else +		for (i = 0; i < adev->uvd.num_uvd_inst; i++) { +			harvest = RREG32_SOC15(UVD, i, mmUVD_PG0_CC_UVD_HARVESTING); +			if (harvest & UVD_PG0_CC_UVD_HARVESTING__UVD_DISABLE_MASK) { +				adev->uvd.harvest_config |= 1 << i; +			} +		} +		if (adev->uvd.harvest_config == (AMDGPU_UVD_HARVEST_UVD0 | +						 AMDGPU_UVD_HARVEST_UVD1)) +			/* both instances are harvested, disable the block */ +			return -ENOENT; +	} else {  		adev->uvd.num_uvd_inst = 1; +	}  	if (amdgpu_sriov_vf(adev))  		adev->uvd.num_enc_rings = 1; @@ -396,19 +410,21 @@ static int uvd_v7_0_early_init(void *handle)  static int uvd_v7_0_sw_init(void *handle)  {  	struct amdgpu_ring *ring; -	struct drm_sched_rq *rq; +  	int i, j, r;  	struct amdgpu_device *adev = (struct amdgpu_device *)handle;  	for (j = 0; j < adev->uvd.num_uvd_inst; j++) { +		if (adev->uvd.harvest_config & (1 << j)) +			continue;  		/* UVD TRAP */ -		r = amdgpu_irq_add_id(adev, amdgpu_ih_clientid_uvds[j], 124, &adev->uvd.inst[j].irq); +		r = amdgpu_irq_add_id(adev, amdgpu_ih_clientid_uvds[j], UVD_7_0__SRCID__UVD_SYSTEM_MESSAGE_INTERRUPT, &adev->uvd.inst[j].irq);  		if (r)  			return r;  		/* UVD ENC TRAP */  		for (i = 0; i < adev->uvd.num_enc_rings; ++i) { -			r = amdgpu_irq_add_id(adev, amdgpu_ih_clientid_uvds[j], i + 119, &adev->uvd.inst[j].irq); +			r = amdgpu_irq_add_id(adev, amdgpu_ih_clientid_uvds[j], i + UVD_7_0__SRCID__UVD_ENC_GEN_PURP, &adev->uvd.inst[j].irq);  			if (r)  				return r;  		} @@ -428,22 +444,13 @@ static int uvd_v7_0_sw_init(void *handle)  		DRM_INFO("PSP loading UVD firmware\n");  	} -	for (j = 0; j < adev->uvd.num_uvd_inst; j++) { -		ring = &adev->uvd.inst[j].ring_enc[0]; -		rq = &ring->sched.sched_rq[DRM_SCHED_PRIORITY_NORMAL]; -		r = drm_sched_entity_init(&ring->sched, &adev->uvd.inst[j].entity_enc, -					  rq, NULL); -		if (r) { -			DRM_ERROR("(%d)Failed setting up UVD ENC run queue.\n", j); -			return r; -		} -	} -  	r = amdgpu_uvd_resume(adev);  	if (r)  		return r;  	for (j = 0; j < adev->uvd.num_uvd_inst; j++) { +		if (adev->uvd.harvest_config & (1 << j)) +			continue;  		if (!amdgpu_sriov_vf(adev)) {  			ring = &adev->uvd.inst[j].ring;  			sprintf(ring->name, "uvd<%d>", j); @@ -472,6 +479,10 @@ static int uvd_v7_0_sw_init(void *handle)  		}  	} +	r = amdgpu_uvd_entity_init(adev); +	if (r) +		return r; +  	r = amdgpu_virt_alloc_mm_table(adev);  	if (r)  		return r; @@ -491,8 +502,8 @@ static int uvd_v7_0_sw_fini(void *handle)  		return r;  	for (j = 0; j < adev->uvd.num_uvd_inst; ++j) { -		drm_sched_entity_fini(&adev->uvd.inst[j].ring_enc[0].sched, &adev->uvd.inst[j].entity_enc); - +		if (adev->uvd.harvest_config & (1 << j)) +			continue;  		for (i = 0; i < adev->uvd.num_enc_rings; ++i)  			amdgpu_ring_fini(&adev->uvd.inst[j].ring_enc[i]);  	} @@ -521,6 +532,8 @@ static int uvd_v7_0_hw_init(void *handle)  		goto done;  	for (j = 0; j < adev->uvd.num_uvd_inst; ++j) { +		if (adev->uvd.harvest_config & (1 << j)) +			continue;  		ring = &adev->uvd.inst[j].ring;  		if (!amdgpu_sriov_vf(adev)) { @@ -600,8 +613,11 @@ static int uvd_v7_0_hw_fini(void *handle)  		DRM_DEBUG("For SRIOV client, shouldn't do anything.\n");  	} -	for (i = 0; i < adev->uvd.num_uvd_inst; ++i) +	for (i = 0; i < adev->uvd.num_uvd_inst; ++i) { +		if (adev->uvd.harvest_config & (1 << i)) +			continue;  		adev->uvd.inst[i].ring.ready = false; +	}  	return 0;  } @@ -644,6 +660,8 @@ static void uvd_v7_0_mc_resume(struct amdgpu_device *adev)  	int i;  	for (i = 0; i < adev->uvd.num_uvd_inst; ++i) { +		if (adev->uvd.harvest_config & (1 << i)) +			continue;  		if (adev->firmware.load_type == AMDGPU_FW_LOAD_PSP) {  			WREG32_SOC15(UVD, i, mmUVD_LMI_VCPU_CACHE_64BIT_BAR_LOW,  				lower_32_bits(adev->firmware.ucode[AMDGPU_UCODE_ID_UVD].mc_addr)); @@ -716,6 +734,8 @@ static int uvd_v7_0_mmsch_start(struct amdgpu_device *adev,  	WREG32_SOC15(VCE, 0, mmVCE_MMSCH_VF_MAILBOX_RESP, 0);  	for (i = 0; i < adev->uvd.num_uvd_inst; ++i) { +		if (adev->uvd.harvest_config & (1 << i)) +			continue;  		WDOORBELL32(adev->uvd.inst[i].ring_enc[0].doorbell_index, 0);  		adev->wb.wb[adev->uvd.inst[i].ring_enc[0].wptr_offs] = 0;  		adev->uvd.inst[i].ring_enc[0].wptr = 0; @@ -772,6 +792,8 @@ static int uvd_v7_0_sriov_start(struct amdgpu_device *adev)  		init_table += header->uvd_table_offset;  		for (i = 0; i < adev->uvd.num_uvd_inst; ++i) { +			if (adev->uvd.harvest_config & (1 << i)) +				continue;  			ring = &adev->uvd.inst[i].ring;  			ring->wptr = 0;  			size = AMDGPU_GPU_PAGE_ALIGN(adev->uvd.fw->size + 4); @@ -911,6 +933,8 @@ static int uvd_v7_0_start(struct amdgpu_device *adev)  	int i, j, k, r;  	for (k = 0; k < adev->uvd.num_uvd_inst; ++k) { +		if (adev->uvd.harvest_config & (1 << k)) +			continue;  		/* disable DPG */  		WREG32_P(SOC15_REG_OFFSET(UVD, k, mmUVD_POWER_STATUS), 0,  				~UVD_POWER_STATUS__UVD_PG_MODE_MASK); @@ -923,6 +947,8 @@ static int uvd_v7_0_start(struct amdgpu_device *adev)  	uvd_v7_0_mc_resume(adev);  	for (k = 0; k < adev->uvd.num_uvd_inst; ++k) { +		if (adev->uvd.harvest_config & (1 << k)) +			continue;  		ring = &adev->uvd.inst[k].ring;  		/* disable clock gating */  		WREG32_P(SOC15_REG_OFFSET(UVD, k, mmUVD_CGC_CTRL), 0, @@ -1090,6 +1116,8 @@ static void uvd_v7_0_stop(struct amdgpu_device *adev)  	uint8_t i = 0;  	for (i = 0; i < adev->uvd.num_uvd_inst; ++i) { +		if (adev->uvd.harvest_config & (1 << i)) +			continue;  		/* force RBC into idle state */  		WREG32_SOC15(UVD, i, mmUVD_RBC_RB_CNTL, 0x11010101); @@ -1227,6 +1255,34 @@ static int uvd_v7_0_ring_test_ring(struct amdgpu_ring *ring)  }  /** + * uvd_v7_0_ring_patch_cs_in_place - Patch the IB for command submission. + * + * @p: the CS parser with the IBs + * @ib_idx: which IB to patch + * + */ +static int uvd_v7_0_ring_patch_cs_in_place(struct amdgpu_cs_parser *p, +					   uint32_t ib_idx) +{ +	struct amdgpu_ib *ib = &p->job->ibs[ib_idx]; +	unsigned i; + +	/* No patching necessary for the first instance */ +	if (!p->ring->me) +		return 0; + +	for (i = 0; i < ib->length_dw; i += 2) { +		uint32_t reg = amdgpu_get_ib_value(p, ib_idx, i); + +		reg -= p->adev->reg_offset[UVD_HWIP][0][1]; +		reg += p->adev->reg_offset[UVD_HWIP][1][1]; + +		amdgpu_set_ib_value(p, ib_idx, i, reg); +	} +	return 0; +} + +/**   * uvd_v7_0_ring_emit_ib - execute indirect buffer   *   * @ring: amdgpu_ring pointer @@ -1718,6 +1774,7 @@ static const struct amdgpu_ring_funcs uvd_v7_0_ring_vm_funcs = {  	.get_rptr = uvd_v7_0_ring_get_rptr,  	.get_wptr = uvd_v7_0_ring_get_wptr,  	.set_wptr = uvd_v7_0_ring_set_wptr, +	.patch_cs_in_place = uvd_v7_0_ring_patch_cs_in_place,  	.emit_frame_size =  		6 + /* hdp invalidate */  		SOC15_FLUSH_GPU_TLB_NUM_WREG * 6 + @@ -1777,6 +1834,8 @@ static void uvd_v7_0_set_ring_funcs(struct amdgpu_device *adev)  	int i;  	for (i = 0; i < adev->uvd.num_uvd_inst; i++) { +		if (adev->uvd.harvest_config & (1 << i)) +			continue;  		adev->uvd.inst[i].ring.funcs = &uvd_v7_0_ring_vm_funcs;  		adev->uvd.inst[i].ring.me = i;  		DRM_INFO("UVD(%d) is enabled in VM mode\n", i); @@ -1788,6 +1847,8 @@ static void uvd_v7_0_set_enc_ring_funcs(struct amdgpu_device *adev)  	int i, j;  	for (j = 0; j < adev->uvd.num_uvd_inst; j++) { +		if (adev->uvd.harvest_config & (1 << j)) +			continue;  		for (i = 0; i < adev->uvd.num_enc_rings; ++i) {  			adev->uvd.inst[j].ring_enc[i].funcs = &uvd_v7_0_enc_ring_vm_funcs;  			adev->uvd.inst[j].ring_enc[i].me = j; @@ -1807,6 +1868,8 @@ static void uvd_v7_0_set_irq_funcs(struct amdgpu_device *adev)  	int i;  	for (i = 0; i < adev->uvd.num_uvd_inst; i++) { +		if (adev->uvd.harvest_config & (1 << i)) +			continue;  		adev->uvd.inst[i].irq.num_types = adev->uvd.num_enc_rings + 1;  		adev->uvd.inst[i].irq.funcs = &uvd_v7_0_irq_funcs;  	} |