aboutsummaryrefslogtreecommitdiffstats
path: root/common/recipes-kernel/linux/linux-yocto-4.19.8/0303-drm-amd-Add-DM-DMCU-support.patch
diff options
context:
space:
mode:
Diffstat (limited to 'common/recipes-kernel/linux/linux-yocto-4.19.8/0303-drm-amd-Add-DM-DMCU-support.patch')
-rw-r--r--common/recipes-kernel/linux/linux-yocto-4.19.8/0303-drm-amd-Add-DM-DMCU-support.patch172
1 files changed, 172 insertions, 0 deletions
diff --git a/common/recipes-kernel/linux/linux-yocto-4.19.8/0303-drm-amd-Add-DM-DMCU-support.patch b/common/recipes-kernel/linux/linux-yocto-4.19.8/0303-drm-amd-Add-DM-DMCU-support.patch
new file mode 100644
index 00000000..0f4f0202
--- /dev/null
+++ b/common/recipes-kernel/linux/linux-yocto-4.19.8/0303-drm-amd-Add-DM-DMCU-support.patch
@@ -0,0 +1,172 @@
+From fe5425903ada15de7a14fab6a6448541f771c710 Mon Sep 17 00:00:00 2001
+From: David Francis <David.Francis@amd.com>
+Date: Tue, 11 Sep 2018 13:49:49 -0400
+Subject: [PATCH 0303/2940] drm/amd: Add DM DMCU support
+
+DMCU (Display Microcontroller Unit) is a GPU chip involved in
+eDP features like Adaptive Backlight Modulation and Panel Self
+Refresh.
+
+DC is already fully equipped to initialize DMCU as long as the
+firmware is loaded.
+
+At the moment only the raven firmware is available.
+
+A single .bin file is loaded by the kernel's loading mechanism
+and split into two ucodes according to the header.
+
+DMCU is optional, so if the firmware is not found, no error or
+warning is raised.
+
+Signed-off-by: David Francis <David.Francis@amd.com>
+Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
+---
+ .../gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c | 91 ++++++++++++++++++-
+ .../gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.h | 2 +
+ 2 files changed, 92 insertions(+), 1 deletion(-)
+
+diff --git a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c
+index 394eea1d1f92..cad0c24c989a 100644
+--- a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c
++++ b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c
+@@ -30,6 +30,7 @@
+ #include "vid.h"
+ #include "amdgpu.h"
+ #include "amdgpu_display.h"
++#include "amdgpu_ucode.h"
+ #include "atom.h"
+ #include "amdgpu_dm.h"
+ #include "amdgpu_pm.h"
+@@ -50,6 +51,7 @@
+ #include <linux/version.h>
+ #include <linux/types.h>
+ #include <linux/pm_runtime.h>
++#include <linux/firmware.h>
+
+ #include <drm/drmP.h>
+ #include <drm/drm_atomic.h>
+@@ -71,6 +73,9 @@
+
+ #include "modules/inc/mod_freesync.h"
+
++#define FIRMWARE_RAVEN_DMCU "amdgpu/raven_dmcu.bin"
++MODULE_FIRMWARE(FIRMWARE_RAVEN_DMCU);
++
+ /* basic init/fini API */
+ static int amdgpu_dm_init(struct amdgpu_device *adev);
+ static void amdgpu_dm_fini(struct amdgpu_device *adev);
+@@ -515,13 +520,97 @@ static void amdgpu_dm_fini(struct amdgpu_device *adev)
+ return;
+ }
+
+-static int dm_sw_init(void *handle)
++static int load_dmcu_fw(struct amdgpu_device *adev)
+ {
++ const char *fw_name_dmcu;
++ int r;
++ const struct dmcu_firmware_header_v1_0 *hdr;
++
++ switch(adev->asic_type) {
++ case CHIP_BONAIRE:
++ case CHIP_HAWAII:
++ case CHIP_KAVERI:
++ case CHIP_KABINI:
++ case CHIP_MULLINS:
++ case CHIP_TONGA:
++ case CHIP_FIJI:
++ case CHIP_CARRIZO:
++ case CHIP_STONEY:
++ case CHIP_POLARIS11:
++ case CHIP_POLARIS10:
++ case CHIP_POLARIS12:
++ case CHIP_VEGAM:
++ case CHIP_VEGA10:
++ case CHIP_VEGA12:
++ case CHIP_VEGA20:
++ return 0;
++ case CHIP_RAVEN:
++ fw_name_dmcu = FIRMWARE_RAVEN_DMCU;
++ break;
++ default:
++ DRM_ERROR("Unsupported ASIC type: 0x%X\n", adev->asic_type);
++ return -1;
++ }
++
++ if (adev->firmware.load_type != AMDGPU_FW_LOAD_PSP) {
++ DRM_DEBUG_KMS("dm: DMCU firmware not supported on direct or SMU loading\n");
++ return 0;
++ }
++
++ r = request_firmware_direct(&adev->dm.fw_dmcu, fw_name_dmcu, adev->dev);
++ if (r == -ENOENT) {
++ /* DMCU firmware is not necessary, so don't raise a fuss if it's missing */
++ DRM_DEBUG_KMS("dm: DMCU firmware not found\n");
++ adev->dm.fw_dmcu = NULL;
++ return 0;
++ }
++ if (r) {
++ dev_err(adev->dev, "amdgpu_dm: Can't load firmware \"%s\"\n",
++ fw_name_dmcu);
++ return r;
++ }
++
++ r = amdgpu_ucode_validate(adev->dm.fw_dmcu);
++ if (r) {
++ dev_err(adev->dev, "amdgpu_dm: Can't validate firmware \"%s\"\n",
++ fw_name_dmcu);
++ release_firmware(adev->dm.fw_dmcu);
++ adev->dm.fw_dmcu = NULL;
++ return r;
++ }
++
++ hdr = (const struct dmcu_firmware_header_v1_0 *)adev->dm.fw_dmcu->data;
++ adev->firmware.ucode[AMDGPU_UCODE_ID_DMCU_ERAM].ucode_id = AMDGPU_UCODE_ID_DMCU_ERAM;
++ adev->firmware.ucode[AMDGPU_UCODE_ID_DMCU_ERAM].fw = adev->dm.fw_dmcu;
++ adev->firmware.fw_size +=
++ ALIGN(le32_to_cpu(hdr->header.ucode_size_bytes) - le32_to_cpu(hdr->intv_size_bytes), PAGE_SIZE);
++
++ adev->firmware.ucode[AMDGPU_UCODE_ID_DMCU_INTV].ucode_id = AMDGPU_UCODE_ID_DMCU_INTV;
++ adev->firmware.ucode[AMDGPU_UCODE_ID_DMCU_INTV].fw = adev->dm.fw_dmcu;
++ adev->firmware.fw_size +=
++ ALIGN(le32_to_cpu(hdr->intv_size_bytes), PAGE_SIZE);
++
++ DRM_DEBUG_KMS("PSP loading DMCU firmware\n");
++
+ return 0;
+ }
+
++static int dm_sw_init(void *handle)
++{
++ struct amdgpu_device *adev = (struct amdgpu_device *)handle;
++
++ return load_dmcu_fw(adev);
++}
++
+ static int dm_sw_fini(void *handle)
+ {
++ struct amdgpu_device *adev = (struct amdgpu_device *)handle;
++
++ if(adev->dm.fw_dmcu) {
++ release_firmware(adev->dm.fw_dmcu);
++ adev->dm.fw_dmcu = NULL;
++ }
++
+ return 0;
+ }
+
+diff --git a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.h b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.h
+index c962a4111883..006c70b53b1b 100644
+--- a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.h
++++ b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.h
+@@ -119,6 +119,8 @@ struct amdgpu_display_manager {
+ struct drm_atomic_state *cached_state;
+
+ struct dm_comressor_info compressor;
++
++ const struct firmware *fw_dmcu;
+ };
+
+ struct amdgpu_dm_connector {
+--
+2.17.1
+