From: Lyude Paul <lyude at redhat.com> Nouveau has finally gotten CRC support, hooray! Well, it's under review at least: https://patchwork.freedesktop.org/series/74804/ (it has a cover letter, but nouveau's mailing list configuration has blocked the email so I'm waiting for a moderator to fix that) So, this series adds the relevant tests for it since nvidia's CRC implementation has some rather interesting design choices that needed to be worked around. Lyude Paul (4): lib/igt_core: Add igt_require_fd() lib/igt_debugfs: Add igt_debugfs_pipe_dir() lib/igt_kms: Hook up connector dithering prop tests: Add nouveau-crc tests lib/drmtest.c | 10 ++ lib/drmtest.h | 2 + lib/igt_core.h | 12 ++ lib/igt_debugfs.c | 29 ++++ lib/igt_debugfs.h | 1 + lib/igt_kms.c | 6 + lib/igt_kms.h | 1 + tests/meson.build | 1 + tests/nouveau_crc.c | 396 ++++++++++++++++++++++++++++++++++++++++++++ 9 files changed, 458 insertions(+) create mode 100644 tests/nouveau_crc.c -- 2.24.1
Lyude
2020-Mar-18 01:00 UTC
[Nouveau] [PATCH i-g-t 1/4] lib/igt_core: Add igt_require_fd()
From: Lyude Paul <lyude at redhat.com> Like igt_assert_fd(), but using igt_require() instead Signed-off-by: Lyude Paul <lyude at redhat.com> --- lib/igt_core.h | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/lib/igt_core.h b/lib/igt_core.h index fae5f59e..b66336cf 100644 --- a/lib/igt_core.h +++ b/lib/igt_core.h @@ -1008,6 +1008,18 @@ void igt_describe_f(const char *fmt, ...); else igt_debug("Test requirement passed: %s\n", #expr); \ } while (0) +/** + * igt_require_fd: + * @fd: file descriptor + * + * Skips (sub-) test if the given file descriptor is invalid. + * + * Like igt_require(), but displays the values being compared on failure instead + * of simply printing the stringified expression.. + */ +#define igt_require_fd(fd) \ + igt_require_f(fd >= 0, "file descriptor " #fd " failed\n"); + /** * igt_skip_on_f: * @expr: condition to test -- 2.24.1
Lyude
2020-Mar-18 01:00 UTC
[Nouveau] [PATCH i-g-t 2/4] lib/igt_debugfs: Add igt_debugfs_pipe_dir()
From: Lyude Paul <lyude at redhat.com> Like igt_debugfs_connector_dir(), but for pipes instead. Signed-off-by: Lyude Paul <lyude at redhat.com> --- lib/igt_debugfs.c | 29 +++++++++++++++++++++++++++++ lib/igt_debugfs.h | 1 + 2 files changed, 30 insertions(+) diff --git a/lib/igt_debugfs.c b/lib/igt_debugfs.c index bf6be552..3c3b11e1 100644 --- a/lib/igt_debugfs.c +++ b/lib/igt_debugfs.c @@ -260,6 +260,35 @@ int igt_debugfs_connector_dir(int device, char *conn_name, int mode) return ret; } +/** + * igt_debugfs_pipe_dir: + * @device: fd of the device + * @pipe: index of pipe + * @mode: mode bits as used by open() + * + * This opens the debugfs directory corresponding to the pipe index on the + * device for use with igt_sysfs_get() and related functions. + * + * Returns: + * The directory fd, or -1 on failure. + */ +int igt_debugfs_pipe_dir(int device, int pipe, int mode) +{ + char buf[128]; + int dir, ret; + + dir = igt_debugfs_dir(device); + if (dir < 0) + return dir; + + snprintf(buf, sizeof(buf), "crtc-%d", pipe); + ret = openat(dir, buf, mode); + + close(dir); + + return ret; +} + /** * igt_debugfs_open: * @filename: name of the debugfs node to open diff --git a/lib/igt_debugfs.h b/lib/igt_debugfs.h index 7d1a6175..15741a40 100644 --- a/lib/igt_debugfs.h +++ b/lib/igt_debugfs.h @@ -36,6 +36,7 @@ char *igt_debugfs_path(int device, char *path, int pathlen); int igt_debugfs_dir(int device); int igt_debugfs_connector_dir(int device, char *conn_name, int mode); +int igt_debugfs_pipe_dir(int device, int pipe, int mode); int igt_debugfs_open(int fd, const char *filename, int mode); void __igt_debugfs_read(int fd, const char *filename, char *buf, int size); -- 2.24.1
Lyude
2020-Mar-18 01:00 UTC
[Nouveau] [PATCH i-g-t 3/4] lib/igt_kms: Hook up connector dithering prop
From: Lyude Paul <lyude at redhat.com> Nvidia display hardware provides a set of flexible dithering options for CRTCs. This dithering is actually noticeable in the CRC output for all available tap points, and can be seen as CRC values for identical frames cycling between either 2 or 4 values repeatedly (each one of these values is a different dithering phase applied to the source output). Of course, this is very likely to break tests using CRC readback since we don't expect the CRC to change if the source content hasn't changed. So, hook up support for configuring the dithering property and reset it to off from igt_display_reset() when applicable. Signed-off-by: Lyude Paul <lyude at redhat.com> --- lib/igt_kms.c | 6 ++++++ lib/igt_kms.h | 1 + 2 files changed, 7 insertions(+) diff --git a/lib/igt_kms.c b/lib/igt_kms.c index 54de45e5..2bbb363e 100644 --- a/lib/igt_kms.c +++ b/lib/igt_kms.c @@ -418,6 +418,7 @@ const char * const igt_connector_prop_names[IGT_NUM_CONNECTOR_PROPS] = { [IGT_CONNECTOR_LINK_STATUS] = "link-status", [IGT_CONNECTOR_MAX_BPC] = "max bpc", [IGT_CONNECTOR_HDR_OUTPUT_METADATA] = "HDR_OUTPUT_METADATA", + [IGT_CONNECTOR_DITHERING_MODE] = "dithering mode", }; /* @@ -1799,6 +1800,10 @@ static void igt_output_reset(igt_output_t *output) if (igt_output_has_prop(output, IGT_CONNECTOR_HDR_OUTPUT_METADATA)) igt_output_set_prop_value(output, IGT_CONNECTOR_HDR_OUTPUT_METADATA, 0); + + if (igt_output_has_prop(output, IGT_CONNECTOR_DITHERING_MODE)) + igt_output_set_prop_enum(output, IGT_CONNECTOR_DITHERING_MODE, + "off"); } /** @@ -1813,6 +1818,7 @@ static void igt_output_reset(igt_output_t *output) * - %IGT_CONNECTOR_BROADCAST_RGB (if applicable) * %IGT_CONNECTOR_CONTENT_PROTECTION (if applicable) * %IGT_CONNECTOR_HDR_OUTPUT_METADATA (if applicable) + * - %IGT_CONNECTOR_DITHERING_MODE (if applicable) * - igt_output_override_mode() to default. * * For pipes: diff --git a/lib/igt_kms.h b/lib/igt_kms.h index cd3fdbc0..998ed241 100644 --- a/lib/igt_kms.h +++ b/lib/igt_kms.h @@ -127,6 +127,7 @@ enum igt_atomic_connector_properties { IGT_CONNECTOR_LINK_STATUS, IGT_CONNECTOR_MAX_BPC, IGT_CONNECTOR_HDR_OUTPUT_METADATA, + IGT_CONNECTOR_DITHERING_MODE, IGT_NUM_CONNECTOR_PROPS }; -- 2.24.1
From: Lyude Paul <lyude at redhat.com> We're finally getting CRC support in nouveau, so let's start testing this in igt as well! While the normal CRC capture tests are nice, there's a number of Nvidia-specific hardware characteristics that we need to test as well. The most important one is known as a "notifier context flip". Basically, Nvidia GPUs store generated CRCs in an allocated memory region, referred to as the notifier context, that the driver programs itself. Strangely, this region can only hold a limited number of CRC entries, and once it runs out of available entries the hardware simply sets an overrun bit and stops writing any new CRC entries. Since igt-gpu-tools doesn't really have an expectation of only being able to grab a limited number of CRCs, we work around this in nouveau by allocating two separate CRC notifier regions each time we start capturing CRCs, and then flip between them whenever we get close to filling our currently programmed notifier context. While this keeps the number of CRC entries we lose to an absolute minimum, we are guaranteed to lose exactly one CRC entry between context flips. Thus, we add some tests to ensure that nouveau handles these flips correctly (pipe-[A-F]-ctx-flip-detection), and that igt itself is also able to handle them correctly (pipe-[A-F]-ctx-flip-skip-current-frame). Since these tests use a debugfs interface to manually control the notifier context flip threshold, we also add one test to ensure that any flip thresholds we set are cleared after a single CRC capture (ctx-flip-threshold-reset-after-capture). In addition, we also add some simple tests to test Nvidia-specific CRC sources. Signed-off-by: Lyude Paul <lyude at redhat.com> --- lib/drmtest.c | 10 ++ lib/drmtest.h | 2 + tests/meson.build | 1 + tests/nouveau_crc.c | 396 ++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 409 insertions(+) create mode 100644 tests/nouveau_crc.c diff --git a/lib/drmtest.c b/lib/drmtest.c index 1fc39925..53c01754 100644 --- a/lib/drmtest.c +++ b/lib/drmtest.c @@ -112,6 +112,11 @@ bool is_i915_device(int fd) return __is_device(fd, "i915"); } +bool is_nouveau_device(int fd) +{ + return __is_device(fd, "nouveau"); +} + bool is_vc4_device(int fd) { return __is_device(fd, "vc4"); @@ -537,6 +542,11 @@ void igt_require_intel(int fd) igt_require(is_i915_device(fd) && has_known_intel_chipset(fd)); } +void igt_require_nouveau(int fd) +{ + igt_require(is_nouveau_device(fd)); +} + void igt_require_vc4(int fd) { igt_require(is_vc4_device(fd)); diff --git a/lib/drmtest.h b/lib/drmtest.h index 632c616b..4937e9d2 100644 --- a/lib/drmtest.h +++ b/lib/drmtest.h @@ -97,10 +97,12 @@ void gem_quiescent_gpu(int fd); void igt_require_amdgpu(int fd); void igt_require_intel(int fd); +void igt_require_nouveau(int fd); void igt_require_vc4(int fd); bool is_amdgpu_device(int fd); bool is_i915_device(int fd); +bool is_nouveau_device(int fd); bool is_vc4_device(int fd); /** diff --git a/tests/meson.build b/tests/meson.build index 7629afde..9ff74cc6 100644 --- a/tests/meson.build +++ b/tests/meson.build @@ -70,6 +70,7 @@ test_progs = [ 'kms_vblank', 'kms_vrr', 'meta_test', + 'nouveau_crc', 'panfrost_get_param', 'panfrost_gem_new', 'panfrost_prime', diff --git a/tests/nouveau_crc.c b/tests/nouveau_crc.c new file mode 100644 index 00000000..09e17a6f --- /dev/null +++ b/tests/nouveau_crc.c @@ -0,0 +1,396 @@ +/* + * Copyright ? 2019 Red Hat 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 (including the next + * paragraph) 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 AUTHORS OR COPYRIGHT HOLDERS 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_sysfs.h" + +IGT_TEST_DESCRIPTION( +"Tests certain aspects of CRC capture that are exclusive to nvidia hardware, " +"such as context flipping."); + +typedef struct { + int pipe; + int drm_fd; + int nv_crc_dir; + igt_display_t display; + igt_output_t *output; + igt_plane_t *primary; + drmModeModeInfo *mode; + igt_fb_t default_fb; +} data_t; + +struct color_fb { + double r, g, b; + igt_crc_t crc; + igt_fb_t fb; +}; + +#define HEX_COLOR(r_, g_, b_) \ + { .r = (r_ / 255.0), .g = (g_ / 255.0), .b = (b_ / 255.0) } + +static void set_crc_flip_threshold(data_t *data, unsigned int threshold) +{ + igt_debug("Setting CRC notifier flip threshold to %d\n", threshold); + igt_assert_lt(0, igt_sysfs_printf(data->nv_crc_dir, "flip_threshold", "%d", threshold)); +} + +static void create_colors(data_t *data, + struct color_fb *colors, + size_t len, + igt_pipe_crc_t *pipe_crc) +{ + char *crc_str; + + igt_pipe_crc_start(pipe_crc); + + for (int i = 0; i < len; i++) { + igt_create_color_fb(data->drm_fd, + data->mode->hdisplay, + data->mode->vdisplay, + DRM_FORMAT_XRGB8888, + LOCAL_DRM_FORMAT_MOD_NONE, + colors[i].r, colors[i].g, colors[i].b, + &colors[i].fb); + + igt_plane_set_fb(data->primary, &colors[i].fb); + igt_display_commit(&data->display); + igt_pipe_crc_get_current(data->drm_fd, pipe_crc, &colors[i].crc); + + crc_str = igt_crc_to_string(&colors[i].crc); + igt_debug("CRC for frame %d of pattern: %s\n", + i, crc_str); + free(crc_str); + } + + igt_pipe_crc_stop(pipe_crc); +} + +static void destroy_colors(data_t *data, struct color_fb *colors, size_t len) +{ + /* So we don't turn off the pipe if we remove it's current fb */ + igt_plane_set_fb(data->primary, &data->default_fb); + + for (int i = 0; i < len; i++) + igt_remove_fb(data->drm_fd, &colors[i].fb); +} + +/* Hard-coded to PIPE_A for now, we don't really need to test this on more then + * one head + */ +static void test_ctx_flip_detection(data_t *data) +{ + struct color_fb colors[] = { + HEX_COLOR(0xFF, 0x00, 0x18), + HEX_COLOR(0xFF, 0xA5, 0x2C), + HEX_COLOR(0xFF, 0xFF, 0x41), + HEX_COLOR(0x00, 0x80, 0x18), + HEX_COLOR(0x00, 0x00, 0xF9), + HEX_COLOR(0x86, 0x00, 0x7D), + }; + igt_output_t *output = data->output; + igt_plane_t *primary = data->primary; + igt_pipe_crc_t *pipe_crc; + const int n_colors = ARRAY_SIZE(colors); + const int n_crcs = 20; + igt_crc_t *crcs = NULL; + int start = -1, frame, start_color = -1, i; + bool found_skip = false; + + pipe_crc = igt_pipe_crc_new(data->drm_fd, data->pipe, "auto"); + + create_colors(data, colors, n_colors, pipe_crc); + + set_crc_flip_threshold(data, n_crcs / 2); + igt_pipe_crc_start(pipe_crc); + + for (i = 0; i < n_crcs; i++) { + const int color_idx = i % n_colors; + + igt_plane_set_fb(primary, &colors[color_idx].fb); + do_or_die(drmModePageFlip(data->drm_fd, + output->config.crtc->crtc_id, + colors[color_idx].fb.fb_id, + DRM_MODE_PAGE_FLIP_EVENT, + NULL)); + kmstest_wait_for_pageflip(data->drm_fd); + } + + igt_pipe_crc_get_crcs(pipe_crc, n_crcs, &crcs); + igt_pipe_crc_stop(pipe_crc); + + /* + * Find the first color in our pattern with a CRC that differs from the + * last CRC, so we can use it to find the start of the pattern + */ + for (i = 0; i < n_colors - 1; i++) { + if (igt_check_crc_equal(&colors[i].crc, &colors[n_colors - 1].crc)) + continue; + + igt_debug("Using frame %d of pattern for finding start\n", i); + start_color = i; + break; + } + igt_assert_lte(0, start_color); + + /* Now, figure out where the pattern starts */ + for (i = 0; i < n_crcs; i++) { + if (!igt_check_crc_equal(&colors[start_color].crc, &crcs[i])) + continue; + + start = i - start_color; + frame = crcs[i].frame; + igt_debug("Pattern started on frame %d\n", frame); + break; + } + igt_assert_lte(0, start); + + /* And finally, assert that according to the CRCs exactly all but one + * frame was displayed in order. The missing frame comes from + * (inevitably) losing a single CRC event when nouveau switches notifier + * contexts + */ + for (i = start; i < n_crcs; i++, frame++) { + igt_crc_t *crc = &crcs[i]; + char *crc_str; + int color_idx; + + crc_str = igt_crc_to_string(crc); + igt_debug("CRC %d: vbl=%d val=%s\n", i, crc->frame, crc_str); + free(crc_str); + + if (!found_skip && crc->frame != frame) { + igt_debug("^^^ Found expected skipped CRC %d ^^^\n", + crc->frame - 1); + found_skip = true; + frame++; + } + + /* We should never skip more then one frame */ + if (found_skip) { + igt_assert_eq(crc->frame, frame); + color_idx = (i - start + 1) % n_colors; + } else { + color_idx = (i - start) % n_colors; + } + + igt_assert_crc_equal(crc, &colors[color_idx].crc); + } + igt_assert(found_skip); + + free(crcs); + igt_pipe_crc_free(pipe_crc); + destroy_colors(data, colors, ARRAY_SIZE(colors)); +} + +/* Test whether or not IGT is able to handle frame skips when requesting the + * CRC for the current frame + */ +static void test_ctx_flip_skip_current_frame(data_t *data) +{ + struct color_fb colors[] = { + { .r = 1.0, .g = 0.0, .b = 0.0 }, + { .r = 0.0, .g = 1.0, .b = 0.0 }, + { .r = 0.0, .g = 0.0, .b = 1.0 }, + }; + igt_output_t *output = data->output; + igt_pipe_crc_t *pipe_crc; + igt_plane_t *primary = data->primary; + const int fd = data->drm_fd; + const int n_colors = ARRAY_SIZE(colors); + const int n_crcs = 30; + + pipe_crc = igt_pipe_crc_new(fd, data->pipe, "auto"); + create_colors(data, colors, n_colors, pipe_crc); + + set_crc_flip_threshold(data, 5); + igt_pipe_crc_start(pipe_crc); + + for (int i = 0; i < n_crcs; i++) { + igt_crc_t crc; + const int color_idx = i % n_colors; + + igt_plane_set_fb(primary, &colors[color_idx].fb); + do_or_die(drmModePageFlip(fd, + output->config.crtc->crtc_id, + colors[color_idx].fb.fb_id, + DRM_MODE_PAGE_FLIP_EVENT, + NULL)); + kmstest_wait_for_pageflip(fd); + + igt_pipe_crc_get_current(fd, pipe_crc, &crc); + igt_assert_crc_equal(&colors[color_idx].crc, &crc); + } + + igt_pipe_crc_stop(pipe_crc); + igt_pipe_crc_free(pipe_crc); + destroy_colors(data, colors, n_colors); +} + +static void test_ctx_flip_threshold_reset_after_capture(data_t *data) +{ + igt_pipe_crc_t *pipe_crc; + const int fd = data->drm_fd; + + pipe_crc = igt_pipe_crc_new(fd, data->pipe, "auto"); + + set_crc_flip_threshold(data, 5); + igt_pipe_crc_start(pipe_crc); + igt_pipe_crc_stop(pipe_crc); + + igt_assert_neq(igt_sysfs_get_u32(data->nv_crc_dir, "flip_threshold"), 5); + igt_pipe_crc_free(pipe_crc); +} + +static void test_source(data_t *data, const char *source) +{ + igt_pipe_crc_t *pipe_crc = igt_pipe_crc_new(data->drm_fd, data->pipe, source); + igt_crc_t *crcs; + + igt_pipe_crc_start(pipe_crc); + igt_pipe_crc_get_crcs(pipe_crc, 2, &crcs); + igt_pipe_crc_stop(pipe_crc); + + /* The CRC shouldn't change if the source content hasn't changed */ + igt_assert_crc_equal(&crcs[0], &crcs[1]); + + igt_pipe_crc_free(pipe_crc); + free(crcs); +} + +static void test_source_outp_inactive(data_t *data) +{ + struct color_fb colors[] = { + { .r = 1.0, .g = 0.0, .b = 0.0 }, + { .r = 0.0, .g = 1.0, .b = 0.0 }, + }; + igt_pipe_crc_t *pipe_crc; + const int fd = data->drm_fd; + const int n_colors = ARRAY_SIZE(colors); + + pipe_crc = igt_pipe_crc_new(fd, data->pipe, "outp-inactive"); + create_colors(data, colors, n_colors, pipe_crc); + + /* Changing the color should not change what's outside the active raster */ + igt_assert_crc_equal(&colors[0].crc, &colors[1].crc); + + igt_pipe_crc_free(pipe_crc); + destroy_colors(data, colors, n_colors); +} + +data_t data = {0, }; + +#define pipe_test(name) igt_subtest_f("pipe-%s-" name, kmstest_pipe_name(pipe)) +igt_main +{ + int pipe; + + igt_fixture { + data.drm_fd = drm_open_driver_master(DRIVER_ANY); + igt_require_nouveau(data.drm_fd); + + kmstest_set_vt_graphics_mode(); + + igt_require_pipe_crc(data.drm_fd); + igt_display_require(&data.display, data.drm_fd); + igt_display_reset(&data.display); + } + + for_each_pipe_static(pipe) { + igt_fixture { + int dir; + + data.pipe = pipe; + igt_display_require_output_on_pipe(&data.display, pipe); + data.output = igt_get_single_output_for_pipe(&data.display, pipe); + data.mode = igt_output_get_mode(data.output); + + /* None of these tests need to perform modesets, + * just page flips. So running display setup + * here is fine + */ + igt_output_set_pipe(data.output, pipe); + data.primary = igt_output_get_plane(data.output, 0); + igt_create_color_fb(data.drm_fd, + data.mode->hdisplay, + data.mode->vdisplay, + DRM_FORMAT_XRGB8888, + LOCAL_DRM_FORMAT_MOD_NONE, + 0.0, 0.0, 0.0, + &data.default_fb); + igt_plane_set_fb(data.primary, &data.default_fb); + igt_display_commit(&data.display); + + dir = igt_debugfs_pipe_dir(data.drm_fd, pipe, O_DIRECTORY); + igt_require_fd(dir); + data.nv_crc_dir = openat(dir, "nv_crc", O_DIRECTORY); + close(dir); + igt_require_fd(data.nv_crc_dir); + } + + /* We don't need to test this on every pipe, but the + * setup is the same */ + if (pipe == PIPE_A) { + igt_describe("Make sure that the CRC notifier context flip threshold " + "is reset to its default value after a single capture."); + igt_subtest("ctx-flip-threshold-reset-after-capture") + test_ctx_flip_threshold_reset_after_capture(&data); + } + + igt_describe("Make sure the association between each CRC and its " + "respective frame index is not broken when the driver " + "performs a notifier context flip."); + pipe_test("ctx-flip-detection") + test_ctx_flip_detection(&data); + + igt_describe("Make sure that igt_pipe_crc_get_current() works even " + "when the CRC for the current frame index is lost."); + pipe_test("ctx-flip-skip-current-frame") + test_ctx_flip_skip_current_frame(&data); + + igt_describe("Check that basic CRC readback using the outp-complete " + "source works."); + pipe_test("source-outp-complete") + test_source(&data, "outp-complete"); + + igt_describe("Check that basic CRC readback using the rg source " + "works."); + pipe_test("source-rg") + test_source(&data, "rg"); + + igt_describe("Make sure that the outp-inactive source is actually " + "capturing the inactive raster."); + pipe_test("source-outp-inactive") + test_source_outp_inactive(&data); + + igt_fixture { + igt_output_set_pipe(data.output, PIPE_NONE); + igt_display_commit(&data.display); + igt_remove_fb(data.drm_fd, &data.default_fb); + close(data.nv_crc_dir); + } + } + igt_fixture + igt_display_fini(&data.display); + +} -- 2.24.1
Petri Latvala
2020-Mar-18 08:12 UTC
[Nouveau] [igt-dev] [PATCH i-g-t 1/4] lib/igt_core: Add igt_require_fd()
On Tue, Mar 17, 2020 at 09:00:44PM -0400, Lyude wrote:> From: Lyude Paul <lyude at redhat.com> > > Like igt_assert_fd(), but using igt_require() instead > > Signed-off-by: Lyude Paul <lyude at redhat.com> > --- > lib/igt_core.h | 12 ++++++++++++ > 1 file changed, 12 insertions(+) > > diff --git a/lib/igt_core.h b/lib/igt_core.h > index fae5f59e..b66336cf 100644 > --- a/lib/igt_core.h > +++ b/lib/igt_core.h > @@ -1008,6 +1008,18 @@ void igt_describe_f(const char *fmt, ...); > else igt_debug("Test requirement passed: %s\n", #expr); \ > } while (0) > > +/** > + * igt_require_fd: > + * @fd: file descriptor > + * > + * Skips (sub-) test if the given file descriptor is invalid. > + * > + * Like igt_require(), but displays the values being compared on failure instead > + * of simply printing the stringified expression..igt_assert_fd has this documentation comment, but that's mistakenly copypasted from igt_assert_eq and pals. Let's not propagate the copypaste error further. -- Petri Latvala> + */ > +#define igt_require_fd(fd) \ > + igt_require_f(fd >= 0, "file descriptor " #fd " failed\n"); > + > /** > * igt_skip_on_f: > * @expr: condition to test > -- > 2.24.1 > > _______________________________________________ > igt-dev mailing list > igt-dev at lists.freedesktop.org > https://lists.freedesktop.org/mailman/listinfo/igt-dev
Petri Latvala
2020-Mar-18 08:19 UTC
[Nouveau] [igt-dev] [PATCH i-g-t 4/4] tests: Add nouveau-crc tests
On Tue, Mar 17, 2020 at 09:00:47PM -0400, Lyude wrote:> From: Lyude Paul <lyude at redhat.com> > > We're finally getting CRC support in nouveau, so let's start testing > this in igt as well! While the normal CRC capture tests are nice, > there's a number of Nvidia-specific hardware characteristics that we > need to test as well. > > The most important one is known as a "notifier context flip". Basically, > Nvidia GPUs store generated CRCs in an allocated memory region, referred > to as the notifier context, that the driver programs itself. Strangely, > this region can only hold a limited number of CRC entries, and once it > runs out of available entries the hardware simply sets an overrun bit > and stops writing any new CRC entries. > > Since igt-gpu-tools doesn't really have an expectation of only being > able to grab a limited number of CRCs, we work around this in nouveau by > allocating two separate CRC notifier regions each time we start > capturing CRCs, and then flip between them whenever we get close to > filling our currently programmed notifier context. While this keeps the > number of CRC entries we lose to an absolute minimum, we are guaranteed > to lose exactly one CRC entry between context flips. Thus, we add some > tests to ensure that nouveau handles these flips correctly > (pipe-[A-F]-ctx-flip-detection), and that igt itself is also able to > handle them correctly (pipe-[A-F]-ctx-flip-skip-current-frame). Since > these tests use a debugfs interface to manually control the notifier > context flip threshold, we also add one test to ensure that any flip > thresholds we set are cleared after a single CRC capture > (ctx-flip-threshold-reset-after-capture). > > In addition, we also add some simple tests to test Nvidia-specific CRC > sources. > > Signed-off-by: Lyude Paul <lyude at redhat.com> > --- > lib/drmtest.c | 10 ++ > lib/drmtest.h | 2 + > tests/meson.build | 1 + > tests/nouveau_crc.c | 396 ++++++++++++++++++++++++++++++++++++++++++++ > 4 files changed, 409 insertions(+) > create mode 100644 tests/nouveau_crc.c > > diff --git a/lib/drmtest.c b/lib/drmtest.c > index 1fc39925..53c01754 100644 > --- a/lib/drmtest.c > +++ b/lib/drmtest.c > @@ -112,6 +112,11 @@ bool is_i915_device(int fd) > return __is_device(fd, "i915"); > } > > +bool is_nouveau_device(int fd) > +{ > + return __is_device(fd, "nouveau"); > +} > + > bool is_vc4_device(int fd) > { > return __is_device(fd, "vc4"); > @@ -537,6 +542,11 @@ void igt_require_intel(int fd) > igt_require(is_i915_device(fd) && has_known_intel_chipset(fd)); > } > > +void igt_require_nouveau(int fd) > +{ > + igt_require(is_nouveau_device(fd)); > +} > + > void igt_require_vc4(int fd) > { > igt_require(is_vc4_device(fd)); > diff --git a/lib/drmtest.h b/lib/drmtest.h > index 632c616b..4937e9d2 100644 > --- a/lib/drmtest.h > +++ b/lib/drmtest.h > @@ -97,10 +97,12 @@ void gem_quiescent_gpu(int fd); > > void igt_require_amdgpu(int fd); > void igt_require_intel(int fd); > +void igt_require_nouveau(int fd); > void igt_require_vc4(int fd); > > bool is_amdgpu_device(int fd); > bool is_i915_device(int fd); > +bool is_nouveau_device(int fd); > bool is_vc4_device(int fd); > > /** > diff --git a/tests/meson.build b/tests/meson.build > index 7629afde..9ff74cc6 100644 > --- a/tests/meson.build > +++ b/tests/meson.build > @@ -70,6 +70,7 @@ test_progs = [ > 'kms_vblank', > 'kms_vrr', > 'meta_test', > + 'nouveau_crc', > 'panfrost_get_param', > 'panfrost_gem_new', > 'panfrost_prime', > diff --git a/tests/nouveau_crc.c b/tests/nouveau_crc.c > new file mode 100644 > index 00000000..09e17a6f > --- /dev/null > +++ b/tests/nouveau_crc.c > @@ -0,0 +1,396 @@ > +/* > + * Copyright ? 2019 Red Hat Inc.Verify that the authoring year is correct. -- Petri Latvala