tests: Add prime_crc for cache coherency

This program can be used to detect when the writes don't land in scanout, due
cache incoherency.

Run it like ./prime_crc --interactive-debug=crc

Signed-off-by: Tiago Vignatti <tiago.vignatti@intel.com>
This commit is contained in:
Tiago Vignatti 2015-08-04 13:38:09 -03:00
parent 84e851cbab
commit 3e130ac2b2
3 changed files with 203 additions and 0 deletions

1
tests/.gitignore vendored
View File

@ -160,6 +160,7 @@ pm_rc6_residency
pm_rpm
pm_rps
pm_sseu
prime_crc
prime_nv_api
prime_nv_pcopy
prime_nv_test

View File

@ -90,6 +90,7 @@ TESTS_progs_M = \
pm_rps \
pm_rc6_residency \
pm_sseu \
prime_crc \
prime_mmap \
prime_self_import \
template \

201
tests/prime_crc.c Normal file
View File

@ -0,0 +1,201 @@
/*
* Copyright © 2015 Intel Corporation
*
* 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.
*
* Authors:
* Tiago Vignatti <tiago.vignatti at intel.com>
*
*/
/* This program can detect when the writes don't land in scanout, due cache
* incoherency. */
#include "drmtest.h"
#include "igt_debugfs.h"
#include "igt_kms.h"
#define MAX_CONNECTORS 32
struct modeset_params {
uint32_t crtc_id;
uint32_t connector_id;
drmModeModeInfoPtr mode;
};
int drm_fd;
drmModeResPtr drm_res;
drmModeConnectorPtr drm_connectors[MAX_CONNECTORS];
drm_intel_bufmgr *bufmgr;
igt_pipe_crc_t *pipe_crc;
struct modeset_params ms;
static void find_modeset_params(void)
{
int i;
uint32_t connector_id = 0, crtc_id;
drmModeModeInfoPtr mode = NULL;
for (i = 0; i < drm_res->count_connectors; i++) {
drmModeConnectorPtr c = drm_connectors[i];
if (c->count_modes) {
connector_id = c->connector_id;
mode = &c->modes[0];
break;
}
}
igt_require(connector_id);
crtc_id = drm_res->crtcs[0];
igt_assert(crtc_id);
igt_assert(mode);
ms.connector_id = connector_id;
ms.crtc_id = crtc_id;
ms.mode = mode;
}
#define BO_SIZE (16*1024)
char pattern[] = {0xff, 0x00, 0x00, 0x00,
0x00, 0xff, 0x00, 0x00,
0x00, 0x00, 0xff, 0x00,
0x00, 0x00, 0x00, 0xff};
static void mess_with_coherency(char *ptr)
{
off_t i;
for (i = 0; i < BO_SIZE; i+=sizeof(pattern)) {
memcpy(ptr + i, pattern, sizeof(pattern));
}
// munmap(ptr, BO_SIZE);
// close(dma_buf_fd);
}
static char *dmabuf_mmap_framebuffer(struct igt_fb *fb)
{
int dma_buf_fd;
char *ptr = NULL;
dma_buf_fd = prime_handle_to_fd(drm_fd, fb->gem_handle);
igt_assert(errno == 0);
ptr = mmap(NULL, BO_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, dma_buf_fd, 0);
igt_assert(ptr != MAP_FAILED);
return ptr;
}
static void get_method_crc(uint64_t tiling, igt_crc_t *crc, bool mess)
{
struct igt_fb fb;
int rc;
char *ptr;
igt_create_fb(drm_fd, ms.mode->hdisplay, ms.mode->vdisplay,
DRM_FORMAT_XRGB8888, tiling, &fb);
if (mess)
ptr = dmabuf_mmap_framebuffer(&fb);
rc = drmModeSetCrtc(drm_fd, ms.crtc_id, fb.fb_id, 0, 0,
&ms.connector_id, 1, ms.mode);
igt_assert(rc == 0);
if (mess)
mess_with_coherency(ptr);
igt_pipe_crc_collect_crc(pipe_crc, crc);
kmstest_unset_all_crtcs(drm_fd, drm_res);
igt_remove_fb(drm_fd, &fb);
}
static void draw_method_subtest(uint64_t tiling)
{
igt_crc_t reference_crc, crc;
kmstest_unset_all_crtcs(drm_fd, drm_res);
find_modeset_params();
get_method_crc(tiling, &reference_crc, false);
get_method_crc(tiling, &crc, true);
// XXX: IIUC if we mess up with the scanout device, through a dma-buf mmap'ed
// pointer, then both the reference crc and the messed up one should be equal
// because the latter wasn't flushed. That's the theory, but it's not what's
// happening and the following is not passing.
igt_assert_crc_equal(&reference_crc, &crc);
}
static void setup_environment(void)
{
int i;
drm_fd = drm_open_any_master();
igt_require(drm_fd >= 0);
drm_res = drmModeGetResources(drm_fd);
igt_assert(drm_res->count_connectors <= MAX_CONNECTORS);
for (i = 0; i < drm_res->count_connectors; i++)
drm_connectors[i] = drmModeGetConnector(drm_fd,
drm_res->connectors[i]);
kmstest_set_vt_graphics_mode();
bufmgr = drm_intel_bufmgr_gem_init(drm_fd, 4096);
igt_assert(bufmgr);
drm_intel_bufmgr_gem_enable_reuse(bufmgr);
pipe_crc = igt_pipe_crc_new(0, INTEL_PIPE_CRC_SOURCE_AUTO);
}
static void teardown_environment(void)
{
int i;
igt_pipe_crc_free(pipe_crc);
drm_intel_bufmgr_destroy(bufmgr);
for (i = 0; i < drm_res->count_connectors; i++)
drmModeFreeConnector(drm_connectors[i]);
drmModeFreeResources(drm_res);
close(drm_fd);
}
igt_main
{
igt_fixture
setup_environment();
igt_subtest_f("draw-method-tiled")
draw_method_subtest(LOCAL_I915_FORMAT_MOD_X_TILED);
igt_fixture
teardown_environment();
}