Re: [PATCH 7/7] lib: add igt_draw

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



2015-03-31 10:07 GMT-03:00 Daniel Vetter <daniel@xxxxxxxx>:
> On Mon, Mar 30, 2015 at 04:45:49PM -0300, Paulo Zanoni wrote:
>> 2015-03-26 7:19 GMT-03:00 Daniel Vetter <daniel@xxxxxxxx>:
>> > On Wed, Mar 25, 2015 at 06:50:39PM -0300, Paulo Zanoni wrote:
>> >> From: Paulo Zanoni <paulo.r.zanoni@xxxxxxxxx>
>> >>
>> >> For all those IGT tests that need an easy way to draw rectangles on
>> >> buffers using different methods. Current planned users: FBC and PSR
>> >> CRC tests.
>> >>
>> >> There is also a lib/tests/igt_draw program to check if the library is
>> >> sane.
>> >
>> > We need to move that to tests/igt_draw. The testcase in lib/tests/* get
>> > run with make check, which must be possible as non-root on a non-intel
>> > (build) machine. If you need an gpu to run your test it must be in tests/*
>> > as a normal igt kernel test.
>> >
>> > Wrt the library itself I'm unsure about the explicit tiling/swizzling.
>> > Your current code only works on gen5-8 and maintaining a full-blown
>> > swizzle/tiling library is real work, and means some of the tests can't be
>> > converted to this.
>>
>> This would just be a problem for the tests that use it, and no test is
>> using it yet. This is just for the cases where you want to use the CPU
>> to write into tiled buffers.
>>
>> If we start using this in a test, then we need to properly test all
>> the affected platforms. I have some local FBC tests that use it -
>> which I was going to submit after getting feedback on this lib -, and
>> I don't think we'll end needing to run these tests on the older
>> platforms.
>>
>>
>> > We do have all the tiling modes encoded in the tiling
>> > tests though, so if you have a lot of time it might be useful to extract
>> > tiling helpers into the igt library which work on all generations.
>>
>> Can you please be more precise here? Where exactly should I look?
>
> gem_tiled_pread has the full-blown tiling/swizzle logic for all platforms.
> This is the one test we have which should work everywhere.
>
> My concern is that by implementing a library which doesn't support
> everywhere we havea  bit a split in the testbase which might surprise
> people. Imo a library function should work everywhere. If you look at the
> code in there compared to yours there's just 2 things missing:
> - Variable tile size/height.
> - Some of the more crazy swizzle modes.
>
> Also if we have a library which works everywhere we could extend it with
> the new fancy gen9+ tiling modes.

I'll take a look at gem_tiled_pread and try to implement what is
missing. I agree that supporting everything is the ideal, but I also
think that we can grow this support as needed instead of all at once.

>
>> > I think we should at least have a fallback mode which allows us to fill an
>> > entire buffer completely (i.e. not just the hxw area) and treat it as
>> > untiled.
>>
>> That is not the goal of the library. Still, if we ever need this
>> function, it would be easy to add.
>
> That was just meant as a fallback for buffers where you don't support the
> exact tiling/swizzling mode.
>>
>> >
>> > More comments below.
>>
>> More below too :)
>>
>> > -Daniel
>> >
>> >>
>> >> Signed-off-by: Paulo Zanoni <paulo.r.zanoni@xxxxxxxxx>
>> >> ---
>> >>  lib/Makefile.sources       |   2 +
>> >>  lib/igt_draw.c             | 467 +++++++++++++++++++++++++++++++++++++++++++++
>> >>  lib/igt_draw.h             |  54 ++++++
>> >>  lib/tests/.gitignore       |   1 +
>> >>  lib/tests/Makefile.sources |   1 +
>> >>  lib/tests/igt_draw.c       | 247 ++++++++++++++++++++++++
>> >>  6 files changed, 772 insertions(+)
>> >>  create mode 100644 lib/igt_draw.c
>> >>  create mode 100644 lib/igt_draw.h
>> >>  create mode 100644 lib/tests/igt_draw.c
>> >>
>> >> diff --git a/lib/Makefile.sources b/lib/Makefile.sources
>> >> index 3d93629..85dc321 100644
>> >> --- a/lib/Makefile.sources
>> >> +++ b/lib/Makefile.sources
>> >> @@ -52,6 +52,8 @@ libintel_tools_la_SOURCES =         \
>> >>       igt_fb.h                \
>> >>       igt_core.c              \
>> >>       igt_core.h              \
>> >> +     igt_draw.c              \
>> >> +     igt_draw.h              \
>> >>       $(NULL)
>> >>
>> >>  .PHONY: version.h.tmp
>> >> diff --git a/lib/igt_draw.c b/lib/igt_draw.c
>> >> new file mode 100644
>> >> index 0000000..4eb7507
>> >> --- /dev/null
>> >> +++ b/lib/igt_draw.c
>> >> @@ -0,0 +1,467 @@
>> >> +/*
>> >> + * 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.
>> >> + *
>> >> + */
>> >> +
>> >> +#include <sys/mman.h>
>> >> +
>> >> +#include "igt_draw.h"
>> >> +
>> >> +#include "drmtest.h"
>> >> +#include "intel_chipset.h"
>> >> +#include "igt_core.h"
>> >> +#include "igt_fb.h"
>> >> +#include "ioctl_wrappers.h"
>> >> +
>> >> +/**
>> >> + * SECTION:igt_draw
>> >> + * @short_description: drawing helpers for tests
>> >> + * @title: i-g-t draw
>> >> + * @include: igt_draw.h
>> >> + *
>> >> + * This library contains some functions for drawing rectangles on buffers using
>> >> + * the many different drawing methods we have. It also contains some wrappers
>> >> + * that make the process easier if you have the abstract objects in hand.
>> >> + *
>> >> + * All functions assume the buffers are in the XRGB 8:8:8 format.
>> >> + *
>> >> + */
>> >> +
>> >> +/* Some internal data structures to avoid having to pass tons of parameters
>> >> + * around everything. */
>> >> +struct cmd_data {
>> >> +     drm_intel_bufmgr *bufmgr;
>> >> +     drm_intel_context *context;
>> >> +};
>> >> +
>> >> +struct buf_data {
>> >> +     uint32_t handle;
>> >> +     uint32_t size;
>> >> +     uint32_t stride;
>> >> +};
>> >> +
>> >> +struct rect {
>> >> +     int x;
>> >> +     int y;
>> >> +     int w;
>> >> +     int h;
>> >> +};
>> >> +
>> >> +const char *igt_draw_get_method_name(enum igt_draw_method method)
>> >> +{
>> >> +     switch (method) {
>> >> +     case IGT_DRAW_MMAP_CPU:
>> >> +             return "mmap-cpu";
>> >> +     case IGT_DRAW_MMAP_GTT:
>> >> +             return "mmap-gtt";
>> >> +     case IGT_DRAW_PWRITE:
>> >> +             return "pwrite";
>> >> +     case IGT_DRAW_BLT:
>> >> +             return "blt";
>> >> +     case IGT_DRAW_RENDER:
>> >> +             return "render";
>> >> +     default:
>> >> +             igt_assert(false);
>> >> +     }
>> >> +}
>> >> +
>> >> +static int swizzle_addr(int addr, int swizzle)
>> >> +{
>> >> +     int bit6;
>> >> +
>> >> +     if (swizzle == I915_BIT_6_SWIZZLE_9_10) {
>> >> +             bit6 = ((addr >> 6) & 1) ^ ((addr >> 9) & 1) ^
>> >> +                    ((addr >> 10) & 1);
>> >> +             addr &= ~(1 << 6);
>> >> +             addr |= (bit6 << 6);
>> >> +     }
>> >> +
>> >> +     return addr;
>> >> +}
>> >> +
>> >> +/* It's all in "pixel coordinates", so make sure you multiply/divide by the bpp
>> >> + * if you need to. */
>> >> +static int linear_x_y_to_tiled_pos(int x, int y, uint32_t stride, int swizzle)
>> >> +{
>> >> +     int x_tile_size, y_tile_size;
>> >> +     int x_tile_n, y_tile_n, x_tile_off, y_tile_off;
>> >> +     int line_size, tile_size;
>> >> +     int tile_n, tile_off;
>> >> +     int tiled_pos, tiles_per_line;
>> >> +     int bpp;
>> >> +
>> >> +     line_size = stride;
>> >> +     x_tile_size = 512;
>> >> +     y_tile_size = 8;
>> >> +     tile_size = x_tile_size * y_tile_size;
>> >> +     tiles_per_line = line_size / x_tile_size;
>> >> +     bpp = sizeof(uint32_t);
>> >> +
>> >> +     y_tile_n = y / y_tile_size;
>> >> +     y_tile_off = y % y_tile_size;
>> >> +
>> >> +     x_tile_n = (x * bpp) / x_tile_size;
>> >> +     x_tile_off = (x * bpp) % x_tile_size;
>> >> +
>> >> +     tile_n = y_tile_n * tiles_per_line + x_tile_n;
>> >> +     tile_off = y_tile_off * x_tile_size + x_tile_off;
>> >> +     tiled_pos = tile_n * tile_size + tile_off;
>> >> +
>> >> +     tiled_pos = swizzle_addr(tiled_pos, swizzle);
>> >> +
>> >> +     return tiled_pos / bpp;
>> >> +}
>> >> +
>> >> +/* It's all in "pixel coordinates", so make sure you multiply/divide by the bpp
>> >> + * if you need to. */
>> >> +static void tiled_pos_to_x_y_linear(int tiled_pos, uint32_t stride,
>> >> +                                 int swizzle, int *x, int *y)
>> >> +{
>> >> +     int tile_n, tile_off, tiles_per_line, line_size;
>> >> +     int x_tile_off, y_tile_off;
>> >> +     int x_tile_n, y_tile_n;
>> >> +     int x_tile_size, y_tile_size, tile_size;
>> >> +     int bpp;
>> >> +
>> >> +     tiled_pos = swizzle_addr(tiled_pos, swizzle);
>> >> +
>> >> +     line_size = stride;
>> >> +     x_tile_size = 512;
>> >> +     y_tile_size = 8;
>> >> +     tile_size = x_tile_size * y_tile_size;
>> >> +     tiles_per_line = line_size / x_tile_size;
>> >> +     bpp = sizeof(uint32_t);
>> >> +
>> >> +     tile_n = tiled_pos / tile_size;
>> >> +     tile_off = tiled_pos % tile_size;
>> >> +
>> >> +     y_tile_off = tile_off / x_tile_size;
>> >> +     x_tile_off = tile_off % x_tile_size;
>> >> +
>> >> +     x_tile_n = tile_n % tiles_per_line;
>> >> +     y_tile_n = tile_n / tiles_per_line;
>> >> +
>> >> +     *x = (x_tile_n * x_tile_size + x_tile_off) / bpp;
>> >> +     *y = y_tile_n * y_tile_size + y_tile_off;
>> >> +}
>> >> +
>> >> +static void draw_rect_mmap_cpu(int fd, struct buf_data *buf, struct rect *rect,
>> >> +                            uint32_t color)
>> >> +{
>> >> +     uint32_t *ptr;
>> >> +     int x, y, pos;
>> >> +     uint32_t tiling, swizzle;
>> >> +
>> >> +     gem_set_domain(fd, buf->handle, I915_GEM_DOMAIN_CPU,
>> >> +                    I915_GEM_DOMAIN_CPU);
>> >> +     gem_get_tiling(fd, buf->handle, &tiling, &swizzle);
>> >> +
>> >> +     ptr = gem_mmap__cpu(fd, buf->handle, 0, buf->size, 0);
>> >> +     igt_assert(ptr);
>> >> +
>> >> +     for (y = rect->y; y < rect->y + rect->h; y++) {
>> >> +             for (x = rect->x; x < rect->x + rect->w; x++) {
>> >> +                     if (tiling)
>> >> +                             pos = linear_x_y_to_tiled_pos(x, y, buf->stride,
>> >> +                                                           swizzle);
>> >> +                     else
>> >> +                             pos = (y * buf->stride / sizeof(uint32_t)) + x;
>> >> +                     ptr[pos] = color;
>> >> +             }
>> >> +     }
>> >> +
>> >> +     gem_sw_finish(fd, buf->handle);
>> >> +
>> >> +     igt_assert(munmap(ptr, buf->size) == 0);
>> >> +}
>> >> +
>> >> +static void draw_rect_mmap_gtt(int fd, struct buf_data *buf, struct rect *rect,
>> >> +                            uint32_t color)
>> >> +{
>> >> +     uint32_t *ptr;
>> >> +     int x, y;
>> >> +
>> >> +     ptr = gem_mmap__gtt(fd, buf->handle, buf->size,
>> >> +                         PROT_READ | PROT_WRITE);
>> >> +     igt_assert(ptr);
>> >> +
>> >> +     for (y = rect->y; y < rect->y + rect->h; y++) {
>> >> +             int line_begin = y * buf->stride / sizeof(uint32_t);
>> >> +             for (x = rect->x; x < rect->x + rect->w; x++)
>> >> +                     ptr[line_begin + x] = color;
>> >> +     }
>> >> +
>> >> +     igt_assert(munmap(ptr, buf->size) == 0);
>> >> +}
>> >> +
>> >> +static void draw_rect_pwrite_untiled(int fd, struct buf_data *buf,
>> >> +                                  struct rect *rect, uint32_t color)
>> >> +{
>> >> +     uint32_t tmp[rect->w];
>> >> +     int i, y, offset, bpp;
>> >> +
>> >> +     bpp = sizeof(uint32_t);
>> >> +
>> >> +     for (i = 0; i < rect->w; i++)
>> >> +             tmp[i] = color;
>> >> +
>> >> +     for (y = rect->y; y < rect->y + rect->h; y++) {
>> >> +             offset = (y * buf->stride) + (rect->x * bpp);
>> >> +             gem_write(fd, buf->handle, offset, tmp, rect->w * bpp);
>> >> +     }
>> >> +}
>> >> +
>> >> +static void draw_rect_pwrite_tiled(int fd, struct buf_data *buf,
>> >> +                                struct rect *rect, uint32_t color,
>> >> +                                uint32_t swizzle)
>> >> +{
>> >> +     int i;
>> >> +     int tiled_pos, bpp, x, y;
>> >> +     uint32_t tmp[1024];
>> >> +     int tmp_used = 0, tmp_size = ARRAY_SIZE(tmp);
>> >> +     bool flush_tmp = false;
>> >> +     int tmp_start_pos = 0;
>> >> +
>> >> +     bpp = sizeof(uint32_t);
>> >> +
>> >> +     /* Instead of doing one pwrite per pixel, we try to group the maximum
>> >> +      * amount of consecutive pixels we can in a single pwrite: that's why we
>> >> +      * use the "tmp" variables. */
>> >> +     for (i = 0; i < tmp_size; i++)
>> >> +             tmp[i] = color;
>> >> +
>> >> +     for (tiled_pos = 0; tiled_pos < buf->size; tiled_pos += bpp) {
>> >> +             tiled_pos_to_x_y_linear(tiled_pos, buf->stride, swizzle, &x, &y);
>> >> +
>> >> +             if (x >= rect->x && x < rect->x + rect->w &&
>> >> +                 y >= rect->y && y < rect->y + rect->h) {
>> >> +                     if (tmp_used == 0)
>> >> +                             tmp_start_pos = tiled_pos;
>> >> +                     tmp_used++;
>> >> +             } else {
>> >> +                     flush_tmp = true;
>> >> +             }
>> >> +
>> >> +             if (tmp_used == tmp_size || (flush_tmp && tmp_used > 0)) {
>> >> +                     gem_write(fd, buf->handle, tmp_start_pos, tmp,
>> >> +                               tmp_used * bpp);
>> >> +                     flush_tmp = false;
>> >> +                     tmp_used = 0;
>> >> +             }
>> >> +     }
>> >> +}
>> >> +
>> >> +static void draw_rect_pwrite(int fd, struct buf_data *buf,
>> >> +                          struct rect *rect, uint32_t color)
>> >> +{
>> >> +     uint32_t tiling, swizzle;
>> >> +
>> >> +     gem_get_tiling(fd, buf->handle, &tiling, &swizzle);
>> >> +
>> >> +     if (tiling)
>> >> +             draw_rect_pwrite_tiled(fd, buf, rect, color, swizzle);
>> >> +     else
>> >> +             draw_rect_pwrite_untiled(fd, buf, rect, color);
>> >> +}
>> >> +
>> >> +static void draw_rect_blt(int fd, struct cmd_data *cmd_data,
>> >> +                       struct buf_data *buf, struct rect *rect,
>> >> +                       uint32_t color)
>> >> +{
>> >> +     drm_intel_bo *dst;
>> >> +     struct intel_batchbuffer *batch;
>> >> +     int blt_cmd_len, blt_cmd_tiling;
>> >> +     uint32_t devid = intel_get_drm_devid(fd);
>> >> +     int gen = intel_gen(devid);
>> >> +     uint32_t tiling, swizzle;
>> >> +     int pitch;
>> >> +
>> >> +     gem_get_tiling(fd, buf->handle, &tiling, &swizzle);
>> >> +
>> >> +     dst = gem_handle_to_libdrm_bo(cmd_data->bufmgr, fd, "", buf->handle);
>> >> +     igt_assert(dst);
>> >> +
>> >> +     batch = intel_batchbuffer_alloc(cmd_data->bufmgr, devid);
>> >> +     igt_assert(batch);
>> >> +
>> >> +     blt_cmd_len = (gen >= 8) ?  0x5 : 0x4;
>> >> +     blt_cmd_tiling = (tiling) ? XY_COLOR_BLT_TILED : 0;
>> >> +     pitch = (tiling) ? buf->stride / 4 : buf->stride;
>> >> +
>> >> +     BEGIN_BATCH(6, 1);
>> >
>> > This breaks on gen8 because there the relocations are 2 dwords. You need
>> > to use the magic macros for that.
>> >
>>
>> Please read the code again. I actually developed and tested everything
>> on gen8 :)
>
> Oh right I missed that OUT_RELOC uses qwords on gen8+ automatically. I
> still think using the same logic as with all other blitter tests would be
> better since blitter cmd copypaste fail has been the bane of igt.

Well, this is one of the main goals of the library: use it instead of
adding yet another implementation of the drawing commands...

>
>>
>> >> +     OUT_BATCH(XY_COLOR_BLT_CMD_NOLEN | XY_COLOR_BLT_WRITE_ALPHA |
>> >> +               XY_COLOR_BLT_WRITE_RGB | blt_cmd_tiling | blt_cmd_len);
>> >> +     OUT_BATCH((3 << 24) | (0xF0 << 16) | pitch);
>> >> +     OUT_BATCH((rect->y << 16) | rect->x);
>> >> +     OUT_BATCH(((rect->y + rect->h) << 16) | (rect->x + rect->w));
>> >> +     if (tiling)
>
> btw this isn't needed OUT_RELOC_FENCED works correctly for untiled
> buffers. It's actually required on gen2/3 for untiled buffers if you reuse
> buffers (which might be the case for a library function) since OUT_RELOC
> does _not_ remove a stale fence if there is one.
>
>> >> +             OUT_RELOC_FENCED(dst, 0, I915_GEM_DOMAIN_RENDER, 0);
>> >> +     else
>> >> +             OUT_RELOC(dst, 0, I915_GEM_DOMAIN_RENDER, 0);
>> >> +     OUT_BATCH(color);
>> >> +     ADVANCE_BATCH();
>> >> +
>> >> +     intel_batchbuffer_flush(batch);
>> >> +     gem_sync(fd, buf->handle);
>> >> +     intel_batchbuffer_free(batch);
>> >> +}
>> >> +
>> >> +static void draw_rect_render(int fd, struct cmd_data *cmd_data,
>> >> +                          struct buf_data *buf, struct rect *rect,
>> >> +                          uint32_t color)
>> >> +{
>> >> +     drm_intel_bo *src, *dst;
>> >> +     uint32_t devid = intel_get_drm_devid(fd);
>> >> +     igt_render_copyfunc_t rendercopy = igt_get_render_copyfunc(devid);
>> >> +     struct igt_buf src_buf, dst_buf;
>> >> +     struct intel_batchbuffer *batch;
>> >> +     uint32_t tiling, swizzle;
>> >> +     struct buf_data tmp;
>> >> +
>> >> +     igt_skip_on(!rendercopy);
>> >> +
>> >> +     gem_get_tiling(fd, buf->handle, &tiling, &swizzle);
>> >> +
>> >> +     /* We create a temporary buffer and copy from it using rendercopy. */
>> >> +     tmp.size = rect->w * rect->h * sizeof(uint32_t);
>> >> +     tmp.handle = gem_create(fd, tmp.size);
>> >> +     tmp.stride = rect->w * sizeof(uint32_t);
>> >> +     draw_rect_mmap_cpu(fd, &tmp, &(struct rect){0, 0, rect->w, rect->h},
>> >> +                        color);
>> >> +
>> >> +     src = gem_handle_to_libdrm_bo(cmd_data->bufmgr, fd, "", tmp.handle);
>> >> +     igt_assert(src);
>> >> +     dst = gem_handle_to_libdrm_bo(cmd_data->bufmgr, fd, "", buf->handle);
>> >> +     igt_assert(dst);
>> >> +
>> >> +     src_buf.bo = src;
>> >> +     src_buf.stride = tmp.stride;
>> >> +     src_buf.tiling = I915_TILING_NONE;
>> >> +     src_buf.size = tmp.size;
>> >> +     dst_buf.bo = dst;
>> >> +     dst_buf.stride = buf->stride;
>> >> +     dst_buf.tiling = tiling;
>> >> +     dst_buf.size = buf->size;
>> >> +
>> >> +     batch = intel_batchbuffer_alloc(cmd_data->bufmgr, devid);
>> >> +     igt_assert(batch);
>> >> +
>> >> +     rendercopy(batch, cmd_data->context, &src_buf, 0, 0, rect->w, rect->h,
>> >> +                &dst_buf, rect->x, rect->y);
>> >> +
>> >> +     gem_sync(fd, buf->handle);
>> >> +     intel_batchbuffer_free(batch);
>> >> +     gem_close(fd, tmp.handle);
>> >> +}
>> >> +
>> >> +/**
>> >> + * igt_draw_rect:
>> >> + * @fd: the DRM file descriptor
>> >> + * @bufmgr: the libdrm bufmgr, only required for IGT_DRAW_BLT and
>> >> + *          IGT_DRAW_RENDER
>> >> + * @context: the context, can be NULL if you don't want to think about it
>> >> + * @buf_handle: the handle of the buffer where you're going to draw to
>> >> + * @buf_size: the size of the buffer
>> >> + * @buf_stride: the stride of the buffer
>> >> + * @method: method you're going to use to write to the buffer
>> >> + * @rect_x: horizontal position on the buffer where your rectangle starts
>> >> + * @rect_y: vertical position on the buffer where your rectangle starts
>> >> + * @rect_w: width of the rectangle
>> >> + * @rect_h: height of the rectangle
>> >> + * @color: color of the rectangle
>> >> + *
>> >> + * This function draws a colored rectangle on the destination buffer, allowing
>> >> + * you to specify the method used to draw the rectangle. We assume 32 bit pixels
>> >> + * with 8 bits per color.
>> >> + */
>> >> +void igt_draw_rect(int fd, drm_intel_bufmgr *bufmgr, drm_intel_context *context,
>> >> +                uint32_t buf_handle, uint32_t buf_size, uint32_t buf_stride,
>> >> +                enum igt_draw_method method, int rect_x, int rect_y,
>> >> +                int rect_w, int rect_h, uint32_t color)
>> >> +{
>> >> +     struct cmd_data cmd_data = {
>> >> +             .bufmgr = bufmgr,
>> >> +             .context = context,
>> >> +     };
>> >> +     struct buf_data buf = {
>> >> +             .handle = buf_handle,
>> >> +             .size = buf_size,
>> >> +             .stride = buf_stride,
>> >> +     };
>> >> +     struct rect rect = {
>> >> +             .x = rect_x,
>> >> +             .y = rect_y,
>> >> +             .w = rect_w,
>> >> +             .h = rect_h,
>> >> +     };
>> >> +
>> >> +     switch (method) {
>> >> +     case IGT_DRAW_MMAP_CPU:
>> >> +             draw_rect_mmap_cpu(fd, &buf, &rect, color);
>> >> +             break;
>> >> +     case IGT_DRAW_MMAP_GTT:
>> >> +             draw_rect_mmap_gtt(fd, &buf, &rect, color);
>> >> +             break;
>> >> +     case IGT_DRAW_PWRITE:
>> >> +             draw_rect_pwrite(fd, &buf, &rect, color);
>> >> +             break;
>> >> +     case IGT_DRAW_BLT:
>> >> +             draw_rect_blt(fd, &cmd_data, &buf, &rect, color);
>> >> +             break;
>> >> +     case IGT_DRAW_RENDER:
>> >> +             draw_rect_render(fd, &cmd_data, &buf, &rect, color);
>> >> +             break;
>> >> +     default:
>> >> +             igt_assert(false);
>> >> +             break;
>> >> +     }
>> >> +}
>> >> +
>> >> +/**
>> >> + * igt_draw_rect_fb:
>> >> + *
>> >> + * This is exactly the same as igt_draw_rect, but you can pass an igt_fb instead
>> >> + * of manually providing its details. See igt_draw_rect.
>> >> + */
>> >> +void igt_draw_rect_fb(int fd, drm_intel_bufmgr *bufmgr,
>> >> +                   drm_intel_context *context, struct igt_fb *fb,
>> >> +                   enum igt_draw_method method, int rect_x, int rect_y,
>> >> +                   int rect_w, int rect_h, uint32_t color)
>> >> +{
>> >> +     igt_draw_rect(fd, bufmgr, context, fb->gem_handle, fb->size, fb->stride,
>> >> +                   method, rect_x, rect_y, rect_w, rect_h, color);
>> >> +}
>> >> +
>> >> +/**
>> >> + * igt_draw_fill_fb:
>> >> + * @fd: the DRM file descriptor
>> >> + * @fb: the FB that is going to be filled
>> >> + * @color: the color you're going to paint it
>> >> + *
>> >> + * This function just paints an igt_fb using the provided color. It assumes 32
>> >> + * bit pixels with 8 bit colors.
>> >> + */
>> >> +void igt_draw_fill_fb(int fd, struct igt_fb *fb, uint32_t color)
>> >> +{
>> >> +     igt_draw_rect_fb(fd, NULL, NULL, fb, IGT_DRAW_MMAP_GTT,
>> >> +                      0, 0, fb->width, fb->height, color);
>> >> +}
>> >> diff --git a/lib/igt_draw.h b/lib/igt_draw.h
>> >> new file mode 100644
>> >> index 0000000..399e17c
>> >> --- /dev/null
>> >> +++ b/lib/igt_draw.h
>> >> @@ -0,0 +1,54 @@
>> >> +/*
>> >> + * 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.
>> >> + *
>> >> + */
>> >> +
>> >> +#ifndef __IGT_DRAW_H__
>> >> +#define __IGT_DRAW_H__
>> >> +
>> >> +#include <intel_bufmgr.h>
>> >> +#include "igt_fb.h"
>> >> +
>> >
>> > gtkdoc for this enum would be nice too I think.
>>
>> Ok.
>>
>>
>> >
>> >> +enum igt_draw_method {
>> >> +     IGT_DRAW_MMAP_CPU,
>> >> +     IGT_DRAW_MMAP_GTT,
>> >> +     IGT_DRAW_PWRITE,
>> >> +     IGT_DRAW_BLT,
>> >> +     IGT_DRAW_RENDER,
>> >> +     IGT_DRAW_METHOD_COUNT,
>> >> +};
>> >> +
>> >> +const char *igt_draw_get_method_name(enum igt_draw_method method);
>> >> +
>> >> +void igt_draw_rect(int fd, drm_intel_bufmgr *bufmgr, drm_intel_context *context,
>> >> +                uint32_t buf_handle, uint32_t buf_size, uint32_t buf_stride,
>> >> +                enum igt_draw_method method, int rect_x, int rect_y,
>> >> +                int rect_w, int rect_h, uint32_t color);
>> >> +
>> >> +void igt_draw_rect_fb(int fd, drm_intel_bufmgr *bufmgr,
>> >> +                   drm_intel_context *context, struct igt_fb *fb,
>> >> +                   enum igt_draw_method method, int rect_x, int rect_y,
>> >> +                   int rect_w, int rect_h, uint32_t color);
>> >> +
>> >> +void igt_draw_fill_fb(int fd, struct igt_fb *fb, uint32_t color);
>> >> +
>> >> +#endif /* __IGT_DRAW_H__ */
>> >> diff --git a/lib/tests/.gitignore b/lib/tests/.gitignore
>> >> index a745a23..88f668a 100644
>> >> --- a/lib/tests/.gitignore
>> >> +++ b/lib/tests/.gitignore
>> >> @@ -1,4 +1,5 @@
>> >>  # Please keep sorted alphabetically
>> >> +igt_draw
>> >>  igt_fork_helper
>> >>  igt_invalid_subtest_name
>> >>  igt_list_only
>> >> diff --git a/lib/tests/Makefile.sources b/lib/tests/Makefile.sources
>> >> index ecd73ae..ff66d9d 100644
>> >> --- a/lib/tests/Makefile.sources
>> >> +++ b/lib/tests/Makefile.sources
>> >> @@ -1,4 +1,5 @@
>> >>  check_PROGRAMS = \
>> >> +     igt_draw \
>> >>       igt_no_exit \
>> >>       igt_no_exit_list_only \
>> >>       igt_fork_helper \
>> >> diff --git a/lib/tests/igt_draw.c b/lib/tests/igt_draw.c
>> >> new file mode 100644
>> >> index 0000000..1630cc2
>> >> --- /dev/null
>> >> +++ b/lib/tests/igt_draw.c
>> >> @@ -0,0 +1,247 @@
>> >> +/*
>> >> + * 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.
>> >> + *
>> >> + */
>> >> +
>> >> +/* This program tests whether the igt_draw library actually works. */
>> >> +
>> >> +#include "drmtest.h"
>> >> +#include "igt_aux.h"
>> >> +#include "igt_draw.h"
>> >> +#include "igt_debugfs.h"
>> >> +#include "igt_fb.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;
>> >> +
>> >> +bool has_method_base_crc = false;
>> >> +igt_crc_t method_base_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;
>> >> +
>> >> +}
>> >> +
>> >> +static void get_method_crc(enum igt_draw_method method, uint64_t tiling,
>> >> +                        igt_crc_t *crc)
>> >> +{
>> >> +     struct igt_fb fb;
>> >> +     int rc;
>> >> +
>> >> +     igt_create_fb(drm_fd, ms.mode->hdisplay, ms.mode->vdisplay,
>> >> +                   DRM_FORMAT_XRGB8888, tiling, &fb);
>> >> +     igt_draw_rect_fb(drm_fd, bufmgr, NULL, &fb, method,
>> >> +                      0, 0, fb.width, fb.height, 0xFF);
>> >> +
>> >> +     igt_draw_rect_fb(drm_fd, bufmgr, NULL, &fb, method,
>> >> +                      fb.width / 4, fb.height / 4,
>> >> +                      fb.width / 2, fb.height / 2, 0xFF00);
>> >> +     igt_draw_rect_fb(drm_fd, bufmgr, NULL, &fb, method,
>> >> +                      fb.width / 8, fb.height / 8,
>> >> +                      fb.width / 4, fb.height / 4, 0xFF0000);
>> >> +     igt_draw_rect_fb(drm_fd, bufmgr, NULL, &fb, method,
>> >> +                      fb.width / 2, fb.height / 2,
>> >> +                      fb.width / 3, fb.height / 3, 0xFF00FF);
>> >> +
>> >> +     rc = drmModeSetCrtc(drm_fd, ms.crtc_id, fb.fb_id, 0, 0,
>> >> +                         &ms.connector_id, 1, ms.mode);
>> >> +     igt_assert(rc == 0);
>> >> +
>> >> +     igt_debug_wait_for_keypress("crc");
>> >> +     igt_pipe_crc_collect_crc(pipe_crc, crc);
>> >
>> > Should we just move this igt_debug_wait_for_keypress into
>> > igt_pipe_crc_collect_crc?
>>
>> I think I prefer to leave these on the programs because you can then
>> opt-out of some cases such as when you're getting the reference CRCs.
>> It doesn't hurt to just add these to the problems, but it can be
>> annoying if the wait is on the lib and you don't want it.
>
> wait_for_keypress takes a key argument, so we could just use a "crc" key.
> Test could then use a "check" or similar (we don't have a standard yet) -
> generally you don't really want to run with --interactive-debug=all anyway
> since that tends to stop awfully often already ;-) Also you can list
> multiple keywords.
>
> So annoying waits from libraries are already taken care of, no harm in
> adding them.
>

Ok, I agree with you here now.


>> >> +
>> >> +     kmstest_unset_all_crtcs(drm_fd, drm_res);
>> >> +     igt_remove_fb(drm_fd, &fb);
>> >> +}
>> >> +
>> >> +static void draw_method_subtest(enum igt_draw_method method, uint64_t tiling)
>> >> +{
>> >> +     igt_crc_t crc;
>> >> +
>> >> +     kmstest_unset_all_crtcs(drm_fd, drm_res);
>> >> +
>> >> +     find_modeset_params();
>> >> +
>> >> +     /* Use IGT_DRAW_MMAP_GTT on an untiled buffer as the parameter for
>> >> +      * comparison. Cache the value so we don't recompute it for every single
>> >> +      * subtest. */
>> >> +     if (!has_method_base_crc) {
>> >> +             get_method_crc(IGT_DRAW_MMAP_GTT, LOCAL_DRM_FORMAT_MOD_NONE,
>> >> +                            &method_base_crc);
>> >> +             has_method_base_crc = true;
>> >> +     }
>> >> +
>> >> +     get_method_crc(method, tiling, &crc);
>> >> +     igt_assert_crc_equal(&crc, &method_base_crc);
>> >> +}
>> >> +
>> >> +static void get_fill_crc(uint64_t tiling, igt_crc_t *crc)
>> >> +{
>> >> +     struct igt_fb fb;
>> >> +     int rc;
>> >> +
>> >> +     igt_create_fb(drm_fd, ms.mode->hdisplay, ms.mode->vdisplay,
>> >> +                   DRM_FORMAT_XRGB8888, tiling, &fb);
>> >> +
>> >> +     igt_draw_fill_fb(drm_fd, &fb, 0xFF);
>> >> +
>> >> +     rc = drmModeSetCrtc(drm_fd, ms.crtc_id, fb.fb_id, 0, 0,
>> >> +                         &ms.connector_id, 1, ms.mode);
>> >> +     igt_assert(rc == 0);
>> >> +
>> >> +     igt_debug_wait_for_keypress("crc");
>> >> +     igt_pipe_crc_collect_crc(pipe_crc, crc);
>> >> +
>> >> +     kmstest_unset_all_crtcs(drm_fd, drm_res);
>> >> +     igt_remove_fb(drm_fd, &fb);
>> >> +}
>> >> +
>> >> +static void fill_fb_subtest(void)
>> >> +{
>> >> +     int rc;
>> >> +     struct igt_fb fb;
>> >> +     igt_crc_t base_crc, crc;
>> >> +
>> >> +     kmstest_unset_all_crtcs(drm_fd, drm_res);
>> >> +
>> >> +     find_modeset_params();
>> >> +
>> >> +     igt_create_fb(drm_fd, ms.mode->hdisplay, ms.mode->vdisplay,
>> >> +                   DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb);
>> >> +
>> >> +     igt_draw_rect_fb(drm_fd, bufmgr, NULL, &fb, IGT_DRAW_MMAP_GTT,
>> >> +                      0, 0, fb.width, fb.height, 0xFF);
>> >> +
>> >> +     rc = drmModeSetCrtc(drm_fd, ms.crtc_id, fb.fb_id, 0, 0,
>> >> +                         &ms.connector_id, 1, ms.mode);
>> >> +     igt_assert(rc == 0);
>> >> +
>> >> +     igt_debug_wait_for_keypress("crc");
>> >> +     igt_pipe_crc_collect_crc(pipe_crc, &base_crc);
>> >> +
>> >> +     get_fill_crc(LOCAL_DRM_FORMAT_MOD_NONE, &crc);
>> >> +     igt_assert_crc_equal(&crc, &base_crc);
>> >> +
>> >> +     get_fill_crc(LOCAL_I915_FORMAT_MOD_X_TILED, &crc);
>> >> +     igt_assert_crc_equal(&crc, &base_crc);
>> >> +
>> >> +     kmstest_unset_all_crtcs(drm_fd, drm_res);
>> >> +     igt_remove_fb(drm_fd, &fb);
>> >> +}
>> >> +
>> >> +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
>> >> +{
>> >> +     enum igt_draw_method method;
>> >> +
>> >> +     igt_fixture
>> >> +             setup_environment();
>> >> +
>> >> +     for (method = 0; method < IGT_DRAW_METHOD_COUNT; method++) {
>> >> +             igt_subtest_f("draw-method-%s-untiled",
>> >> +                           igt_draw_get_method_name(method))
>> >> +                     draw_method_subtest(method, LOCAL_DRM_FORMAT_MOD_NONE);
>> >> +             igt_subtest_f("draw-method-%s-tiled",
>> >> +                           igt_draw_get_method_name(method))
>> >> +                     draw_method_subtest(method,
>> >> +                                         LOCAL_I915_FORMAT_MOD_X_TILED);
>> >> +     }
>> >> +
>> >> +     igt_subtest("fill-fb")
>> >> +             fill_fb_subtest();
>> >> +
>> >> +     igt_fixture
>> >> +             teardown_environment();
>> >> +}
>> >> --
>> >> 2.1.4
>> >>
>> >> _______________________________________________
>> >> Intel-gfx mailing list
>> >> Intel-gfx@xxxxxxxxxxxxxxxxxxxxx
>> >> http://lists.freedesktop.org/mailman/listinfo/intel-gfx
>> >
>> > --
>> > Daniel Vetter
>> > Software Engineer, Intel Corporation
>> > http://blog.ffwll.ch
>>
>>
>>
>> --
>> Paulo Zanoni
>
> --
> Daniel Vetter
> Software Engineer, Intel Corporation
> http://blog.ffwll.ch



-- 
Paulo Zanoni
_______________________________________________
Intel-gfx mailing list
Intel-gfx@xxxxxxxxxxxxxxxxxxxxx
http://lists.freedesktop.org/mailman/listinfo/intel-gfx





[Index of Archives]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]
  Powered by Linux