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_acp.c | 47 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_acpi.c | 131 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_atpx_handler.c | 7 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c | 4 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_device.c | 13 | ||||
| -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_object.c | 24 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_pm.c | 2 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_ttm.h | 1 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.c | 14 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.h | 2 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_vcn.c | 33 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c | 12 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_vram_mgr.c | 39 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/vce_v3_0.c | 4 | 
17 files changed, 276 insertions, 117 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_acp.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_acp.c index f4c474a95875..71efcf38f11b 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_acp.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_acp.c @@ -57,6 +57,10 @@  #define ACP_I2S_COMP2_CAP_REG_OFFSET		0xa8  #define ACP_I2S_COMP1_PLAY_REG_OFFSET		0x6c  #define ACP_I2S_COMP2_PLAY_REG_OFFSET		0x68 +#define ACP_BT_PLAY_REGS_START			0x14970 +#define ACP_BT_PLAY_REGS_END			0x14a24 +#define ACP_BT_COMP1_REG_OFFSET			0xac +#define ACP_BT_COMP2_REG_OFFSET			0xa8  #define mmACP_PGFSM_RETAIN_REG			0x51c9  #define mmACP_PGFSM_CONFIG_REG			0x51ca @@ -77,7 +81,7 @@  #define ACP_SOFT_RESET_DONE_TIME_OUT_VALUE	0x000000FF  #define ACP_TIMEOUT_LOOP			0x000000FF -#define ACP_DEVS				3 +#define ACP_DEVS				4  #define ACP_SRC_ID				162  enum { @@ -316,14 +320,13 @@ static int acp_hw_init(void *handle)  	if (adev->acp.acp_cell == NULL)  		return -ENOMEM; -	adev->acp.acp_res = kcalloc(4, sizeof(struct resource), GFP_KERNEL); - +	adev->acp.acp_res = kcalloc(5, sizeof(struct resource), GFP_KERNEL);  	if (adev->acp.acp_res == NULL) {  		kfree(adev->acp.acp_cell);  		return -ENOMEM;  	} -	i2s_pdata = kcalloc(2, sizeof(struct i2s_platform_data), GFP_KERNEL); +	i2s_pdata = kcalloc(3, sizeof(struct i2s_platform_data), GFP_KERNEL);  	if (i2s_pdata == NULL) {  		kfree(adev->acp.acp_res);  		kfree(adev->acp.acp_cell); @@ -358,6 +361,20 @@ static int acp_hw_init(void *handle)  	i2s_pdata[1].i2s_reg_comp1 = ACP_I2S_COMP1_CAP_REG_OFFSET;  	i2s_pdata[1].i2s_reg_comp2 = ACP_I2S_COMP2_CAP_REG_OFFSET; +	i2s_pdata[2].quirks = DW_I2S_QUIRK_COMP_REG_OFFSET; +	switch (adev->asic_type) { +	case CHIP_STONEY: +		i2s_pdata[2].quirks |= DW_I2S_QUIRK_16BIT_IDX_OVERRIDE; +		break; +	default: +		break; +	} + +	i2s_pdata[2].cap = DWC_I2S_PLAY | DWC_I2S_RECORD; +	i2s_pdata[2].snd_rates = SNDRV_PCM_RATE_8000_96000; +	i2s_pdata[2].i2s_reg_comp1 = ACP_BT_COMP1_REG_OFFSET; +	i2s_pdata[2].i2s_reg_comp2 = ACP_BT_COMP2_REG_OFFSET; +  	adev->acp.acp_res[0].name = "acp2x_dma";  	adev->acp.acp_res[0].flags = IORESOURCE_MEM;  	adev->acp.acp_res[0].start = acp_base; @@ -373,13 +390,18 @@ static int acp_hw_init(void *handle)  	adev->acp.acp_res[2].start = acp_base + ACP_I2S_CAP_REGS_START;  	adev->acp.acp_res[2].end = acp_base + ACP_I2S_CAP_REGS_END; -	adev->acp.acp_res[3].name = "acp2x_dma_irq"; -	adev->acp.acp_res[3].flags = IORESOURCE_IRQ; -	adev->acp.acp_res[3].start = amdgpu_irq_create_mapping(adev, 162); -	adev->acp.acp_res[3].end = adev->acp.acp_res[3].start; +	adev->acp.acp_res[3].name = "acp2x_dw_bt_i2s_play_cap"; +	adev->acp.acp_res[3].flags = IORESOURCE_MEM; +	adev->acp.acp_res[3].start = acp_base + ACP_BT_PLAY_REGS_START; +	adev->acp.acp_res[3].end = acp_base + ACP_BT_PLAY_REGS_END; + +	adev->acp.acp_res[4].name = "acp2x_dma_irq"; +	adev->acp.acp_res[4].flags = IORESOURCE_IRQ; +	adev->acp.acp_res[4].start = amdgpu_irq_create_mapping(adev, 162); +	adev->acp.acp_res[4].end = adev->acp.acp_res[4].start;  	adev->acp.acp_cell[0].name = "acp_audio_dma"; -	adev->acp.acp_cell[0].num_resources = 4; +	adev->acp.acp_cell[0].num_resources = 5;  	adev->acp.acp_cell[0].resources = &adev->acp.acp_res[0];  	adev->acp.acp_cell[0].platform_data = &adev->asic_type;  	adev->acp.acp_cell[0].pdata_size = sizeof(adev->asic_type); @@ -396,6 +418,12 @@ static int acp_hw_init(void *handle)  	adev->acp.acp_cell[2].platform_data = &i2s_pdata[1];  	adev->acp.acp_cell[2].pdata_size = sizeof(struct i2s_platform_data); +	adev->acp.acp_cell[3].name = "designware-i2s"; +	adev->acp.acp_cell[3].num_resources = 1; +	adev->acp.acp_cell[3].resources = &adev->acp.acp_res[3]; +	adev->acp.acp_cell[3].platform_data = &i2s_pdata[2]; +	adev->acp.acp_cell[3].pdata_size = sizeof(struct i2s_platform_data); +  	r = mfd_add_hotplug_devices(adev->acp.parent, adev->acp.acp_cell,  								ACP_DEVS);  	if (r) @@ -451,7 +479,6 @@ static int acp_hw_init(void *handle)  	val = cgs_read_register(adev->acp.cgs_device, mmACP_SOFT_RESET);  	val &= ~ACP_SOFT_RESET__SoftResetAud_MASK;  	cgs_write_register(adev->acp.cgs_device, mmACP_SOFT_RESET, val); -  	return 0;  } 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..ca8bf1c9a98e 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   * @@ -569,6 +575,7 @@ static const struct amdgpu_px_quirk amdgpu_px_quirk_list[] = {  	{ 0x1002, 0x6900, 0x1002, 0x0124, AMDGPU_PX_QUIRK_FORCE_ATPX },  	{ 0x1002, 0x6900, 0x1028, 0x0812, AMDGPU_PX_QUIRK_FORCE_ATPX },  	{ 0x1002, 0x6900, 0x1028, 0x0813, AMDGPU_PX_QUIRK_FORCE_ATPX }, +	{ 0x1002, 0x6900, 0x1025, 0x125A, AMDGPU_PX_QUIRK_FORCE_ATPX },  	{ 0, 0, 0, 0, 0 },  }; diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c index 82312a7bc6ad..9c85a90be293 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c @@ -927,6 +927,10 @@ static int amdgpu_cs_ib_vm_chunk(struct amdgpu_device *adev,  		r = amdgpu_bo_vm_update_pte(p);  		if (r)  			return r; + +		r = reservation_object_reserve_shared(vm->root.base.bo->tbo.resv); +		if (r) +			return r;  	}  	return amdgpu_cs_sync_rings(p); diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c index 3317d1536f4f..2c5f093e79e3 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_device.c @@ -2158,10 +2158,18 @@ bool amdgpu_device_asic_has_dc_support(enum amd_asic_type asic_type)  	switch (asic_type) {  #if defined(CONFIG_DRM_AMD_DC)  	case CHIP_BONAIRE: -	case CHIP_HAWAII:  	case CHIP_KAVERI:  	case CHIP_KABINI:  	case CHIP_MULLINS: +		/* +		 * We have systems in the wild with these ASICs that require +		 * LVDS and VGA support which is not supported with DC. +		 * +		 * Fallback to the non-DC driver here by default so as not to +		 * cause regressions. +		 */ +		return amdgpu_dc > 0; +	case CHIP_HAWAII:  	case CHIP_CARRIZO:  	case CHIP_STONEY:  	case CHIP_POLARIS10: @@ -2739,6 +2747,9 @@ int amdgpu_device_resume(struct drm_device *dev, bool resume, bool fbcon)  	if (r)  		return r; +	/* Make sure IB tests flushed */ +	flush_delayed_work(&adev->late_init_work); +  	/* blat the mode back in */  	if (fbcon) {  		if (!amdgpu_device_has_dc_support(adev)) { 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_object.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_object.c index 5e4e1bd90383..3526efa8960e 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_object.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_object.c @@ -762,8 +762,7 @@ int amdgpu_bo_pin_restricted(struct amdgpu_bo *bo, u32 domain,  	domain = amdgpu_mem_type_to_domain(bo->tbo.mem.mem_type);  	if (domain == AMDGPU_GEM_DOMAIN_VRAM) {  		adev->vram_pin_size += amdgpu_bo_size(bo); -		if (bo->flags & AMDGPU_GEM_CREATE_NO_CPU_ACCESS) -			adev->invisible_pin_size += amdgpu_bo_size(bo); +		adev->invisible_pin_size += amdgpu_vram_mgr_bo_invisible_size(bo);  	} else if (domain == AMDGPU_GEM_DOMAIN_GTT) {  		adev->gart_pin_size += amdgpu_bo_size(bo);  	} @@ -790,25 +789,22 @@ int amdgpu_bo_unpin(struct amdgpu_bo *bo)  	bo->pin_count--;  	if (bo->pin_count)  		return 0; -	for (i = 0; i < bo->placement.num_placement; i++) { -		bo->placements[i].lpfn = 0; -		bo->placements[i].flags &= ~TTM_PL_FLAG_NO_EVICT; -	} -	r = ttm_bo_validate(&bo->tbo, &bo->placement, &ctx); -	if (unlikely(r)) { -		dev_err(adev->dev, "%p validate failed for unpin\n", bo); -		goto error; -	}  	if (bo->tbo.mem.mem_type == TTM_PL_VRAM) {  		adev->vram_pin_size -= amdgpu_bo_size(bo); -		if (bo->flags & AMDGPU_GEM_CREATE_NO_CPU_ACCESS) -			adev->invisible_pin_size -= amdgpu_bo_size(bo); +		adev->invisible_pin_size -= amdgpu_vram_mgr_bo_invisible_size(bo);  	} else if (bo->tbo.mem.mem_type == TTM_PL_TT) {  		adev->gart_pin_size -= amdgpu_bo_size(bo);  	} -error: +	for (i = 0; i < bo->placement.num_placement; i++) { +		bo->placements[i].lpfn = 0; +		bo->placements[i].flags &= ~TTM_PL_FLAG_NO_EVICT; +	} +	r = ttm_bo_validate(&bo->tbo, &bo->placement, &ctx); +	if (unlikely(r)) +		dev_err(adev->dev, "%p validate failed for unpin\n", bo); +  	return r;  } 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_ttm.h b/drivers/gpu/drm/amd/amdgpu/amdgpu_ttm.h index e969c879d87e..e5da4654b630 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_ttm.h +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_ttm.h @@ -73,6 +73,7 @@ bool amdgpu_gtt_mgr_has_gart_addr(struct ttm_mem_reg *mem);  uint64_t amdgpu_gtt_mgr_usage(struct ttm_mem_type_manager *man);  int amdgpu_gtt_mgr_recover(struct ttm_mem_type_manager *man); +u64 amdgpu_vram_mgr_bo_invisible_size(struct amdgpu_bo *bo);  uint64_t amdgpu_vram_mgr_usage(struct ttm_mem_type_manager *man);  uint64_t amdgpu_vram_mgr_vis_usage(struct ttm_mem_type_manager *man); diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.c index bcf68f80bbf0..3ff08e326838 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.c @@ -130,7 +130,7 @@ int amdgpu_uvd_sw_init(struct amdgpu_device *adev)  	unsigned version_major, version_minor, family_id;  	int i, j, r; -	INIT_DELAYED_WORK(&adev->uvd.inst->idle_work, amdgpu_uvd_idle_work_handler); +	INIT_DELAYED_WORK(&adev->uvd.idle_work, amdgpu_uvd_idle_work_handler);  	switch (adev->asic_type) {  #ifdef CONFIG_DRM_AMDGPU_CIK @@ -314,12 +314,12 @@ int amdgpu_uvd_suspend(struct amdgpu_device *adev)  	void *ptr;  	int i, j; +	cancel_delayed_work_sync(&adev->uvd.idle_work); +  	for (j = 0; j < adev->uvd.num_uvd_inst; ++j) {  		if (adev->uvd.inst[j].vcpu_bo == NULL)  			continue; -		cancel_delayed_work_sync(&adev->uvd.inst[j].idle_work); -  		/* only valid for physical mode */  		if (adev->asic_type < CHIP_POLARIS10) {  			for (i = 0; i < adev->uvd.max_handles; ++i) @@ -1145,7 +1145,7 @@ int amdgpu_uvd_get_destroy_msg(struct amdgpu_ring *ring, uint32_t handle,  static void amdgpu_uvd_idle_work_handler(struct work_struct *work)  {  	struct amdgpu_device *adev = -		container_of(work, struct amdgpu_device, uvd.inst->idle_work.work); +		container_of(work, struct amdgpu_device, uvd.idle_work.work);  	unsigned fences = 0, i, j;  	for (i = 0; i < adev->uvd.num_uvd_inst; ++i) { @@ -1167,7 +1167,7 @@ static void amdgpu_uvd_idle_work_handler(struct work_struct *work)  							       AMD_CG_STATE_GATE);  		}  	} else { -		schedule_delayed_work(&adev->uvd.inst->idle_work, UVD_IDLE_TIMEOUT); +		schedule_delayed_work(&adev->uvd.idle_work, UVD_IDLE_TIMEOUT);  	}  } @@ -1179,7 +1179,7 @@ void amdgpu_uvd_ring_begin_use(struct amdgpu_ring *ring)  	if (amdgpu_sriov_vf(adev))  		return; -	set_clocks = !cancel_delayed_work_sync(&adev->uvd.inst->idle_work); +	set_clocks = !cancel_delayed_work_sync(&adev->uvd.idle_work);  	if (set_clocks) {  		if (adev->pm.dpm_enabled) {  			amdgpu_dpm_enable_uvd(adev, true); @@ -1196,7 +1196,7 @@ void amdgpu_uvd_ring_begin_use(struct amdgpu_ring *ring)  void amdgpu_uvd_ring_end_use(struct amdgpu_ring *ring)  {  	if (!amdgpu_sriov_vf(ring->adev)) -		schedule_delayed_work(&ring->adev->uvd.inst->idle_work, UVD_IDLE_TIMEOUT); +		schedule_delayed_work(&ring->adev->uvd.idle_work, UVD_IDLE_TIMEOUT);  }  /** diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.h b/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.h index b1579fba134c..8b23a1b00c76 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.h +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_uvd.h @@ -44,7 +44,6 @@ struct amdgpu_uvd_inst {  	void			*saved_bo;  	atomic_t		handles[AMDGPU_MAX_UVD_HANDLES];  	struct drm_file		*filp[AMDGPU_MAX_UVD_HANDLES]; -	struct delayed_work	idle_work;  	struct amdgpu_ring	ring;  	struct amdgpu_ring	ring_enc[AMDGPU_MAX_UVD_ENC_RINGS];  	struct amdgpu_irq_src	irq; @@ -62,6 +61,7 @@ struct amdgpu_uvd {  	bool			address_64_bit;  	bool			use_ctx_buf;  	struct amdgpu_uvd_inst		inst[AMDGPU_MAX_UVD_INSTANCES]; +	struct delayed_work	idle_work;  };  int amdgpu_uvd_sw_init(struct amdgpu_device *adev); 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..fdcb498f6d19 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_vm.c @@ -107,6 +107,9 @@ static void amdgpu_vm_bo_base_init(struct amdgpu_vm_bo_base *base,  		return;  	list_add_tail(&base->bo_list, &bo->va); +	if (bo->tbo.type == ttm_bo_type_kernel) +		list_move(&base->vm_status, &vm->relocated); +  	if (bo->tbo.resv != vm->root.base.bo->tbo.resv)  		return; @@ -468,7 +471,6 @@ static int amdgpu_vm_alloc_levels(struct amdgpu_device *adev,  			pt->parent = amdgpu_bo_ref(parent->base.bo);  			amdgpu_vm_bo_base_init(&entry->base, vm, pt); -			list_move(&entry->base.vm_status, &vm->relocated);  		}  		if (level < AMDGPU_VM_PTB) { @@ -1463,7 +1465,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 +1480,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 +1495,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/amdgpu_vram_mgr.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_vram_mgr.c index 9aca653bec07..b6333f92ba45 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_vram_mgr.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_vram_mgr.c @@ -97,6 +97,38 @@ static u64 amdgpu_vram_mgr_vis_size(struct amdgpu_device *adev,  }  /** + * amdgpu_vram_mgr_bo_invisible_size - CPU invisible BO size + * + * @bo: &amdgpu_bo buffer object (must be in VRAM) + * + * Returns: + * How much of the given &amdgpu_bo buffer object lies in CPU invisible VRAM. + */ +u64 amdgpu_vram_mgr_bo_invisible_size(struct amdgpu_bo *bo) +{ +	struct amdgpu_device *adev = amdgpu_ttm_adev(bo->tbo.bdev); +	struct ttm_mem_reg *mem = &bo->tbo.mem; +	struct drm_mm_node *nodes = mem->mm_node; +	unsigned pages = mem->num_pages; +	u64 usage = 0; + +	if (adev->gmc.visible_vram_size == adev->gmc.real_vram_size) +		return 0; + +	if (mem->start >= adev->gmc.visible_vram_size >> PAGE_SHIFT) +		return amdgpu_bo_size(bo); + +	while (nodes && pages) { +		usage += nodes->size << PAGE_SHIFT; +		usage -= amdgpu_vram_mgr_vis_size(adev, nodes); +		pages -= nodes->size; +		++nodes; +	} + +	return usage; +} + +/**   * amdgpu_vram_mgr_new - allocate new ranges   *   * @man: TTM memory type manager @@ -135,7 +167,8 @@ static int amdgpu_vram_mgr_new(struct ttm_mem_type_manager *man,  		num_nodes = DIV_ROUND_UP(mem->num_pages, pages_per_node);  	} -	nodes = kcalloc(num_nodes, sizeof(*nodes), GFP_KERNEL); +	nodes = kvmalloc_array(num_nodes, sizeof(*nodes), +			       GFP_KERNEL | __GFP_ZERO);  	if (!nodes)  		return -ENOMEM; @@ -190,7 +223,7 @@ error:  		drm_mm_remove_node(&nodes[i]);  	spin_unlock(&mgr->lock); -	kfree(nodes); +	kvfree(nodes);  	return r == -ENOSPC ? 0 : r;  } @@ -229,7 +262,7 @@ static void amdgpu_vram_mgr_del(struct ttm_mem_type_manager *man,  	atomic64_sub(usage, &mgr->usage);  	atomic64_sub(vis_usage, &mgr->vis_usage); -	kfree(mem->mm_node); +	kvfree(mem->mm_node);  	mem->mm_node = NULL;  } 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, | 
