drm/amd/display: Add tunneling IRQ handler
authorCruise Hung <Cruise.Hung@amd.com>
Thu, 20 Feb 2025 03:29:50 +0000 (11:29 +0800)
committerAlex Deucher <alexander.deucher@amd.com>
Wed, 5 Mar 2025 15:41:08 +0000 (10:41 -0500)
USB4 DP BW Allocation uses DP_TUNNELING_IRQ to indicate the status update.
The DP_TUNNELING_IRQ is defined in LINK_SERVICE_IRQ_VECTOR_ESI0. When
receiving DP HPD IRQ in USB4, read the LINK_SERVICE_IRQ_VECTOR_ESI0.

Reviewed-by: Wenjing Liu <wenjing.liu@amd.com>
Signed-off-by: Cruise Hung <Cruise.Hung@amd.com>
Signed-off-by: Wayne Lin <wayne.lin@amd.com>
Tested-by: Daniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
drivers/gpu/drm/amd/display/dc/dc_dp_types.h
drivers/gpu/drm/amd/display/dc/link/protocols/link_dp_dpia_bw.c
drivers/gpu/drm/amd/display/dc/link/protocols/link_dp_dpia_bw.h
drivers/gpu/drm/amd/display/dc/link/protocols/link_dp_irq_handler.c

index d988c00f5ca47d67aee375d6d78e09fd6c44dce5..1f4f11adc491f462d076f0acd27c020da6ce1c29 100644 (file)
@@ -300,6 +300,19 @@ union lane_align_status_updated {
        uint8_t raw;
 };
 
+union link_service_irq_vector_esi0 {
+       struct {
+               uint8_t DP_LINK_RX_CAP_CHANGED:1;
+               uint8_t DP_LINK_STATUS_CHANGED:1;
+               uint8_t DP_LINK_STREAM_STATUS_CHANGED:1;
+               uint8_t DP_LINK_HDMI_LINK_STATUS_CHANGED:1;
+               uint8_t DP_LINK_CONNECTED_OFF_ENTRY_REQUESTED:1;
+               uint8_t DP_LINK_TUNNELING_IRQ:1;
+               uint8_t reserved:2;
+       } bits;
+       uint8_t raw;
+};
+
 union lane_adjust {
        struct {
                uint8_t VOLTAGE_SWING_LANE:2;
@@ -462,8 +475,10 @@ union sink_status {
        uint8_t raw;
 };
 
-/*6-byte structure corresponding to 6 registers (200h-205h)
-read during handling of HPD-IRQ*/
+/* 7-byte structure corresponding to 6 registers (200h-205h)
+ * and LINK_SERVICE_IRQ_ESI0 (2005h) for tunneling IRQ
+ * read during handling of HPD-IRQ
+ */
 union hpd_irq_data {
        struct {
                union sink_count sink_cnt;/* 200h */
@@ -471,9 +486,10 @@ union hpd_irq_data {
                union lane_status lane01_status;/* 202h */
                union lane_status lane23_status;/* 203h */
                union lane_align_status_updated lane_status_updated;/* 204h */
-               union sink_status sink_status;
+               union sink_status sink_status;/* 205h */
+               union link_service_irq_vector_esi0 link_service_irq_esi0;/* 2005h */
        } bytes;
-       uint8_t raw[6];
+       uint8_t raw[7];
 };
 
 union down_stream_port_count {
@@ -1430,4 +1446,20 @@ struct dp_trace {
 #ifndef REQUESTED_BW
 #define REQUESTED_BW                                   0xE0031 /* 1.4a */
 #endif
+# ifndef DP_TUNNELING_BW_ALLOC_BITS_MASK
+# define DP_TUNNELING_BW_ALLOC_BITS_MASK               (0x0F << 0)
+# endif
+# ifndef DP_TUNNELING_BW_REQUEST_FAILED
+# define DP_TUNNELING_BW_REQUEST_FAILED                        (1 << 0)
+# endif
+# ifndef DP_TUNNELING_BW_REQUEST_SUCCEEDED
+# define DP_TUNNELING_BW_REQUEST_SUCCEEDED             (1 << 1)
+# endif
+# ifndef DP_TUNNELING_ESTIMATED_BW_CHANGED
+# define DP_TUNNELING_ESTIMATED_BW_CHANGED             (1 << 2)
+# endif
+# ifndef DP_TUNNELING_BW_ALLOC_CAP_CHANGED
+# define DP_TUNNELING_BW_ALLOC_CAP_CHANGED             (1 << 3)
+# endif
+
 #endif /* DC_DP_TYPES_H */
index 0f1c411523a235ea9409c8511a396b527b0deda6..a5541b8fc95bdc355fdc2ca42a7380934c36995d 100644 (file)
@@ -356,6 +356,32 @@ out:
        return ret;
 }
 
+/*
+ * Handle DP BW allocation status register
+ *
+ * @link: pointer to the dc_link struct instance
+ * @status: content of DP tunneling status DPCD register
+ *
+ * return: none
+ */
+void link_dp_dpia_handle_bw_alloc_status(struct dc_link *link, uint8_t status)
+{
+       if (status & DP_TUNNELING_BW_REQUEST_SUCCEEDED) {
+               DC_LOG_DEBUG("%s: BW Allocation request succeeded on link(%d)",
+                               __func__, link->link_index);
+       } else if (status & DP_TUNNELING_BW_REQUEST_FAILED) {
+               DC_LOG_DEBUG("%s: BW Allocation request failed on link(%d)  allocated/estimated BW=%d",
+                               __func__, link->link_index, link->dpia_bw_alloc_config.estimated_bw);
+       } else if (status & DP_TUNNELING_ESTIMATED_BW_CHANGED) {
+               DC_LOG_DEBUG("%s: Estimated BW changed on link(%d)  new estimated BW=%d",
+                               __func__, link->link_index, link->dpia_bw_alloc_config.estimated_bw);
+       }
+
+       core_link_write_dpcd(
+               link, DP_TUNNELING_STATUS,
+               &status, sizeof(status));
+}
+
 void dpia_handle_bw_alloc_response(struct dc_link *link, uint8_t bw, uint8_t result)
 {
        int bw_needed = 0;
index 3b6d8494f9d5da4ceb05711c9596007ac73f08a2..1b240a2f6ce0dbaa2c2319c6216e95e9ddf365b6 100644 (file)
@@ -108,4 +108,14 @@ bool dpia_validate_usb4_bw(struct dc_link **link, int *bw_needed, const unsigned
  */
 int link_dp_dpia_get_dp_overhead_in_dp_tunneling(struct dc_link *link);
 
+/*
+ * Handle DP BW allocation status register
+ *
+ * @link: pointer to the dc_link struct instance
+ * @status: content of DP tunneling status register
+ *
+ * return: none
+ */
+void link_dp_dpia_handle_bw_alloc_status(struct dc_link *link, uint8_t status);
+
 #endif /* DC_INC_LINK_DP_DPIA_BW_H_ */
index a08403c022eae2e30b274ad6f965d62b55f703b8..5be00e4ce10b31aa303193ac55344c90ed5a60d5 100644 (file)
@@ -37,6 +37,7 @@
 #include "link/accessories/link_dp_trace.h"
 #include "link/link_dpms.h"
 #include "dm_helpers.h"
+#include "link_dp_dpia_bw.h"
 
 #define DC_LOGGER \
        link->ctx->logger
@@ -286,6 +287,30 @@ void dp_handle_link_loss(struct dc_link *link)
        }
 }
 
+static void dp_handle_tunneling_irq(struct dc_link *link)
+{
+       enum dc_status retval;
+       uint8_t tunneling_status = 0;
+
+       retval = core_link_read_dpcd(
+                       link, DP_TUNNELING_STATUS,
+                       &tunneling_status,
+                       sizeof(tunneling_status));
+
+       if (retval == DC_OK) {
+               DC_LOG_HW_HPD_IRQ("%s: Got DP tunneling status on link %d status=0x%x",
+                               __func__, link->link_index, tunneling_status);
+
+               if (tunneling_status & DP_TUNNELING_BW_ALLOC_BITS_MASK)
+                       link_dp_dpia_handle_bw_alloc_status(link, tunneling_status);
+       }
+
+       tunneling_status = DP_TUNNELING_IRQ;
+       core_link_write_dpcd(
+               link, DP_LINK_SERVICE_IRQ_VECTOR_ESI0,
+               &tunneling_status, 1);
+}
+
 static void read_dpcd204h_on_irq_hpd(struct dc_link *link, union hpd_irq_data *irq_data)
 {
        enum dc_status retval;
@@ -319,13 +344,19 @@ enum dc_status dp_read_hpd_rx_irq_data(
         *
         * For DP 1.4 we need to read those from 2002h range.
         */
-       if (link->dpcd_caps.dpcd_rev.raw < DPCD_REV_14)
+       if (link->dpcd_caps.dpcd_rev.raw < DPCD_REV_14) {
                retval = core_link_read_dpcd(
                        link,
                        DP_SINK_COUNT,
                        irq_data->raw,
-                       sizeof(union hpd_irq_data));
-       else {
+                       DP_SINK_STATUS - DP_SINK_COUNT + 1);
+
+               if (link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA) {
+                       retval = core_link_read_dpcd(
+                                       link, DP_LINK_SERVICE_IRQ_VECTOR_ESI0,
+                                       &irq_data->bytes.link_service_irq_esi0.raw, 1);
+               }
+       } else {
                /* Read 14 bytes in a single read and then copy only the required fields.
                 * This is more efficient than doing it in two separate AUX reads. */
 
@@ -346,6 +377,7 @@ enum dc_status dp_read_hpd_rx_irq_data(
                irq_data->bytes.lane23_status.raw = tmp[DP_LANE2_3_STATUS_ESI - DP_SINK_COUNT_ESI];
                irq_data->bytes.lane_status_updated.raw = tmp[DP_LANE_ALIGN_STATUS_UPDATED_ESI - DP_SINK_COUNT_ESI];
                irq_data->bytes.sink_status.raw = tmp[DP_SINK_STATUS_ESI - DP_SINK_COUNT_ESI];
+               irq_data->bytes.link_service_irq_esi0.raw = tmp[DP_LINK_SERVICE_IRQ_VECTOR_ESI0 - DP_SINK_COUNT_ESI];
 
                /*
                 * This display doesn't have correct values in DPCD200Eh.
@@ -488,6 +520,11 @@ bool dp_handle_hpd_rx_irq(struct dc_link *link,
                dp_trace_link_loss_increment(link);
        }
 
+       if (link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA) {
+               if (hpd_irq_dpcd_data.bytes.link_service_irq_esi0.bits.DP_LINK_TUNNELING_IRQ)
+                       dp_handle_tunneling_irq(link);
+       }
+
        if (link->type == dc_connection_sst_branch &&
                hpd_irq_dpcd_data.bytes.sink_cnt.bits.SINK_COUNT
                        != link->dpcd_sink_count)