Re: [PATCH v2 4/4] drm/amdgpu: Implement virt req_ras_err_count

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




On 11/6/2024 12:34 AM, Victor Skvortsov wrote:
> Enable RAS late init  if VF RAS Telemetry is supported.
> 
> When enabled, the VF can use this interface to query total
> RAS error counts from the host.
> 
> The VF FB access may abruptly end due to a fatal error,
> therefore the VF must cache and sanitize the input.
> 
> The Host allows 15 Telemetry messages every 60 seconds, afterwhich
> the host will ignore any more in-coming telemetry messages. The VF will
> rate limit its msg calling to once every 5 seconds (12 times in 60 seconds).
> While the VF is rate limited, it will continue to report the last
> good cached data.
> 
> v2: Flip generate report & update statistics order for VF
> 
> Signed-off-by: Victor Skvortsov <victor.skvortsov@xxxxxxx>
> ---
>  drivers/gpu/drm/amd/amdgpu/amdgpu_device.c |   6 +
>  drivers/gpu/drm/amd/amdgpu/amdgpu_gfx.c    |   3 +
>  drivers/gpu/drm/amd/amdgpu/amdgpu_ras.c    |  72 +++++++++--
>  drivers/gpu/drm/amd/amdgpu/amdgpu_ras.h    |   1 +
>  drivers/gpu/drm/amd/amdgpu/amdgpu_umc.c    |   3 +
>  drivers/gpu/drm/amd/amdgpu/amdgpu_virt.c   | 138 +++++++++++++++++++++
>  drivers/gpu/drm/amd/amdgpu/amdgpu_virt.h   |  15 +++
>  7 files changed, 231 insertions(+), 7 deletions(-)
> 
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c
> index 383bbee87df5..c22a9925cba7 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c
> @@ -4235,7 +4235,10 @@ int amdgpu_device_init(struct amdgpu_device *adev,
>  	 * for throttling interrupt) = 60 seconds.
>  	 */
>  	ratelimit_state_init(&adev->throttling_logging_rs, (60 - 1) * HZ, 1);
> +	ratelimit_state_init(&adev->virt.ras_telemetry_rs, 5 * HZ, 1);
> +
>  	ratelimit_set_flags(&adev->throttling_logging_rs, RATELIMIT_MSG_ON_RELEASE);
> +	ratelimit_set_flags(&adev->virt.ras_telemetry_rs, RATELIMIT_MSG_ON_RELEASE);
>  
>  	/* Registers mapping */
>  	/* TODO: block userspace mapping of io register */
> @@ -5185,6 +5188,9 @@ static int amdgpu_device_reset_sriov(struct amdgpu_device *adev,
>  	    amdgpu_ip_version(adev, GC_HWIP, 0) == IP_VERSION(9, 4, 4) ||
>  	    amdgpu_ip_version(adev, GC_HWIP, 0) == IP_VERSION(11, 0, 3))
>  		amdgpu_ras_resume(adev);
> +
> +	amdgpu_virt_ras_telemetry_post_reset(adev);
> +
>  	return 0;
>  }
>  
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_gfx.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_gfx.c
> index b8cc4b146bdc..8c9fcfb7f22e 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_gfx.c
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_gfx.c
> @@ -885,6 +885,9 @@ int amdgpu_gfx_ras_late_init(struct amdgpu_device *adev, struct ras_common_if *r
>  		if (r)
>  			return r;
>  
> +		if (amdgpu_sriov_vf(adev))
> +			return r;
> +
>  		if (adev->gfx.cp_ecc_error_irq.funcs) {
>  			r = amdgpu_irq_get(adev, &adev->gfx.cp_ecc_error_irq, 0);
>  			if (r)
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.c
> index 89d87dc17ac1..a8b4f4178e5a 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.c
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.c
> @@ -1214,6 +1214,42 @@ static void amdgpu_ras_error_generate_report(struct amdgpu_device *adev,
>  	}
>  }
>  
> +static void amdgpu_ras_virt_error_generate_report(struct amdgpu_device *adev,
> +						  struct ras_query_if *query_if,
> +						  struct ras_err_data *err_data,
> +						  struct ras_query_context *qctx)
> +{
> +	unsigned long new_ue, new_ce, new_de;
> +	struct ras_manager *obj = amdgpu_ras_find_obj(adev, &query_if->head);
> +	const char *blk_name = get_ras_block_str(&query_if->head);
> +	u64 event_id = qctx->event_id;
> +
> +	new_ce = err_data->ce_count - obj->err_data.ce_count;
> +	new_ue = err_data->ue_count - obj->err_data.ue_count;
> +	new_de = err_data->de_count - obj->err_data.de_count;
> +
> +	if (new_ce) {
> +		RAS_EVENT_LOG(adev, event_id, "%lu correctable hardware errors "
> +			      "detected in %s block\n",
> +			      new_ce,
> +			      blk_name);
> +	}
> +
> +	if (new_ue) {
> +		RAS_EVENT_LOG(adev, event_id, "%lu uncorrectable hardware errors "
> +			      "detected in %s block\n",
> +			      new_ue,
> +			      blk_name);
> +	}
> +
> +	if (new_de) {
> +		RAS_EVENT_LOG(adev, event_id, "%lu deferred hardware errors "
> +			      "detected in %s block\n",
> +			      new_de,
> +			      blk_name);
> +	}
> +}
> +
>  static void amdgpu_rasmgr_error_data_statistic_update(struct ras_manager *obj, struct ras_err_data *err_data)
>  {
>  	struct ras_err_node *err_node;
> @@ -1237,6 +1273,15 @@ static void amdgpu_rasmgr_error_data_statistic_update(struct ras_manager *obj, s
>  	}
>  }
>  
> +static void amdgpu_ras_mgr_virt_error_data_statistics_update(struct ras_manager *obj,
> +							     struct ras_err_data *err_data)
> +{
> +	/* Host reports absolute counts */
> +	obj->err_data.ue_count = err_data->ue_count;
> +	obj->err_data.ce_count = err_data->ce_count;
> +	obj->err_data.de_count = err_data->de_count;
> +}
> +
>  static struct ras_manager *get_ras_manager(struct amdgpu_device *adev, enum amdgpu_ras_block blk)
>  {
>  	struct ras_common_if head;
> @@ -1323,7 +1368,9 @@ static int amdgpu_ras_query_error_status_helper(struct amdgpu_device *adev,
>  	if (error_query_mode == AMDGPU_RAS_INVALID_ERROR_QUERY)
>  		return -EINVAL;
>  
> -	if (error_query_mode == AMDGPU_RAS_DIRECT_ERROR_QUERY) {
> +	if (error_query_mode == AMDGPU_RAS_VIRT_ERROR_COUNT_QUERY) {
> +		return amdgpu_virt_req_ras_err_count(adev, blk, err_data);
> +	} else if (error_query_mode == AMDGPU_RAS_DIRECT_ERROR_QUERY) {
>  		if (info->head.block == AMDGPU_RAS_BLOCK__UMC) {
>  			amdgpu_ras_get_ecc_info(adev, err_data);
>  		} else {
> @@ -1405,14 +1452,22 @@ static int amdgpu_ras_query_error_status_with_event(struct amdgpu_device *adev,
>  	if (ret)
>  		goto out_fini_err_data;
>  
> -	amdgpu_rasmgr_error_data_statistic_update(obj, &err_data);
> +	if (error_query_mode != AMDGPU_RAS_VIRT_ERROR_COUNT_QUERY) {
> +		amdgpu_rasmgr_error_data_statistic_update(obj, &err_data);
> +		amdgpu_ras_error_generate_report(adev, info, &err_data, &qctx);
> +	} else {
> +		/* Host provides absolute error counts. First generate the report
> +		 * using the previous VF internal count against new host count.
> +		 * Then Update VF internal count.
> +		 */
> +		amdgpu_ras_virt_error_generate_report(adev, info, &err_data, &qctx);
> +		amdgpu_ras_mgr_virt_error_data_statistics_update(obj, &err_data);
> +	}
>  
>  	info->ue_count = obj->err_data.ue_count;
>  	info->ce_count = obj->err_data.ce_count;
>  	info->de_count = obj->err_data.de_count;
>  
> -	amdgpu_ras_error_generate_report(adev, info, &err_data, &qctx);
> -
>  out_fini_err_data:
>  	amdgpu_ras_error_data_fini(&err_data);
>  
> @@ -3929,7 +3984,7 @@ int amdgpu_ras_late_init(struct amdgpu_device *adev)
>  	}
>  
>  	/* Guest side doesn't need init ras feature */
> -	if (amdgpu_sriov_vf(adev))
> +	if (amdgpu_sriov_vf(adev) && !amdgpu_sriov_ras_telemetry_en(adev))
>  		return 0;
>  
>  	list_for_each_entry_safe(node, tmp, &adev->ras_list, node) {
> @@ -4396,11 +4451,14 @@ bool amdgpu_ras_get_error_query_mode(struct amdgpu_device *adev,
>  		return false;
>  	}
>  
> -	if ((smu_funcs && smu_funcs->set_debug_mode) || (mca_funcs && mca_funcs->mca_set_debug_mode))
> +	if (amdgpu_sriov_vf(adev)) {
> +		*error_query_mode = AMDGPU_RAS_VIRT_ERROR_COUNT_QUERY;
> +	} else if ((smu_funcs && smu_funcs->set_debug_mode) || (mca_funcs && mca_funcs->mca_set_debug_mode)) {
>  		*error_query_mode =
>  			(con->is_aca_debug_mode) ? AMDGPU_RAS_DIRECT_ERROR_QUERY : AMDGPU_RAS_FIRMWARE_ERROR_QUERY;
> -	else
> +	} else {
>  		*error_query_mode = AMDGPU_RAS_DIRECT_ERROR_QUERY;
> +	}
>  
>  	return true;
>  }
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.h b/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.h
> index 871b2d6278e0..6db772ecfee4 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.h
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_ras.h
> @@ -365,6 +365,7 @@ enum amdgpu_ras_error_query_mode {
>  	AMDGPU_RAS_INVALID_ERROR_QUERY		= 0,
>  	AMDGPU_RAS_DIRECT_ERROR_QUERY		= 1,
>  	AMDGPU_RAS_FIRMWARE_ERROR_QUERY		= 2,
> +	AMDGPU_RAS_VIRT_ERROR_COUNT_QUERY	= 3,
>  };
>  
>  /* ras error status reisger fields */
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_umc.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_umc.c
> index bb7b9b2eaac1..896f3609b0ee 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_umc.c
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_umc.c
> @@ -318,6 +318,9 @@ int amdgpu_umc_ras_late_init(struct amdgpu_device *adev, struct ras_common_if *r
>  	if (r)
>  		return r;
>  
> +	if (amdgpu_sriov_vf(adev))
> +		return r;
> +
>  	if (amdgpu_ras_is_supported(adev, ras_block->block)) {
>  		r = amdgpu_irq_get(adev, &adev->gmc.ecc_irq, 0);
>  		if (r)
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.c
> index 53297c40f09c..b1e584e4ef13 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.c
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.c
> @@ -524,6 +524,8 @@ static int amdgpu_virt_read_pf2vf_data(struct amdgpu_device *adev)
>  		adev->unique_id =
>  			((struct amd_sriov_msg_pf2vf_info *)pf2vf_info)->uuid;
>  		adev->virt.ras_en_caps.all = ((struct amd_sriov_msg_pf2vf_info *)pf2vf_info)->ras_en_caps.all;
> +		adev->virt.ras_telemetry_en_caps.all =
> +			((struct amd_sriov_msg_pf2vf_info *)pf2vf_info)->ras_telemetry_en_caps.all;
>  		break;
>  	default:
>  		dev_err(adev->dev, "invalid pf2vf version: 0x%x\n", pf2vf_info->version);
> @@ -704,6 +706,8 @@ void amdgpu_virt_exchange_data(struct amdgpu_device *adev)
>  			adev->virt.fw_reserve.p_vf2pf =
>  				(struct amd_sriov_msg_vf2pf_info_header *)
>  				(adev->mman.fw_vram_usage_va + (AMD_SRIOV_MSG_VF2PF_OFFSET_KB << 10));
> +			adev->virt.fw_reserve.ras_telemetry =
> +				(adev->mman.fw_vram_usage_va + (AMD_SRIOV_MSG_RAS_TELEMETRY_OFFSET_KB << 10));

Shouldn't it check if it's telemetry capable before this assignment?

Overall, it feels like it would be better to have a ras manager/ras
control block under virt which maintains

	count_cache
	ras_telemetry_rs
	req_ras_err_count
	ras_telemetry_en_caps

It gets initialized only if ras capability and ras telemetry capability
is detected - amdgpu_virt_ras_init and then go from there. Also, makes
it easier for read, if all starts like amdgpu_virt_ras

	amdgpu_virt_req_ras_err_count => amdgpu_virt_ras_req_err_count
	amdgpu_virt_cache_host_ => amdgpu_virt_ras_cache_host_errors

and so forth.

>  		} else if (adev->mman.drv_vram_usage_va) {
>  			adev->virt.fw_reserve.p_pf2vf =
>  				(struct amd_sriov_msg_pf2vf_info_header *)
> @@ -711,6 +715,8 @@ void amdgpu_virt_exchange_data(struct amdgpu_device *adev)
>  			adev->virt.fw_reserve.p_vf2pf =
>  				(struct amd_sriov_msg_vf2pf_info_header *)
>  				(adev->mman.drv_vram_usage_va + (AMD_SRIOV_MSG_VF2PF_OFFSET_KB << 10));
> +			adev->virt.fw_reserve.ras_telemetry =
> +				(adev->mman.drv_vram_usage_va + (AMD_SRIOV_MSG_RAS_TELEMETRY_OFFSET_KB << 10));
>  		}
>  
>  		amdgpu_virt_read_pf2vf_data(adev);
> @@ -1197,3 +1203,135 @@ bool amdgpu_virt_get_ras_capability(struct amdgpu_device *adev)
>  
>  	return true;
>  }
> +
> +static inline enum amd_sriov_ras_telemetry_gpu_block
> +amdgpu_ras_block_to_sriov(struct amdgpu_device *adev, enum amdgpu_ras_block block) {
> +	switch (block) {
> +	case AMDGPU_RAS_BLOCK__UMC:
> +		return RAS_TELEMETRY_GPU_BLOCK_UMC;
> +	case AMDGPU_RAS_BLOCK__SDMA:
> +		return RAS_TELEMETRY_GPU_BLOCK_SDMA;
> +	case AMDGPU_RAS_BLOCK__GFX:
> +		return RAS_TELEMETRY_GPU_BLOCK_GFX;
> +	case AMDGPU_RAS_BLOCK__MMHUB:
> +		return RAS_TELEMETRY_GPU_BLOCK_MMHUB;
> +	case AMDGPU_RAS_BLOCK__ATHUB:
> +		return RAS_TELEMETRY_GPU_BLOCK_ATHUB;
> +	case AMDGPU_RAS_BLOCK__PCIE_BIF:
> +		return RAS_TELEMETRY_GPU_BLOCK_PCIE_BIF;
> +	case AMDGPU_RAS_BLOCK__HDP:
> +		return RAS_TELEMETRY_GPU_BLOCK_HDP;
> +	case AMDGPU_RAS_BLOCK__XGMI_WAFL:
> +		return RAS_TELEMETRY_GPU_BLOCK_XGMI_WAFL;
> +	case AMDGPU_RAS_BLOCK__DF:
> +		return RAS_TELEMETRY_GPU_BLOCK_DF;
> +	case AMDGPU_RAS_BLOCK__SMN:
> +		return RAS_TELEMETRY_GPU_BLOCK_SMN;
> +	case AMDGPU_RAS_BLOCK__SEM:
> +		return RAS_TELEMETRY_GPU_BLOCK_SEM;
> +	case AMDGPU_RAS_BLOCK__MP0:
> +		return RAS_TELEMETRY_GPU_BLOCK_MP0;
> +	case AMDGPU_RAS_BLOCK__MP1:
> +		return RAS_TELEMETRY_GPU_BLOCK_MP1;
> +	case AMDGPU_RAS_BLOCK__FUSE:
> +		return RAS_TELEMETRY_GPU_BLOCK_FUSE;
> +	case AMDGPU_RAS_BLOCK__MCA:
> +		return RAS_TELEMETRY_GPU_BLOCK_MCA;
> +	case AMDGPU_RAS_BLOCK__VCN:
> +		return RAS_TELEMETRY_GPU_BLOCK_VCN;
> +	case AMDGPU_RAS_BLOCK__JPEG:
> +		return RAS_TELEMETRY_GPU_BLOCK_JPEG;
> +	case AMDGPU_RAS_BLOCK__IH:
> +		return RAS_TELEMETRY_GPU_BLOCK_IH;
> +	case AMDGPU_RAS_BLOCK__MPIO:
> +		return RAS_TELEMETRY_GPU_BLOCK_MPIO;
> +	default:
> +		dev_err(adev->dev, "Unsupported SRIOV RAS telemetry block 0x%x\n", block);
> +		return RAS_TELEMETRY_GPU_BLOCK_COUNT;
> +	}
> +}
> +
> +static int amdgpu_virt_cache_host_error_counts(struct amdgpu_device *adev,
> +					       struct amdsriov_ras_telemetry *host_telemetry)
> +{
> +	struct amd_sriov_ras_telemetry_error_count *tmp = { 0 };
> +	uint32_t checksum, used_size, tmp_check;
> +
> +	checksum = host_telemetry->header.checksum;
> +	used_size = host_telemetry->header.used_size;
> +
> +	if (used_size > (AMD_SRIOV_RAS_TELEMETRY_SIZE_KB << 10))
> +		return 0;
> +
> +	tmp = kmalloc(used_size, GFP_KERNEL);
> +	if (!tmp)
> +		return -ENOMEM;
> +
> +	memcpy(tmp, &host_telemetry->body.error_count, used_size);
> +
> +	tmp_check = amd_sriov_msg_checksum(tmp, used_size, 0, 0);
> +	if (checksum != amd_sriov_msg_checksum(tmp, used_size, 0, 0))
> +		goto out;
> +
> +	/* Copy sanitized data to guest cache */
> +	memcpy(&adev->virt.count_cache, tmp, sizeof(adev->virt.count_cache));
> +out:
> +	kfree(tmp);
> +
> +	return 0;
> +}
> +
> +static int amdgpu_virt_req_ras_err_count_internal(struct amdgpu_device *adev, bool force_update)
> +{
> +	struct amdgpu_virt *virt = &adev->virt;
> +
> +	/* Host allows 15 ras telemetry requests per 60 seconds. Afterwhich, the Host
> +	 * will ignore incoming guest messages. Ratelimit the guest messages to
> +	 * prevent guest self DOS.
> +	 */
> +	if (__ratelimit(&adev->virt.ras_telemetry_rs) || force_update) {
> +		if (!virt->ops->req_ras_err_count(adev))
> +			amdgpu_virt_cache_host_error_counts(adev,
> +				adev->virt.fw_reserve.ras_telemetry);
> +	}
> +
> +	return 0;
> +}
> +
> +/* Bypass ACA interface and query ECC counts directly from host */
> +int amdgpu_virt_req_ras_err_count(struct amdgpu_device *adev, enum amdgpu_ras_block block,
> +				  struct ras_err_data *err_data)
> +{
> +	int ret;
> +	enum amd_sriov_ras_telemetry_gpu_block sriov_block;
> +
> +	sriov_block = amdgpu_ras_block_to_sriov(adev, block);
> +
> +	if (sriov_block >= RAS_TELEMETRY_GPU_BLOCK_COUNT ||
> +	    !amdgpu_sriov_ras_telemetry_block_en(adev, sriov_block))
> +		return -EOPNOTSUPP;
> +
> +	/* Host Access may be lost during reset, just return last cached data. */
> +	if (down_read_trylock(&adev->reset_domain->sem)) {
> +		amdgpu_virt_req_ras_err_count_internal(adev, false);
> +		up_read(&adev->reset_domain->sem);
> +	}
> +
> +	err_data->ue_count = adev->virt.count_cache.block[sriov_block].ue_count;
> +	err_data->ce_count = adev->virt.count_cache.block[sriov_block].ce_count;
> +	err_data->de_count = adev->virt.count_cache.block[sriov_block].de_count;
> +
> +	return 0;
> +}
> +
> +int amdgpu_virt_ras_telemetry_post_reset(struct amdgpu_device *adev)
> +{
> +	unsigned long ue_count, ce_count;
> +
> +	if (amdgpu_sriov_ras_telemetry_en(adev)) {
> +		amdgpu_virt_req_ras_err_count_internal(adev, true);
> +		amdgpu_ras_query_error_count(adev, &ce_count, &ue_count, NULL);
> +	}
> +
> +	return 0;
> +}
> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.h b/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.h
> index f0ff84add692..5381b8d596e6 100644
> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.h
> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_virt.h
> @@ -104,6 +104,7 @@ struct amdgpu_virt_ops {
>  struct amdgpu_virt_fw_reserve {
>  	struct amd_sriov_msg_pf2vf_info_header *p_pf2vf;
>  	struct amd_sriov_msg_vf2pf_info_header *p_vf2pf;
> +	void *ras_telemetry;
>  	unsigned int checksum_key;
>  };
>  
> @@ -138,6 +139,7 @@ enum AMDGIM_FEATURE_FLAG {
>  	/* MES info */
>  	AMDGIM_FEATURE_MES_INFO_ENABLE = (1 << 8),
>  	AMDGIM_FEATURE_RAS_CAPS = (1 << 9),
> +	AMDGIM_FEATURE_RAS_TELEMETRY = (1 << 10),
>  };
>  
>  enum AMDGIM_REG_ACCESS_FLAG {
> @@ -280,6 +282,10 @@ struct amdgpu_virt {
>  	struct mutex rlcg_reg_lock;
>  
>  	union amd_sriov_ras_caps ras_en_caps;
> +	union amd_sriov_ras_caps ras_telemetry_en_caps;
> +
> +	struct ratelimit_state ras_telemetry_rs;
> +	struct amd_sriov_ras_telemetry_error_count count_cache;
>  };
>  
>  struct amdgpu_video_codec_info;
> @@ -327,6 +333,12 @@ struct amdgpu_video_codec_info;
>  #define amdgpu_sriov_ras_caps_en(adev) \
>  ((adev)->virt.gim_feature & AMDGIM_FEATURE_RAS_CAPS)
>  
> +#define amdgpu_sriov_ras_telemetry_en(adev) \
> +(((adev)->virt.gim_feature & AMDGIM_FEATURE_RAS_TELEMETRY) && (adev)->virt.fw_reserve.ras_telemetry)
> +

As a matter of coding style/easier read,  would be good if you could
maintain gim_feature checks at one place. Now it's mixed across.

Thanks,
Lijo

> +#define amdgpu_sriov_ras_telemetry_block_en(adev, sriov_blk) \
> +(amdgpu_sriov_ras_telemetry_en((adev)) && (adev)->virt.ras_telemetry_en_caps.all & BIT(sriov_blk))
> +
>  static inline bool is_virtual_machine(void)
>  {
>  #if defined(CONFIG_X86)
> @@ -391,4 +403,7 @@ bool amdgpu_virt_get_rlcg_reg_access_flag(struct amdgpu_device *adev,
>  					  bool write, u32 *rlcg_flag);
>  u32 amdgpu_virt_rlcg_reg_rw(struct amdgpu_device *adev, u32 offset, u32 v, u32 flag, u32 xcc_id);
>  bool amdgpu_virt_get_ras_capability(struct amdgpu_device *adev);
> +int amdgpu_virt_req_ras_err_count(struct amdgpu_device *adev, enum amdgpu_ras_block block,
> +				  struct ras_err_data *err_data);
> +int amdgpu_virt_ras_telemetry_post_reset(struct amdgpu_device *adev);
>  #endif



[Index of Archives]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux