diff options
Diffstat (limited to 'drivers/gpu/drm/amd/amdgpu')
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu.h | 46 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_acpi.c | 131 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_atpx_handler.c | 6 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c | 2 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_ib.c | 12 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_pm.c | 2 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_vcn.c | 33 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c | 8 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/vce_v3_0.c | 4 | 
9 files changed, 164 insertions, 80 deletions
| diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu.h b/drivers/gpu/drm/amd/amdgpu/amdgpu.h index a59c07590cee..7dcbac8af9a7 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu.h +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu.h @@ -190,6 +190,7 @@ struct amdgpu_job;  struct amdgpu_irq_src;  struct amdgpu_fpriv;  struct amdgpu_bo_va_mapping; +struct amdgpu_atif;  enum amdgpu_cp_irq {  	AMDGPU_CP_IRQ_GFX_EOP = 0, @@ -1269,43 +1270,6 @@ struct amdgpu_vram_scratch {  /*   * ACPI   */ -struct amdgpu_atif_notification_cfg { -	bool enabled; -	int command_code; -}; - -struct amdgpu_atif_notifications { -	bool display_switch; -	bool expansion_mode_change; -	bool thermal_state; -	bool forced_power_state; -	bool system_power_state; -	bool display_conf_change; -	bool px_gfx_switch; -	bool brightness_change; -	bool dgpu_display_event; -}; - -struct amdgpu_atif_functions { -	bool system_params; -	bool sbios_requests; -	bool select_active_disp; -	bool lid_state; -	bool get_tv_standard; -	bool set_tv_standard; -	bool get_panel_expansion_mode; -	bool set_panel_expansion_mode; -	bool temperature_change; -	bool graphics_device_types; -}; - -struct amdgpu_atif { -	struct amdgpu_atif_notifications notifications; -	struct amdgpu_atif_functions functions; -	struct amdgpu_atif_notification_cfg notification_cfg; -	struct amdgpu_encoder *encoder_for_bl; -}; -  struct amdgpu_atcs_functions {  	bool get_ext_state;  	bool pcie_perf_req; @@ -1466,7 +1430,7 @@ struct amdgpu_device {  #if defined(CONFIG_DEBUG_FS)  	struct dentry			*debugfs_regs[AMDGPU_DEBUGFS_MAX_COMPONENTS];  #endif -	struct amdgpu_atif		atif; +	struct amdgpu_atif		*atif;  	struct amdgpu_atcs		atcs;  	struct mutex			srbm_mutex;  	/* GRBM index mutex. Protects concurrent access to GRBM index */ @@ -1894,6 +1858,12 @@ static inline bool amdgpu_atpx_dgpu_req_power_for_displays(void) { return false;  static inline bool amdgpu_has_atpx(void) { return false; }  #endif +#if defined(CONFIG_VGA_SWITCHEROO) && defined(CONFIG_ACPI) +void *amdgpu_atpx_get_dhandle(void); +#else +static inline void *amdgpu_atpx_get_dhandle(void) { return NULL; } +#endif +  /*   * KMS   */ diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_acpi.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_acpi.c index 8fa850a070e0..0d8c3fc6eace 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_acpi.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_acpi.c @@ -34,6 +34,45 @@  #include "amd_acpi.h"  #include "atom.h" +struct amdgpu_atif_notification_cfg { +	bool enabled; +	int command_code; +}; + +struct amdgpu_atif_notifications { +	bool display_switch; +	bool expansion_mode_change; +	bool thermal_state; +	bool forced_power_state; +	bool system_power_state; +	bool display_conf_change; +	bool px_gfx_switch; +	bool brightness_change; +	bool dgpu_display_event; +}; + +struct amdgpu_atif_functions { +	bool system_params; +	bool sbios_requests; +	bool select_active_disp; +	bool lid_state; +	bool get_tv_standard; +	bool set_tv_standard; +	bool get_panel_expansion_mode; +	bool set_panel_expansion_mode; +	bool temperature_change; +	bool graphics_device_types; +}; + +struct amdgpu_atif { +	acpi_handle handle; + +	struct amdgpu_atif_notifications notifications; +	struct amdgpu_atif_functions functions; +	struct amdgpu_atif_notification_cfg notification_cfg; +	struct amdgpu_encoder *encoder_for_bl; +}; +  /* Call the ATIF method   */  /** @@ -46,8 +85,9 @@   * Executes the requested ATIF function (all asics).   * Returns a pointer to the acpi output buffer.   */ -static union acpi_object *amdgpu_atif_call(acpi_handle handle, int function, -		struct acpi_buffer *params) +static union acpi_object *amdgpu_atif_call(struct amdgpu_atif *atif, +					   int function, +					   struct acpi_buffer *params)  {  	acpi_status status;  	union acpi_object atif_arg_elements[2]; @@ -70,7 +110,8 @@ static union acpi_object *amdgpu_atif_call(acpi_handle handle, int function,  		atif_arg_elements[1].integer.value = 0;  	} -	status = acpi_evaluate_object(handle, "ATIF", &atif_arg, &buffer); +	status = acpi_evaluate_object(atif->handle, NULL, &atif_arg, +				      &buffer);  	/* Fail only if calling the method fails and ATIF is supported */  	if (ACPI_FAILURE(status) && status != AE_NOT_FOUND) { @@ -141,15 +182,14 @@ static void amdgpu_atif_parse_functions(struct amdgpu_atif_functions *f, u32 mas   * (all asics).   * returns 0 on success, error on failure.   */ -static int amdgpu_atif_verify_interface(acpi_handle handle, -		struct amdgpu_atif *atif) +static int amdgpu_atif_verify_interface(struct amdgpu_atif *atif)  {  	union acpi_object *info;  	struct atif_verify_interface output;  	size_t size;  	int err = 0; -	info = amdgpu_atif_call(handle, ATIF_FUNCTION_VERIFY_INTERFACE, NULL); +	info = amdgpu_atif_call(atif, ATIF_FUNCTION_VERIFY_INTERFACE, NULL);  	if (!info)  		return -EIO; @@ -176,6 +216,35 @@ out:  	return err;  } +static acpi_handle amdgpu_atif_probe_handle(acpi_handle dhandle) +{ +	acpi_handle handle = NULL; +	char acpi_method_name[255] = { 0 }; +	struct acpi_buffer buffer = { sizeof(acpi_method_name), acpi_method_name }; +	acpi_status status; + +	/* For PX/HG systems, ATIF and ATPX are in the iGPU's namespace, on dGPU only +	 * systems, ATIF is in the dGPU's namespace. +	 */ +	status = acpi_get_handle(dhandle, "ATIF", &handle); +	if (ACPI_SUCCESS(status)) +		goto out; + +	if (amdgpu_has_atpx()) { +		status = acpi_get_handle(amdgpu_atpx_get_dhandle(), "ATIF", +					 &handle); +		if (ACPI_SUCCESS(status)) +			goto out; +	} + +	DRM_DEBUG_DRIVER("No ATIF handle found\n"); +	return NULL; +out: +	acpi_get_name(handle, ACPI_FULL_PATHNAME, &buffer); +	DRM_DEBUG_DRIVER("Found ATIF handle %s\n", acpi_method_name); +	return handle; +} +  /**   * amdgpu_atif_get_notification_params - determine notify configuration   * @@ -188,15 +257,16 @@ out:   * where n is specified in the result if a notifier is used.   * Returns 0 on success, error on failure.   */ -static int amdgpu_atif_get_notification_params(acpi_handle handle, -		struct amdgpu_atif_notification_cfg *n) +static int amdgpu_atif_get_notification_params(struct amdgpu_atif *atif)  {  	union acpi_object *info; +	struct amdgpu_atif_notification_cfg *n = &atif->notification_cfg;  	struct atif_system_params params;  	size_t size;  	int err = 0; -	info = amdgpu_atif_call(handle, ATIF_FUNCTION_GET_SYSTEM_PARAMETERS, NULL); +	info = amdgpu_atif_call(atif, ATIF_FUNCTION_GET_SYSTEM_PARAMETERS, +				NULL);  	if (!info) {  		err = -EIO;  		goto out; @@ -250,14 +320,15 @@ out:   * (all asics).   * Returns 0 on success, error on failure.   */ -static int amdgpu_atif_get_sbios_requests(acpi_handle handle, -		struct atif_sbios_requests *req) +static int amdgpu_atif_get_sbios_requests(struct amdgpu_atif *atif, +					  struct atif_sbios_requests *req)  {  	union acpi_object *info;  	size_t size;  	int count = 0; -	info = amdgpu_atif_call(handle, ATIF_FUNCTION_GET_SYSTEM_BIOS_REQUESTS, NULL); +	info = amdgpu_atif_call(atif, ATIF_FUNCTION_GET_SYSTEM_BIOS_REQUESTS, +				NULL);  	if (!info)  		return -EIO; @@ -290,11 +361,10 @@ out:   * Returns NOTIFY code   */  static int amdgpu_atif_handler(struct amdgpu_device *adev, -			struct acpi_bus_event *event) +			       struct acpi_bus_event *event)  { -	struct amdgpu_atif *atif = &adev->atif; +	struct amdgpu_atif *atif = adev->atif;  	struct atif_sbios_requests req; -	acpi_handle handle;  	int count;  	DRM_DEBUG_DRIVER("event, device_class = %s, type = %#x\n", @@ -303,14 +373,14 @@ static int amdgpu_atif_handler(struct amdgpu_device *adev,  	if (strcmp(event->device_class, ACPI_VIDEO_CLASS) != 0)  		return NOTIFY_DONE; -	if (!atif->notification_cfg.enabled || +	if (!atif || +	    !atif->notification_cfg.enabled ||  	    event->type != atif->notification_cfg.command_code)  		/* Not our event */  		return NOTIFY_DONE;  	/* Check pending SBIOS requests */ -	handle = ACPI_HANDLE(&adev->pdev->dev); -	count = amdgpu_atif_get_sbios_requests(handle, &req); +	count = amdgpu_atif_get_sbios_requests(atif, &req);  	if (count <= 0)  		return NOTIFY_DONE; @@ -641,8 +711,8 @@ static int amdgpu_acpi_event(struct notifier_block *nb,   */  int amdgpu_acpi_init(struct amdgpu_device *adev)  { -	acpi_handle handle; -	struct amdgpu_atif *atif = &adev->atif; +	acpi_handle handle, atif_handle; +	struct amdgpu_atif *atif;  	struct amdgpu_atcs *atcs = &adev->atcs;  	int ret; @@ -658,12 +728,26 @@ int amdgpu_acpi_init(struct amdgpu_device *adev)  		DRM_DEBUG_DRIVER("Call to ATCS verify_interface failed: %d\n", ret);  	} +	/* Probe for ATIF, and initialize it if found */ +	atif_handle = amdgpu_atif_probe_handle(handle); +	if (!atif_handle) +		goto out; + +	atif = kzalloc(sizeof(*atif), GFP_KERNEL); +	if (!atif) { +		DRM_WARN("Not enough memory to initialize ATIF\n"); +		goto out; +	} +	atif->handle = atif_handle; +  	/* Call the ATIF method */ -	ret = amdgpu_atif_verify_interface(handle, atif); +	ret = amdgpu_atif_verify_interface(atif);  	if (ret) {  		DRM_DEBUG_DRIVER("Call to ATIF verify_interface failed: %d\n", ret); +		kfree(atif);  		goto out;  	} +	adev->atif = atif;  	if (atif->notifications.brightness_change) {  		struct drm_encoder *tmp; @@ -693,8 +777,7 @@ int amdgpu_acpi_init(struct amdgpu_device *adev)  	}  	if (atif->functions.system_params) { -		ret = amdgpu_atif_get_notification_params(handle, -				&atif->notification_cfg); +		ret = amdgpu_atif_get_notification_params(atif);  		if (ret) {  			DRM_DEBUG_DRIVER("Call to GET_SYSTEM_PARAMS failed: %d\n",  					ret); @@ -720,4 +803,6 @@ out:  void amdgpu_acpi_fini(struct amdgpu_device *adev)  {  	unregister_acpi_notifier(&adev->acpi_nb); +	if (adev->atif) +		kfree(adev->atif);  } diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_atpx_handler.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_atpx_handler.c index daa06e7c5bb7..9ab89371d9e8 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_atpx_handler.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_atpx_handler.c @@ -90,6 +90,12 @@ bool amdgpu_atpx_dgpu_req_power_for_displays(void) {  	return amdgpu_atpx_priv.atpx.dgpu_req_power_for_displays;  } +#if defined(CONFIG_ACPI) +void *amdgpu_atpx_get_dhandle(void) { +	return amdgpu_atpx_priv.dhandle; +} +#endif +  /**   * amdgpu_atpx_call - call an ATPX method   * diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c index 39ec6b8890a1..e74d620d9699 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c @@ -376,7 +376,7 @@ int amdgpu_fence_driver_start_ring(struct amdgpu_ring *ring,  	struct amdgpu_device *adev = ring->adev;  	uint64_t index; -	if (ring != &adev->uvd.inst[ring->me].ring) { +	if (ring->funcs->type != AMDGPU_RING_TYPE_UVD) {  		ring->fence_drv.cpu_addr = &adev->wb.wb[ring->fence_offs];  		ring->fence_drv.gpu_addr = adev->wb.gpu_addr + (ring->fence_offs * 4);  	} else { diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_ib.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_ib.c index f70eeed9ed76..7aaa263ad8c7 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_ib.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_ib.c @@ -231,6 +231,12 @@ int amdgpu_ib_schedule(struct amdgpu_ring *ring, unsigned num_ibs,  	if (ib->flags & AMDGPU_IB_FLAG_TC_WB_NOT_INVALIDATE)  		fence_flags |= AMDGPU_FENCE_FLAG_TC_WB_ONLY; +	/* wrap the last IB with fence */ +	if (job && job->uf_addr) { +		amdgpu_ring_emit_fence(ring, job->uf_addr, job->uf_sequence, +				       fence_flags | AMDGPU_FENCE_FLAG_64BIT); +	} +  	r = amdgpu_fence_emit(ring, f, fence_flags);  	if (r) {  		dev_err(adev->dev, "failed to emit fence (%d)\n", r); @@ -243,12 +249,6 @@ int amdgpu_ib_schedule(struct amdgpu_ring *ring, unsigned num_ibs,  	if (ring->funcs->insert_end)  		ring->funcs->insert_end(ring); -	/* wrap the last IB with fence */ -	if (job && job->uf_addr) { -		amdgpu_ring_emit_fence(ring, job->uf_addr, job->uf_sequence, -				       fence_flags | AMDGPU_FENCE_FLAG_64BIT); -	} -  	if (patch_offset != ~0 && ring->funcs->patch_cond_exec)  		amdgpu_ring_patch_cond_exec(ring, patch_offset); diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_pm.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_pm.c index b455da487782..fc818b4d849c 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_pm.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_pm.c @@ -1882,7 +1882,7 @@ void amdgpu_pm_compute_clocks(struct amdgpu_device *adev)  		if (!amdgpu_device_has_dc_support(adev)) {  			mutex_lock(&adev->pm.mutex);  			amdgpu_dpm_get_active_displays(adev); -			adev->pm.pm_display_cfg.num_display = adev->pm.dpm.new_active_crtcs; +			adev->pm.pm_display_cfg.num_display = adev->pm.dpm.new_active_crtc_count;  			adev->pm.pm_display_cfg.vrefresh = amdgpu_dpm_get_vrefresh(adev);  			adev->pm.pm_display_cfg.min_vblank_time = amdgpu_dpm_get_vblank_time(adev);  			/* we have issues with mclk switching with refresh rates over 120 hz on the non-DC code. */ diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_vcn.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_vcn.c index 127e87b470ff..1b4ad9b2a755 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_vcn.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_vcn.c @@ -52,7 +52,7 @@ int amdgpu_vcn_sw_init(struct amdgpu_device *adev)  	unsigned long bo_size;  	const char *fw_name;  	const struct common_firmware_header *hdr; -	unsigned version_major, version_minor, family_id; +	unsigned char fw_check;  	int r;  	INIT_DELAYED_WORK(&adev->vcn.idle_work, amdgpu_vcn_idle_work_handler); @@ -83,12 +83,33 @@ int amdgpu_vcn_sw_init(struct amdgpu_device *adev)  	hdr = (const struct common_firmware_header *)adev->vcn.fw->data;  	adev->vcn.fw_version = le32_to_cpu(hdr->ucode_version); -	family_id = le32_to_cpu(hdr->ucode_version) & 0xff; -	version_major = (le32_to_cpu(hdr->ucode_version) >> 24) & 0xff; -	version_minor = (le32_to_cpu(hdr->ucode_version) >> 8) & 0xff; -	DRM_INFO("Found VCN firmware Version: %hu.%hu Family ID: %hu\n", -		version_major, version_minor, family_id); +	/* Bit 20-23, it is encode major and non-zero for new naming convention. +	 * This field is part of version minor and DRM_DISABLED_FLAG in old naming +	 * convention. Since the l:wq!atest version minor is 0x5B and DRM_DISABLED_FLAG +	 * is zero in old naming convention, this field is always zero so far. +	 * These four bits are used to tell which naming convention is present. +	 */ +	fw_check = (le32_to_cpu(hdr->ucode_version) >> 20) & 0xf; +	if (fw_check) { +		unsigned int dec_ver, enc_major, enc_minor, vep, fw_rev; + +		fw_rev = le32_to_cpu(hdr->ucode_version) & 0xfff; +		enc_minor = (le32_to_cpu(hdr->ucode_version) >> 12) & 0xff; +		enc_major = fw_check; +		dec_ver = (le32_to_cpu(hdr->ucode_version) >> 24) & 0xf; +		vep = (le32_to_cpu(hdr->ucode_version) >> 28) & 0xf; +		DRM_INFO("Found VCN firmware Version ENC: %hu.%hu DEC: %hu VEP: %hu Revision: %hu\n", +			enc_major, enc_minor, dec_ver, vep, fw_rev); +	} else { +		unsigned int version_major, version_minor, family_id; + +		family_id = le32_to_cpu(hdr->ucode_version) & 0xff; +		version_major = (le32_to_cpu(hdr->ucode_version) >> 24) & 0xff; +		version_minor = (le32_to_cpu(hdr->ucode_version) >> 8) & 0xff; +		DRM_INFO("Found VCN firmware Version: %hu.%hu Family ID: %hu\n", +			version_major, version_minor, family_id); +	}  	bo_size = AMDGPU_GPU_PAGE_ALIGN(le32_to_cpu(hdr->ucode_size_bytes) + 8)  		  +  AMDGPU_VCN_STACK_SIZE + AMDGPU_VCN_HEAP_SIZE diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c index b0eb2f537392..edf16b2b957a 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c @@ -1463,7 +1463,9 @@ static int amdgpu_vm_bo_split_mapping(struct amdgpu_device *adev,  			uint64_t count;  			max_entries = min(max_entries, 16ull * 1024ull); -			for (count = 1; count < max_entries; ++count) { +			for (count = 1; +			     count < max_entries / (PAGE_SIZE / AMDGPU_GPU_PAGE_SIZE); +			     ++count) {  				uint64_t idx = pfn + count;  				if (pages_addr[idx] != @@ -1476,7 +1478,7 @@ static int amdgpu_vm_bo_split_mapping(struct amdgpu_device *adev,  				dma_addr = pages_addr;  			} else {  				addr = pages_addr[pfn]; -				max_entries = count; +				max_entries = count * (PAGE_SIZE / AMDGPU_GPU_PAGE_SIZE);  			}  		} else if (flags & AMDGPU_PTE_VALID) { @@ -1491,7 +1493,7 @@ static int amdgpu_vm_bo_split_mapping(struct amdgpu_device *adev,  		if (r)  			return r; -		pfn += last - start + 1; +		pfn += (last - start + 1) / (PAGE_SIZE / AMDGPU_GPU_PAGE_SIZE);  		if (nodes && nodes->size == pfn) {  			pfn = 0;  			++nodes; diff --git a/drivers/gpu/drm/amd/amdgpu/vce_v3_0.c b/drivers/gpu/drm/amd/amdgpu/vce_v3_0.c index 0999c843f623..a71b97519cc0 100644 --- a/drivers/gpu/drm/amd/amdgpu/vce_v3_0.c +++ b/drivers/gpu/drm/amd/amdgpu/vce_v3_0.c @@ -900,7 +900,7 @@ static const struct amdgpu_ring_funcs vce_v3_0_ring_phys_funcs = {  	.emit_frame_size =  		4 + /* vce_v3_0_emit_pipeline_sync */  		6, /* amdgpu_vce_ring_emit_fence x1 no user fence */ -	.emit_ib_size = 5, /* vce_v3_0_ring_emit_ib */ +	.emit_ib_size = 4, /* amdgpu_vce_ring_emit_ib */  	.emit_ib = amdgpu_vce_ring_emit_ib,  	.emit_fence = amdgpu_vce_ring_emit_fence,  	.test_ring = amdgpu_vce_ring_test_ring, @@ -924,7 +924,7 @@ static const struct amdgpu_ring_funcs vce_v3_0_ring_vm_funcs = {  		6 + /* vce_v3_0_emit_vm_flush */  		4 + /* vce_v3_0_emit_pipeline_sync */  		6 + 6, /* amdgpu_vce_ring_emit_fence x2 vm fence */ -	.emit_ib_size = 4, /* amdgpu_vce_ring_emit_ib */ +	.emit_ib_size = 5, /* vce_v3_0_ring_emit_ib */  	.emit_ib = vce_v3_0_ring_emit_ib,  	.emit_vm_flush = vce_v3_0_emit_vm_flush,  	.emit_pipeline_sync = vce_v3_0_emit_pipeline_sync, | 
