[PATCH] drm/amdgpu:partially revert 1cfd8e237f0318e330190ac21d63c58ae6a1f66c

Christian König ckoenig.leichtzumerken at gmail.com
Tue Nov 21 08:26:33 UTC 2017


Am 21.11.2017 um 06:29 schrieb Monk Liu:
> found RING0 test fail after S3 resume regression, which is
> introduced by 1cfd8e237f0318e330190ac21d63c58ae6a1f66c
>
> Because after suspend VRAM will be cleared, so driver must
> unpin the GART table(resident in VRAM) during suspend so it
> can be evicted to system ram and must correspondingly pin it
> during resume so the GART table could be restored to VRAM.
>
> Change-Id: I0c0f9dc89f38a903caf114094433fb00bd6c93fb
> Signed-off-by: Monk Liu <Monk.Liu at amd.com>

Reviewed-by: Christian König <christian.koenig at amd.com>

> ---
>   drivers/gpu/drm/amd/amdgpu/amdgpu_gart.c | 79 +++++++++++++++++++++++++++++---
>   drivers/gpu/drm/amd/amdgpu/amdgpu_gart.h |  2 +
>   drivers/gpu/drm/amd/amdgpu/gmc_v6_0.c    |  7 ++-
>   drivers/gpu/drm/amd/amdgpu/gmc_v7_0.c    |  7 ++-
>   drivers/gpu/drm/amd/amdgpu/gmc_v8_0.c    |  7 ++-
>   drivers/gpu/drm/amd/amdgpu/gmc_v9_0.c    |  4 ++
>   6 files changed, 94 insertions(+), 12 deletions(-)
>
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.c
> index 707f858..1f51897 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.c
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.c
> @@ -68,9 +68,75 @@
>    */
>   int amdgpu_gart_table_vram_alloc(struct amdgpu_device *adev)
>   {
> -	return amdgpu_bo_create_kernel(adev, adev->gart.table_size, PAGE_SIZE,
> -					AMDGPU_GEM_DOMAIN_VRAM, &adev->gart.robj,
> -					&adev->gart.table_addr, &adev->gart.ptr);
> +	int r;
> +
> +	if (adev->gart.robj == NULL) {
> +		r = amdgpu_bo_create(adev, adev->gart.table_size,
> +				     PAGE_SIZE, true, AMDGPU_GEM_DOMAIN_VRAM,
> +				     AMDGPU_GEM_CREATE_CPU_ACCESS_REQUIRED |
> +				     AMDGPU_GEM_CREATE_VRAM_CONTIGUOUS,
> +				     NULL, NULL, 0, &adev->gart.robj);
> +		if (r) {
> +			return r;
> +		}
> +	}
> +	return 0;
> +}
> +
> +/**
> + * amdgpu_gart_table_vram_pin - pin gart page table in vram
> + *
> + * @adev: amdgpu_device pointer
> + *
> + * Pin the GART page table in vram so it will not be moved
> + * by the memory manager (pcie r4xx, r5xx+).  These asics require the
> + * gart table to be in video memory.
> + * Returns 0 for success, error for failure.
> + */
> +int amdgpu_gart_table_vram_pin(struct amdgpu_device *adev)
> +{
> +	uint64_t gpu_addr;
> +	int r;
> +
> +	r = amdgpu_bo_reserve(adev->gart.robj, false);
> +	if (unlikely(r != 0))
> +		return r;
> +	r = amdgpu_bo_pin(adev->gart.robj,
> +				AMDGPU_GEM_DOMAIN_VRAM, &gpu_addr);
> +	if (r) {
> +		amdgpu_bo_unreserve(adev->gart.robj);
> +		return r;
> +	}
> +	r = amdgpu_bo_kmap(adev->gart.robj, &adev->gart.ptr);
> +	if (r)
> +		amdgpu_bo_unpin(adev->gart.robj);
> +	amdgpu_bo_unreserve(adev->gart.robj);
> +	adev->gart.table_addr = gpu_addr;
> +	return r;
> +}
> +
> +/**
> + * amdgpu_gart_table_vram_unpin - unpin gart page table in vram
> + *
> + * @adev: amdgpu_device pointer
> + *
> + * Unpin the GART page table in vram (pcie r4xx, r5xx+).
> + * These asics require the gart table to be in video memory.
> + */
> +void amdgpu_gart_table_vram_unpin(struct amdgpu_device *adev)
> +{
> +	int r;
> +
> +	if (adev->gart.robj == NULL) {
> +		return;
> +	}
> +	r = amdgpu_bo_reserve(adev->gart.robj, true);
> +	if (likely(r == 0)) {
> +		amdgpu_bo_kunmap(adev->gart.robj);
> +		amdgpu_bo_unpin(adev->gart.robj);
> +		amdgpu_bo_unreserve(adev->gart.robj);
> +		adev->gart.ptr = NULL;
> +	}
>   }
>   
>   /**
> @@ -84,9 +150,10 @@ int amdgpu_gart_table_vram_alloc(struct amdgpu_device *adev)
>    */
>   void amdgpu_gart_table_vram_free(struct amdgpu_device *adev)
>   {
> -	amdgpu_bo_free_kernel(&adev->gart.robj,
> -				&adev->gart.table_addr,
> -				&adev->gart.ptr);
> +	if (adev->gart.robj == NULL) {
> +		return;
> +	}
> +	amdgpu_bo_unref(&adev->gart.robj);
>   }
>   
>   /*
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.h b/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.h
> index f15e319..fe2e76b 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.h
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_gart.h
> @@ -58,6 +58,8 @@ struct amdgpu_gart {
>   
>   int amdgpu_gart_table_vram_alloc(struct amdgpu_device *adev);
>   void amdgpu_gart_table_vram_free(struct amdgpu_device *adev);
> +int amdgpu_gart_table_vram_pin(struct amdgpu_device *adev);
> +void amdgpu_gart_table_vram_unpin(struct amdgpu_device *adev);
>   int amdgpu_gart_init(struct amdgpu_device *adev);
>   void amdgpu_gart_fini(struct amdgpu_device *adev);
>   int amdgpu_gart_unbind(struct amdgpu_device *adev, uint64_t offset,
> diff --git a/drivers/gpu/drm/amd/amdgpu/gmc_v6_0.c b/drivers/gpu/drm/amd/amdgpu/gmc_v6_0.c
> index 9c672ec..aea3d1b 100644
> --- a/drivers/gpu/drm/amd/amdgpu/gmc_v6_0.c
> +++ b/drivers/gpu/drm/amd/amdgpu/gmc_v6_0.c
> @@ -483,14 +483,16 @@ static void gmc_v6_0_set_prt(struct amdgpu_device *adev, bool enable)
>   
>   static int gmc_v6_0_gart_enable(struct amdgpu_device *adev)
>   {
> -	int i;
> +	int r, i;
>   	u32 field;
>   
>   	if (adev->gart.robj == NULL) {
>   		dev_err(adev->dev, "No VRAM object for PCIE GART.\n");
>   		return -EINVAL;
>   	}
> -
> +	r = amdgpu_gart_table_vram_pin(adev);
> +	if (r)
> +		return r;
>   	/* Setup TLB control */
>   	WREG32(mmMC_VM_MX_L1_TLB_CNTL,
>   	       (0xA << 7) |
> @@ -617,6 +619,7 @@ static void gmc_v6_0_gart_disable(struct amdgpu_device *adev)
>   	WREG32(mmVM_L2_CNTL3,
>   	       VM_L2_CNTL3__L2_CACHE_BIGK_ASSOCIATIVITY_MASK |
>   	       (0UL << VM_L2_CNTL3__L2_CACHE_BIGK_FRAGMENT_SIZE__SHIFT));
> +	amdgpu_gart_table_vram_unpin(adev);
>   }
>   
>   static void gmc_v6_0_gart_fini(struct amdgpu_device *adev)
> diff --git a/drivers/gpu/drm/amd/amdgpu/gmc_v7_0.c b/drivers/gpu/drm/amd/amdgpu/gmc_v7_0.c
> index de7a249..b2e1176 100644
> --- a/drivers/gpu/drm/amd/amdgpu/gmc_v7_0.c
> +++ b/drivers/gpu/drm/amd/amdgpu/gmc_v7_0.c
> @@ -588,14 +588,16 @@ static void gmc_v7_0_set_prt(struct amdgpu_device *adev, bool enable)
>    */
>   static int gmc_v7_0_gart_enable(struct amdgpu_device *adev)
>   {
> -	int i;
> +	int r, i;
>   	u32 tmp, field;
>   
>   	if (adev->gart.robj == NULL) {
>   		dev_err(adev->dev, "No VRAM object for PCIE GART.\n");
>   		return -EINVAL;
>   	}
> -
> +	r = amdgpu_gart_table_vram_pin(adev);
> +	if (r)
> +		return r;
>   	/* Setup TLB control */
>   	tmp = RREG32(mmMC_VM_MX_L1_TLB_CNTL);
>   	tmp = REG_SET_FIELD(tmp, MC_VM_MX_L1_TLB_CNTL, ENABLE_L1_TLB, 1);
> @@ -728,6 +730,7 @@ static void gmc_v7_0_gart_disable(struct amdgpu_device *adev)
>   	tmp = REG_SET_FIELD(tmp, VM_L2_CNTL, ENABLE_L2_CACHE, 0);
>   	WREG32(mmVM_L2_CNTL, tmp);
>   	WREG32(mmVM_L2_CNTL2, 0);
> +	amdgpu_gart_table_vram_unpin(adev);
>   }
>   
>   /**
> diff --git a/drivers/gpu/drm/amd/amdgpu/gmc_v8_0.c b/drivers/gpu/drm/amd/amdgpu/gmc_v8_0.c
> index 6777874..728b501 100644
> --- a/drivers/gpu/drm/amd/amdgpu/gmc_v8_0.c
> +++ b/drivers/gpu/drm/amd/amdgpu/gmc_v8_0.c
> @@ -787,14 +787,16 @@ static void gmc_v8_0_set_prt(struct amdgpu_device *adev, bool enable)
>    */
>   static int gmc_v8_0_gart_enable(struct amdgpu_device *adev)
>   {
> -	int i;
> +	int r, i;
>   	u32 tmp, field;
>   
>   	if (adev->gart.robj == NULL) {
>   		dev_err(adev->dev, "No VRAM object for PCIE GART.\n");
>   		return -EINVAL;
>   	}
> -
> +	r = amdgpu_gart_table_vram_pin(adev);
> +	if (r)
> +		return r;
>   	/* Setup TLB control */
>   	tmp = RREG32(mmMC_VM_MX_L1_TLB_CNTL);
>   	tmp = REG_SET_FIELD(tmp, MC_VM_MX_L1_TLB_CNTL, ENABLE_L1_TLB, 1);
> @@ -944,6 +946,7 @@ static void gmc_v8_0_gart_disable(struct amdgpu_device *adev)
>   	tmp = REG_SET_FIELD(tmp, VM_L2_CNTL, ENABLE_L2_CACHE, 0);
>   	WREG32(mmVM_L2_CNTL, tmp);
>   	WREG32(mmVM_L2_CNTL2, 0);
> +	amdgpu_gart_table_vram_unpin(adev);
>   }
>   
>   /**
> diff --git a/drivers/gpu/drm/amd/amdgpu/gmc_v9_0.c b/drivers/gpu/drm/amd/amdgpu/gmc_v9_0.c
> index 798f7fc..e48265e 100644
> --- a/drivers/gpu/drm/amd/amdgpu/gmc_v9_0.c
> +++ b/drivers/gpu/drm/amd/amdgpu/gmc_v9_0.c
> @@ -933,6 +933,9 @@ static int gmc_v9_0_gart_enable(struct amdgpu_device *adev)
>   		dev_err(adev->dev, "No VRAM object for PCIE GART.\n");
>   		return -EINVAL;
>   	}
> +	r = amdgpu_gart_table_vram_pin(adev);
> +	if (r)
> +		return r;
>   
>   	switch (adev->asic_type) {
>   	case CHIP_RAVEN:
> @@ -1010,6 +1013,7 @@ static void gmc_v9_0_gart_disable(struct amdgpu_device *adev)
>   {
>   	gfxhub_v1_0_gart_disable(adev);
>   	mmhub_v1_0_gart_disable(adev);
> +	amdgpu_gart_table_vram_unpin(adev);
>   }
>   
>   static int gmc_v9_0_hw_fini(void *handle)




More information about the amd-gfx mailing list