All of lore.kernel.org
 help / color / mirror / Atom feed
* [igt-dev] [PATCH] test/amdgpu: Add DP mst test
@ 2022-07-11  3:50 Wayne Lin
  2022-07-11  3:53 ` [igt-dev] ✗ Fi.CI.BUILD: failure for " Patchwork
  2022-07-20 16:26 ` [igt-dev] [PATCH] " Rodrigo Siqueira Jordao
  0 siblings, 2 replies; 5+ messages in thread
From: Wayne Lin @ 2022-07-11  3:50 UTC (permalink / raw)
  To: igt-dev; +Cc: jerry.zuo, hersenxs.wu

[Why & How]
Add new igt test amd_mst for DP mst feature.

MST (Multi-Stream Transport) is the feature introduced from DP 1.2.
By mst, source can transport multiple streams to different stream
sinks through the same DP Tx.

This new igt test "amd_mst" validate mst feature from two perspectives:

* Trigger soft hotplug to see whether all mst end sinks successfully go
  through mst light up procedure
* Check if all mst end sinks successfully go through mst light up
  procedure after suspend-resume

Signed-off-by: Wayne Lin <Wayne.Lin@amd.com>
---
 lib/igt_amd.c            | 160 +++++++++++++++++++++
 lib/igt_amd.h            |  43 ++++++
 tests/amdgpu/amd_mst.c   | 301 +++++++++++++++++++++++++++++++++++++++
 tests/amdgpu/meson.build |   1 +
 4 files changed, 505 insertions(+)
 create mode 100644 tests/amdgpu/amd_mst.c

diff --git a/lib/igt_amd.c b/lib/igt_amd.c
index bef9c193..5de26691 100644
--- a/lib/igt_amd.c
+++ b/lib/igt_amd.c
@@ -1175,3 +1175,163 @@ bool igt_amd_set_visual_confirm(int drm_fd, enum amdgpu_debug_visual_confirm opt
 
 	return res;
 }
+
+/**
+ * igt_amd_output_has_mst: check if connector has MST debugfs entries
+ * @drm_fd: DRM file descriptor
+ * @connector_name: The connector's name, on which we're reading the status
+ */
+static bool igt_amd_output_has_mst(int drm_fd, char *connector_name)
+{
+	bool ret = true;
+
+	ret = igt_amd_output_has_debugfs(drm_fd, connector_name, DEBUGFS_MST_IS_MST_CONNECTOR) &
+			igt_amd_output_has_debugfs(drm_fd, connector_name, DEBUGFS_MST_PROGRESS_STATUS);
+    return ret;
+}
+
+/**
+ * igt_amd_require_mst: Checks if connectors have mst debugfs entries
+ * @display: A pointer to an #igt_display_t structure
+ * @drm_fd: DRM file descriptor
+ *
+ * Checks if the AMDGPU driver supports the 'is_mst_connector'
+ * and 'mst_progress_status' entry for mst. Skip test if there
+ * are no relevant debugfs entries.
+ */
+void igt_amd_require_mst(igt_display_t *display, int drm_fd)
+{
+	igt_output_t *output;
+
+	for_each_connected_output(display, output) {
+		if (igt_amd_output_has_mst(drm_fd, output->name))
+			return;
+	}
+
+	igt_skip("No MST relevant debugfs support.\n");
+}
+
+/**
+ * igt_amd_get_mst_role: Get the mst role of this output
+ * @drm_fd: DRM file descriptor
+ * @output: the output to be checked
+ *
+ * DRM enumerates a drm connector to each MST output port. This is used to
+ * determine the mst role of this specific output. The mst role in the
+ * mst topology shoule be 'no', 'root', 'branch' and 'end'
+ */
+enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output)
+{
+	int fd, ret;
+	char buf[256] = {'\0'};
+	int i = 0;
+	enum dp_mst_role role = MST_NONE;
+
+	fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
+	if (fd < 0) {
+		igt_info("Could not open connector %s debugfs directory\n",
+			 output->name);
+		return MST_NONE;
+	}
+
+	if (output->config.connector->connector_type != DRM_MODE_CONNECTOR_DisplayPort)
+		return MST_NONE;
+
+	ret = igt_debugfs_simple_read(fd, DEBUGFS_MST_IS_MST_CONNECTOR, buf, sizeof(buf));
+	igt_assert_f(ret >= 0, "Reading %s for connector %s failed.\n",
+		     DEBUGFS_MST_IS_MST_CONNECTOR, output->name);
+
+	close(fd);
+
+	for (i = 0; i < sizeof(is_mst_connector)/sizeof(char *); i++) {
+		if (strstr(buf, is_mst_connector[i])) {
+			role = i;
+			break;
+		}
+	}
+
+	if (role != MST_NONE)
+		igt_info("Connector %s: is a mst %s",
+		 output->name, buf);
+	else
+		igt_info("Connector %s: is not a mst device\n",
+			 output->name);
+
+	return role;
+}
+
+/**
+ * igt_amd_get_mst_progress_status: Get the mst progress status of a mst device
+ * @drm_fd: DRM file descriptor
+ * @output: the output to be checked
+ *
+ * Get the mst progress status of this specific output. The mst progress status
+ * is indicated by 'MST_PROBE', 'MST_REMOTE_EDID' , 'MST_ALLOCATE_NEW_PAYLOAD'
+ * and 'MST_CLEAR_ALLOCATED_PAYLOAD'
+ */
+uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t *output)
+{
+	int fd, ret;
+	char buf[256] = {'\0'};
+	int idx = 0;
+	char *token_end;
+	uint8_t status = MST_STATUS_DEFAULT;
+
+	fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
+	if (fd < 0) {
+		igt_info("Could not open connector %s debugfs directory\n",
+			 output->name);
+		return MST_STATUS_DEFAULT;
+	}
+
+	if (output->config.connector->connector_type != DRM_MODE_CONNECTOR_DisplayPort)
+		return MST_STATUS_DEFAULT;
+
+	ret = igt_debugfs_simple_read(fd, DEBUGFS_MST_PROGRESS_STATUS, buf, sizeof(buf));
+	igt_assert_f(ret >= 0, "Reading %s for connector %s failed.\n",
+		     DEBUGFS_MST_PROGRESS_STATUS, output->name);
+
+	close(fd);
+
+	/* Parse values read from file. */
+	for (char *token = strtok_r(buf, "\n", &token_end);
+	     token != NULL;
+	     token = strtok_r(NULL, "\n", &token_end), idx++)
+	{
+		if (strstr(token, "not") == NULL)
+			status |= BIT(idx);
+	}
+
+	return status;
+}
+
+/**
+ * mst_trigger_hotplug: Trigger mst hotplug
+ * @drm_fd: DRM file descriptor
+ * @output: #igt_output_t to check.
+ *
+ * Trigger soft hotplug on mst connector. Caller should make sure the
+ * passed output is mst root by igt_amd_get_mst_role(). We can't generate
+ * soft hotplug on mst end device.
+ */
+void mst_trigger_hotplug(int drm_fd, igt_output_t *output)
+{
+    int fd, hpd_fd;
+	int wr_len;
+	const char *enable_hpd = "1", *disable_hpd = "0";
+
+	fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
+	igt_assert(fd >= 0);
+	hpd_fd = openat(fd, DEBUGFS_HPD_TRIGGER, O_WRONLY);
+	close(fd);
+	igt_assert(hpd_fd >= 0);
+
+	wr_len = write(hpd_fd, disable_hpd, strlen(disable_hpd));
+	igt_assert_eq(wr_len, strlen(disable_hpd));
+
+	sleep(3);
+
+	wr_len = write(hpd_fd, enable_hpd, strlen(enable_hpd));
+	igt_assert_eq(wr_len, strlen(enable_hpd));
+	close(hpd_fd);
+}
\ No newline at end of file
diff --git a/lib/igt_amd.h b/lib/igt_amd.h
index 428bfe6f..b88755b3 100644
--- a/lib/igt_amd.h
+++ b/lib/igt_amd.h
@@ -52,6 +52,12 @@
 /* amdgpu DM interface entries */
 #define DEBUGFS_DM_VISUAL_CONFIRM "amdgpu_dm_visual_confirm"
 
+/* mst related read only status*/
+#define DEBUGFS_MST_IS_MST_CONNECTOR "is_mst_connector"
+#define DEBUGFS_MST_PROGRESS_STATUS "mst_progress_status"
+
+#define BIT(x) (1ul <<(x))
+
 enum amd_dsc_clock_force {
 	DSC_AUTOMATIC = 0,
 	DSC_FORCE_ON,
@@ -131,6 +137,37 @@ enum amdgpu_debug_visual_confirm {
 	VISUAL_CONFIRM_SWIZZLE	= 9
 };
 
+/*
+ * Enumeration of the role in MST topology
+ */
+enum dp_mst_role {
+	MST_NONE = 0,
+	MST_ROOT,
+	MST_BRANCH,
+	MST_END,
+};
+
+static const char *const is_mst_connector[] = {
+    "no",
+    "root",
+    "branch",
+    "end",
+};
+
+/*
+ * Enumeration of MST progress state. The bit order should align
+ * with the 'enum mst_progress_statusto' in amdgpu_dm.h of
+ * upstreamed amdgpu kernel driver
+ */
+enum mst_progress_status {
+	MST_STATUS_DEFAULT = 0,
+	MST_PROBE = BIT(0),
+	MST_REMOTE_EDID = BIT(1),
+	MST_ALLOCATE_NEW_PAYLOAD = BIT(2),
+	MST_CLEAR_ALLOCATED_PAYLOAD = BIT(3),
+	NUM_MST_STATUS = 4,
+};
+
 uint32_t igt_amd_create_bo(int fd, uint64_t size);
 void *igt_amd_mmap_bo(int fd, uint32_t handle, uint64_t size, int prot);
 unsigned int igt_amd_compute_offset(unsigned int* swizzle_pattern,
@@ -192,4 +229,10 @@ int  igt_amd_read_psr_state(int drm_fd, char *connector_name);
 bool igt_amd_has_visual_confirm(int drm_fd);
 int  igt_amd_get_visual_confirm(int drm_fd);
 bool igt_amd_set_visual_confirm(int drm_fd, enum amdgpu_debug_visual_confirm option);
+
+/* DP MST debugfs helpers*/
+void igt_amd_require_mst(igt_display_t *display, int drm_fd);
+enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output);
+uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t *output);
+void mst_trigger_hotplug(int drm_fd, igt_output_t *output);
 #endif /* IGT_AMD_H */
diff --git a/tests/amdgpu/amd_mst.c b/tests/amdgpu/amd_mst.c
new file mode 100644
index 00000000..3b65f465
--- /dev/null
+++ b/tests/amdgpu/amd_mst.c
@@ -0,0 +1,301 @@
+/*
+ * Copyright 2022 Advanced Micro Devices, Inc.
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a
+ * copy of this software and associated documentation files (the "Software"),
+ * to deal in the Software without restriction, including without limitation
+ * the rights to use, copy, modify, merge, publish, distribute, sublicense,
+ * and/or sell copies of the Software, and to permit persons to whom the
+ * Software is furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
+ * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
+ * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
+ * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
+ * OTHER DEALINGS IN THE SOFTWARE.
+ */
+
+#include "igt.h"
+#include "igt_amd.h"
+
+IGT_TEST_DESCRIPTION("This igt test validates DP MST feature by:"
+	"1. Check each mst end sink successfully go through mst light up procedure"
+	"after hotplug root branch."
+	"2. Check each mst end sink successfully go through mst light up procedure"
+	"after suspend-resume");
+
+/* Maximum pipes on any AMD ASIC. */
+#define MAX_PIPES 6
+
+typedef struct data {
+	igt_display_t display;
+	igt_plane_t *primary[MAX_PIPES];
+	igt_output_t *output_mst_end[MAX_PIPES];
+	igt_output_t *output_mst_root[MAX_PIPES];
+	igt_pipe_t *pipe[MAX_PIPES];
+	igt_fb_t ref_fb[MAX_PIPES];
+	drmModeModeInfo mode[MAX_PIPES];
+	enum pipe pipe_id[MAX_PIPES];
+	int w[MAX_PIPES];
+	int h[MAX_PIPES];
+	int fd;
+	int num_mst_end;
+	int usr_num_mst_end;
+} data_t;
+
+static void test_init(data_t *data)
+{
+	igt_display_t *display = &data->display;
+	int i, m, n, max_pipes = display->n_pipes;
+	enum dp_mst_role role;
+	drmModeConnector *conn;
+
+	for_each_pipe(display, i) {
+		data->pipe_id[i] = PIPE_A + i;
+		data->pipe[i] = &data->display.pipes[data->pipe_id[i]];
+		data->primary[i] = igt_pipe_get_plane_type(
+			data->pipe[i], DRM_PLANE_TYPE_PRIMARY);
+	}
+
+	/* Find mst connected root/end connector*/
+	for (i = 0, n = 0, m = 0, data->num_mst_end = 0;
+		i < display->n_outputs && n < max_pipes && m < max_pipes; i++) {
+		igt_output_t *output = &display->outputs[i];
+
+		conn = output->config.connector;
+		if (conn->connector_type != DRM_MODE_CONNECTOR_DisplayPort)
+			continue;
+
+		role = igt_amd_get_mst_role(data->fd, output);
+
+		switch(role) {
+			case MST_ROOT:
+				data->output_mst_root[n++] = output;
+				break;
+			case MST_END:
+				if (!igt_output_is_connected(output))
+					continue;
+				data->output_mst_end[m] = output;
+				igt_assert(kmstest_get_connector_default_mode(
+				data->fd, output->config.connector, &data->mode[m]));
+
+				data->w[m] = data->mode[m].hdisplay;
+				data->h[m] = data->mode[m].vdisplay;
+				m++;
+				data->num_mst_end++;
+				break;
+			default:
+				continue;
+				break;
+		}
+	}
+
+	igt_require(data->output_mst_root[0]);
+	igt_require(data->output_mst_end[0]);
+	if (data->usr_num_mst_end)
+		igt_assert_eq(data->usr_num_mst_end, data->num_mst_end);
+	igt_display_reset(display);
+}
+
+static void test_fini(data_t *data)
+{
+	igt_display_t *display = &data->display;
+
+	igt_display_reset(display);
+	igt_display_commit_atomic(display, DRM_MODE_ATOMIC_ALLOW_MODESET, 0);
+}
+
+static bool show_mst_status_result(igt_output_t *output, uint8_t status)
+{
+	uint8_t bit_idx = 0;
+	bool result = true;
+
+	igt_info("Connector:%s mst progress status:\n", output->name);
+
+	for (bit_idx = 0; bit_idx < NUM_MST_STATUS; bit_idx++) {
+		switch(BIT(bit_idx)) {
+			case MST_PROBE:
+				if ((status & MST_PROBE) == MST_STATUS_DEFAULT)
+					result = false;
+				igt_info("Probe:%s\n", (status & MST_PROBE) ? "PASS" : "FAIL");
+				break;
+			case MST_REMOTE_EDID:
+				if ((status & MST_REMOTE_EDID) == MST_STATUS_DEFAULT)
+					result = false;
+				igt_info("Remote EDID:%s\n", (status & MST_REMOTE_EDID) ? "PASS" : "FAIL");
+				break;
+			case MST_ALLOCATE_NEW_PAYLOAD:
+				if ((status & MST_ALLOCATE_NEW_PAYLOAD) == MST_STATUS_DEFAULT)
+					result = false;
+				igt_info("Allocate new payload:%s\n",
+					(status & MST_ALLOCATE_NEW_PAYLOAD) ? "PASS" : "FAIL");
+				break;
+			default:
+				break;
+		}
+	}
+
+	return result;
+}
+
+static void reset_resource(data_t *data)
+{
+	igt_display_reset(&data->display);
+	igt_display_fini(&data->display);
+	close(data->fd);
+
+	memset(data->primary, 0, sizeof(igt_plane_t *)*MAX_PIPES);
+	memset(data->output_mst_end, 0, sizeof(igt_output_t *)*MAX_PIPES);
+	memset(data->output_mst_root, 0, sizeof(igt_output_t *)*MAX_PIPES);
+	memset(data->pipe, 0, sizeof(igt_pipe_t *)*MAX_PIPES);
+	memset(data->mode, 0, sizeof(drmModeModeInfo)*MAX_PIPES);
+	memset(data->pipe_id, 0, sizeof(enum pipe)*MAX_PIPES);
+	memset(data->w, 0, sizeof(int)*MAX_PIPES);
+	memset(data->h, 0, sizeof(int)*MAX_PIPES);
+
+	data->fd = drm_open_driver_master(DRIVER_AMDGPU);
+	igt_display_require(&data->display, data->fd);
+}
+
+static void set_up_streams(data_t *data)
+{
+	igt_output_t *output;
+	int i;
+
+	for (i = 0; i < MAX_PIPES; i++) {
+		output = data->output_mst_end[i];
+		if (!output || !igt_output_is_connected(output))
+			continue;
+
+		igt_create_pattern_fb(data->fd, data->w[i], data->h[i],
+				      DRM_FORMAT_XRGB8888, 0, &data->ref_fb[i]);
+		igt_output_set_pipe(output, data->pipe_id[i]);
+		igt_plane_set_fb(data->primary[i], &data->ref_fb[i]);
+	}
+	igt_display_commit_atomic(&data->display, DRM_MODE_ATOMIC_ALLOW_MODESET, 0);
+	sleep(5);
+}
+
+static void validate_mst_progress_result(data_t *data)
+{
+	int i = 0;
+	uint8_t status = 0;
+	igt_output_t *output;
+
+	for (i = 0; i < data->num_mst_end; i++) {
+		output = data->output_mst_end[i];
+
+		if (!output)
+			continue;
+
+		status = igt_amd_get_mst_progress_status(data->fd, output);
+
+		/* make sure all mst end device get lit up correctly*/
+		igt_assert(show_mst_status_result(output, status));
+	}
+}
+
+static void test_mst_hotplug_basic(data_t *data)
+{
+	igt_output_t *output;
+	int i = 0;
+
+	test_init(data);
+
+	/* Setup all end mst output */
+	set_up_streams(data);
+
+	/*confirm mst progress is done*/
+	validate_mst_progress_result(data);
+
+	/* Trigger hotplug*/
+	for (i = 0; i < MAX_PIPES; i++) {
+		output = data->output_mst_root[i];
+
+		if (!output)
+			continue;
+
+		mst_trigger_hotplug(data->fd, output);
+	}
+
+	/* mst connectors are dynamically created/destroyed. Need
+	 * to free and reallocate resources then commit streams again
+	 */
+	reset_resource(data);
+
+	test_init(data);
+
+	set_up_streams(data);
+
+	/*confirm mst progress status again*/
+	validate_mst_progress_result(data);
+
+	/*do suspend resume*/
+	igt_system_suspend_autoresume(SUSPEND_STATE_MEM, SUSPEND_TEST_NONE);
+	sleep(5);
+
+	/*confirm mst progress status again*/
+	validate_mst_progress_result(data);
+
+	for (i = 0; i < MAX_PIPES; i++) {
+		output = data->output_mst_end[i];
+
+		if (!output)
+			continue;
+
+		igt_remove_fb(data->fd, &data->ref_fb[i]);
+	}
+
+	test_fini(data);
+}
+
+const char *optstr = "hn:";
+static void usage(const char *name)
+{
+	igt_info("Usage: %s options\n", name);
+	igt_info("-h			Show help\n");
+	igt_info("-n mst_num	Set how many monitors connected in the mst topology\n");
+	igt_info("NOTE: if -n is not specified, number of monitors will be detected automatically"
+		"by current connection status\n");
+}
+
+int main(int argc, char **argv)
+{
+	data_t data;
+	int c;
+
+	igt_skip_on_simulation();
+
+	memset(&data, 0, sizeof(data));
+
+	data.fd = drm_open_driver_master(DRIVER_AMDGPU);
+	kmstest_set_vt_graphics_mode();
+
+	igt_display_require(&data.display, data.fd);
+	igt_require(data.display.is_atomic);
+	igt_display_require_output(&data.display);
+
+	igt_amd_require_hpd(&data.display, data.fd);
+	igt_amd_require_mst(&data.display, data.fd);
+
+	while((c = getopt(argc, argv, optstr)) != -1) {
+		switch(c) {
+		case 'n':
+			data.usr_num_mst_end = atoi(optarg);
+			break;
+		case 'h':
+		default:
+			usage(argv[0]);
+			exit(EXIT_SUCCESS);
+		}
+	}
+
+	test_mst_hotplug_basic(&data);
+
+	igt_display_fini(&data.display);
+}
diff --git a/tests/amdgpu/meson.build b/tests/amdgpu/meson.build
index f4a0b9fd..5964c080 100644
--- a/tests/amdgpu/meson.build
+++ b/tests/amdgpu/meson.build
@@ -21,6 +21,7 @@ if libdrm_amdgpu.found()
 			  'amd_psr',
 			  'amd_plane',
 			  'amd_ilr',
+			  'amd_mst',
 			]
 	amdgpu_deps += libdrm_amdgpu
 endif
-- 
2.25.1

^ permalink raw reply related	[flat|nested] 5+ messages in thread

* [igt-dev] ✗ Fi.CI.BUILD: failure for test/amdgpu: Add DP mst test
  2022-07-11  3:50 [igt-dev] [PATCH] test/amdgpu: Add DP mst test Wayne Lin
@ 2022-07-11  3:53 ` Patchwork
  2022-07-20 16:26 ` [igt-dev] [PATCH] " Rodrigo Siqueira Jordao
  1 sibling, 0 replies; 5+ messages in thread
From: Patchwork @ 2022-07-11  3:53 UTC (permalink / raw)
  To: Wayne Lin; +Cc: igt-dev

== Series Details ==

Series: test/amdgpu: Add DP mst test
URL   : https://patchwork.freedesktop.org/series/106166/
State : failure

== Summary ==

IGT patchset build failed on latest successful build
7d43b49bf10788d4870668f93a800888fc8ab339 Revert "lib/i915: request CPU_ACCESS for fb objects"

311/330 testcase check amdgpu/amd_psr                 OK               0.12s
312/330 testcase check amdgpu/amd_ilr                 OK               0.11s
313/330 testcase check amdgpu/amd_plane               OK               0.11s
314/330 testcase check amdgpu/amd_mst                 FAIL             0.10s   exit status 1
>>> MALLOC_PERTURB_=71 /home/cidrm/igt-gpu-tools/tests/igt_command_line.sh amdgpu/amd_mst
――――――――――――――――――――――――――――――――――――― ✀  ―――――――――――――――――――――――――――――――――――――
tests/amdgpu/amd_mst:
  Checking invalid option handling...
Test requirement not met in function drm_open_driver, file ../lib/drmtest.c:573:
Test requirement: !(fd<0)
No known gpu found for chipset flags 0x8 (amdgpu)
Last errno: 2, No such file or directory
SKIP (-1.000s)
  Checking valid option handling...
FAIL: tests/amdgpu/amd_mst
――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――

315/330 runner_json                                   OK               0.09s
316/330 assembler test/mov                            OK               0.08s
317/330 assembler test/frc                            OK               0.08s
318/330 assembler test/regtype                        OK               0.07s
319/330 assembler test/rndd                           OK               0.07s
320/330 assembler test/rndu                           OK               0.07s
321/330 assembler test/rnde                           OK               0.06s
322/330 assembler test/rnde-intsrc                    OK               0.06s
323/330 assembler test/rndz                           OK               0.05s
324/330 assembler test/lzd                            OK               0.04s
325/330 assembler test/not                            OK               0.04s
326/330 assembler test/immediate                      OK               0.03s
327/330 lib igt_nesting                               OK               2.19s
328/330 lib igt_fork                                  OK               2.73s
329/330 runner                                        OK               4.57s
330/330 testcase check gem_concurrent_all             OK               5.89s

Summary of Failures:

314/330 testcase check amdgpu/amd_mst                 FAIL             0.10s   exit status 1


Ok:                 326 
Expected Fail:      3   
Fail:               1   
Unexpected Pass:    0   
Skipped:            0   
Timeout:            0   

Full log written to /home/cidrm/igt-gpu-tools/build/meson-logs/testlog.txt
FAILED: meson-test 
/usr/bin/meson test --no-rebuild --print-errorlogs
ninja: build stopped: subcommand failed.


^ permalink raw reply	[flat|nested] 5+ messages in thread

* Re: [igt-dev] [PATCH] test/amdgpu: Add DP mst test
  2022-07-11  3:50 [igt-dev] [PATCH] test/amdgpu: Add DP mst test Wayne Lin
  2022-07-11  3:53 ` [igt-dev] ✗ Fi.CI.BUILD: failure for " Patchwork
@ 2022-07-20 16:26 ` Rodrigo Siqueira Jordao
  2022-07-21 17:45   ` Lyude Paul
  1 sibling, 1 reply; 5+ messages in thread
From: Rodrigo Siqueira Jordao @ 2022-07-20 16:26 UTC (permalink / raw)
  To: Wayne Lin, igt-dev, Lyude Paul, Wentland, Harry; +Cc: jerry.zuo, hersenxs.wu

Hi Wayne,

(+Lyude, +Harry)

In general, I'm ok with this patch... but tbh, I'm not sure if I 
understand MST really well to provide a meaningful review, but follows 
some of my comments, and I also Cc other people that know more about MST.

On 2022-07-10 23:50, Wayne Lin wrote:
> [Why & How]
> Add new igt test amd_mst for DP mst feature.
> 
> MST (Multi-Stream Transport) is the feature introduced from DP 1.2.
> By mst, source can transport multiple streams to different stream
> sinks through the same DP Tx.
> 
> This new igt test "amd_mst" validate mst feature from two perspectives:
> 
> * Trigger soft hotplug to see whether all mst end sinks successfully go
>    through mst light up procedure
> * Check if all mst end sinks successfully go through mst light up
>    procedure after suspend-resume

Maybe you could describe the hardware setup you used to test this patch. 
I think this could be useful in case we want to enable this test in our CI.

> 
> Signed-off-by: Wayne Lin <Wayne.Lin@amd.com>
> ---
>   lib/igt_amd.c            | 160 +++++++++++++++++++++
>   lib/igt_amd.h            |  43 ++++++
>   tests/amdgpu/amd_mst.c   | 301 +++++++++++++++++++++++++++++++++++++++
>   tests/amdgpu/meson.build |   1 +
>   4 files changed, 505 insertions(+)
>   create mode 100644 tests/amdgpu/amd_mst.c
> 
> diff --git a/lib/igt_amd.c b/lib/igt_amd.c
> index bef9c193..5de26691 100644
> --- a/lib/igt_amd.c
> +++ b/lib/igt_amd.c
> @@ -1175,3 +1175,163 @@ bool igt_amd_set_visual_confirm(int drm_fd, enum amdgpu_debug_visual_confirm opt
>   
>   	return res;
>   }
> +
> +/**
> + * igt_amd_output_has_mst: check if connector has MST debugfs entries
> + * @drm_fd: DRM file descriptor
> + * @connector_name: The connector's name, on which we're reading the status
> + */
> +static bool igt_amd_output_has_mst(int drm_fd, char *connector_name)
> +{
> +	bool ret = true;
> +

Do we need this ret variable? Maybe you can return the 
igt_amd_output_has_debugfs directly.

> +	ret = igt_amd_output_has_debugfs(drm_fd, connector_name, DEBUGFS_MST_IS_MST_CONNECTOR) &
> +			igt_amd_output_has_debugfs(drm_fd, connector_name, DEBUGFS_MST_PROGRESS_STATUS);
> +    return ret;

nit:
I think you mixed space with tabs here.

> +}
> +
> +/**
> + * igt_amd_require_mst: Checks if connectors have mst debugfs entries
> + * @display: A pointer to an #igt_display_t structure
> + * @drm_fd: DRM file descriptor
> + *
> + * Checks if the AMDGPU driver supports the 'is_mst_connector'
> + * and 'mst_progress_status' entry for mst. Skip test if there
> + * are no relevant debugfs entries.
> + */
> +void igt_amd_require_mst(igt_display_t *display, int drm_fd)
> +{
> +	igt_output_t *output;
> +
> +	for_each_connected_output(display, output) {
> +		if (igt_amd_output_has_mst(drm_fd, output->name))
> +			return;
> +	}

nit:
You can drop `{}` in the for_each part.

> +
> +	igt_skip("No MST relevant debugfs support.\n");
> +}
> +
> +/**
> + * igt_amd_get_mst_role: Get the mst role of this output
> + * @drm_fd: DRM file descriptor
> + * @output: the output to be checked
> + *
> + * DRM enumerates a drm connector to each MST output port. This is used to
> + * determine the mst role of this specific output. The mst role in the
> + * mst topology shoule be 'no', 'root', 'branch' and 'end'
> + */
> +enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output)
> +{
> +	int fd, ret;
> +	char buf[256] = {'\0'};
> +	int i = 0;
> +	enum dp_mst_role role = MST_NONE;
> +
> +	fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
> +	if (fd < 0) {
> +		igt_info("Could not open connector %s debugfs directory\n",
> +			 output->name);
> +		return MST_NONE;
> +	}
> +
> +	if (output->config.connector->connector_type != DRM_MODE_CONNECTOR_DisplayPort)
> +		return MST_NONE;
> +
> +	ret = igt_debugfs_simple_read(fd, DEBUGFS_MST_IS_MST_CONNECTOR, buf, sizeof(buf));
> +	igt_assert_f(ret >= 0, "Reading %s for connector %s failed.\n",
> +		     DEBUGFS_MST_IS_MST_CONNECTOR, output->name);
> +
> +	close(fd);
> +
> +	for (i = 0; i < sizeof(is_mst_connector)/sizeof(char *); i++) {

The first time that I read this `is_mst_connector`, I was a little bit 
confused... maybe you can rename it to something like 
`mst_connector_array`. Also, don't we have IGT helpers for this kind of 
thing?

> +		if (strstr(buf, is_mst_connector[i])) {
> +			role = i;
> +			break;
> +		}
> +	}
> +
> +	if (role != MST_NONE)
> +		igt_info("Connector %s: is a mst %s",
> +		 output->name, buf);
> +	else
> +		igt_info("Connector %s: is not a mst device\n",
> +			 output->name);

nit:
Maybe you can keep the igt_info in a single line.

> +
> +	return role;
> +}
> +
> +/**
> + * igt_amd_get_mst_progress_status: Get the mst progress status of a mst device
> + * @drm_fd: DRM file descriptor
> + * @output: the output to be checked
> + *
> + * Get the mst progress status of this specific output. The mst progress status
> + * is indicated by 'MST_PROBE', 'MST_REMOTE_EDID' , 'MST_ALLOCATE_NEW_PAYLOAD'
> + * and 'MST_CLEAR_ALLOCATED_PAYLOAD'

Maybe you can add each of these statuses as a bullet and add a tiny 
summary of the status meaning?

> + */
> +uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t *output)
> +{
> +	int fd, ret;
> +	char buf[256] = {'\0'};
> +	int idx = 0;
> +	char *token_end;
> +	uint8_t status = MST_STATUS_DEFAULT;
> +
> +	fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
> +	if (fd < 0) {
> +		igt_info("Could not open connector %s debugfs directory\n",
> +			 output->name);
> +		return MST_STATUS_DEFAULT;
> +	}
> +
> +	if (output->config.connector->connector_type != DRM_MODE_CONNECTOR_DisplayPort)
> +		return MST_STATUS_DEFAULT;
> +
> +	ret = igt_debugfs_simple_read(fd, DEBUGFS_MST_PROGRESS_STATUS, buf, sizeof(buf));
> +	igt_assert_f(ret >= 0, "Reading %s for connector %s failed.\n",
> +		     DEBUGFS_MST_PROGRESS_STATUS, output->name);
> +
> +	close(fd);
> +
> +	/* Parse values read from file. */
> +	for (char *token = strtok_r(buf, "\n", &token_end);
> +	     token != NULL;
> +	     token = strtok_r(NULL, "\n", &token_end), idx++)
> +	{
> +		if (strstr(token, "not") == NULL)
> +			status |= BIT(idx);
> +	}
> +
> +	return status;
> +}
> +
> +/**
> + * mst_trigger_hotplug: Trigger mst hotplug
> + * @drm_fd: DRM file descriptor
> + * @output: #igt_output_t to check.
> + *
> + * Trigger soft hotplug on mst connector. Caller should make sure the

just for curiosity, what "soft hotplug" means?

> + * passed output is mst root by igt_amd_get_mst_role(). We can't generate
> + * soft hotplug on mst end device.
> + */
> +void mst_trigger_hotplug(int drm_fd, igt_output_t *output)
> +{
> +    int fd, hpd_fd;

nit:
Use tab instead of space.

> +	int wr_len;
> +	const char *enable_hpd = "1", *disable_hpd = "0";
> +
> +	fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
> +	igt_assert(fd >= 0);
> +	hpd_fd = openat(fd, DEBUGFS_HPD_TRIGGER, O_WRONLY);
> +	close(fd);
> +	igt_assert(hpd_fd >= 0);
> +
> +	wr_len = write(hpd_fd, disable_hpd, strlen(disable_hpd));
> +	igt_assert_eq(wr_len, strlen(disable_hpd));
> +
> +	sleep(3);

Why did you decide on 3 here? Also, a retry attempt mechanism would be 
more reliable, no? I mean, something like tries 5 times, and each time 
you wait X amount of time.

> +
> +	wr_len = write(hpd_fd, enable_hpd, strlen(enable_hpd));
> +	igt_assert_eq(wr_len, strlen(enable_hpd));
> +	close(hpd_fd);
> +}
> \ No newline at end of file
> diff --git a/lib/igt_amd.h b/lib/igt_amd.h
> index 428bfe6f..b88755b3 100644
> --- a/lib/igt_amd.h
> +++ b/lib/igt_amd.h
> @@ -52,6 +52,12 @@
>   /* amdgpu DM interface entries */
>   #define DEBUGFS_DM_VISUAL_CONFIRM "amdgpu_dm_visual_confirm"
>   
> +/* mst related read only status*/
> +#define DEBUGFS_MST_IS_MST_CONNECTOR "is_mst_connector"
> +#define DEBUGFS_MST_PROGRESS_STATUS "mst_progress_status"
> +
> +#define BIT(x) (1ul <<(x))

I think i915 also defines this. Maybe it is time to make this generic 
for the entire IGT.

> +
>   enum amd_dsc_clock_force {
>   	DSC_AUTOMATIC = 0,
>   	DSC_FORCE_ON,
> @@ -131,6 +137,37 @@ enum amdgpu_debug_visual_confirm {
>   	VISUAL_CONFIRM_SWIZZLE	= 9
>   };
>   
> +/*
> + * Enumeration of the role in MST topology
> + */

Maybe this is obvious for everyone that knows MST, but how about adding 
documentation for each MST topology? Something with a short description.

> +enum dp_mst_role {
> +	MST_NONE = 0,
> +	MST_ROOT,
> +	MST_BRANCH,
> +	MST_END,
> +};
> +
> +static const char *const is_mst_connector[] = {
> +    "no",
> +    "root",
> +    "branch",
> +    "end",
> +};
> +
> +/*
> + * Enumeration of MST progress state. The bit order should align
> + * with the 'enum mst_progress_statusto' in amdgpu_dm.h of
> + * upstreamed amdgpu kernel driver
> + */
> +enum mst_progress_status {
> +	MST_STATUS_DEFAULT = 0,
> +	MST_PROBE = BIT(0),
> +	MST_REMOTE_EDID = BIT(1),
> +	MST_ALLOCATE_NEW_PAYLOAD = BIT(2),
> +	MST_CLEAR_ALLOCATED_PAYLOAD = BIT(3),
> +	NUM_MST_STATUS = 4,
> +};
> +
>   uint32_t igt_amd_create_bo(int fd, uint64_t size);
>   void *igt_amd_mmap_bo(int fd, uint32_t handle, uint64_t size, int prot);
>   unsigned int igt_amd_compute_offset(unsigned int* swizzle_pattern,
> @@ -192,4 +229,10 @@ int  igt_amd_read_psr_state(int drm_fd, char *connector_name);
>   bool igt_amd_has_visual_confirm(int drm_fd);
>   int  igt_amd_get_visual_confirm(int drm_fd);
>   bool igt_amd_set_visual_confirm(int drm_fd, enum amdgpu_debug_visual_confirm option);
> +
> +/* DP MST debugfs helpers*/
> +void igt_amd_require_mst(igt_display_t *display, int drm_fd);
> +enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output);
> +uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t *output);
> +void mst_trigger_hotplug(int drm_fd, igt_output_t *output);
>   #endif /* IGT_AMD_H */
> diff --git a/tests/amdgpu/amd_mst.c b/tests/amdgpu/amd_mst.c
> new file mode 100644
> index 00000000..3b65f465
> --- /dev/null
> +++ b/tests/amdgpu/amd_mst.c
> @@ -0,0 +1,301 @@
> +/*
> + * Copyright 2022 Advanced Micro Devices, Inc.
> + *
> + * Permission is hereby granted, free of charge, to any person obtaining a
> + * copy of this software and associated documentation files (the "Software"),
> + * to deal in the Software without restriction, including without limitation
> + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
> + * and/or sell copies of the Software, and to permit persons to whom the
> + * Software is furnished to do so, subject to the following conditions:
> + *
> + * The above copyright notice and this permission notice shall be included in
> + * all copies or substantial portions of the Software.
> + *
> + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
> + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
> + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
> + * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
> + * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
> + * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
> + * OTHER DEALINGS IN THE SOFTWARE.
> + */
> +
> +#include "igt.h"
> +#include "igt_amd.h"
> +
> +IGT_TEST_DESCRIPTION("This igt test validates DP MST feature by:"
> +	"1. Check each mst end sink successfully go through mst light up procedure"
> +	"after hotplug root branch."
> +	"2. Check each mst end sink successfully go through mst light up procedure"
> +	"after suspend-resume");
> +
> +/* Maximum pipes on any AMD ASIC. */
> +#define MAX_PIPES 6
> +
> +typedef struct data {
> +	igt_display_t display;
> +	igt_plane_t *primary[MAX_PIPES];
> +	igt_output_t *output_mst_end[MAX_PIPES];
> +	igt_output_t *output_mst_root[MAX_PIPES];
> +	igt_pipe_t *pipe[MAX_PIPES];
> +	igt_fb_t ref_fb[MAX_PIPES];
> +	drmModeModeInfo mode[MAX_PIPES];
> +	enum pipe pipe_id[MAX_PIPES];
> +	int w[MAX_PIPES];
> +	int h[MAX_PIPES];
> +	int fd;
> +	int num_mst_end;
> +	int usr_num_mst_end;
> +} data_t;
> +
> +static void test_init(data_t *data)
> +{
> +	igt_display_t *display = &data->display;
> +	int i, m, n, max_pipes = display->n_pipes;
> +	enum dp_mst_role role;
> +	drmModeConnector *conn;
> +
> +	for_each_pipe(display, i) {
> +		data->pipe_id[i] = PIPE_A + i;
> +		data->pipe[i] = &data->display.pipes[data->pipe_id[i]];
> +		data->primary[i] = igt_pipe_get_plane_type(
> +			data->pipe[i], DRM_PLANE_TYPE_PRIMARY);
> +	}
> +
> +	/* Find mst connected root/end connector*/
> +	for (i = 0, n = 0, m = 0, data->num_mst_end = 0;
> +		i < display->n_outputs && n < max_pipes && m < max_pipes; i++) {
> +		igt_output_t *output = &display->outputs[i];
> +
> +		conn = output->config.connector;
> +		if (conn->connector_type != DRM_MODE_CONNECTOR_DisplayPort)
> +			continue;
> +
> +		role = igt_amd_get_mst_role(data->fd, output);
> +
> +		switch(role) {
> +			case MST_ROOT:
> +				data->output_mst_root[n++] = output;
> +				break;
> +			case MST_END:
> +				if (!igt_output_is_connected(output))
> +					continue;
> +				data->output_mst_end[m] = output;
> +				igt_assert(kmstest_get_connector_default_mode(
> +				data->fd, output->config.connector, &data->mode[m]));
> +
> +				data->w[m] = data->mode[m].hdisplay;
> +				data->h[m] = data->mode[m].vdisplay;
> +				m++;
> +				data->num_mst_end++;
> +				break;
> +			default:
> +				continue;
> +				break;
> +		}
> +	}
> +
> +	igt_require(data->output_mst_root[0]);
> +	igt_require(data->output_mst_end[0]);
> +	if (data->usr_num_mst_end)
> +		igt_assert_eq(data->usr_num_mst_end, data->num_mst_end);
> +	igt_display_reset(display);
> +}
> +
> +static void test_fini(data_t *data)
> +{
> +	igt_display_t *display = &data->display;
> +
> +	igt_display_reset(display);
> +	igt_display_commit_atomic(display, DRM_MODE_ATOMIC_ALLOW_MODESET, 0);
> +}
> +
> +static bool show_mst_status_result(igt_output_t *output, uint8_t status)
> +{
> +	uint8_t bit_idx = 0;
> +	bool result = true;
> +
> +	igt_info("Connector:%s mst progress status:\n", output->name);
> +
> +	for (bit_idx = 0; bit_idx < NUM_MST_STATUS; bit_idx++) {
> +		switch(BIT(bit_idx)) {
> +			case MST_PROBE:
> +				if ((status & MST_PROBE) == MST_STATUS_DEFAULT)
> +					result = false;
> +				igt_info("Probe:%s\n", (status & MST_PROBE) ? "PASS" : "FAIL");
> +				break;
> +			case MST_REMOTE_EDID:
> +				if ((status & MST_REMOTE_EDID) == MST_STATUS_DEFAULT)
> +					result = false;
> +				igt_info("Remote EDID:%s\n", (status & MST_REMOTE_EDID) ? "PASS" : "FAIL");
> +				break;
> +			case MST_ALLOCATE_NEW_PAYLOAD:
> +				if ((status & MST_ALLOCATE_NEW_PAYLOAD) == MST_STATUS_DEFAULT)
> +					result = false;
> +				igt_info("Allocate new payload:%s\n",
> +					(status & MST_ALLOCATE_NEW_PAYLOAD) ? "PASS" : "FAIL");
> +				break;
> +			default:
> +				break;
> +		}
> +	}
> +
> +	return result;
> +}
> +
> +static void reset_resource(data_t *data)
> +{
> +	igt_display_reset(&data->display);
> +	igt_display_fini(&data->display);
> +	close(data->fd);
> +
> +	memset(data->primary, 0, sizeof(igt_plane_t *)*MAX_PIPES);
> +	memset(data->output_mst_end, 0, sizeof(igt_output_t *)*MAX_PIPES);
> +	memset(data->output_mst_root, 0, sizeof(igt_output_t *)*MAX_PIPES);
> +	memset(data->pipe, 0, sizeof(igt_pipe_t *)*MAX_PIPES);
> +	memset(data->mode, 0, sizeof(drmModeModeInfo)*MAX_PIPES);
> +	memset(data->pipe_id, 0, sizeof(enum pipe)*MAX_PIPES);
> +	memset(data->w, 0, sizeof(int)*MAX_PIPES);
> +	memset(data->h, 0, sizeof(int)*MAX_PIPES);
> +
> +	data->fd = drm_open_driver_master(DRIVER_AMDGPU);
> +	igt_display_require(&data->display, data->fd);
> +}
> +
> +static void set_up_streams(data_t *data)
> +{
> +	igt_output_t *output;
> +	int i;
> +
> +	for (i = 0; i < MAX_PIPES; i++) {
> +		output = data->output_mst_end[i];
> +		if (!output || !igt_output_is_connected(output))
> +			continue;
> +
> +		igt_create_pattern_fb(data->fd, data->w[i], data->h[i],
> +				      DRM_FORMAT_XRGB8888, 0, &data->ref_fb[i]);
> +		igt_output_set_pipe(output, data->pipe_id[i]);
> +		igt_plane_set_fb(data->primary[i], &data->ref_fb[i]);
> +	}
> +	igt_display_commit_atomic(&data->display, DRM_MODE_ATOMIC_ALLOW_MODESET, 0);
> +	sleep(5);
> +}
> +
> +static void validate_mst_progress_result(data_t *data)
> +{
> +	int i = 0;
> +	uint8_t status = 0;
> +	igt_output_t *output;
> +
> +	for (i = 0; i < data->num_mst_end; i++) {
> +		output = data->output_mst_end[i];
> +
> +		if (!output)
> +			continue;
> +
> +		status = igt_amd_get_mst_progress_status(data->fd, output);
> +
> +		/* make sure all mst end device get lit up correctly*/
> +		igt_assert(show_mst_status_result(output, status));
> +	}
> +}
> +
> +static void test_mst_hotplug_basic(data_t *data)
> +{
> +	igt_output_t *output;
> +	int i = 0;
> +
> +	test_init(data);
> +
> +	/* Setup all end mst output */
> +	set_up_streams(data);
> +
> +	/*confirm mst progress is done*/
> +	validate_mst_progress_result(data);
> +
> +	/* Trigger hotplug*/
> +	for (i = 0; i < MAX_PIPES; i++) {
> +		output = data->output_mst_root[i];
> +
> +		if (!output)
> +			continue;
> +
> +		mst_trigger_hotplug(data->fd, output);
> +	}
> +
> +	/* mst connectors are dynamically created/destroyed. Need
> +	 * to free and reallocate resources then commit streams again
> +	 */
> +	reset_resource(data);
> +
> +	test_init(data);
> +
> +	set_up_streams(data);
> +
> +	/*confirm mst progress status again*/
> +	validate_mst_progress_result(data);
> +
> +	/*do suspend resume*/
> +	igt_system_suspend_autoresume(SUSPEND_STATE_MEM, SUSPEND_TEST_NONE);
> +	sleep(5);

Why 5?

Thanks
Siqueira

> +
> +	/*confirm mst progress status again*/
> +	validate_mst_progress_result(data);
> +
> +	for (i = 0; i < MAX_PIPES; i++) {
> +		output = data->output_mst_end[i];
> +
> +		if (!output)
> +			continue;
> +
> +		igt_remove_fb(data->fd, &data->ref_fb[i]);
> +	}
> +
> +	test_fini(data);
> +}
> +
> +const char *optstr = "hn:";
> +static void usage(const char *name)
> +{
> +	igt_info("Usage: %s options\n", name);
> +	igt_info("-h			Show help\n");
> +	igt_info("-n mst_num	Set how many monitors connected in the mst topology\n");
> +	igt_info("NOTE: if -n is not specified, number of monitors will be detected automatically"
> +		"by current connection status\n");
> +}
> +
> +int main(int argc, char **argv)
> +{
> +	data_t data;
> +	int c;
> +
> +	igt_skip_on_simulation();
> +
> +	memset(&data, 0, sizeof(data));
> +
> +	data.fd = drm_open_driver_master(DRIVER_AMDGPU);
> +	kmstest_set_vt_graphics_mode();
> +
> +	igt_display_require(&data.display, data.fd);
> +	igt_require(data.display.is_atomic);
> +	igt_display_require_output(&data.display);
> +
> +	igt_amd_require_hpd(&data.display, data.fd);
> +	igt_amd_require_mst(&data.display, data.fd);
> +
> +	while((c = getopt(argc, argv, optstr)) != -1) {
> +		switch(c) {
> +		case 'n':
> +			data.usr_num_mst_end = atoi(optarg);
> +			break;
> +		case 'h':
> +		default:
> +			usage(argv[0]);
> +			exit(EXIT_SUCCESS);
> +		}
> +	}
> +
> +	test_mst_hotplug_basic(&data);
> +
> +	igt_display_fini(&data.display);
> +}
> diff --git a/tests/amdgpu/meson.build b/tests/amdgpu/meson.build
> index f4a0b9fd..5964c080 100644
> --- a/tests/amdgpu/meson.build
> +++ b/tests/amdgpu/meson.build
> @@ -21,6 +21,7 @@ if libdrm_amdgpu.found()
>   			  'amd_psr',
>   			  'amd_plane',
>   			  'amd_ilr',
> +			  'amd_mst',
>   			]
>   	amdgpu_deps += libdrm_amdgpu
>   endif

^ permalink raw reply	[flat|nested] 5+ messages in thread

* Re: [igt-dev] [PATCH] test/amdgpu: Add DP mst test
  2022-07-20 16:26 ` [igt-dev] [PATCH] " Rodrigo Siqueira Jordao
@ 2022-07-21 17:45   ` Lyude Paul
  2022-08-02  9:23     ` Lin, Wayne
  0 siblings, 1 reply; 5+ messages in thread
From: Lyude Paul @ 2022-07-21 17:45 UTC (permalink / raw)
  To: Rodrigo Siqueira Jordao, Wayne Lin, igt-dev, Wentland, Harry
  Cc: jerry.zuo, hersenxs.wu

Will take a look at this probably early next week, feel free to poke if it
falls through the cracks

On Wed, 2022-07-20 at 12:26 -0400, Rodrigo Siqueira Jordao wrote:
> Hi Wayne,
> 
> (+Lyude, +Harry)
> 
> In general, I'm ok with this patch... but tbh, I'm not sure if I 
> understand MST really well to provide a meaningful review, but follows 
> some of my comments, and I also Cc other people that know more about MST.
> 
> On 2022-07-10 23:50, Wayne Lin wrote:
> > [Why & How]
> > Add new igt test amd_mst for DP mst feature.
> > 
> > MST (Multi-Stream Transport) is the feature introduced from DP 1.2.
> > By mst, source can transport multiple streams to different stream
> > sinks through the same DP Tx.
> > 
> > This new igt test "amd_mst" validate mst feature from two perspectives:
> > 
> > * Trigger soft hotplug to see whether all mst end sinks successfully go
> >    through mst light up procedure
> > * Check if all mst end sinks successfully go through mst light up
> >    procedure after suspend-resume
> 
> Maybe you could describe the hardware setup you used to test this patch. 
> I think this could be useful in case we want to enable this test in our CI.
> 
> > 
> > Signed-off-by: Wayne Lin <Wayne.Lin@amd.com>
> > ---
> >   lib/igt_amd.c            | 160 +++++++++++++++++++++
> >   lib/igt_amd.h            |  43 ++++++
> >   tests/amdgpu/amd_mst.c   | 301 +++++++++++++++++++++++++++++++++++++++
> >   tests/amdgpu/meson.build |   1 +
> >   4 files changed, 505 insertions(+)
> >   create mode 100644 tests/amdgpu/amd_mst.c
> > 
> > diff --git a/lib/igt_amd.c b/lib/igt_amd.c
> > index bef9c193..5de26691 100644
> > --- a/lib/igt_amd.c
> > +++ b/lib/igt_amd.c
> > @@ -1175,3 +1175,163 @@ bool igt_amd_set_visual_confirm(int drm_fd, enum
> > amdgpu_debug_visual_confirm opt
> >   
> >         return res;
> >   }
> > +
> > +/**
> > + * igt_amd_output_has_mst: check if connector has MST debugfs entries
> > + * @drm_fd: DRM file descriptor
> > + * @connector_name: The connector's name, on which we're reading the
> > status
> > + */
> > +static bool igt_amd_output_has_mst(int drm_fd, char *connector_name)
> > +{
> > +       bool ret = true;
> > +
> 
> Do we need this ret variable? Maybe you can return the 
> igt_amd_output_has_debugfs directly.
> 
> > +       ret = igt_amd_output_has_debugfs(drm_fd, connector_name,
> > DEBUGFS_MST_IS_MST_CONNECTOR) &
> > +                       igt_amd_output_has_debugfs(drm_fd, connector_name,
> > DEBUGFS_MST_PROGRESS_STATUS);
> > +    return ret;
> 
> nit:
> I think you mixed space with tabs here.
> 
> > +}
> > +
> > +/**
> > + * igt_amd_require_mst: Checks if connectors have mst debugfs entries
> > + * @display: A pointer to an #igt_display_t structure
> > + * @drm_fd: DRM file descriptor
> > + *
> > + * Checks if the AMDGPU driver supports the 'is_mst_connector'
> > + * and 'mst_progress_status' entry for mst. Skip test if there
> > + * are no relevant debugfs entries.
> > + */
> > +void igt_amd_require_mst(igt_display_t *display, int drm_fd)
> > +{
> > +       igt_output_t *output;
> > +
> > +       for_each_connected_output(display, output) {
> > +               if (igt_amd_output_has_mst(drm_fd, output->name))
> > +                       return;
> > +       }
> 
> nit:
> You can drop `{}` in the for_each part.
> 
> > +
> > +       igt_skip("No MST relevant debugfs support.\n");
> > +}
> > +
> > +/**
> > + * igt_amd_get_mst_role: Get the mst role of this output
> > + * @drm_fd: DRM file descriptor
> > + * @output: the output to be checked
> > + *
> > + * DRM enumerates a drm connector to each MST output port. This is used
> > to
> > + * determine the mst role of this specific output. The mst role in the
> > + * mst topology shoule be 'no', 'root', 'branch' and 'end'
> > + */
> > +enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output)
> > +{
> > +       int fd, ret;
> > +       char buf[256] = {'\0'};
> > +       int i = 0;
> > +       enum dp_mst_role role = MST_NONE;
> > +
> > +       fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
> > +       if (fd < 0) {
> > +               igt_info("Could not open connector %s debugfs
> > directory\n",
> > +                        output->name);
> > +               return MST_NONE;
> > +       }
> > +
> > +       if (output->config.connector->connector_type !=
> > DRM_MODE_CONNECTOR_DisplayPort)
> > +               return MST_NONE;
> > +
> > +       ret = igt_debugfs_simple_read(fd, DEBUGFS_MST_IS_MST_CONNECTOR,
> > buf, sizeof(buf));
> > +       igt_assert_f(ret >= 0, "Reading %s for connector %s failed.\n",
> > +                    DEBUGFS_MST_IS_MST_CONNECTOR, output->name);
> > +
> > +       close(fd);
> > +
> > +       for (i = 0; i < sizeof(is_mst_connector)/sizeof(char *); i++) {
> 
> The first time that I read this `is_mst_connector`, I was a little bit 
> confused... maybe you can rename it to something like 
> `mst_connector_array`. Also, don't we have IGT helpers for this kind of 
> thing?
> 
> > +               if (strstr(buf, is_mst_connector[i])) {
> > +                       role = i;
> > +                       break;
> > +               }
> > +       }
> > +
> > +       if (role != MST_NONE)
> > +               igt_info("Connector %s: is a mst %s",
> > +                output->name, buf);
> > +       else
> > +               igt_info("Connector %s: is not a mst device\n",
> > +                        output->name);
> 
> nit:
> Maybe you can keep the igt_info in a single line.
> 
> > +
> > +       return role;
> > +}
> > +
> > +/**
> > + * igt_amd_get_mst_progress_status: Get the mst progress status of a mst
> > device
> > + * @drm_fd: DRM file descriptor
> > + * @output: the output to be checked
> > + *
> > + * Get the mst progress status of this specific output. The mst progress
> > status
> > + * is indicated by 'MST_PROBE', 'MST_REMOTE_EDID' ,
> > 'MST_ALLOCATE_NEW_PAYLOAD'
> > + * and 'MST_CLEAR_ALLOCATED_PAYLOAD'
> 
> Maybe you can add each of these statuses as a bullet and add a tiny 
> summary of the status meaning?
> 
> > + */
> > +uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t *output)
> > +{
> > +       int fd, ret;
> > +       char buf[256] = {'\0'};
> > +       int idx = 0;
> > +       char *token_end;
> > +       uint8_t status = MST_STATUS_DEFAULT;
> > +
> > +       fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
> > +       if (fd < 0) {
> > +               igt_info("Could not open connector %s debugfs
> > directory\n",
> > +                        output->name);
> > +               return MST_STATUS_DEFAULT;
> > +       }
> > +
> > +       if (output->config.connector->connector_type !=
> > DRM_MODE_CONNECTOR_DisplayPort)
> > +               return MST_STATUS_DEFAULT;
> > +
> > +       ret = igt_debugfs_simple_read(fd, DEBUGFS_MST_PROGRESS_STATUS,
> > buf, sizeof(buf));
> > +       igt_assert_f(ret >= 0, "Reading %s for connector %s failed.\n",
> > +                    DEBUGFS_MST_PROGRESS_STATUS, output->name);
> > +
> > +       close(fd);
> > +
> > +       /* Parse values read from file. */
> > +       for (char *token = strtok_r(buf, "\n", &token_end);
> > +            token != NULL;
> > +            token = strtok_r(NULL, "\n", &token_end), idx++)
> > +       {
> > +               if (strstr(token, "not") == NULL)
> > +                       status |= BIT(idx);
> > +       }
> > +
> > +       return status;
> > +}
> > +
> > +/**
> > + * mst_trigger_hotplug: Trigger mst hotplug
> > + * @drm_fd: DRM file descriptor
> > + * @output: #igt_output_t to check.
> > + *
> > + * Trigger soft hotplug on mst connector. Caller should make sure the
> 
> just for curiosity, what "soft hotplug" means?
> 
> > + * passed output is mst root by igt_amd_get_mst_role(). We can't generate
> > + * soft hotplug on mst end device.
> > + */
> > +void mst_trigger_hotplug(int drm_fd, igt_output_t *output)
> > +{
> > +    int fd, hpd_fd;
> 
> nit:
> Use tab instead of space.
> 
> > +       int wr_len;
> > +       const char *enable_hpd = "1", *disable_hpd = "0";
> > +
> > +       fd = igt_debugfs_connector_dir(drm_fd, output->name, O_RDONLY);
> > +       igt_assert(fd >= 0);
> > +       hpd_fd = openat(fd, DEBUGFS_HPD_TRIGGER, O_WRONLY);
> > +       close(fd);
> > +       igt_assert(hpd_fd >= 0);
> > +
> > +       wr_len = write(hpd_fd, disable_hpd, strlen(disable_hpd));
> > +       igt_assert_eq(wr_len, strlen(disable_hpd));
> > +
> > +       sleep(3);
> 
> Why did you decide on 3 here? Also, a retry attempt mechanism would be 
> more reliable, no? I mean, something like tries 5 times, and each time 
> you wait X amount of time.
> 
> > +
> > +       wr_len = write(hpd_fd, enable_hpd, strlen(enable_hpd));
> > +       igt_assert_eq(wr_len, strlen(enable_hpd));
> > +       close(hpd_fd);
> > +}
> > \ No newline at end of file
> > diff --git a/lib/igt_amd.h b/lib/igt_amd.h
> > index 428bfe6f..b88755b3 100644
> > --- a/lib/igt_amd.h
> > +++ b/lib/igt_amd.h
> > @@ -52,6 +52,12 @@
> >   /* amdgpu DM interface entries */
> >   #define DEBUGFS_DM_VISUAL_CONFIRM "amdgpu_dm_visual_confirm"
> >   
> > +/* mst related read only status*/
> > +#define DEBUGFS_MST_IS_MST_CONNECTOR "is_mst_connector"
> > +#define DEBUGFS_MST_PROGRESS_STATUS "mst_progress_status"
> > +
> > +#define BIT(x) (1ul <<(x))
> 
> I think i915 also defines this. Maybe it is time to make this generic 
> for the entire IGT.
> 
> > +
> >   enum amd_dsc_clock_force {
> >         DSC_AUTOMATIC = 0,
> >         DSC_FORCE_ON,
> > @@ -131,6 +137,37 @@ enum amdgpu_debug_visual_confirm {
> >         VISUAL_CONFIRM_SWIZZLE  = 9
> >   };
> >   
> > +/*
> > + * Enumeration of the role in MST topology
> > + */
> 
> Maybe this is obvious for everyone that knows MST, but how about adding 
> documentation for each MST topology? Something with a short description.
> 
> > +enum dp_mst_role {
> > +       MST_NONE = 0,
> > +       MST_ROOT,
> > +       MST_BRANCH,
> > +       MST_END,
> > +};
> > +
> > +static const char *const is_mst_connector[] = {
> > +    "no",
> > +    "root",
> > +    "branch",
> > +    "end",
> > +};
> > +
> > +/*
> > + * Enumeration of MST progress state. The bit order should align
> > + * with the 'enum mst_progress_statusto' in amdgpu_dm.h of
> > + * upstreamed amdgpu kernel driver
> > + */
> > +enum mst_progress_status {
> > +       MST_STATUS_DEFAULT = 0,
> > +       MST_PROBE = BIT(0),
> > +       MST_REMOTE_EDID = BIT(1),
> > +       MST_ALLOCATE_NEW_PAYLOAD = BIT(2),
> > +       MST_CLEAR_ALLOCATED_PAYLOAD = BIT(3),
> > +       NUM_MST_STATUS = 4,
> > +};
> > +
> >   uint32_t igt_amd_create_bo(int fd, uint64_t size);
> >   void *igt_amd_mmap_bo(int fd, uint32_t handle, uint64_t size, int prot);
> >   unsigned int igt_amd_compute_offset(unsigned int* swizzle_pattern,
> > @@ -192,4 +229,10 @@ int  igt_amd_read_psr_state(int drm_fd, char
> > *connector_name);
> >   bool igt_amd_has_visual_confirm(int drm_fd);
> >   int  igt_amd_get_visual_confirm(int drm_fd);
> >   bool igt_amd_set_visual_confirm(int drm_fd, enum
> > amdgpu_debug_visual_confirm option);
> > +
> > +/* DP MST debugfs helpers*/
> > +void igt_amd_require_mst(igt_display_t *display, int drm_fd);
> > +enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output);
> > +uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t
> > *output);
> > +void mst_trigger_hotplug(int drm_fd, igt_output_t *output);
> >   #endif /* IGT_AMD_H */
> > diff --git a/tests/amdgpu/amd_mst.c b/tests/amdgpu/amd_mst.c
> > new file mode 100644
> > index 00000000..3b65f465
> > --- /dev/null
> > +++ b/tests/amdgpu/amd_mst.c
> > @@ -0,0 +1,301 @@
> > +/*
> > + * Copyright 2022 Advanced Micro Devices, Inc.
> > + *
> > + * Permission is hereby granted, free of charge, to any person obtaining
> > a
> > + * copy of this software and associated documentation files (the
> > "Software"),
> > + * to deal in the Software without restriction, including without
> > limitation
> > + * the rights to use, copy, modify, merge, publish, distribute,
> > sublicense,
> > + * and/or sell copies of the Software, and to permit persons to whom the
> > + * Software is furnished to do so, subject to the following conditions:
> > + *
> > + * The above copyright notice and this permission notice shall be
> > included in
> > + * all copies or substantial portions of the Software.
> > + *
> > + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
> > EXPRESS OR
> > + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
> > MERCHANTABILITY,
> > + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT
> > SHALL
> > + * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES
> > OR
> > + * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
> > + * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
> > + * OTHER DEALINGS IN THE SOFTWARE.
> > + */
> > +
> > +#include "igt.h"
> > +#include "igt_amd.h"
> > +
> > +IGT_TEST_DESCRIPTION("This igt test validates DP MST feature by:"
> > +       "1. Check each mst end sink successfully go through mst light up
> > procedure"
> > +       "after hotplug root branch."
> > +       "2. Check each mst end sink successfully go through mst light up
> > procedure"
> > +       "after suspend-resume");
> > +
> > +/* Maximum pipes on any AMD ASIC. */
> > +#define MAX_PIPES 6
> > +
> > +typedef struct data {
> > +       igt_display_t display;
> > +       igt_plane_t *primary[MAX_PIPES];
> > +       igt_output_t *output_mst_end[MAX_PIPES];
> > +       igt_output_t *output_mst_root[MAX_PIPES];
> > +       igt_pipe_t *pipe[MAX_PIPES];
> > +       igt_fb_t ref_fb[MAX_PIPES];
> > +       drmModeModeInfo mode[MAX_PIPES];
> > +       enum pipe pipe_id[MAX_PIPES];
> > +       int w[MAX_PIPES];
> > +       int h[MAX_PIPES];
> > +       int fd;
> > +       int num_mst_end;
> > +       int usr_num_mst_end;
> > +} data_t;
> > +
> > +static void test_init(data_t *data)
> > +{
> > +       igt_display_t *display = &data->display;
> > +       int i, m, n, max_pipes = display->n_pipes;
> > +       enum dp_mst_role role;
> > +       drmModeConnector *conn;
> > +
> > +       for_each_pipe(display, i) {
> > +               data->pipe_id[i] = PIPE_A + i;
> > +               data->pipe[i] = &data->display.pipes[data->pipe_id[i]];
> > +               data->primary[i] = igt_pipe_get_plane_type(
> > +                       data->pipe[i], DRM_PLANE_TYPE_PRIMARY);
> > +       }
> > +
> > +       /* Find mst connected root/end connector*/
> > +       for (i = 0, n = 0, m = 0, data->num_mst_end = 0;
> > +               i < display->n_outputs && n < max_pipes && m < max_pipes;
> > i++) {
> > +               igt_output_t *output = &display->outputs[i];
> > +
> > +               conn = output->config.connector;
> > +               if (conn->connector_type !=
> > DRM_MODE_CONNECTOR_DisplayPort)
> > +                       continue;
> > +
> > +               role = igt_amd_get_mst_role(data->fd, output);
> > +
> > +               switch(role) {
> > +                       case MST_ROOT:
> > +                               data->output_mst_root[n++] = output;
> > +                               break;
> > +                       case MST_END:
> > +                               if (!igt_output_is_connected(output))
> > +                                       continue;
> > +                               data->output_mst_end[m] = output;
> > +                               igt_assert(kmstest_get_connector_default_m
> > ode(
> > +                               data->fd, output->config.connector, &data-
> > >mode[m]));
> > +
> > +                               data->w[m] = data->mode[m].hdisplay;
> > +                               data->h[m] = data->mode[m].vdisplay;
> > +                               m++;
> > +                               data->num_mst_end++;
> > +                               break;
> > +                       default:
> > +                               continue;
> > +                               break;
> > +               }
> > +       }
> > +
> > +       igt_require(data->output_mst_root[0]);
> > +       igt_require(data->output_mst_end[0]);
> > +       if (data->usr_num_mst_end)
> > +               igt_assert_eq(data->usr_num_mst_end, data->num_mst_end);
> > +       igt_display_reset(display);
> > +}
> > +
> > +static void test_fini(data_t *data)
> > +{
> > +       igt_display_t *display = &data->display;
> > +
> > +       igt_display_reset(display);
> > +       igt_display_commit_atomic(display, DRM_MODE_ATOMIC_ALLOW_MODESET,
> > 0);
> > +}
> > +
> > +static bool show_mst_status_result(igt_output_t *output, uint8_t status)
> > +{
> > +       uint8_t bit_idx = 0;
> > +       bool result = true;
> > +
> > +       igt_info("Connector:%s mst progress status:\n", output->name);
> > +
> > +       for (bit_idx = 0; bit_idx < NUM_MST_STATUS; bit_idx++) {
> > +               switch(BIT(bit_idx)) {
> > +                       case MST_PROBE:
> > +                               if ((status & MST_PROBE) ==
> > MST_STATUS_DEFAULT)
> > +                                       result = false;
> > +                               igt_info("Probe:%s\n", (status &
> > MST_PROBE) ? "PASS" : "FAIL");
> > +                               break;
> > +                       case MST_REMOTE_EDID:
> > +                               if ((status & MST_REMOTE_EDID) ==
> > MST_STATUS_DEFAULT)
> > +                                       result = false;
> > +                               igt_info("Remote EDID:%s\n", (status &
> > MST_REMOTE_EDID) ? "PASS" : "FAIL");
> > +                               break;
> > +                       case MST_ALLOCATE_NEW_PAYLOAD:
> > +                               if ((status & MST_ALLOCATE_NEW_PAYLOAD) ==
> > MST_STATUS_DEFAULT)
> > +                                       result = false;
> > +                               igt_info("Allocate new payload:%s\n",
> > +                                       (status &
> > MST_ALLOCATE_NEW_PAYLOAD) ? "PASS" : "FAIL");
> > +                               break;
> > +                       default:
> > +                               break;
> > +               }
> > +       }
> > +
> > +       return result;
> > +}
> > +
> > +static void reset_resource(data_t *data)
> > +{
> > +       igt_display_reset(&data->display);
> > +       igt_display_fini(&data->display);
> > +       close(data->fd);
> > +
> > +       memset(data->primary, 0, sizeof(igt_plane_t *)*MAX_PIPES);
> > +       memset(data->output_mst_end, 0, sizeof(igt_output_t *)*MAX_PIPES);
> > +       memset(data->output_mst_root, 0, sizeof(igt_output_t
> > *)*MAX_PIPES);
> > +       memset(data->pipe, 0, sizeof(igt_pipe_t *)*MAX_PIPES);
> > +       memset(data->mode, 0, sizeof(drmModeModeInfo)*MAX_PIPES);
> > +       memset(data->pipe_id, 0, sizeof(enum pipe)*MAX_PIPES);
> > +       memset(data->w, 0, sizeof(int)*MAX_PIPES);
> > +       memset(data->h, 0, sizeof(int)*MAX_PIPES);
> > +
> > +       data->fd = drm_open_driver_master(DRIVER_AMDGPU);
> > +       igt_display_require(&data->display, data->fd);
> > +}
> > +
> > +static void set_up_streams(data_t *data)
> > +{
> > +       igt_output_t *output;
> > +       int i;
> > +
> > +       for (i = 0; i < MAX_PIPES; i++) {
> > +               output = data->output_mst_end[i];
> > +               if (!output || !igt_output_is_connected(output))
> > +                       continue;
> > +
> > +               igt_create_pattern_fb(data->fd, data->w[i], data->h[i],
> > +                                     DRM_FORMAT_XRGB8888, 0, &data-
> > >ref_fb[i]);
> > +               igt_output_set_pipe(output, data->pipe_id[i]);
> > +               igt_plane_set_fb(data->primary[i], &data->ref_fb[i]);
> > +       }
> > +       igt_display_commit_atomic(&data->display,
> > DRM_MODE_ATOMIC_ALLOW_MODESET, 0);
> > +       sleep(5);
> > +}
> > +
> > +static void validate_mst_progress_result(data_t *data)
> > +{
> > +       int i = 0;
> > +       uint8_t status = 0;
> > +       igt_output_t *output;
> > +
> > +       for (i = 0; i < data->num_mst_end; i++) {
> > +               output = data->output_mst_end[i];
> > +
> > +               if (!output)
> > +                       continue;
> > +
> > +               status = igt_amd_get_mst_progress_status(data->fd,
> > output);
> > +
> > +               /* make sure all mst end device get lit up correctly*/
> > +               igt_assert(show_mst_status_result(output, status));
> > +       }
> > +}
> > +
> > +static void test_mst_hotplug_basic(data_t *data)
> > +{
> > +       igt_output_t *output;
> > +       int i = 0;
> > +
> > +       test_init(data);
> > +
> > +       /* Setup all end mst output */
> > +       set_up_streams(data);
> > +
> > +       /*confirm mst progress is done*/
> > +       validate_mst_progress_result(data);
> > +
> > +       /* Trigger hotplug*/
> > +       for (i = 0; i < MAX_PIPES; i++) {
> > +               output = data->output_mst_root[i];
> > +
> > +               if (!output)
> > +                       continue;
> > +
> > +               mst_trigger_hotplug(data->fd, output);
> > +       }
> > +
> > +       /* mst connectors are dynamically created/destroyed. Need
> > +        * to free and reallocate resources then commit streams again
> > +        */
> > +       reset_resource(data);
> > +
> > +       test_init(data);
> > +
> > +       set_up_streams(data);
> > +
> > +       /*confirm mst progress status again*/
> > +       validate_mst_progress_result(data);
> > +
> > +       /*do suspend resume*/
> > +       igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
> > SUSPEND_TEST_NONE);
> > +       sleep(5);
> 
> Why 5?
> 
> Thanks
> Siqueira
> 
> > +
> > +       /*confirm mst progress status again*/
> > +       validate_mst_progress_result(data);
> > +
> > +       for (i = 0; i < MAX_PIPES; i++) {
> > +               output = data->output_mst_end[i];
> > +
> > +               if (!output)
> > +                       continue;
> > +
> > +               igt_remove_fb(data->fd, &data->ref_fb[i]);
> > +       }
> > +
> > +       test_fini(data);
> > +}
> > +
> > +const char *optstr = "hn:";
> > +static void usage(const char *name)
> > +{
> > +       igt_info("Usage: %s options\n", name);
> > +       igt_info("-h                    Show help\n");
> > +       igt_info("-n mst_num    Set how many monitors connected in the mst
> > topology\n");
> > +       igt_info("NOTE: if -n is not specified, number of monitors will be
> > detected automatically"
> > +               "by current connection status\n");
> > +}
> > +
> > +int main(int argc, char **argv)
> > +{
> > +       data_t data;
> > +       int c;
> > +
> > +       igt_skip_on_simulation();
> > +
> > +       memset(&data, 0, sizeof(data));
> > +
> > +       data.fd = drm_open_driver_master(DRIVER_AMDGPU);
> > +       kmstest_set_vt_graphics_mode();
> > +
> > +       igt_display_require(&data.display, data.fd);
> > +       igt_require(data.display.is_atomic);
> > +       igt_display_require_output(&data.display);
> > +
> > +       igt_amd_require_hpd(&data.display, data.fd);
> > +       igt_amd_require_mst(&data.display, data.fd);
> > +
> > +       while((c = getopt(argc, argv, optstr)) != -1) {
> > +               switch(c) {
> > +               case 'n':
> > +                       data.usr_num_mst_end = atoi(optarg);
> > +                       break;
> > +               case 'h':
> > +               default:
> > +                       usage(argv[0]);
> > +                       exit(EXIT_SUCCESS);
> > +               }
> > +       }
> > +
> > +       test_mst_hotplug_basic(&data);
> > +
> > +       igt_display_fini(&data.display);
> > +}
> > diff --git a/tests/amdgpu/meson.build b/tests/amdgpu/meson.build
> > index f4a0b9fd..5964c080 100644
> > --- a/tests/amdgpu/meson.build
> > +++ b/tests/amdgpu/meson.build
> > @@ -21,6 +21,7 @@ if libdrm_amdgpu.found()
> >                           'amd_psr',
> >                           'amd_plane',
> >                           'amd_ilr',
> > +                         'amd_mst',
> >                         ]
> >         amdgpu_deps += libdrm_amdgpu
> >   endif
> 

-- 
Cheers,
 Lyude Paul (she/her)
 Software Engineer at Red Hat

^ permalink raw reply	[flat|nested] 5+ messages in thread

* Re: [igt-dev] [PATCH] test/amdgpu: Add DP mst test
  2022-07-21 17:45   ` Lyude Paul
@ 2022-08-02  9:23     ` Lin, Wayne
  0 siblings, 0 replies; 5+ messages in thread
From: Lin, Wayne @ 2022-08-02  9:23 UTC (permalink / raw)
  To: Lyude Paul, Siqueira, Rodrigo, igt-dev, Wentland, Harry
  Cc: Zuo, Jerry, Wu, Hersen

[Public]

Thanks, Lyude and Siqueira : )

Reply inline to Siqueira's comments:

> -----Original Message-----
> From: Lyude Paul <lyude@redhat.com>
> Sent: Friday, July 22, 2022 1:46 AM
> To: Siqueira, Rodrigo <Rodrigo.Siqueira@amd.com>; Lin, Wayne
> <Wayne.Lin@amd.com>; igt-dev@lists.freedesktop.org; Wentland, Harry
> <Harry.Wentland@amd.com>
> Cc: Kazlauskas, Nicholas <Nicholas.Kazlauskas@amd.com>; Li, Sun peng (Leo)
> <Sunpeng.Li@amd.com>; Pillai, Aurabindo <Aurabindo.Pillai@amd.com>; Wu,
> Hersen <hersenxs.wu@amd.com>; Zuo, Jerry <Jerry.Zuo@amd.com>
> Subject: Re: [PATCH] test/amdgpu: Add DP mst test
> 
> Will take a look at this probably early next week, feel free to poke if it falls
> through the cracks
> 
> On Wed, 2022-07-20 at 12:26 -0400, Rodrigo Siqueira Jordao wrote:
> > Hi Wayne,
> >
> > (+Lyude, +Harry)
> >
> > In general, I'm ok with this patch... but tbh, I'm not sure if I
> > understand MST really well to provide a meaningful review, but follows
> > some of my comments, and I also Cc other people that know more about
> MST.
> >
> > On 2022-07-10 23:50, Wayne Lin wrote:
> > > [Why & How]
> > > Add new igt test amd_mst for DP mst feature.
> > >
> > > MST (Multi-Stream Transport) is the feature introduced from DP 1.2.
> > > By mst, source can transport multiple streams to different stream
> > > sinks through the same DP Tx.
> > >
> > > This new igt test "amd_mst" validate mst feature from two perspectives:
> > >
> > > * Trigger soft hotplug to see whether all mst end sinks successfully
> > > go
> > >    through mst light up procedure
> > > * Check if all mst end sinks successfully go through mst light up
> > >    procedure after suspend-resume
> >
> > Maybe you could describe the hardware setup you used to test this patch.
> > I think this could be useful in case we want to enable this test in our CI.

Thanks, Siqueira. Will include that in next version.

> >
> > >
> > > Signed-off-by: Wayne Lin <Wayne.Lin@amd.com>
> > > ---
> > >   lib/igt_amd.c            | 160 +++++++++++++++++++++
> > >   lib/igt_amd.h            |  43 ++++++
> > >   tests/amdgpu/amd_mst.c   | 301
> > > +++++++++++++++++++++++++++++++++++++++
> > >   tests/amdgpu/meson.build |   1 +
> > >   4 files changed, 505 insertions(+)
> > >   create mode 100644 tests/amdgpu/amd_mst.c
> > >
> > > diff --git a/lib/igt_amd.c b/lib/igt_amd.c index bef9c193..5de26691
> > > 100644
> > > --- a/lib/igt_amd.c
> > > +++ b/lib/igt_amd.c
> > > @@ -1175,3 +1175,163 @@ bool igt_amd_set_visual_confirm(int drm_fd,
> > > enum amdgpu_debug_visual_confirm opt
> > >
> > >         return res;
> > >   }
> > > +
> > > +/**
> > > + * igt_amd_output_has_mst: check if connector has MST debugfs
> > > +entries
> > > + * @drm_fd: DRM file descriptor
> > > + * @connector_name: The connector's name, on which we're reading
> > > +the
> > > status
> > > + */
> > > +static bool igt_amd_output_has_mst(int drm_fd, char
> > > +*connector_name) {
> > > +       bool ret = true;
> > > +
> >
> > Do we need this ret variable? Maybe you can return the
> > igt_amd_output_has_debugfs directly.

Thanks. Will revise.
> >
> > > +       ret = igt_amd_output_has_debugfs(drm_fd, connector_name,
> > > DEBUGFS_MST_IS_MST_CONNECTOR) &
> > > +                       igt_amd_output_has_debugfs(drm_fd,
> > > +connector_name,
> > > DEBUGFS_MST_PROGRESS_STATUS);
> > > +    return ret;
> >
> > nit:
> > I think you mixed space with tabs here.

Ack. Thanks!
> >
> > > +}
> > > +
> > > +/**
> > > + * igt_amd_require_mst: Checks if connectors have mst debugfs
> > > +entries
> > > + * @display: A pointer to an #igt_display_t structure
> > > + * @drm_fd: DRM file descriptor
> > > + *
> > > + * Checks if the AMDGPU driver supports the 'is_mst_connector'
> > > + * and 'mst_progress_status' entry for mst. Skip test if there
> > > + * are no relevant debugfs entries.
> > > + */
> > > +void igt_amd_require_mst(igt_display_t *display, int drm_fd) {
> > > +       igt_output_t *output;
> > > +
> > > +       for_each_connected_output(display, output) {
> > > +               if (igt_amd_output_has_mst(drm_fd, output->name))
> > > +                       return;
> > > +       }
> >
> > nit:
> > You can drop `{}` in the for_each part.

Thanks. Will revise.
> >
> > > +
> > > +       igt_skip("No MST relevant debugfs support.\n"); }
> > > +
> > > +/**
> > > + * igt_amd_get_mst_role: Get the mst role of this output
> > > + * @drm_fd: DRM file descriptor
> > > + * @output: the output to be checked
> > > + *
> > > + * DRM enumerates a drm connector to each MST output port. This is
> > > +used
> > > to
> > > + * determine the mst role of this specific output. The mst role in
> > > +the
> > > + * mst topology shoule be 'no', 'root', 'branch' and 'end'
> > > + */
> > > +enum dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t
> > > +*output) {
> > > +       int fd, ret;
> > > +       char buf[256] = {'\0'};
> > > +       int i = 0;
> > > +       enum dp_mst_role role = MST_NONE;
> > > +
> > > +       fd = igt_debugfs_connector_dir(drm_fd, output->name,
> > > +O_RDONLY);
> > > +       if (fd < 0) {
> > > +               igt_info("Could not open connector %s debugfs
> > > directory\n",
> > > +                        output->name);
> > > +               return MST_NONE;
> > > +       }
> > > +
> > > +       if (output->config.connector->connector_type !=
> > > DRM_MODE_CONNECTOR_DisplayPort)
> > > +               return MST_NONE;
> > > +
> > > +       ret = igt_debugfs_simple_read(fd,
> > > +DEBUGFS_MST_IS_MST_CONNECTOR,
> > > buf, sizeof(buf));
> > > +       igt_assert_f(ret >= 0, "Reading %s for connector %s
> > > +failed.\n",
> > > +                    DEBUGFS_MST_IS_MST_CONNECTOR, output->name);
> > > +
> > > +       close(fd);
> > > +
> > > +       for (i = 0; i < sizeof(is_mst_connector)/sizeof(char *);
> > > +i++) {
> >
> > The first time that I read this `is_mst_connector`, I was a little bit
> > confused... maybe you can rename it to something like
> > `mst_connector_array`. Also, don't we have IGT helpers for this kind
> > of thing?

Rename it to mst_roles. 
I'm not aware if there is a similar helper to identify the mst role of a specific
drm connector. Currently, we create drm connectors for each mst output ports
but which doesn't mean each of them is a stream sink. An output port might
connect to another mst branch device. This function is trying to identify the mst
role of a specific drm connector. Then we can take appropriate action/expectation
on that connector.
 
> >
> > > +               if (strstr(buf, is_mst_connector[i])) {
> > > +                       role = i;
> > > +                       break;
> > > +               }
> > > +       }
> > > +
> > > +       if (role != MST_NONE)
> > > +               igt_info("Connector %s: is a mst %s",
> > > +                output->name, buf);
> > > +       else
> > > +               igt_info("Connector %s: is not a mst device\n",
> > > +                        output->name);
> >
> > nit:
> > Maybe you can keep the igt_info in a single line.

Ack. Thanks!
> >
> > > +
> > > +       return role;
> > > +}
> > > +
> > > +/**
> > > + * igt_amd_get_mst_progress_status: Get the mst progress status of
> > > +a mst
> > > device
> > > + * @drm_fd: DRM file descriptor
> > > + * @output: the output to be checked
> > > + *
> > > + * Get the mst progress status of this specific output. The mst
> > > + progress
> > > status
> > > + * is indicated by 'MST_PROBE', 'MST_REMOTE_EDID' ,
> > > 'MST_ALLOCATE_NEW_PAYLOAD'
> > > + * and 'MST_CLEAR_ALLOCATED_PAYLOAD'
> >
> > Maybe you can add each of these statuses as a bullet and add a tiny
> > summary of the status meaning?

Thanks! Will revise.
> >
> > > + */
> > > +uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t
> > > +*output) {
> > > +       int fd, ret;
> > > +       char buf[256] = {'\0'};
> > > +       int idx = 0;
> > > +       char *token_end;
> > > +       uint8_t status = MST_STATUS_DEFAULT;
> > > +
> > > +       fd = igt_debugfs_connector_dir(drm_fd, output->name,
> > > +O_RDONLY);
> > > +       if (fd < 0) {
> > > +               igt_info("Could not open connector %s debugfs
> > > directory\n",
> > > +                        output->name);
> > > +               return MST_STATUS_DEFAULT;
> > > +       }
> > > +
> > > +       if (output->config.connector->connector_type !=
> > > DRM_MODE_CONNECTOR_DisplayPort)
> > > +               return MST_STATUS_DEFAULT;
> > > +
> > > +       ret = igt_debugfs_simple_read(fd,
> > > +DEBUGFS_MST_PROGRESS_STATUS,
> > > buf, sizeof(buf));
> > > +       igt_assert_f(ret >= 0, "Reading %s for connector %s
> > > +failed.\n",
> > > +                    DEBUGFS_MST_PROGRESS_STATUS, output->name);
> > > +
> > > +       close(fd);
> > > +
> > > +       /* Parse values read from file. */
> > > +       for (char *token = strtok_r(buf, "\n", &token_end);
> > > +            token != NULL;
> > > +            token = strtok_r(NULL, "\n", &token_end), idx++)
> > > +       {
> > > +               if (strstr(token, "not") == NULL)
> > > +                       status |= BIT(idx);
> > > +       }
> > > +
> > > +       return status;
> > > +}
> > > +
> > > +/**
> > > + * mst_trigger_hotplug: Trigger mst hotplug
> > > + * @drm_fd: DRM file descriptor
> > > + * @output: #igt_output_t to check.
> > > + *
> > > + * Trigger soft hotplug on mst connector. Caller should make sure
> > > +the
> >
> > just for curiosity, what "soft hotplug" means?

I meant to trigger sw hotplug handling procedure by debugfs. Not from
physical HPD pin : p

> >
> > > + * passed output is mst root by igt_amd_get_mst_role(). We can't
> > > +generate
> > > + * soft hotplug on mst end device.
> > > + */
> > > +void mst_trigger_hotplug(int drm_fd, igt_output_t *output) {
> > > +    int fd, hpd_fd;
> >
> > nit:
> > Use tab instead of space.

Oops. Thanks! Will revise.
> >
> > > +       int wr_len;
> > > +       const char *enable_hpd = "1", *disable_hpd = "0";
> > > +
> > > +       fd = igt_debugfs_connector_dir(drm_fd, output->name,
> > > +O_RDONLY);
> > > +       igt_assert(fd >= 0);
> > > +       hpd_fd = openat(fd, DEBUGFS_HPD_TRIGGER, O_WRONLY);
> > > +       close(fd);
> > > +       igt_assert(hpd_fd >= 0);
> > > +
> > > +       wr_len = write(hpd_fd, disable_hpd, strlen(disable_hpd));
> > > +       igt_assert_eq(wr_len, strlen(disable_hpd));
> > > +
> > > +       sleep(3);
> >
> > Why did you decide on 3 here? Also, a retry attempt mechanism would be
> > more reliable, no? I mean, something like tries 5 times, and each time
> > you wait X amount of time.

This function is for simulating unplugging mst root connector and plug it 
in again. The reason here I chose 3 is because after unplugging the mst root
connector, we still have work to destroy stale mst drm connectors. I just wanna
have next time plug in event be clean enough. I supposed that hotplug should be 
triggered immediately so didn’t want to use retry attempt mechanism. If it fails,
should directly take a look of that.
> >
> > > +
> > > +       wr_len = write(hpd_fd, enable_hpd, strlen(enable_hpd));
> > > +       igt_assert_eq(wr_len, strlen(enable_hpd));
> > > +       close(hpd_fd);
> > > +}
> > > \ No newline at end of file
> > > diff --git a/lib/igt_amd.h b/lib/igt_amd.h index 428bfe6f..b88755b3
> > > 100644
> > > --- a/lib/igt_amd.h
> > > +++ b/lib/igt_amd.h
> > > @@ -52,6 +52,12 @@
> > >   /* amdgpu DM interface entries */
> > >   #define DEBUGFS_DM_VISUAL_CONFIRM
> "amdgpu_dm_visual_confirm"
> > >
> > > +/* mst related read only status*/
> > > +#define DEBUGFS_MST_IS_MST_CONNECTOR "is_mst_connector"
> > > +#define DEBUGFS_MST_PROGRESS_STATUS "mst_progress_status"
> > > +
> > > +#define BIT(x) (1ul <<(x))
> >
> > I think i915 also defines this. Maybe it is time to make this generic
> > for the entire IGT.

Thanks and agree on that. I'll create another thread for moving this into
Igt_core.h
> >
> > > +
> > >   enum amd_dsc_clock_force {
> > >         DSC_AUTOMATIC = 0,
> > >         DSC_FORCE_ON,
> > > @@ -131,6 +137,37 @@ enum amdgpu_debug_visual_confirm {
> > >         VISUAL_CONFIRM_SWIZZLE  = 9
> > >   };
> > >
> > > +/*
> > > + * Enumeration of the role in MST topology  */
> >
> > Maybe this is obvious for everyone that knows MST, but how about
> > adding documentation for each MST topology? Something with a short
> description.

Will add short description of these. Thanks.
> >
> > > +enum dp_mst_role {
> > > +       MST_NONE = 0,
> > > +       MST_ROOT,
> > > +       MST_BRANCH,
> > > +       MST_END,
> > > +};
> > > +
> > > +static const char *const is_mst_connector[] = {
> > > +    "no",
> > > +    "root",
> > > +    "branch",
> > > +    "end",
> > > +};
> > > +
> > > +/*
> > > + * Enumeration of MST progress state. The bit order should align
> > > + * with the 'enum mst_progress_statusto' in amdgpu_dm.h of
> > > + * upstreamed amdgpu kernel driver
> > > + */
> > > +enum mst_progress_status {
> > > +       MST_STATUS_DEFAULT = 0,
> > > +       MST_PROBE = BIT(0),
> > > +       MST_REMOTE_EDID = BIT(1),
> > > +       MST_ALLOCATE_NEW_PAYLOAD = BIT(2),
> > > +       MST_CLEAR_ALLOCATED_PAYLOAD = BIT(3),
> > > +       NUM_MST_STATUS = 4,
> > > +};
> > > +
> > >   uint32_t igt_amd_create_bo(int fd, uint64_t size);
> > >   void *igt_amd_mmap_bo(int fd, uint32_t handle, uint64_t size, int
> > > prot);
> > >   unsigned int igt_amd_compute_offset(unsigned int* swizzle_pattern,
> > > @@ -192,4 +229,10 @@ int  igt_amd_read_psr_state(int drm_fd, char
> > > *connector_name);
> > >   bool igt_amd_has_visual_confirm(int drm_fd);
> > >   int  igt_amd_get_visual_confirm(int drm_fd);
> > >   bool igt_amd_set_visual_confirm(int drm_fd, enum
> > > amdgpu_debug_visual_confirm option);
> > > +
> > > +/* DP MST debugfs helpers*/
> > > +void igt_amd_require_mst(igt_display_t *display, int drm_fd); enum
> > > +dp_mst_role igt_amd_get_mst_role(int drm_fd, igt_output_t *output);
> > > +uint8_t igt_amd_get_mst_progress_status(int drm_fd, igt_output_t
> > > *output);
> > > +void mst_trigger_hotplug(int drm_fd, igt_output_t *output);
> > >   #endif /* IGT_AMD_H */
> > > diff --git a/tests/amdgpu/amd_mst.c b/tests/amdgpu/amd_mst.c new
> > > file mode 100644 index 00000000..3b65f465
> > > --- /dev/null
> > > +++ b/tests/amdgpu/amd_mst.c
> > > @@ -0,0 +1,301 @@
> > > +/*
> > > + * Copyright 2022 Advanced Micro Devices, Inc.
> > > + *
> > > + * Permission is hereby granted, free of charge, to any person
> > > +obtaining
> > > a
> > > + * copy of this software and associated documentation files (the
> > > "Software"),
> > > + * to deal in the Software without restriction, including without
> > > limitation
> > > + * the rights to use, copy, modify, merge, publish, distribute,
> > > sublicense,
> > > + * and/or sell copies of the Software, and to permit persons to
> > > + whom the
> > > + * Software is furnished to do so, subject to the following conditions:
> > > + *
> > > + * The above copyright notice and this permission notice shall be
> > > included in
> > > + * all copies or substantial portions of the Software.
> > > + *
> > > + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
> KIND,
> > > EXPRESS OR
> > > + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
> > > MERCHANTABILITY,
> > > + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN
> NO
> > > + EVENT
> > > SHALL
> > > + * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM,
> > > + DAMAGES
> > > OR
> > > + * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
> > > + OTHERWISE,
> > > + * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR
> THE
> > > + USE OR
> > > + * OTHER DEALINGS IN THE SOFTWARE.
> > > + */
> > > +
> > > +#include "igt.h"
> > > +#include "igt_amd.h"
> > > +
> > > +IGT_TEST_DESCRIPTION("This igt test validates DP MST feature by:"
> > > +       "1. Check each mst end sink successfully go through mst
> > > +light up
> > > procedure"
> > > +       "after hotplug root branch."
> > > +       "2. Check each mst end sink successfully go through mst
> > > +light up
> > > procedure"
> > > +       "after suspend-resume");
> > > +
> > > +/* Maximum pipes on any AMD ASIC. */ #define MAX_PIPES 6
> > > +
> > > +typedef struct data {
> > > +       igt_display_t display;
> > > +       igt_plane_t *primary[MAX_PIPES];
> > > +       igt_output_t *output_mst_end[MAX_PIPES];
> > > +       igt_output_t *output_mst_root[MAX_PIPES];
> > > +       igt_pipe_t *pipe[MAX_PIPES];
> > > +       igt_fb_t ref_fb[MAX_PIPES];
> > > +       drmModeModeInfo mode[MAX_PIPES];
> > > +       enum pipe pipe_id[MAX_PIPES];
> > > +       int w[MAX_PIPES];
> > > +       int h[MAX_PIPES];
> > > +       int fd;
> > > +       int num_mst_end;
> > > +       int usr_num_mst_end;
> > > +} data_t;
> > > +
> > > +static void test_init(data_t *data) {
> > > +       igt_display_t *display = &data->display;
> > > +       int i, m, n, max_pipes = display->n_pipes;
> > > +       enum dp_mst_role role;
> > > +       drmModeConnector *conn;
> > > +
> > > +       for_each_pipe(display, i) {
> > > +               data->pipe_id[i] = PIPE_A + i;
> > > +               data->pipe[i] =
> > > +&data->display.pipes[data->pipe_id[i]];
> > > +               data->primary[i] = igt_pipe_get_plane_type(
> > > +                       data->pipe[i], DRM_PLANE_TYPE_PRIMARY);
> > > +       }
> > > +
> > > +       /* Find mst connected root/end connector*/
> > > +       for (i = 0, n = 0, m = 0, data->num_mst_end = 0;
> > > +               i < display->n_outputs && n < max_pipes && m <
> > > +max_pipes;
> > > i++) {
> > > +               igt_output_t *output = &display->outputs[i];
> > > +
> > > +               conn = output->config.connector;
> > > +               if (conn->connector_type !=
> > > DRM_MODE_CONNECTOR_DisplayPort)
> > > +                       continue;
> > > +
> > > +               role = igt_amd_get_mst_role(data->fd, output);
> > > +
> > > +               switch(role) {
> > > +                       case MST_ROOT:
> > > +                               data->output_mst_root[n++] = output;
> > > +                               break;
> > > +                       case MST_END:
> > > +                               if
> > > +(!igt_output_is_connected(output))
> > > +                                       continue;
> > > +                               data->output_mst_end[m] = output;
> > > +
> > > +igt_assert(kmstest_get_connector_default_m
> > > ode(
> > > +                               data->fd, output->config.connector,
> > > +&data-
> > > >mode[m]));
> > > +
> > > +                               data->w[m] = data->mode[m].hdisplay;
> > > +                               data->h[m] = data->mode[m].vdisplay;
> > > +                               m++;
> > > +                               data->num_mst_end++;
> > > +                               break;
> > > +                       default:
> > > +                               continue;
> > > +                               break;
> > > +               }
> > > +       }
> > > +
> > > +       igt_require(data->output_mst_root[0]);
> > > +       igt_require(data->output_mst_end[0]);
> > > +       if (data->usr_num_mst_end)
> > > +               igt_assert_eq(data->usr_num_mst_end,
> > > +data->num_mst_end);
> > > +       igt_display_reset(display);
> > > +}
> > > +
> > > +static void test_fini(data_t *data) {
> > > +       igt_display_t *display = &data->display;
> > > +
> > > +       igt_display_reset(display);
> > > +       igt_display_commit_atomic(display,
> > > +DRM_MODE_ATOMIC_ALLOW_MODESET,
> > > 0);
> > > +}
> > > +
> > > +static bool show_mst_status_result(igt_output_t *output, uint8_t
> > > +status) {
> > > +       uint8_t bit_idx = 0;
> > > +       bool result = true;
> > > +
> > > +       igt_info("Connector:%s mst progress status:\n",
> > > +output->name);
> > > +
> > > +       for (bit_idx = 0; bit_idx < NUM_MST_STATUS; bit_idx++) {
> > > +               switch(BIT(bit_idx)) {
> > > +                       case MST_PROBE:
> > > +                               if ((status & MST_PROBE) ==
> > > MST_STATUS_DEFAULT)
> > > +                                       result = false;
> > > +                               igt_info("Probe:%s\n", (status &
> > > MST_PROBE) ? "PASS" : "FAIL");
> > > +                               break;
> > > +                       case MST_REMOTE_EDID:
> > > +                               if ((status & MST_REMOTE_EDID) ==
> > > MST_STATUS_DEFAULT)
> > > +                                       result = false;
> > > +                               igt_info("Remote EDID:%s\n", (status
> > > +&
> > > MST_REMOTE_EDID) ? "PASS" : "FAIL");
> > > +                               break;
> > > +                       case MST_ALLOCATE_NEW_PAYLOAD:
> > > +                               if ((status &
> > > +MST_ALLOCATE_NEW_PAYLOAD) ==
> > > MST_STATUS_DEFAULT)
> > > +                                       result = false;
> > > +                               igt_info("Allocate new
> > > +payload:%s\n",
> > > +                                       (status &
> > > MST_ALLOCATE_NEW_PAYLOAD) ? "PASS" : "FAIL");
> > > +                               break;
> > > +                       default:
> > > +                               break;
> > > +               }
> > > +       }
> > > +
> > > +       return result;
> > > +}
> > > +
> > > +static void reset_resource(data_t *data) {
> > > +       igt_display_reset(&data->display);
> > > +       igt_display_fini(&data->display);
> > > +       close(data->fd);
> > > +
> > > +       memset(data->primary, 0, sizeof(igt_plane_t *)*MAX_PIPES);
> > > +       memset(data->output_mst_end, 0, sizeof(igt_output_t
> > > +*)*MAX_PIPES);
> > > +       memset(data->output_mst_root, 0, sizeof(igt_output_t
> > > *)*MAX_PIPES);
> > > +       memset(data->pipe, 0, sizeof(igt_pipe_t *)*MAX_PIPES);
> > > +       memset(data->mode, 0, sizeof(drmModeModeInfo)*MAX_PIPES);
> > > +       memset(data->pipe_id, 0, sizeof(enum pipe)*MAX_PIPES);
> > > +       memset(data->w, 0, sizeof(int)*MAX_PIPES);
> > > +       memset(data->h, 0, sizeof(int)*MAX_PIPES);
> > > +
> > > +       data->fd = drm_open_driver_master(DRIVER_AMDGPU);
> > > +       igt_display_require(&data->display, data->fd); }
> > > +
> > > +static void set_up_streams(data_t *data) {
> > > +       igt_output_t *output;
> > > +       int i;
> > > +
> > > +       for (i = 0; i < MAX_PIPES; i++) {
> > > +               output = data->output_mst_end[i];
> > > +               if (!output || !igt_output_is_connected(output))
> > > +                       continue;
> > > +
> > > +               igt_create_pattern_fb(data->fd, data->w[i],
> > > +data->h[i],
> > > +                                     DRM_FORMAT_XRGB8888, 0, &data-
> > > >ref_fb[i]);
> > > +               igt_output_set_pipe(output, data->pipe_id[i]);
> > > +               igt_plane_set_fb(data->primary[i],
> > > +&data->ref_fb[i]);
> > > +       }
> > > +       igt_display_commit_atomic(&data->display,
> > > DRM_MODE_ATOMIC_ALLOW_MODESET, 0);
> > > +       sleep(5);
> > > +}
> > > +
> > > +static void validate_mst_progress_result(data_t *data) {
> > > +       int i = 0;
> > > +       uint8_t status = 0;
> > > +       igt_output_t *output;
> > > +
> > > +       for (i = 0; i < data->num_mst_end; i++) {
> > > +               output = data->output_mst_end[i];
> > > +
> > > +               if (!output)
> > > +                       continue;
> > > +
> > > +               status = igt_amd_get_mst_progress_status(data->fd,
> > > output);
> > > +
> > > +               /* make sure all mst end device get lit up
> > > +correctly*/
> > > +               igt_assert(show_mst_status_result(output, status));
> > > +       }
> > > +}
> > > +
> > > +static void test_mst_hotplug_basic(data_t *data) {
> > > +       igt_output_t *output;
> > > +       int i = 0;
> > > +
> > > +       test_init(data);
> > > +
> > > +       /* Setup all end mst output */
> > > +       set_up_streams(data);
> > > +
> > > +       /*confirm mst progress is done*/
> > > +       validate_mst_progress_result(data);
> > > +
> > > +       /* Trigger hotplug*/
> > > +       for (i = 0; i < MAX_PIPES; i++) {
> > > +               output = data->output_mst_root[i];
> > > +
> > > +               if (!output)
> > > +                       continue;
> > > +
> > > +               mst_trigger_hotplug(data->fd, output);
> > > +       }
> > > +
> > > +       /* mst connectors are dynamically created/destroyed. Need
> > > +        * to free and reallocate resources then commit streams
> > > +again
> > > +        */
> > > +       reset_resource(data);
> > > +
> > > +       test_init(data);
> > > +
> > > +       set_up_streams(data);
> > > +
> > > +       /*confirm mst progress status again*/
> > > +       validate_mst_progress_result(data);
> > > +
> > > +       /*do suspend resume*/
> > > +       igt_system_suspend_autoresume(SUSPEND_STATE_MEM,
> > > SUSPEND_TEST_NONE);
> > > +       sleep(5);
> >
> > Why 5?

Just a rule of thumb. The timeout for one sideband msg is 4s. Here just want
to give a reasonable waiting time before checking mst progress status after resume.

If the depth of the MST topology is deeper, then it probably takes longer to light up
a mst monitor. It also depends on how busy the branch devices are to handle mst 
sideband msg. I'll add an option to specify this delay. Thanks!

> >
> > Thanks
> > Siqueira
> >
> > > +
> > > +       /*confirm mst progress status again*/
> > > +       validate_mst_progress_result(data);
> > > +
> > > +       for (i = 0; i < MAX_PIPES; i++) {
> > > +               output = data->output_mst_end[i];
> > > +
> > > +               if (!output)
> > > +                       continue;
> > > +
> > > +               igt_remove_fb(data->fd, &data->ref_fb[i]);
> > > +       }
> > > +
> > > +       test_fini(data);
> > > +}
> > > +
> > > +const char *optstr = "hn:";
> > > +static void usage(const char *name) {
> > > +       igt_info("Usage: %s options\n", name);
> > > +       igt_info("-h                    Show help\n");
> > > +       igt_info("-n mst_num    Set how many monitors connected in
> > > +the mst
> > > topology\n");
> > > +       igt_info("NOTE: if -n is not specified, number of monitors
> > > +will be
> > > detected automatically"
> > > +               "by current connection status\n"); }
> > > +
> > > +int main(int argc, char **argv)
> > > +{
> > > +       data_t data;
> > > +       int c;
> > > +
> > > +       igt_skip_on_simulation();
> > > +
> > > +       memset(&data, 0, sizeof(data));
> > > +
> > > +       data.fd = drm_open_driver_master(DRIVER_AMDGPU);
> > > +       kmstest_set_vt_graphics_mode();
> > > +
> > > +       igt_display_require(&data.display, data.fd);
> > > +       igt_require(data.display.is_atomic);
> > > +       igt_display_require_output(&data.display);
> > > +
> > > +       igt_amd_require_hpd(&data.display, data.fd);
> > > +       igt_amd_require_mst(&data.display, data.fd);
> > > +
> > > +       while((c = getopt(argc, argv, optstr)) != -1) {
> > > +               switch(c) {
> > > +               case 'n':
> > > +                       data.usr_num_mst_end = atoi(optarg);
> > > +                       break;
> > > +               case 'h':
> > > +               default:
> > > +                       usage(argv[0]);
> > > +                       exit(EXIT_SUCCESS);
> > > +               }
> > > +       }
> > > +
> > > +       test_mst_hotplug_basic(&data);
> > > +
> > > +       igt_display_fini(&data.display); }
> > > diff --git a/tests/amdgpu/meson.build b/tests/amdgpu/meson.build
> > > index f4a0b9fd..5964c080 100644
> > > --- a/tests/amdgpu/meson.build
> > > +++ b/tests/amdgpu/meson.build
> > > @@ -21,6 +21,7 @@ if libdrm_amdgpu.found()
> > >                           'amd_psr',
> > >                           'amd_plane',
> > >                           'amd_ilr',
> > > +                         'amd_mst',
> > >                         ]
> > >         amdgpu_deps += libdrm_amdgpu
> > >   endif
> >
> 
> --
> Cheers,
>  Lyude Paul (she/her)
>  Software Engineer at Red Hat
-- 
Regards,
Wayne Lin

^ permalink raw reply	[flat|nested] 5+ messages in thread

end of thread, other threads:[~2022-08-02  9:23 UTC | newest]

Thread overview: 5+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2022-07-11  3:50 [igt-dev] [PATCH] test/amdgpu: Add DP mst test Wayne Lin
2022-07-11  3:53 ` [igt-dev] ✗ Fi.CI.BUILD: failure for " Patchwork
2022-07-20 16:26 ` [igt-dev] [PATCH] " Rodrigo Siqueira Jordao
2022-07-21 17:45   ` Lyude Paul
2022-08-02  9:23     ` Lin, Wayne

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.