From: Kunal Joshi <kunal1.joshi@intel.com>
To: igt-dev@lists.freedesktop.org
Cc: Kunal Joshi <kunal1.joshi@intel.com>
Subject: [PATCH i-g-t 6/6] tests/intel/kms_usb4_switch: Add USB4 switch test suite
Date: Thu, 9 Apr 2026 10:07:14 +0530 [thread overview]
Message-ID: <20260409043714.284108-7-kunal1.joshi@intel.com> (raw)
In-Reply-To: <20260409043714.284108-1-kunal1.joshi@intel.com>
Add a comprehensive test suite for USB4/Thunderbolt dock/undock and
port switching scenarios using the Microsoft USB4 Switch 3141:
- dock-undock: Basic dock/undock cycles with display verification
- dock-undock-sr: Dock/undock with suspend/resume stability
- dock-undock-during-suspend: Dock while system is suspended
- switch: Port-to-port switching with display verification
- switch-sr: Port switching with suspend/resume stability
- switch-during-suspend: Port switch during suspend via HW delay
v2: Make timeout const (Arun)
Signed-off-by: Kunal Joshi <kunal1.joshi@intel.com>
---
tests/intel/kms_usb4_switch.c | 1254 +++++++++++++++++++++++++++++++++
tests/meson.build | 2 +
2 files changed, 1256 insertions(+)
create mode 100644 tests/intel/kms_usb4_switch.c
diff --git a/tests/intel/kms_usb4_switch.c b/tests/intel/kms_usb4_switch.c
new file mode 100644
index 000000000..d26cacaf1
--- /dev/null
+++ b/tests/intel/kms_usb4_switch.c
@@ -0,0 +1,1254 @@
+// SPDX-License-Identifier: MIT
+/*
+ * Copyright © 2026 Intel Corporation
+ */
+
+/**
+ * TEST: kms usb4 switch
+ * Category: Display
+ * Description: USB4/Thunderbolt dock/undock and port switching tests
+ * Driver requirement: i915, xe
+ * Mega feature: General Display Features
+ *
+ * SUBTEST: dock-undock
+ * Description: Test dock/undock cycles with display verification.
+ * Verifies hotplug events, EDID serial matching, modeset with
+ * max non-joiner mode, and pipe CRC stability.
+ *
+ * SUBTEST: dock-undock-sr
+ * Description: Test dock/undock with suspend/resume stability.
+ * Docks, verifies displays with modeset and CRC, suspends/resumes,
+ * then verifies displays still produce valid CRC after resume.
+ *
+ * SUBTEST: dock-during-suspend
+ * Description: Test docking while system is suspended.
+ * Simulates user plugging into a suspended laptop by triggering
+ * dock during suspend using hardware delay command. Verifies
+ * modeset and pipe CRC after resume.
+ *
+ * SUBTEST: undock-during-suspend
+ * Description: Test undocking while system is suspended.
+ * Docks first, verifies displays, then schedules a delayed
+ * undock during suspend. After resume, verifies all port
+ * displays have been removed.
+ *
+ * SUBTEST: switch
+ * Description: Test switching between USB4 switch ports.
+ * Verifies hotplug events, display verification with modeset
+ * and pipe CRC when switching from one port to another.
+ *
+ * SUBTEST: switch-sr
+ * Description: Test port switching with suspend/resume stability.
+ * Switches ports, verifies displays with modeset and CRC,
+ * suspends/resumes, then verifies CRC stability after resume.
+ *
+ * SUBTEST: switch-during-suspend
+ * Description: Test port switching during suspend using hardware delay.
+ * Schedules a delayed switch, suspends system, switch occurs during
+ * suspend, then verifies modeset and pipe CRC on new port after resume.
+ */
+
+#include <ctype.h>
+#include <sys/poll.h>
+
+#include "igt.h"
+#include "igt_edid.h"
+#include "igt_kms.h"
+#include "igt_pipe_crc.h"
+#include "igt_usb4_switch.h"
+#include "igt_connector_helper.h"
+#include "kms_joiner_helper.h"
+
+/*
+ * Extended timeout for switch operations.
+ * Port switching requires longer stabilization time than simple dock/undock
+ * due to hardware state transitions and MST topology rebuilds.
+ */
+#define USB4_SWITCH_TIMEOUT_S 20
+
+/* Number of CRC samples for stability check (solid color FB should be stable) */
+#define CRC_STABILITY_SAMPLES 3
+
+/* Maximum displays expected on a single USB4 switch port (MST hub) */
+#define MAX_DISPLAYS_PER_PORT 4
+
+/* Number of reprobe cycles for MST topology stabilization after resume */
+#define MST_STABILIZE_REPROBE_COUNT 10
+
+/* Inter-reprobe delay in microseconds (500 ms) */
+#define MST_STABILIZE_DELAY_US 500000
+
+/*
+ * Iteration helpers for dynamic subtest generation.
+ * Skip ports (or pairs) that have no displays configured.
+ *
+ * for_each_usb4_port_pair iterates adjacent cyclic pairs (0->1, 1->2, ...,
+ * N-1->0). For the 2-port Switch 3141 this covers the only possible pair.
+ * If the hardware grows beyond 2 ports, consider generating all distinct
+ * pairs instead.
+ */
+#define for_each_usb4_port(sw, count, p, pcfg) \
+ for ((p) = 0, \
+ (pcfg) = usb4switch_get_port_config((sw), 0); \
+ (p) < (count) && (pcfg) && (pcfg)->display_count > 0; \
+ (pcfg) = usb4switch_get_port_config((sw), ++(p)))
+
+#define for_each_usb4_port_pair(sw, count, p, pa, pb) \
+ for ((p) = 0, \
+ (pa) = usb4switch_get_port_config((sw), 0), \
+ (pb) = usb4switch_get_port_config((sw), 1 % (count)); \
+ (p) < (count) && (pa) && (pb) && \
+ (pa)->display_count > 0 && (pb)->display_count > 0; \
+ ++(p), \
+ (pa) = usb4switch_get_port_config((sw), (p)), \
+ (pb) = usb4switch_get_port_config((sw), ((p) + 1) % (count)))
+
+typedef struct {
+ int drm_fd;
+ igt_display_t display;
+ struct usb4switch *sw;
+ struct udev_monitor *hotplug_mon;
+ int max_dotclock;
+ uint32_t master_pipes;
+ uint32_t valid_pipes;
+} data_t;
+
+/*
+ * Per-display modeset state, used by the composable display building blocks.
+ * Holds everything needed to arm, commit, collect CRC, and disarm one display.
+ */
+struct display_ctx {
+ uint32_t conn_id;
+ igt_output_t *output;
+ igt_plane_t *primary;
+ struct igt_fb fb;
+ drmModeModeInfo mode;
+ bool valid;
+};
+
+/*
+ * find_output_by_id - Find an igt_output_t by DRM connector ID.
+ */
+static igt_output_t *find_output_by_id(igt_display_t *display,
+ uint32_t connector_id)
+{
+ int i;
+
+ for (i = 0; i < display->n_outputs; i++) {
+ if (display->outputs[i].id == connector_id)
+ return &display->outputs[i];
+ }
+
+ return NULL;
+}
+
+/*
+ * select_max_non_joiner_mode - Pick the largest mode that fits a single pipe.
+ *
+ * "Non-joiner" means hdisplay <= max_pipe_hdisplay AND clock <= max_dotclock.
+ * Among qualifying modes, pick by:
+ * 1. Highest pixel area (hdisplay * vdisplay)
+ * 2. Highest vrefresh (tie-break)
+ * 3. Highest clock (tie-break)
+ *
+ * If max_dotclock is 0 (debugfs unavailable), only hdisplay is checked.
+ *
+ * TODO: Remove this filter when joiner CRC collection is supported;
+ * the pipe allocator already handles joiner pipe counts.
+ *
+ * Returns: Pointer to a function-static copy of the best mode, or NULL
+ * if none found. Not reentrant — single-threaded callers only.
+ */
+static drmModeModeInfo *select_max_non_joiner_mode(int drm_fd,
+ igt_output_t *output,
+ int max_dotclock)
+{
+ static drmModeModeInfo best;
+ drmModeConnector *conn = output->config.connector;
+ uint64_t best_area = 0;
+ uint32_t best_vrefresh = 0;
+ int best_clock = 0;
+ bool found = false;
+ int i;
+
+ if (!conn || conn->count_modes == 0)
+ return NULL;
+
+ for (i = 0; i < conn->count_modes; i++) {
+ drmModeModeInfo *m = &conn->modes[i];
+ uint64_t area;
+
+ if (igt_bigjoiner_possible(drm_fd, m, max_dotclock))
+ continue;
+
+ area = (uint64_t)m->hdisplay * m->vdisplay;
+
+ if (area > best_area ||
+ (area == best_area && m->vrefresh > best_vrefresh) ||
+ (area == best_area && m->vrefresh == best_vrefresh &&
+ m->clock > best_clock)) {
+ best = *m;
+ best_area = area;
+ best_vrefresh = m->vrefresh;
+ best_clock = m->clock;
+ found = true;
+ }
+ }
+
+ return found ? &best : NULL;
+}
+
+/*
+ * find_connector - Find a connector ID for the given display config.
+ * Prefers PATH property (stable for MST) with name as fallback.
+ */
+static bool find_connector(int drm_fd,
+ const struct usb4switch_display *disp,
+ uint32_t *connector_id)
+{
+ if (!disp || !connector_id)
+ return false;
+
+ if (disp->connector_path &&
+ igt_connector_find_by_path(drm_fd, disp->connector_path,
+ connector_id))
+ return true;
+
+ if (disp->connector_name &&
+ igt_connector_find_by_name(drm_fd, disp->connector_name,
+ connector_id))
+ return true;
+
+ return false;
+}
+
+/*
+ * reprobe_connectors - Force reprobe of connectors and wait for MST topology.
+ * Critical after suspend/resume — without this MST connectors may take 90+ s.
+ *
+ * Timing rationale: MST hubs need ~3 s after reprobe to fully enumerate
+ * their downstream topology, based on empirical testing with USB4/TBT docks.
+ */
+static void reprobe_connectors(int drm_fd)
+{
+ igt_connector_reprobe_all(drm_fd);
+ igt_debug("reprobe: Connector reprobe completed\n");
+}
+
+static bool verify_port_displays(data_t *data,
+ const struct usb4switch_port *port_cfg)
+{
+ int i;
+
+ for (i = 0; i < port_cfg->display_count; i++) {
+ const struct usb4switch_display *disp = &port_cfg->displays[i];
+ uint32_t conn_id;
+ char name[32];
+ char serial[64];
+
+ if (!find_connector(data->drm_fd, disp, &conn_id)) {
+ igt_warn("Display %d not found for port %d\n",
+ i + 1, port_cfg->port_num);
+ return false;
+ }
+
+ if (disp->edid_serial) {
+ if (!igt_connector_get_info(data->drm_fd, conn_id,
+ name, sizeof(name),
+ serial, sizeof(serial),
+ NULL, 0)) {
+ igt_warn("Failed to get EDID serial for display %d\n",
+ i + 1);
+ return false;
+ }
+
+ if (strcmp(serial, disp->edid_serial) != 0) {
+ igt_warn("EDID serial mismatch for display %d: expected '%s', got '%s'\n",
+ i + 1, disp->edid_serial, serial);
+ return false;
+ }
+
+ igt_debug("Display %d EDID serial verified: %s\n",
+ i + 1, serial);
+ } else {
+ igt_debug("Display %d: No EDID serial configured, "
+ "skipping verification\n", i + 1);
+ }
+ }
+
+ igt_info("Port %d: All %d displays verified\n",
+ port_cfg->port_num, port_cfg->display_count);
+ return true;
+}
+
+/*
+ * init_display_ctx - Resolve a usb4switch_display to an output and select mode.
+ *
+ * Finds the connector, resolves to igt_output_t, and selects the max
+ * non-joiner mode. Must be called before arm_display().
+ */
+static void init_display_ctx(data_t *data, struct display_ctx *ctx,
+ const struct usb4switch_display *disp)
+{
+ drmModeModeInfo *mode;
+
+ memset(ctx, 0, sizeof(*ctx));
+
+ igt_assert_f(find_connector(data->drm_fd, disp, &ctx->conn_id),
+ "Display not found for pipeline test\n");
+
+ ctx->output = find_output_by_id(&data->display, ctx->conn_id);
+ igt_assert_f(ctx->output,
+ "No output for connector %u\n", ctx->conn_id);
+
+ mode = select_max_non_joiner_mode(data->drm_fd, ctx->output,
+ data->max_dotclock);
+ igt_skip_on_f(!mode,
+ "No non-joiner mode for connector %u\n", ctx->conn_id);
+
+ ctx->mode = *mode;
+ ctx->valid = true;
+}
+
+/*
+ * arm_display - Create FB, set primary plane, configure mode override.
+ *
+ * Pipe assignment must be done before this (by setup_port_displays).
+ * After arming, caller must commit with igt_display_commit2().
+ */
+static void arm_display(data_t *data, struct display_ctx *ctx)
+{
+ igt_assert(ctx->valid);
+
+ igt_output_override_mode(ctx->output, &ctx->mode);
+
+ ctx->primary = igt_output_get_plane_type(ctx->output,
+ DRM_PLANE_TYPE_PRIMARY);
+ igt_assert(ctx->primary);
+
+ igt_create_color_fb(data->drm_fd,
+ ctx->mode.hdisplay, ctx->mode.vdisplay,
+ DRM_FORMAT_XRGB8888, DRM_FORMAT_MOD_LINEAR,
+ 0.0, 1.0, 0.0, /* green */
+ &ctx->fb);
+ igt_plane_set_fb(ctx->primary, &ctx->fb);
+
+ igt_info(" Display %s: armed %dx%d@%d (clock %d)\n",
+ igt_output_name(ctx->output),
+ ctx->mode.hdisplay, ctx->mode.vdisplay,
+ ctx->mode.vrefresh, ctx->mode.clock);
+}
+
+/*
+ * collect_display_crc - Collect CRC samples and verify stability.
+ *
+ * Collects CRC_STABILITY_SAMPLES, asserts they are identical (expected
+ * for a solid color FB), and returns the representative CRC in *out_crc.
+ * Display must be armed and committed before calling.
+ */
+static void collect_display_crc(data_t *data, struct display_ctx *ctx,
+ igt_crc_t *out_crc)
+{
+ igt_crtc_t *crtc = igt_output_get_driving_crtc(ctx->output);
+ igt_pipe_crc_t *pipe_crc;
+ igt_crc_t samples[CRC_STABILITY_SAMPLES];
+ int i;
+
+ igt_assert(crtc);
+
+ pipe_crc = igt_crtc_crc_new(crtc, IGT_PIPE_CRC_SOURCE_AUTO);
+ igt_assert(pipe_crc);
+
+ igt_pipe_crc_start(pipe_crc);
+ for (i = 0; i < CRC_STABILITY_SAMPLES; i++)
+ igt_pipe_crc_get_single(pipe_crc, &samples[i]);
+ igt_pipe_crc_stop(pipe_crc);
+
+ /* Solid color FB: all samples must be identical */
+ for (i = 1; i < CRC_STABILITY_SAMPLES; i++)
+ igt_assert_crc_equal(&samples[0], &samples[i]);
+
+ *out_crc = samples[0];
+
+ igt_info(" Display %s: CRC stable (%d samples, pipe %s)\n",
+ igt_output_name(ctx->output), CRC_STABILITY_SAMPLES,
+ igt_crtc_name(crtc));
+
+ igt_pipe_crc_free(pipe_crc);
+}
+
+/*
+ * disarm_display - Clear plane, output, and remove FB.
+ *
+ * After disarming all outputs, caller must commit to apply changes.
+ */
+static void disarm_display(data_t *data, struct display_ctx *ctx)
+{
+ if (!ctx->valid)
+ return;
+
+ if (ctx->primary)
+ igt_plane_set_fb(ctx->primary, NULL);
+
+ igt_output_set_crtc(ctx->output, NULL);
+ igt_output_override_mode(ctx->output, NULL);
+
+ if (ctx->fb.fb_id)
+ igt_remove_fb(data->drm_fd, &ctx->fb);
+
+ ctx->primary = NULL;
+ ctx->fb.fb_id = 0;
+ ctx->valid = false;
+}
+
+/*
+ * setup_port_displays - Init contexts, allocate pipes, arm all displays.
+ *
+ * Resolves all displays on a port, selects modes, allocates pipes using
+ * the joiner-aware pipe allocator, and arms all displays.
+ * Caller must commit with igt_display_commit2() after this returns.
+ *
+ * Returns: Number of displays set up.
+ */
+static int setup_port_displays(data_t *data, struct display_ctx *ctxs,
+ const struct usb4switch_port *port_cfg)
+{
+ igt_output_t *outputs[MAX_DISPLAYS_PER_PORT];
+ uint32_t used_pipes = 0;
+ int i;
+
+ igt_assert(port_cfg->display_count <= MAX_DISPLAYS_PER_PORT);
+
+ for (i = 0; i < port_cfg->display_count; i++) {
+ init_display_ctx(data, &ctxs[i], &port_cfg->displays[i]);
+ outputs[i] = ctxs[i].output;
+ }
+
+ /* Joiner-aware pipe allocation */
+ igt_assert_f(igt_assign_pipes_for_outputs(data->drm_fd, outputs,
+ port_cfg->display_count,
+ data->display.n_crtcs,
+ &used_pipes,
+ data->master_pipes,
+ data->valid_pipes),
+ "Failed to allocate pipes for port %d displays\n",
+ port_cfg->port_num);
+
+ for (i = 0; i < port_cfg->display_count; i++)
+ arm_display(data, &ctxs[i]);
+
+ return port_cfg->display_count;
+}
+
+static void teardown_port_displays(data_t *data, struct display_ctx *ctxs,
+ int count)
+{
+ int i;
+
+ for (i = 0; i < count; i++)
+ disarm_display(data, &ctxs[i]);
+}
+
+/*
+ * verify_port_display_pipeline - Modeset all port displays and verify CRC.
+ *
+ * Used in non-suspend paths to confirm the display pipeline is functioning.
+ * Arms all displays, commits, collects stable CRCs, then tears down.
+ */
+static void verify_port_display_pipeline(data_t *data,
+ const struct usb4switch_port *port_cfg)
+{
+ struct display_ctx ctxs[MAX_DISPLAYS_PER_PORT];
+ igt_crc_t crc;
+ int count, i;
+
+ count = setup_port_displays(data, ctxs, port_cfg);
+ igt_display_commit2(&data->display, COMMIT_ATOMIC);
+
+ for (i = 0; i < count; i++)
+ collect_display_crc(data, &ctxs[i], &crc);
+
+ teardown_port_displays(data, ctxs, count);
+
+ igt_info("Port %d: Pipeline verification passed for %d displays\n",
+ port_cfg->port_num, count);
+}
+
+/*
+ * get_port_reference_crcs - Modeset and collect reference CRCs before suspend.
+ *
+ * Arms all displays, commits, collects a baseline CRC per display into
+ * ref_crcs[], then tears down. The reference CRCs are compared with
+ * post-resume CRCs to detect display corruption.
+ */
+static void get_port_reference_crcs(data_t *data,
+ const struct usb4switch_port *port_cfg,
+ igt_crc_t *ref_crcs)
+{
+ struct display_ctx ctxs[MAX_DISPLAYS_PER_PORT];
+ int count, i;
+
+ count = setup_port_displays(data, ctxs, port_cfg);
+ igt_display_commit2(&data->display, COMMIT_ATOMIC);
+
+ for (i = 0; i < count; i++)
+ collect_display_crc(data, &ctxs[i], &ref_crcs[i]);
+
+ teardown_port_displays(data, ctxs, count);
+
+ igt_info("Port %d: Collected %d reference CRCs for suspend comparison\n",
+ port_cfg->port_num, count);
+}
+
+/*
+ * verify_port_crcs_after_resume - Compare post-resume CRCs with pre-suspend.
+ *
+ * Arms all displays with the same mode/FB as before suspend, commits,
+ * collects new CRCs, and asserts each matches the corresponding reference.
+ * A mismatch indicates the display is showing garbage after resume.
+ */
+static void verify_port_crcs_after_resume(data_t *data,
+ const struct usb4switch_port *port_cfg,
+ const igt_crc_t *ref_crcs)
+{
+ struct display_ctx ctxs[MAX_DISPLAYS_PER_PORT];
+ igt_crc_t resume_crc;
+ int count, i;
+
+ count = setup_port_displays(data, ctxs, port_cfg);
+ igt_display_commit2(&data->display, COMMIT_ATOMIC);
+
+ for (i = 0; i < count; i++) {
+ collect_display_crc(data, &ctxs[i], &resume_crc);
+ igt_assert_crc_equal(&ref_crcs[i], &resume_crc);
+ igt_info(" Display %s: CRC matches pre-suspend reference\n",
+ igt_output_name(ctxs[i].output));
+ }
+
+ teardown_port_displays(data, ctxs, count);
+
+ igt_info("Port %d: All %d CRCs match pre-suspend reference\n",
+ port_cfg->port_num, count);
+}
+
+/*
+ * refresh_display - Reinitialize display structure to pick up new connectors.
+ * After hotplug events (dock/undock), new MST connectors may be created.
+ *
+ * WARNING: Any previously cached igt_output_t pointers become invalid
+ * after this call. Callers must re-resolve outputs via find_output_by_id()
+ * or init_display_ctx() before using them.
+ */
+static void refresh_display(data_t *data)
+{
+ igt_display_fini(&data->display);
+ igt_display_require(&data->display, data->drm_fd);
+}
+
+/*
+ * wait_for_displays - Wait for all displays on a port to connect.
+ */
+static bool wait_for_displays(data_t *data,
+ const struct usb4switch_port *port,
+ int timeout_s)
+{
+ int elapsed = 0;
+ int found = 0;
+ int i;
+
+ if (!port)
+ return false;
+
+ while (elapsed < timeout_s) {
+ reprobe_connectors(data->drm_fd);
+
+ found = 0;
+ for (i = 0; i < port->display_count; i++) {
+ uint32_t id;
+
+ if (find_connector(data->drm_fd,
+ &port->displays[i], &id))
+ found++;
+ }
+
+ if (found == port->display_count) {
+ igt_debug("All %d displays found for port %d\n",
+ port->display_count, port->port_num);
+ /*
+ * Reprobe cycles above may have created new MST
+ * connectors. Rebuild igt_display_t so callers
+ * see up-to-date connector IDs and outputs.
+ */
+ refresh_display(data);
+ return true;
+ }
+
+ sleep(1);
+ elapsed++;
+ }
+
+ igt_warn("Timeout waiting for displays (found %d/%d)\n",
+ found, port->display_count);
+ return false;
+}
+
+static void wait_for_hotplug(data_t *data, int timeout_s)
+{
+ bool detected;
+
+ detected = igt_hotplug_detected(data->hotplug_mon, timeout_s);
+
+ if (detected)
+ igt_debug("Hotplug detected\n");
+ else
+ igt_warn("Hotplug uevent not detected within %ds\n",
+ timeout_s);
+
+ reprobe_connectors(data->drm_fd);
+ refresh_display(data);
+}
+
+/*
+ * mst_stabilize - Extended MST topology stabilization after resume.
+ * MST hubs need additional time and reprobe cycles to rebuild their
+ * topology after suspend/resume. Ten iterations with 500 ms spacing
+ * gives the hub firmware enough time to re-enumerate all downstream
+ * ports, based on empirical testing with USB4/TBT docks.
+ */
+static void mst_stabilize(data_t *data)
+{
+ int i;
+
+ for (i = 0; i < MST_STABILIZE_REPROBE_COUNT; i++) {
+ reprobe_connectors(data->drm_fd);
+ usleep(MST_STABILIZE_DELAY_US);
+ }
+}
+
+/*
+ * port_dynamic_name - Format dynamic subtest name for a port.
+ * Uses port name if configured, otherwise falls back to "port-N".
+ *
+ * IGT requires subtest names to contain only [a-zA-Z0-9_-] and to be
+ * lower-case. Any upper-case letter is lowered, any other invalid
+ * character (e.g. spaces) is replaced with '-'.
+ */
+static const char *port_dynamic_name(const struct usb4switch_port *port,
+ char *buf, size_t len)
+{
+ size_t i;
+
+ if (port->name) {
+ snprintf(buf, len, "%s", port->name);
+ for (i = 0; buf[i]; i++) {
+ buf[i] = tolower((unsigned char)buf[i]);
+ if (!isalnum((unsigned char)buf[i]) &&
+ buf[i] != '-' && buf[i] != '_')
+ buf[i] = '-';
+ }
+ } else {
+ snprintf(buf, len, "port-%d", port->port_num);
+ }
+ return buf;
+}
+
+/*
+ * pair_dynamic_name - Format dynamic subtest name for a port pair.
+ * Combines both port names with "-to-" separator.
+ */
+static const char *pair_dynamic_name(const struct usb4switch_port *a,
+ const struct usb4switch_port *b,
+ char *buf, size_t len)
+{
+ char na[32], nb[32];
+
+ port_dynamic_name(a, na, sizeof(na));
+ port_dynamic_name(b, nb, sizeof(nb));
+ snprintf(buf, len, "%s-to-%s", na, nb);
+ return buf;
+}
+
+static void test_dock_undock(data_t *data,
+ const struct usb4switch_port *port_cfg)
+{
+ int iterations = usb4switch_get_iterations(data->sw);
+ int timeout = usb4switch_get_hotplug_timeout(data->sw);
+ int i;
+
+ igt_info("Testing port %d (%s) with %d displays, %d iterations\n",
+ port_cfg->port_num,
+ port_cfg->name ? port_cfg->name : "unnamed",
+ port_cfg->display_count, iterations);
+
+ for (i = 0; i < iterations; i++) {
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /* Dock */
+ igt_info(" Docking port %d...\n",
+ port_cfg->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_cfg->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_cfg, timeout),
+ "Displays did not enumerate on port %d\n",
+ port_cfg->port_num);
+
+ igt_assert_f(verify_port_displays(data, port_cfg),
+ "Display verification failed on port %d\n",
+ port_cfg->port_num);
+
+ verify_port_display_pipeline(data, port_cfg);
+
+ /* Undock */
+ igt_info(" Undocking...\n");
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ wait_for_hotplug(data, timeout);
+ }
+}
+
+static void test_dock_undock_sr(data_t *data,
+ const struct usb4switch_port *port_cfg)
+{
+ igt_crc_t ref_crcs[MAX_DISPLAYS_PER_PORT];
+ int iterations = usb4switch_get_iterations(data->sw);
+ int timeout = usb4switch_get_hotplug_timeout(data->sw);
+ int i;
+
+ igt_info("Testing port %d (%s) dock/undock with S/R, %d iterations\n",
+ port_cfg->port_num,
+ port_cfg->name ? port_cfg->name : "unnamed",
+ iterations);
+
+ for (i = 0; i < iterations; i++) {
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /* Dock */
+ igt_info(" Docking port %d...\n",
+ port_cfg->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_cfg->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_cfg,
+ timeout),
+ "Displays did not enumerate\n");
+
+ igt_info(" Verifying displays before suspend...\n");
+ igt_assert_f(verify_port_displays(data, port_cfg),
+ "Display verification failed before suspend\n");
+
+ /* Collect reference CRCs before suspend */
+ get_port_reference_crcs(data, port_cfg, ref_crcs);
+
+ /* Suspend/Resume while docked */
+ igt_info(" Suspending while docked...\n");
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ mst_stabilize(data);
+
+ igt_assert_f(wait_for_displays(data, port_cfg,
+ timeout),
+ "Displays did not enumerate after resume\n");
+ igt_info(" Verifying displays after resume...\n");
+ igt_assert_f(verify_port_displays(data, port_cfg),
+ "Display verification failed after resume\n");
+
+ /* Compare post-resume CRCs with pre-suspend reference */
+ verify_port_crcs_after_resume(data, port_cfg, ref_crcs);
+
+ /* Undock */
+ igt_info(" Undocking...\n");
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ wait_for_hotplug(data, timeout);
+ }
+}
+
+static void test_dock_during_suspend(data_t *data,
+ const struct usb4switch_port *port_cfg)
+{
+ int iterations = usb4switch_get_iterations(data->sw);
+ int timeout = usb4switch_get_hotplug_timeout(data->sw);
+ int i;
+
+ igt_info("Testing port %d (%s) dock-during-suspend, %d iterations\n",
+ port_cfg->port_num,
+ port_cfg->name ? port_cfg->name : "unnamed",
+ iterations);
+
+ for (i = 0; i < iterations; i++) {
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /*
+ * Schedule dock during suspend using hardware delay.
+ * Port change executes at T+7s while system is
+ * suspended (15s suspend cycle).
+ */
+ igt_info(" Scheduling dock during suspend (T+7s)...\n");
+ igt_assert(usb4switch_port_enable_delayed(data->sw,
+ port_cfg->port_num,
+ 7));
+
+ igt_info(" Suspending (15s, dock at T+7s)...\n");
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ igt_info(" Reprobing connectors for MST discovery...\n");
+ mst_stabilize(data);
+
+ igt_assert_f(wait_for_displays(data, port_cfg,
+ timeout),
+ "Displays did not enumerate after dock-during-suspend\n");
+
+ igt_info(" Verifying displays after dock-during-suspend...\n");
+ igt_assert_f(verify_port_displays(data, port_cfg),
+ "Display verification failed after dock-during-suspend\n");
+
+ verify_port_display_pipeline(data, port_cfg);
+
+ /* Undock to restore disconnected state for next iteration */
+ igt_info(" Undocking...\n");
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ wait_for_hotplug(data, timeout);
+ }
+}
+
+static void test_undock_during_suspend(data_t *data,
+ const struct usb4switch_port *port_cfg)
+{
+ int iterations = usb4switch_get_iterations(data->sw);
+ int timeout = usb4switch_get_hotplug_timeout(data->sw);
+ int i;
+
+ igt_info("Testing port %d (%s) undock-during-suspend, %d iterations\n",
+ port_cfg->port_num,
+ port_cfg->name ? port_cfg->name : "unnamed",
+ iterations);
+
+ for (i = 0; i < iterations; i++) {
+ int j, found;
+
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /* Dock first */
+ igt_info(" Docking port %d...\n", port_cfg->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_cfg->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_cfg, timeout),
+ "Displays did not enumerate on port %d\n",
+ port_cfg->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg),
+ "Display verification failed on port %d\n",
+ port_cfg->port_num);
+
+ verify_port_display_pipeline(data, port_cfg);
+
+ /*
+ * Schedule undock during suspend using hardware delay.
+ * Port disable executes at T+7s while system is
+ * suspended (15s suspend cycle).
+ */
+ igt_info(" Scheduling undock during suspend (T+7s)...\n");
+ igt_assert(usb4switch_port_disable_delayed(data->sw, 7));
+
+ igt_info(" Suspending (15s, undock at T+7s)...\n");
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ mst_stabilize(data);
+
+ /* Verify displays are gone after undock-during-suspend */
+ reprobe_connectors(data->drm_fd);
+ found = 0;
+ for (j = 0; j < port_cfg->display_count; j++) {
+ uint32_t id;
+
+ if (find_connector(data->drm_fd,
+ &port_cfg->displays[j], &id))
+ found++;
+ }
+ igt_assert_f(found == 0,
+ "Port %d: %d/%d displays still present after undock-during-suspend\n",
+ port_cfg->port_num, found,
+ port_cfg->display_count);
+
+ igt_info("Port %d: All displays removed after undock-during-suspend\n",
+ port_cfg->port_num);
+ }
+}
+
+static void test_switch(data_t *data,
+ const struct usb4switch_port *port_a,
+ const struct usb4switch_port *port_b)
+{
+ int iterations = usb4switch_get_iterations(data->sw);
+ const int timeout = USB4_SWITCH_TIMEOUT_S;
+ int i;
+
+ igt_info("Testing switch port %d -> port %d, %d iterations\n",
+ port_a->port_num, port_b->port_num, iterations);
+
+ for (i = 0; i < iterations; i++) {
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /* Enable port A */
+ igt_info(" Enabling port %d...\n",
+ port_a->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_a->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_a,
+ timeout),
+ "Displays did not enumerate on port %d\n",
+ port_a->port_num);
+ igt_assert_f(verify_port_displays(data, port_a),
+ "Display verification failed on port %d\n",
+ port_a->port_num);
+ verify_port_display_pipeline(data, port_a);
+
+ /* Switch to port B */
+ igt_info(" Switching to port %d...\n",
+ port_b->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_b->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_b,
+ timeout),
+ "Displays did not enumerate after switch to port %d\n",
+ port_b->port_num);
+ igt_assert_f(verify_port_displays(data, port_b),
+ "Display verification failed after switch to port %d\n",
+ port_b->port_num);
+
+ verify_port_display_pipeline(data, port_b);
+ }
+
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ wait_for_hotplug(data, timeout);
+}
+
+static void test_switch_during_suspend(data_t *data,
+ const struct usb4switch_port *port_cfg_a,
+ const struct usb4switch_port *port_cfg_b)
+{
+ int iterations = usb4switch_get_iterations(data->sw);
+ const int timeout = USB4_SWITCH_TIMEOUT_S;
+ int i;
+
+ igt_info("Testing switch during suspend port %d <-> port %d, %d iterations\n",
+ port_cfg_a->port_num, port_cfg_b->port_num, iterations);
+
+ for (i = 0; i < iterations; i++) {
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /* Start on port A */
+ igt_info(" Enabling port %d...\n", port_cfg_a->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_cfg_a->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_a, timeout),
+ "Displays did not enumerate on port %d\n",
+ port_cfg_a->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_a),
+ "Display verification failed on port %d\n",
+ port_cfg_a->port_num);
+
+ verify_port_display_pipeline(data, port_cfg_a);
+
+ /* Schedule switch to B during suspend */
+ igt_info(" Scheduling switch to port %d during suspend (T+7s)...\n",
+ port_cfg_b->port_num);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ igt_assert(usb4switch_port_enable_delayed(data->sw,
+ port_cfg_b->port_num,
+ 7));
+
+ igt_info(" Suspending (15s, switch at T+7s)...\n");
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ igt_info(" Reprobing connectors for MST discovery...\n");
+ mst_stabilize(data);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_b, timeout),
+ "Displays did not enumerate on port %d after switch-during-suspend\n",
+ port_cfg_b->port_num);
+ igt_info(" Verifying displays on port %d...\n",
+ port_cfg_b->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_b),
+ "Display verification failed on port %d after switch-during-suspend\n",
+ port_cfg_b->port_num);
+
+ verify_port_display_pipeline(data, port_cfg_b);
+
+ /* Schedule switch back to A during suspend */
+ igt_info(" Scheduling switch to port %d during suspend (T+7s)...\n",
+ port_cfg_a->port_num);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ igt_assert(usb4switch_port_enable_delayed(data->sw,
+ port_cfg_a->port_num,
+ 7));
+
+ igt_info(" Suspending (15s, switch at T+7s)...\n");
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ mst_stabilize(data);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_a, timeout),
+ "Displays did not enumerate on port %d after switch-during-suspend\n",
+ port_cfg_a->port_num);
+ igt_info(" Verifying displays on port %d...\n",
+ port_cfg_a->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_a),
+ "Display verification failed on port %d after switch-during-suspend\n",
+ port_cfg_a->port_num);
+
+ verify_port_display_pipeline(data, port_cfg_a);
+ }
+
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ wait_for_hotplug(data, timeout);
+}
+
+static void test_switch_sr(data_t *data,
+ const struct usb4switch_port *port_cfg_a,
+ const struct usb4switch_port *port_cfg_b)
+{
+ igt_crc_t ref_crcs[MAX_DISPLAYS_PER_PORT];
+ int iterations = usb4switch_get_iterations(data->sw);
+ const int timeout = USB4_SWITCH_TIMEOUT_S;
+ int i;
+
+ igt_info("Testing switch with S/R port %d <-> port %d, %d iterations\n",
+ port_cfg_a->port_num, port_cfg_b->port_num, iterations);
+
+ for (i = 0; i < iterations; i++) {
+ igt_info("Iteration %d/%d\n", i + 1, iterations);
+
+ /* Enable port A */
+ igt_info(" Enabling port %d...\n", port_cfg_a->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_cfg_a->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_a, timeout),
+ "Displays did not enumerate on port %d\n",
+ port_cfg_a->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_a),
+ "Display verification failed on port %d\n",
+ port_cfg_a->port_num);
+
+ /* Collect reference CRCs on port A before suspend */
+ get_port_reference_crcs(data, port_cfg_a, ref_crcs);
+
+ /* Suspend/Resume */
+ igt_info(" Suspending with port %d active...\n",
+ port_cfg_a->port_num);
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ mst_stabilize(data);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_a, timeout),
+ "Displays did not enumerate on port %d after resume\n",
+ port_cfg_a->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_a),
+ "Display verification failed on port %d after resume\n",
+ port_cfg_a->port_num);
+
+ /* Compare post-resume CRCs with pre-suspend reference */
+ verify_port_crcs_after_resume(data, port_cfg_a, ref_crcs);
+
+ /* Switch to port B */
+ igt_info(" Switching to port %d...\n",
+ port_cfg_b->port_num);
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_switch(data->sw,
+ port_cfg_b->port_num));
+ wait_for_hotplug(data, timeout);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_b, timeout),
+ "Displays did not enumerate on port %d\n",
+ port_cfg_b->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_b),
+ "Display verification failed on port %d\n",
+ port_cfg_b->port_num);
+
+ /* Collect reference CRCs on port B before suspend */
+ get_port_reference_crcs(data, port_cfg_b, ref_crcs);
+
+ /* Suspend/Resume with port B */
+ igt_info(" Suspending with port %d active...\n",
+ port_cfg_b->port_num);
+ igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
+ SUSPEND_TEST_NONE);
+ igt_info(" Resumed\n");
+
+ mst_stabilize(data);
+
+ igt_assert_f(wait_for_displays(data, port_cfg_b, timeout),
+ "Displays did not enumerate on port %d after resume\n",
+ port_cfg_b->port_num);
+ igt_assert_f(verify_port_displays(data, port_cfg_b),
+ "Display verification failed on port %d after resume\n",
+ port_cfg_b->port_num);
+
+ /* Compare post-resume CRCs with pre-suspend reference */
+ verify_port_crcs_after_resume(data, port_cfg_b, ref_crcs);
+ }
+
+ igt_flush_uevents(data->hotplug_mon);
+ igt_assert(usb4switch_port_disable_and_wait(data->sw));
+ wait_for_hotplug(data, timeout);
+}
+
+int igt_main()
+{
+ const struct usb4switch_port *pcfg, *pa, *pb;
+ data_t data = {};
+ igt_crtc_t *crtc;
+ char name[80];
+ int port_count;
+ int p;
+
+ igt_fixture() {
+ data.drm_fd = drm_open_driver_master(DRIVER_INTEL | DRIVER_XE);
+ igt_require(data.drm_fd >= 0);
+
+ kmstest_set_vt_graphics_mode();
+ igt_display_require(&data.display, data.drm_fd);
+
+ data.sw = usb4switch_init(data.drm_fd);
+ igt_require_f(data.sw, "USB4 Switch 3141 not available\n");
+
+ igt_require_pipe_crc(data.drm_fd);
+ data.max_dotclock = igt_get_max_dotclock(data.drm_fd);
+
+ data.hotplug_mon = igt_watch_uevents();
+ igt_require(data.hotplug_mon);
+
+ /* Compute pipe masks for joiner-aware allocation */
+ igt_set_all_master_pipes_for_platform(&data.display,
+ &data.master_pipes);
+ data.valid_pipes = 0;
+ for_each_crtc(&data.display, crtc)
+ data.valid_pipes |= BIT(crtc->pipe);
+
+ /* Ensure all ports are disconnected */
+ igt_assert(usb4switch_port_disable_and_wait(data.sw));
+ }
+
+ igt_describe("Dock/undock cycles with display verification");
+ igt_subtest_with_dynamic("dock-undock") {
+ port_count = usb4switch_get_port_count(data.sw);
+
+ for_each_usb4_port(data.sw, port_count, p, pcfg) {
+ port_dynamic_name(pcfg, name, sizeof(name));
+ igt_dynamic(name)
+ test_dock_undock(&data, pcfg);
+ }
+ }
+
+ igt_describe("Dock/undock with suspend/resume stability");
+ igt_subtest_with_dynamic("dock-undock-sr") {
+ port_count = usb4switch_get_port_count(data.sw);
+
+ for_each_usb4_port(data.sw, port_count, p, pcfg) {
+ port_dynamic_name(pcfg, name, sizeof(name));
+ igt_dynamic(name)
+ test_dock_undock_sr(&data, pcfg);
+ }
+ }
+
+ igt_describe("Dock during suspend with display verification");
+ igt_subtest_with_dynamic("dock-during-suspend") {
+ port_count = usb4switch_get_port_count(data.sw);
+
+ for_each_usb4_port(data.sw, port_count, p, pcfg) {
+ port_dynamic_name(pcfg, name, sizeof(name));
+ igt_dynamic(name)
+ test_dock_during_suspend(&data, pcfg);
+ }
+ }
+
+ igt_describe("Undock during suspend with display verification");
+ igt_subtest_with_dynamic("undock-during-suspend") {
+ port_count = usb4switch_get_port_count(data.sw);
+
+ for_each_usb4_port(data.sw, port_count, p, pcfg) {
+ port_dynamic_name(pcfg, name, sizeof(name));
+ igt_dynamic(name)
+ test_undock_during_suspend(&data, pcfg);
+ }
+ }
+
+ igt_describe("Port switching with display verification");
+ igt_subtest_with_dynamic("switch") {
+ port_count = usb4switch_get_port_count(data.sw);
+ igt_require(port_count > 1);
+
+ for_each_usb4_port_pair(data.sw, port_count, p, pa, pb) {
+ pair_dynamic_name(pa, pb, name, sizeof(name));
+ igt_dynamic(name)
+ test_switch(&data, pa, pb);
+ }
+ }
+
+ igt_describe("Port switching with suspend/resume stability");
+ igt_subtest_with_dynamic("switch-sr") {
+ port_count = usb4switch_get_port_count(data.sw);
+ igt_require(port_count > 1);
+
+ for_each_usb4_port_pair(data.sw, port_count, p, pa, pb) {
+ pair_dynamic_name(pa, pb, name, sizeof(name));
+ igt_dynamic(name)
+ test_switch_sr(&data, pa, pb);
+ }
+ }
+
+ igt_describe("Port switching during suspend via hardware delay");
+ igt_subtest_with_dynamic("switch-during-suspend") {
+ port_count = usb4switch_get_port_count(data.sw);
+ igt_require(port_count > 1);
+
+ for_each_usb4_port_pair(data.sw, port_count, p, pa, pb) {
+ pair_dynamic_name(pa, pb, name, sizeof(name));
+ igt_dynamic(name)
+ test_switch_during_suspend(&data, pa, pb);
+ }
+ }
+
+ igt_fixture() {
+ if (!usb4switch_port_disable_and_wait(data.sw))
+ igt_warn("Failed to disable ports during cleanup\n");
+ igt_cleanup_uevents(data.hotplug_mon);
+ usb4switch_deinit(data.sw);
+ igt_display_fini(&data.display);
+ drm_close_driver(data.drm_fd);
+ }
+}
diff --git a/tests/meson.build b/tests/meson.build
index 26d9345ec..e1d17b002 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -277,6 +277,7 @@ intel_kms_progs = [
'kms_psr_stress_test',
'kms_pwrite_crc',
'kms_sharpness_filter',
+ 'kms_usb4_switch',
]
intel_xe_progs = [
@@ -404,6 +405,7 @@ extra_sources = {
'kms_dsc': [ join_paths ('intel', 'kms_dsc_helper.c') ],
'kms_joiner': [ join_paths ('intel', 'kms_joiner_helper.c') ],
'kms_psr2_sf': [ join_paths ('intel', 'kms_dsc_helper.c') ],
+ 'kms_usb4_switch': [ join_paths ('intel', 'kms_joiner_helper.c') ],
}
# Extra dependencies used on core and Intel drivers
--
2.25.1
next prev parent reply other threads:[~2026-04-09 4:16 UTC|newest]
Thread overview: 13+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-04-09 4:37 [PATCH i-g-t 0/6] add test to validate dock/undock and switch Kunal Joshi
2026-04-09 4:37 ` [PATCH i-g-t 1/6] lib/igt_edid: add EDID serial extraction helpers Kunal Joshi
2026-04-09 4:37 ` [PATCH i-g-t 2/6] lib/igt_connector_helper: Add generic connector helpers Kunal Joshi
2026-04-09 4:37 ` [PATCH i-g-t 3/6] lib/igt_serial: add generic serial communication helper Kunal Joshi
2026-04-09 4:37 ` [PATCH i-g-t 4/6] lib/igt_usb4_switch: add helper library for USB4 Switch 3141 Kunal Joshi
2026-04-09 4:37 ` [PATCH i-g-t 5/6] tests/kms_feature_discovery: add basic usb4 switch discovery Kunal Joshi
2026-04-09 4:37 ` Kunal Joshi [this message]
2026-04-10 0:18 ` ✓ i915.CI.BAT: success for add test to validate dock/undock and switch (rev4) Patchwork
2026-04-10 0:19 ` ✓ Xe.CI.BAT: " Patchwork
2026-04-10 2:45 ` ✗ Xe.CI.FULL: failure " Patchwork
2026-04-10 17:16 ` ✗ i915.CI.Full: " Patchwork
-- strict thread matches above, loose matches on Subject: below --
2026-02-25 21:28 [PATCH i-g-t 0/6] add test to validate dock/undock and switch Kunal Joshi
2026-02-25 21:28 ` [PATCH i-g-t 6/6] tests/intel/kms_usb4_switch: Add USB4 switch test suite Kunal Joshi
2026-02-25 19:42 [PATCH i-g-t 0/6] add test to validate dock/undock and switch Kunal Joshi
2026-02-25 19:42 ` [PATCH i-g-t 6/6] tests/intel/kms_usb4_switch: Add USB4 switch test suite Kunal Joshi
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20260409043714.284108-7-kunal1.joshi@intel.com \
--to=kunal1.joshi@intel.com \
--cc=igt-dev@lists.freedesktop.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox