[PATCH libdrm] tests/modetest: Add modetest_atomic tool

Eric Engestrom eric.engestrom at intel.com
Mon Jul 23 16:30:00 UTC 2018


On Friday, 2018-07-20 13:33:29 +0200, Benjamin Gaignard wrote:
> This is a modetest like tool but using atomic API.
> 
> With modetest_atomic it is mandatory to specify a mode ("-s")
> and a plane ("-P") to display a pattern on screen.
> 
> "-v" does a loop swapping between two framebuffers for each
> active planes.
> 
> modetest_atomic doesn't offer cursor support
> 
> Signed-off-by: Benjamin Gaignard <benjamin.gaignard at linaro.org>
> ---
> 
> The code is based on modetest and keep most of it infrastructure
> like arguments parsing, finding properties id from their name,
> resources dumping or the general way of working. 
> It duplicates modetest code but adding compilation flags or 
> conditional tests everywhere in modetest would have made it 
> more complex and unreadable.

I don't have an opinion on whether duplicating the test is the right
thing, but if you do, please also duplicate the lines in
tests/modetest/meson.build :)

> 
> Creating modetest_atomic could allow to test atomic API without
> need to use "big" frameworks like weston, drm_hwcomposer or igt
> with all their dependencies.
> kmscube could also be used to test atomic API but it need EGL.
> 
> It have been tested (only) on stm driver with one or two planes
> actived.
> 
>  tests/modetest/Makefile.am       |   13 +-
>  tests/modetest/Makefile.sources  |    7 +
>  tests/modetest/modetest_atomic.c | 1546 ++++++++++++++++++++++++++++++++++++++
>  3 files changed, 1564 insertions(+), 2 deletions(-)
>  create mode 100644 tests/modetest/modetest_atomic.c
> 
> diff --git a/tests/modetest/Makefile.am b/tests/modetest/Makefile.am
> index 4b296c83..8f697bb3 100644
> --- a/tests/modetest/Makefile.am
> +++ b/tests/modetest/Makefile.am
> @@ -10,10 +10,12 @@ AM_CFLAGS += \
>  
>  if HAVE_INSTALL_TESTS
>  bin_PROGRAMS = \
> -	modetest
> +	modetest \
> +	modetest_atomic
>  else
>  noinst_PROGRAMS = \
> -	modetest
> +	modetest \
> +	modetest_atomic
>  endif
>  
>  modetest_SOURCES = $(MODETEST_FILES)
> @@ -22,3 +24,10 @@ modetest_LDADD = \
>  	$(top_builddir)/libdrm.la \
>  	$(top_builddir)/tests/util/libutil.la \
>  	$(CAIRO_LIBS)
> +
> +modetest_atomic_SOURCES = $(MODETEST_ATOMIC_FILES)
> +
> +modetest_atomic_LDADD = \
> +	$(top_builddir)/libdrm.la \
> +	$(top_builddir)/tests/util/libutil.la \
> +	$(CAIRO_LIBS)
> diff --git a/tests/modetest/Makefile.sources b/tests/modetest/Makefile.sources
> index 399af0df..0a1df4c0 100644
> --- a/tests/modetest/Makefile.sources
> +++ b/tests/modetest/Makefile.sources
> @@ -4,3 +4,10 @@ MODETEST_FILES := \
>  	cursor.c \
>  	cursor.h \
>  	modetest.c
> +
> +MODETEST_ATOMIC_FILES := \
> +	buffers.c \
> +	buffers.h \
> +	cursor.c \
> +	cursor.h \
> +	modetest_atomic.c
> diff --git a/tests/modetest/modetest_atomic.c b/tests/modetest/modetest_atomic.c
> new file mode 100644
> index 00000000..8c877860
> --- /dev/null
> +++ b/tests/modetest/modetest_atomic.c
> @@ -0,0 +1,1546 @@
> +/*
> + * DRM based mode setting test program
> + * Copyright 2008 Tungsten Graphics
> + *   Jakob Bornecrantz <jakob at tungstengraphics.com>
> + * Copyright 2008 Intel Corporation
> + *   Jesse Barnes <jesse.barnes at intel.com>
> + *
> + * 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
> + * 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 fairly simple test program dumps output in a similar format to the
> + * "xrandr" tool everyone knows & loves.  It's necessarily slightly different
> + * since the kernel separates outputs into encoder and connector structures,
> + * each with their own unique ID.  The program also allows test testing of the
> + * memory management and mode setting APIs by allowing the user to specify a
> + * connector and mode to use for mode setting.  If all works as expected, a
> + * blue background should be painted on the monitor attached to the specified
> + * connector after the selected mode is set.
> + *
> + * TODO: use cairo to write the mode info on the selected output once
> + *       the mode has been programmed, along with possible test patterns.
> + */
> +
> +#ifdef HAVE_CONFIG_H
> +#include "config.h"
> +#endif
> +
> +#include <assert.h>
> +#include <ctype.h>
> +#include <stdbool.h>
> +#include <stdio.h>
> +#include <stdlib.h>
> +#include <stdint.h>
> +#include <inttypes.h>
> +#include <unistd.h>
> +#include <string.h>
> +#include <strings.h>
> +#include <errno.h>
> +#include <poll.h>
> +#include <sys/time.h>
> +#ifdef HAVE_SYS_SELECT_H
> +#include <sys/select.h>
> +#endif
> +
> +#include "xf86drm.h"
> +#include "xf86drmMode.h"
> +#include "drm_fourcc.h"
> +
> +#include "util/common.h"
> +#include "util/format.h"
> +#include "util/kms.h"
> +#include "util/pattern.h"
> +
> +#include "buffers.h"
> +
> +struct crtc {
> +	drmModeCrtc *crtc;
> +	drmModeObjectProperties *props;
> +	drmModePropertyRes **props_info;
> +	drmModeModeInfo *mode;
> +};
> +
> +struct encoder {
> +	drmModeEncoder *encoder;
> +};
> +
> +struct connector {
> +	drmModeConnector *connector;
> +	drmModeObjectProperties *props;
> +	drmModePropertyRes **props_info;
> +	char *name;
> +};
> +
> +struct fb {
> +	drmModeFB *fb;
> +};
> +
> +struct plane {
> +	drmModePlane *plane;
> +	drmModeObjectProperties *props;
> +	drmModePropertyRes **props_info;
> +};
> +
> +struct resources {
> +	drmModeRes *res;
> +	drmModePlaneRes *plane_res;
> +
> +	struct crtc *crtcs;
> +	struct encoder *encoders;
> +	struct connector *connectors;
> +	struct fb *fbs;
> +	struct plane *planes;
> +};
> +
> +struct device {
> +	int fd;
> +
> +	struct resources *resources;
> +	drmModeAtomicReq *req;
> +};
> +
> +static inline int64_t U642I64(uint64_t val)
> +{
> +	return (int64_t)*((int64_t *)&val);
> +}
> +
> +#define bit_name_fn(res)					\
> +const char * res##_str(int type) {				\
> +	unsigned int i;						\
> +	const char *sep = "";					\
> +	for (i = 0; i < ARRAY_SIZE(res##_names); i++) {		\
> +		if (type & (1 << i)) {				\
> +			printf("%s%s", sep, res##_names[i]);	\
> +			sep = ", ";				\
> +		}						\
> +	}							\
> +	return NULL;						\
> +}
> +
> +static const char *mode_type_names[] = {
> +	"builtin",
> +	"clock_c",
> +	"crtc_c",
> +	"preferred",
> +	"default",
> +	"userdef",
> +	"driver",
> +};
> +
> +static bit_name_fn(mode_type)
> +
> +static const char *mode_flag_names[] = {
> +	"phsync",
> +	"nhsync",
> +	"pvsync",
> +	"nvsync",
> +	"interlace",
> +	"dblscan",
> +	"csync",
> +	"pcsync",
> +	"ncsync",
> +	"hskew",
> +	"bcast",
> +	"pixmux",
> +	"dblclk",
> +	"clkdiv2"
> +};
> +
> +static bit_name_fn(mode_flag)
> +
> +static void dump_fourcc(uint32_t fourcc)
> +{
> +	printf(" %c%c%c%c",
> +		fourcc,
> +		fourcc >> 8,
> +		fourcc >> 16,
> +		fourcc >> 24);
> +}
> +
> +static void dump_encoders(struct device *dev)
> +{
> +	drmModeEncoder *encoder;
> +	int i;
> +
> +	printf("Encoders:\n");
> +	printf("id\tcrtc\ttype\tpossible crtcs\tpossible clones\t\n");
> +	for (i = 0; i < dev->resources->res->count_encoders; i++) {
> +		encoder = dev->resources->encoders[i].encoder;
> +		if (!encoder)
> +			continue;
> +
> +		printf("%d\t%d\t%s\t0x%08x\t0x%08x\n",
> +		       encoder->encoder_id,
> +		       encoder->crtc_id,
> +		       util_lookup_encoder_type_name(encoder->encoder_type),
> +		       encoder->possible_crtcs,
> +		       encoder->possible_clones);
> +	}
> +	printf("\n");
> +}
> +
> +static void dump_mode(drmModeModeInfo *mode)
> +{
> +	printf("  %s %d %d %d %d %d %d %d %d %d %d",
> +	       mode->name,
> +	       mode->vrefresh,
> +	       mode->hdisplay,
> +	       mode->hsync_start,
> +	       mode->hsync_end,
> +	       mode->htotal,
> +	       mode->vdisplay,
> +	       mode->vsync_start,
> +	       mode->vsync_end,
> +	       mode->vtotal,
> +	       mode->clock);
> +
> +	printf(" flags: ");
> +	mode_flag_str(mode->flags);
> +	printf("; type: ");
> +	mode_type_str(mode->type);
> +	printf("\n");
> +}
> +
> +static void dump_blob(struct device *dev, uint32_t blob_id)
> +{
> +	uint32_t i;
> +	unsigned char *blob_data;
> +	drmModePropertyBlobPtr blob;
> +
> +	blob = drmModeGetPropertyBlob(dev->fd, blob_id);
> +	if (!blob) {
> +		printf("\n");
> +		return;
> +	}
> +
> +	blob_data = blob->data;
> +
> +	for (i = 0; i < blob->length; i++) {
> +		if (i % 16 == 0)
> +			printf("\n\t\t\t");
> +		printf("%.2hhx", blob_data[i]);
> +	}
> +	printf("\n");
> +
> +	drmModeFreePropertyBlob(blob);
> +}
> +
> +static void dump_prop(struct device *dev, drmModePropertyPtr prop,
> +		      uint32_t prop_id, uint64_t value)
> +{
> +	int i;
> +	printf("\t%d", prop_id);
> +	if (!prop) {
> +		printf("\n");
> +		return;
> +	}
> +
> +	printf(" %s:\n", prop->name);
> +
> +	printf("\t\tflags:");
> +	if (prop->flags & DRM_MODE_PROP_PENDING)
> +		printf(" pending");
> +	if (prop->flags & DRM_MODE_PROP_IMMUTABLE)
> +		printf(" immutable");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_SIGNED_RANGE))
> +		printf(" signed range");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_RANGE))
> +		printf(" range");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_ENUM))
> +		printf(" enum");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_BITMASK))
> +		printf(" bitmask");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_BLOB))
> +		printf(" blob");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_OBJECT))
> +		printf(" object");
> +	printf("\n");
> +
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_SIGNED_RANGE)) {
> +		printf("\t\tvalues:");
> +		for (i = 0; i < prop->count_values; i++)
> +			printf(" %"PRId64, U642I64(prop->values[i]));
> +		printf("\n");
> +	}
> +
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_RANGE)) {
> +		printf("\t\tvalues:");
> +		for (i = 0; i < prop->count_values; i++)
> +			printf(" %"PRIu64, prop->values[i]);
> +		printf("\n");
> +	}
> +
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_ENUM)) {
> +		printf("\t\tenums:");
> +		for (i = 0; i < prop->count_enums; i++)
> +			printf(" %s=%llu", prop->enums[i].name,
> +			       prop->enums[i].value);
> +		printf("\n");
> +	} else if (drm_property_type_is(prop, DRM_MODE_PROP_BITMASK)) {
> +		printf("\t\tvalues:");
> +		for (i = 0; i < prop->count_enums; i++)
> +			printf(" %s=0x%llx", prop->enums[i].name,
> +			       (1LL << prop->enums[i].value));
> +		printf("\n");
> +	} else {
> +		assert(prop->count_enums == 0);
> +	}
> +
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_BLOB)) {
> +		printf("\t\tblobs:\n");
> +		for (i = 0; i < prop->count_blobs; i++)
> +			dump_blob(dev, prop->blob_ids[i]);
> +		printf("\n");
> +	} else {
> +		assert(prop->count_blobs == 0);
> +	}
> +
> +	printf("\t\tvalue:");
> +	if (drm_property_type_is(prop, DRM_MODE_PROP_BLOB))
> +		dump_blob(dev, value);
> +	else if (drm_property_type_is(prop, DRM_MODE_PROP_SIGNED_RANGE))
> +		printf(" %"PRId64"\n", value);
> +	else
> +		printf(" %"PRIu64"\n", value);
> +}
> +
> +static void dump_connectors(struct device *dev)
> +{
> +	int i, j;
> +
> +	printf("Connectors:\n");
> +	printf("id\tencoder\tstatus\t\tname\t\tsize (mm)\tmodes\tencoders\n");
> +	for (i = 0; i < dev->resources->res->count_connectors; i++) {
> +		struct connector *_connector = &dev->resources->connectors[i];
> +		drmModeConnector *connector = _connector->connector;
> +		if (!connector)
> +			continue;
> +
> +		printf("%d\t%d\t%s\t%-15s\t%dx%d\t\t%d\t",
> +		       connector->connector_id,
> +		       connector->encoder_id,
> +		       util_lookup_connector_status_name(connector->connection),
> +		       _connector->name,
> +		       connector->mmWidth, connector->mmHeight,
> +		       connector->count_modes);
> +
> +		for (j = 0; j < connector->count_encoders; j++)
> +			printf("%s%d", j > 0 ? ", " : "", connector->encoders[j]);
> +		printf("\n");
> +
> +		if (connector->count_modes) {
> +			printf("  modes:\n");
> +			printf("\tname refresh (Hz) hdisp hss hse htot vdisp "
> +			       "vss vse vtot)\n");
> +			for (j = 0; j < connector->count_modes; j++)
> +				dump_mode(&connector->modes[j]);
> +		}
> +
> +		if (_connector->props) {
> +			printf("  props:\n");
> +			for (j = 0; j < (int)_connector->props->count_props; j++)
> +				dump_prop(dev, _connector->props_info[j],
> +					  _connector->props->props[j],
> +					  _connector->props->prop_values[j]);
> +		}
> +	}
> +	printf("\n");
> +}
> +
> +static void dump_crtcs(struct device *dev)
> +{
> +	int i;
> +	uint32_t j;
> +
> +	printf("CRTCs:\n");
> +	printf("id\tfb\tpos\tsize\n");
> +	for (i = 0; i < dev->resources->res->count_crtcs; i++) {
> +		struct crtc *_crtc = &dev->resources->crtcs[i];
> +		drmModeCrtc *crtc = _crtc->crtc;
> +		if (!crtc)
> +			continue;
> +
> +		printf("%d\t%d\t(%d,%d)\t(%dx%d)\n",
> +		       crtc->crtc_id,
> +		       crtc->buffer_id,
> +		       crtc->x, crtc->y,
> +		       crtc->width, crtc->height);
> +		dump_mode(&crtc->mode);
> +
> +		if (_crtc->props) {
> +			printf("  props:\n");
> +			for (j = 0; j < _crtc->props->count_props; j++)
> +				dump_prop(dev, _crtc->props_info[j],
> +					  _crtc->props->props[j],
> +					  _crtc->props->prop_values[j]);
> +		} else {
> +			printf("  no properties found\n");
> +		}
> +	}
> +	printf("\n");
> +}
> +
> +static void dump_framebuffers(struct device *dev)
> +{
> +	drmModeFB *fb;
> +	int i;
> +
> +	printf("Frame buffers:\n");
> +	printf("id\tsize\tpitch\n");
> +	for (i = 0; i < dev->resources->res->count_fbs; i++) {
> +		fb = dev->resources->fbs[i].fb;
> +		if (!fb)
> +			continue;
> +
> +		printf("%u\t(%ux%u)\t%u\n",
> +		       fb->fb_id,
> +		       fb->width, fb->height,
> +		       fb->pitch);
> +	}
> +	printf("\n");
> +}
> +
> +static void dump_planes(struct device *dev)
> +{
> +	unsigned int i, j;
> +
> +	printf("Planes:\n");
> +	printf("id\tcrtc\tfb\tCRTC x,y\tx,y\tgamma size\tpossible crtcs\n");
> +
> +	if (!dev->resources->plane_res)
> +		return;
> +
> +	for (i = 0; i < dev->resources->plane_res->count_planes; i++) {
> +		struct plane *plane = &dev->resources->planes[i];
> +		drmModePlane *ovr = plane->plane;
> +		if (!ovr)
> +			continue;
> +
> +		printf("%d\t%d\t%d\t%d,%d\t\t%d,%d\t%-8d\t0x%08x\n",
> +		       ovr->plane_id, ovr->crtc_id, ovr->fb_id,
> +		       ovr->crtc_x, ovr->crtc_y, ovr->x, ovr->y,
> +		       ovr->gamma_size, ovr->possible_crtcs);
> +
> +		if (!ovr->count_formats)
> +			continue;
> +
> +		printf("  formats:");
> +		for (j = 0; j < ovr->count_formats; j++)
> +			dump_fourcc(ovr->formats[j]);
> +		printf("\n");
> +
> +		if (plane->props) {
> +			printf("  props:\n");
> +			for (j = 0; j < plane->props->count_props; j++)
> +				dump_prop(dev, plane->props_info[j],
> +					  plane->props->props[j],
> +					  plane->props->prop_values[j]);
> +		} else {
> +			printf("  no properties found\n");
> +		}
> +	}
> +	printf("\n");
> +
> +	return;
> +}
> +
> +static void free_resources(struct resources *res)
> +{
> +	int i;
> +
> +	if (!res)
> +		return;
> +
> +#define free_resource(_res, __res, type, Type)					\
> +	do {									\
> +		if (!(_res)->type##s)						\
> +			break;							\
> +		for (i = 0; i < (int)(_res)->__res->count_##type##s; ++i) {	\
> +			if (!(_res)->type##s[i].type)				\
> +				break;						\
> +			drmModeFree##Type((_res)->type##s[i].type);		\
> +		}								\
> +		free((_res)->type##s);						\
> +	} while (0)
> +
> +#define free_properties(_res, __res, type)					\
> +	do {									\
> +		for (i = 0; i < (int)(_res)->__res->count_##type##s; ++i) {	\
> +			drmModeFreeObjectProperties(res->type##s[i].props);	\
> +			free(res->type##s[i].props_info);			\
> +		}								\
> +	} while (0)
> +
> +	if (res->res) {
> +		free_properties(res, res, crtc);
> +
> +		free_resource(res, res, crtc, Crtc);
> +		free_resource(res, res, encoder, Encoder);
> +
> +		for (i = 0; i < res->res->count_connectors; i++)
> +			free(res->connectors[i].name);
> +
> +		free_resource(res, res, connector, Connector);
> +		free_resource(res, res, fb, FB);
> +
> +		drmModeFreeResources(res->res);
> +	}
> +
> +	if (res->plane_res) {
> +		free_properties(res, plane_res, plane);
> +
> +		free_resource(res, plane_res, plane, Plane);
> +
> +		drmModeFreePlaneResources(res->plane_res);
> +	}
> +
> +	free(res);
> +}
> +
> +static struct resources *get_resources(struct device *dev)
> +{
> +	struct resources *res;
> +	int i;
> +
> +	res = calloc(1, sizeof(*res));
> +	if (res == 0)
> +		return NULL;
> +
> +	drmSetClientCap(dev->fd, DRM_CLIENT_CAP_UNIVERSAL_PLANES, 1);
> +
> +	res->res = drmModeGetResources(dev->fd);
> +	if (!res->res) {
> +		fprintf(stderr, "drmModeGetResources failed: %s\n",
> +			strerror(errno));
> +		goto error;
> +	}
> +
> +	res->crtcs = calloc(res->res->count_crtcs, sizeof(*res->crtcs));
> +	res->encoders = calloc(res->res->count_encoders, sizeof(*res->encoders));
> +	res->connectors = calloc(res->res->count_connectors, sizeof(*res->connectors));
> +	res->fbs = calloc(res->res->count_fbs, sizeof(*res->fbs));
> +
> +	if (!res->crtcs || !res->encoders || !res->connectors || !res->fbs)
> +		goto error;
> +
> +#define get_resource(_res, __res, type, Type)					\
> +	do {									\
> +		for (i = 0; i < (int)(_res)->__res->count_##type##s; ++i) {	\
> +			(_res)->type##s[i].type =				\
> +				drmModeGet##Type(dev->fd, (_res)->__res->type##s[i]); \
> +			if (!(_res)->type##s[i].type)				\
> +				fprintf(stderr, "could not get %s %i: %s\n",	\
> +					#type, (_res)->__res->type##s[i],	\
> +					strerror(errno));			\
> +		}								\
> +	} while (0)
> +
> +	get_resource(res, res, crtc, Crtc);
> +	get_resource(res, res, encoder, Encoder);
> +	get_resource(res, res, connector, Connector);
> +	get_resource(res, res, fb, FB);
> +
> +	/* Set the name of all connectors based on the type name and the per-type ID. */
> +	for (i = 0; i < res->res->count_connectors; i++) {
> +		struct connector *connector = &res->connectors[i];
> +		drmModeConnector *conn = connector->connector;
> +		int num;
> +
> +		num = asprintf(&connector->name, "%s-%u",
> +			 util_lookup_connector_type_name(conn->connector_type),
> +			 conn->connector_type_id);
> +		if (num < 0)
> +			goto error;
> +	}
> +
> +#define get_properties(_res, __res, type, Type)					\
> +	do {									\
> +		for (i = 0; i < (int)(_res)->__res->count_##type##s; ++i) {	\
> +			struct type *obj = &res->type##s[i];			\
> +			unsigned int j;						\
> +			obj->props =						\
> +				drmModeObjectGetProperties(dev->fd, obj->type->type##_id, \
> +							   DRM_MODE_OBJECT_##Type); \
> +			if (!obj->props) {					\
> +				fprintf(stderr,					\
> +					"could not get %s %i properties: %s\n", \
> +					#type, obj->type->type##_id,		\
> +					strerror(errno));			\
> +				continue;					\
> +			}							\
> +			obj->props_info = calloc(obj->props->count_props,	\
> +						 sizeof(*obj->props_info));	\
> +			if (!obj->props_info)					\
> +				continue;					\
> +			for (j = 0; j < obj->props->count_props; ++j)		\
> +				obj->props_info[j] =				\
> +					drmModeGetProperty(dev->fd, obj->props->props[j]); \
> +		}								\
> +	} while (0)
> +
> +	get_properties(res, res, crtc, CRTC);
> +	get_properties(res, res, connector, CONNECTOR);
> +
> +	for (i = 0; i < res->res->count_crtcs; ++i)
> +		res->crtcs[i].mode = &res->crtcs[i].crtc->mode;
> +
> +	res->plane_res = drmModeGetPlaneResources(dev->fd);
> +	if (!res->plane_res) {
> +		fprintf(stderr, "drmModeGetPlaneResources failed: %s\n",
> +			strerror(errno));
> +		return res;
> +	}
> +
> +	res->planes = calloc(res->plane_res->count_planes, sizeof(*res->planes));
> +	if (!res->planes)
> +		goto error;
> +
> +	get_resource(res, plane_res, plane, Plane);
> +	get_properties(res, plane_res, plane, PLANE);
> +
> +	return res;
> +
> +error:
> +	free_resources(res);
> +	return NULL;
> +}
> +
> +static int get_crtc_index(struct device *dev, uint32_t id)
> +{
> +	int i;
> +
> +	for (i = 0; i < dev->resources->res->count_crtcs; ++i) {
> +		drmModeCrtc *crtc = dev->resources->crtcs[i].crtc;
> +		if (crtc && crtc->crtc_id == id)
> +			return i;
> +	}
> +
> +	return -1;
> +}
> +
> +static drmModeConnector *get_connector_by_name(struct device *dev, const char *name)
> +{
> +	struct connector *connector;
> +	int i;
> +
> +	for (i = 0; i < dev->resources->res->count_connectors; i++) {
> +		connector = &dev->resources->connectors[i];
> +
> +		if (strcmp(connector->name, name) == 0)
> +			return connector->connector;
> +	}
> +
> +	return NULL;
> +}
> +
> +static drmModeConnector *get_connector_by_id(struct device *dev, uint32_t id)
> +{
> +	drmModeConnector *connector;
> +	int i;
> +
> +	for (i = 0; i < dev->resources->res->count_connectors; i++) {
> +		connector = dev->resources->connectors[i].connector;
> +		if (connector && connector->connector_id == id)
> +			return connector;
> +	}
> +
> +	return NULL;
> +}
> +
> +static drmModeEncoder *get_encoder_by_id(struct device *dev, uint32_t id)
> +{
> +	drmModeEncoder *encoder;
> +	int i;
> +
> +	for (i = 0; i < dev->resources->res->count_encoders; i++) {
> +		encoder = dev->resources->encoders[i].encoder;
> +		if (encoder && encoder->encoder_id == id)
> +			return encoder;
> +	}
> +
> +	return NULL;
> +}
> +
> +/* -----------------------------------------------------------------------------
> + * Pipes and planes
> + */
> +
> +/*
> + * Mode setting with the kernel interfaces is a bit of a chore.
> + * First you have to find the connector in question and make sure the
> + * requested mode is available.
> + * Then you need to find the encoder attached to that connector so you
> + * can bind it with a free crtc.
> + */
> +struct pipe_arg {
> +	const char **cons;
> +	uint32_t *con_ids;
> +	unsigned int num_cons;
> +	uint32_t crtc_id;
> +	char mode_str[64];
> +	char format_str[5];
> +	unsigned int vrefresh;
> +	unsigned int fourcc;
> +	drmModeModeInfo *mode;
> +	struct crtc *crtc;
> +	unsigned int fb_id[2], current_fb_id;
> +	struct timeval start;
> +
> +	int swap_count;
> +};
> +
> +struct plane_arg {
> +	uint32_t plane_id;  /* the id of plane to use */
> +	uint32_t crtc_id;  /* the id of CRTC to bind to */
> +	bool has_position;
> +	int32_t x, y;
> +	uint32_t w, h;
> +	double scale;
> +	unsigned int fb_id;
> +	unsigned int old_fb_id;
> +	struct bo *bo;
> +	struct bo *old_bo;
> +	char format_str[5]; /* need to leave room for terminating \0 */
> +	unsigned int fourcc;
> +};
> +
> +static drmModeModeInfo *
> +connector_find_mode(struct device *dev, uint32_t con_id, const char *mode_str,
> +        const unsigned int vrefresh)
> +{
> +	drmModeConnector *connector;
> +	drmModeModeInfo *mode;
> +	int i;
> +
> +	connector = get_connector_by_id(dev, con_id);
> +	if (!connector || !connector->count_modes)
> +		return NULL;
> +
> +	for (i = 0; i < connector->count_modes; i++) {
> +		mode = &connector->modes[i];
> +		if (!strcmp(mode->name, mode_str)) {
> +			/* If the vertical refresh frequency is not specified then return the
> +			 * first mode that match with the name. Else, return the mode that match
> +			 * the name and the specified vertical refresh frequency.
> +			 */
> +			if (vrefresh == 0)
> +				return mode;
> +			else if (mode->vrefresh == vrefresh)
> +				return mode;
> +		}
> +	}
> +
> +	return NULL;
> +}
> +
> +static struct crtc *pipe_find_crtc(struct device *dev, struct pipe_arg *pipe)
> +{
> +	uint32_t possible_crtcs = ~0;
> +	uint32_t active_crtcs = 0;
> +	unsigned int crtc_idx;
> +	unsigned int i;
> +	int j;
> +
> +	for (i = 0; i < pipe->num_cons; ++i) {
> +		uint32_t crtcs_for_connector = 0;
> +		drmModeConnector *connector;
> +		drmModeEncoder *encoder;
> +		int idx;
> +
> +		connector = get_connector_by_id(dev, pipe->con_ids[i]);
> +		if (!connector)
> +			return NULL;
> +
> +		for (j = 0; j < connector->count_encoders; ++j) {
> +			encoder = get_encoder_by_id(dev, connector->encoders[j]);
> +			if (!encoder)
> +				continue;
> +
> +			crtcs_for_connector |= encoder->possible_crtcs;
> +
> +			idx = get_crtc_index(dev, encoder->crtc_id);
> +			if (idx >= 0)
> +				active_crtcs |= 1 << idx;
> +		}
> +
> +		possible_crtcs &= crtcs_for_connector;
> +	}
> +
> +	if (!possible_crtcs)
> +		return NULL;
> +
> +	/* Return the first possible and active CRTC if one exists, or the first
> +	 * possible CRTC otherwise.
> +	 */
> +	if (possible_crtcs & active_crtcs)
> +		crtc_idx = ffs(possible_crtcs & active_crtcs);
> +	else
> +		crtc_idx = ffs(possible_crtcs);
> +
> +	return &dev->resources->crtcs[crtc_idx - 1];
> +}
> +
> +static int pipe_find_crtc_and_mode(struct device *dev, struct pipe_arg *pipe)
> +{
> +	drmModeModeInfo *mode = NULL;
> +	int i;
> +
> +	pipe->mode = NULL;
> +
> +	for (i = 0; i < (int)pipe->num_cons; i++) {
> +		mode = connector_find_mode(dev, pipe->con_ids[i],
> +					   pipe->mode_str, pipe->vrefresh);
> +		if (mode == NULL) {
> +			fprintf(stderr,
> +				"failed to find mode \"%s\" for connector %s\n",
> +				pipe->mode_str, pipe->cons[i]);
> +			return -EINVAL;
> +		}
> +	}
> +
> +	/* If the CRTC ID was specified, get the corresponding CRTC. Otherwise
> +	 * locate a CRTC that can be attached to all the connectors.
> +	 */
> +	if (pipe->crtc_id != (uint32_t)-1) {
> +		for (i = 0; i < dev->resources->res->count_crtcs; i++) {
> +			struct crtc *crtc = &dev->resources->crtcs[i];
> +
> +			if (pipe->crtc_id == crtc->crtc->crtc_id) {
> +				pipe->crtc = crtc;
> +				break;
> +			}
> +		}
> +	} else {
> +		pipe->crtc = pipe_find_crtc(dev, pipe);
> +	}
> +
> +	if (!pipe->crtc) {
> +		fprintf(stderr, "failed to find CRTC for pipe\n");
> +		return -EINVAL;
> +	}
> +
> +	pipe->mode = mode;
> +	pipe->crtc->mode = mode;
> +
> +	return 0;
> +}
> +
> +/* -----------------------------------------------------------------------------
> + * Properties
> + */
> +
> +struct property_arg {
> +	uint32_t obj_id;
> +	uint32_t obj_type;
> +	char name[DRM_PROP_NAME_LEN+1];
> +	uint32_t prop_id;
> +	uint64_t value;
> +};
> +
> +static void set_property(struct device *dev, struct property_arg *p)
> +{
> +	drmModeObjectProperties *props = NULL;
> +	drmModePropertyRes **props_info = NULL;
> +	const char *obj_type;
> +	int ret;
> +	int i;
> +
> +	p->obj_type = 0;
> +	p->prop_id = 0;
> +
> +#define find_object(_res, __res, type, Type)					\
> +	do {									\
> +		for (i = 0; i < (int)(_res)->__res->count_##type##s; ++i) {	\
> +			struct type *obj = &(_res)->type##s[i];			\
> +			if (obj->type->type##_id != p->obj_id)			\
> +				continue;					\
> +			p->obj_type = DRM_MODE_OBJECT_##Type;			\
> +			obj_type = #Type;					\
> +			props = obj->props;					\
> +			props_info = obj->props_info;				\
> +		}								\
> +	} while(0)								\
> +
> +	find_object(dev->resources, res, crtc, CRTC);
> +	if (p->obj_type == 0)
> +		find_object(dev->resources, res, connector, CONNECTOR);
> +	if (p->obj_type == 0)
> +		find_object(dev->resources, plane_res, plane, PLANE);
> +	if (p->obj_type == 0) {
> +		fprintf(stderr, "Object %i not found, can't set property\n",
> +			p->obj_id);
> +			return;
> +	}
> +
> +	if (!props) {
> +		fprintf(stderr, "%s %i has no properties\n",
> +			obj_type, p->obj_id);
> +		return;
> +	}
> +
> +	for (i = 0; i < (int)props->count_props; ++i) {
> +		if (!props_info[i])
> +			continue;
> +		if (strcmp(props_info[i]->name, p->name) == 0)
> +			break;
> +	}
> +
> +	if (i == (int)props->count_props) {
> +		fprintf(stderr, "%s %i has no %s property\n",
> +			obj_type, p->obj_id, p->name);
> +		return;
> +	}
> +
> +	p->prop_id = props->props[i];
> +
> +	ret = drmModeAtomicAddProperty(dev->req, p->obj_id, p->prop_id, p->value);
> +	if (ret < 0)
> +		fprintf(stderr, "failed to set %s %i property %s to %" PRIu64 ": %s\n",
> +			obj_type, p->obj_id, p->name, p->value, strerror(errno));
> +}
> +
> +/* -------------------------------------------------------------------------- */
> +
> +/*static bool format_support(const drmModePlanePtr ovr, uint32_t fmt)
> +{
> +	unsigned int i;
> +
> +	for (i = 0; i < ovr->count_formats; ++i) {
> +		if (ovr->formats[i] == fmt)
> +			return true;
> +	}
> +
> +	return false;
> +}*/
> +
> +static void add_property(struct device *dev, uint32_t obj_id,
> +			       const char *name, uint64_t value)
> +{
> +	struct property_arg p;
> +
> +	p.obj_id = obj_id;
> +	strcpy(p.name, name);
> +	p.value = value;
> +
> +	set_property(dev, &p);
> +}
> +
> +static int set_plane(struct device *dev, struct plane_arg *p,
> +		     int pattern, bool update)
> +{
> +	uint32_t handles[4] = {0}, pitches[4] = {0}, offsets[4] = {0};
> +	struct bo *plane_bo;
> +	int crtc_x, crtc_y, crtc_w, crtc_h;
> +	struct crtc *crtc = NULL;
> +	unsigned int i;
> +	unsigned int old_fb_id;
> +
> +	/* Find an unused plane which can be connected to our CRTC. Find the
> +	 * CRTC index first, then iterate over available planes.
> +	 */
> +	for (i = 0; i < (unsigned int)dev->resources->res->count_crtcs; i++) {
> +		if (p->crtc_id == dev->resources->res->crtcs[i]) {
> +			crtc = &dev->resources->crtcs[i];
> +			break;
> +		}
> +	}
> +
> +	if (!crtc) {
> +		fprintf(stderr, "CRTC %u not found\n", p->crtc_id);
> +		return -1;
> +	}
> +
> +	if (!update)
> +		fprintf(stderr, "testing %dx%d@%s on plane %u, crtc %u\n",
> +			p->w, p->h, p->format_str, p->plane_id, p->crtc_id);
> +
> +	plane_bo = p->old_bo;
> +	p->old_bo = p->bo;
> +
> +	if (!plane_bo) {
> +		plane_bo = bo_create(dev->fd, p->fourcc, p->w, p->h,
> +				     handles, pitches, offsets, pattern);
> +
> +		if (plane_bo == NULL)
> +			return -1;
> +
> +		if (drmModeAddFB2(dev->fd, p->w, p->h, p->fourcc,
> +			handles, pitches, offsets, &p->fb_id, 0)) {
> +			fprintf(stderr, "failed to add fb: %s\n", strerror(errno));
> +			return -1;
> +		}
> +	}
> +
> +	p->bo = plane_bo;
> +
> +	old_fb_id = p->fb_id;
> +	p->old_fb_id = old_fb_id;
> +
> +	crtc_w = p->w * p->scale;
> +	crtc_h = p->h * p->scale;
> +	if (!p->has_position) {
> +		/* Default to the middle of the screen */
> +		crtc_x = (crtc->mode->hdisplay - crtc_w) / 2;
> +		crtc_y = (crtc->mode->vdisplay - crtc_h) / 2;
> +	} else {
> +		crtc_x = p->x;
> +		crtc_y = p->y;
> +	}
> +
> +	add_property(dev, p->plane_id, "FB_ID", p->fb_id);
> +	add_property(dev, p->plane_id, "CRTC_ID", p->crtc_id);
> +	add_property(dev, p->plane_id, "SRC_X", 0);
> +	add_property(dev, p->plane_id, "SRC_Y", 0);
> +	add_property(dev, p->plane_id, "SRC_W", p->w << 16);
> +	add_property(dev, p->plane_id, "SRC_H", p->h << 16);
> +	add_property(dev, p->plane_id, "CRTC_X", crtc_x);
> +	add_property(dev, p->plane_id, "CRTC_Y", crtc_y);
> +	add_property(dev, p->plane_id, "CRTC_W", crtc_w);
> +	add_property(dev, p->plane_id, "CRTC_H", crtc_h);
> +
> +	return 0;
> +}
> +
> +static void set_planes(struct device *dev, struct plane_arg *p,
> +		       unsigned int count, bool update)
> +{
> +	unsigned int i, pattern = UTIL_PATTERN_SMPTE;
> +
> +	/* set up planes */
> +	for (i = 0; i < count; i++) {
> +		if (i > 0)
> +			pattern = UTIL_PATTERN_TILES;
> +
> +		if (set_plane(dev, &p[i], pattern, update))
> +			return;
> +	}
> +}
> +
> +static void clear_planes(struct device *dev, struct plane_arg *p, unsigned int count)
> +{
> +	unsigned int i;
> +
> +	for (i = 0; i < count; i++) {
> +		add_property(dev, p[i].plane_id, "FB_ID", 0);
> +		add_property(dev, p[i].plane_id, "CRTC_ID", 0);
> +		add_property(dev, p[i].plane_id, "SRC_X", 0);
> +		add_property(dev, p[i].plane_id, "SRC_Y", 0);
> +		add_property(dev, p[i].plane_id, "SRC_W", 0);
> +		add_property(dev, p[i].plane_id, "SRC_H", 0);
> +		add_property(dev, p[i].plane_id, "CRTC_X", 0);
> +		add_property(dev, p[i].plane_id, "CRTC_Y", 0);
> +		add_property(dev, p[i].plane_id, "CRTC_W", 0);
> +		add_property(dev, p[i].plane_id, "CRTC_H", 0);
> +	}
> +}
> +
> +static void clear_FB(struct device *dev, struct plane_arg *p, unsigned int count)
> +{
> +	unsigned int i;
> +
> +	for (i = 0; i < count; i++) {
> +		if (p[i].fb_id) {
> +			drmModeRmFB(dev->fd, p[i].fb_id);
> +			p[i].fb_id = 0;
> +		}
> +		if (p[i].old_fb_id) {
> +			drmModeRmFB(dev->fd, p[i].old_fb_id);
> +			p[i].old_fb_id = 0;
> +		}
> +		if (p[i].bo) {
> +			bo_destroy(p[i].bo);
> +			p[i].bo = NULL;
> +		}
> +		if (p[i].old_bo) {
> +			bo_destroy(p[i].old_bo);
> +			p[i].old_bo = NULL;
> +		}
> +
> +	}
> +}
> +
> +static void set_mode(struct device *dev, struct pipe_arg *pipes, unsigned int count)
> +{
> +	unsigned int i;
> +	unsigned int j;
> +	int ret;
> +
> +	for (i = 0; i < count; i++) {
> +		struct pipe_arg *pipe = &pipes[i];
> +
> +		ret = pipe_find_crtc_and_mode(dev, pipe);
> +		if (ret < 0)
> +			continue;
> +	}
> +
> +	for (i = 0; i < count; i++) {
> +		struct pipe_arg *pipe = &pipes[i];
> +		uint32_t blob_id;
> +
> +		if (pipe->mode == NULL)
> +			continue;
> +
> +		printf("setting mode %s-%dHz@%s on connectors ",
> +		       pipe->mode_str, pipe->mode->vrefresh, pipe->format_str);
> +		for (j = 0; j < pipe->num_cons; ++j) {
> +			printf("%s, ", pipe->cons[j]);
> +			add_property(dev, pipe->con_ids[j], "CRTC_ID", pipe->crtc->crtc->crtc_id);
> +		}
> +		printf("crtc %d\n", pipe->crtc->crtc->crtc_id);
> +
> +		drmModeCreatePropertyBlob(dev->fd, pipe->mode, sizeof(*pipe->mode), &blob_id);
> +		add_property(dev, pipe->crtc->crtc->crtc_id, "MODE_ID", blob_id);
> +		add_property(dev, pipe->crtc->crtc->crtc_id, "ACTIVE", 1);
> +	}
> +}
> +
> +static void clear_mode(struct device *dev, struct pipe_arg *pipes, unsigned int count)
> +{
> +	unsigned int i;
> +	unsigned int j;
> +
> +	for (i = 0; i < count; i++) {
> +		struct pipe_arg *pipe = &pipes[i];
> +
> +		if (pipe->mode == NULL)
> +			continue;
> +
> +		for (j = 0; j < pipe->num_cons; ++j)
> +			add_property(dev, pipe->con_ids[j], "CRTC_ID",0);
> +
> +		add_property(dev, pipe->crtc->crtc->crtc_id, "MODE_ID", 0);
> +		add_property(dev, pipe->crtc->crtc->crtc_id, "ACTIVE", 0);
> +	}
> +
> +}
> +
> +#define min(a, b)	((a) < (b) ? (a) : (b))
> +
> +static int parse_connector(struct pipe_arg *pipe, const char *arg)
> +{
> +	unsigned int len;
> +	unsigned int i;
> +	const char *p;
> +	char *endp;
> +
> +	pipe->vrefresh = 0;
> +	pipe->crtc_id = (uint32_t)-1;
> +	strcpy(pipe->format_str, "XR24");
> +
> +	/* Count the number of connectors and allocate them. */
> +	pipe->num_cons = 1;
> +	for (p = arg; *p && *p != ':' && *p != '@'; ++p) {
> +		if (*p == ',')
> +			pipe->num_cons++;
> +	}
> +
> +	pipe->con_ids = calloc(pipe->num_cons, sizeof(*pipe->con_ids));
> +	pipe->cons = calloc(pipe->num_cons, sizeof(*pipe->cons));
> +	if (pipe->con_ids == NULL || pipe->cons == NULL)
> +		return -1;
> +
> +	/* Parse the connectors. */
> +	for (i = 0, p = arg; i < pipe->num_cons; ++i, p = endp + 1) {
> +		endp = strpbrk(p, ",@:");
> +		if (!endp)
> +			break;
> +
> +		pipe->cons[i] = strndup(p, endp - p);
> +
> +		if (*endp != ',')
> +			break;
> +	}
> +
> +	if (i != pipe->num_cons - 1)
> +		return -1;
> +
> +	/* Parse the remaining parameters. */
> +	if (*endp == '@') {
> +		arg = endp + 1;
> +		pipe->crtc_id = strtoul(arg, &endp, 10);
> +	}
> +	if (*endp != ':')
> +		return -1;
> +
> +	arg = endp + 1;
> +
> +	/* Search for the vertical refresh or the format. */
> +	p = strpbrk(arg, "-@");
> +	if (p == NULL)
> +		p = arg + strlen(arg);
> +	len = min(sizeof pipe->mode_str - 1, (unsigned int)(p - arg));
> +	strncpy(pipe->mode_str, arg, len);
> +	pipe->mode_str[len] = '\0';
> +
> +	if (*p == '-') {
> +		pipe->vrefresh = strtoul(p + 1, &endp, 10);
> +		p = endp;
> +	}
> +
> +	if (*p == '@') {
> +		strncpy(pipe->format_str, p + 1, 4);
> +		pipe->format_str[4] = '\0';
> +	}
> +
> +	pipe->fourcc = util_format_fourcc(pipe->format_str);
> +	if (pipe->fourcc == 0)  {
> +		fprintf(stderr, "unknown format %s\n", pipe->format_str);
> +		return -1;
> +	}
> +
> +	return 0;
> +}
> +
> +static int parse_plane(struct plane_arg *plane, const char *p)
> +{
> +	char *end;
> +
> +	plane->plane_id = strtoul(p, &end, 10);
> +	if (*end != '@')
> +		return -EINVAL;
> +
> +	p = end + 1;
> +	plane->crtc_id = strtoul(p, &end, 10);
> +	if (*end != ':')
> +		return -EINVAL;
> +
> +	p = end + 1;
> +	plane->w = strtoul(p, &end, 10);
> +	if (*end != 'x')
> +		return -EINVAL;
> +
> +	p = end + 1;
> +	plane->h = strtoul(p, &end, 10);
> +
> +	if (*end == '+' || *end == '-') {
> +		plane->x = strtol(end, &end, 10);
> +		if (*end != '+' && *end != '-')
> +			return -EINVAL;
> +		plane->y = strtol(end, &end, 10);
> +
> +		plane->has_position = true;
> +	}
> +
> +	if (*end == '*') {
> +		p = end + 1;
> +		plane->scale = strtod(p, &end);
> +		if (plane->scale <= 0.0)
> +			return -EINVAL;
> +	} else {
> +		plane->scale = 1.0;
> +	}
> +
> +	if (*end == '@') {
> +		p = end + 1;
> +		if (strlen(p) != 4)
> +			return -EINVAL;
> +
> +		strcpy(plane->format_str, p);
> +	} else {
> +		strcpy(plane->format_str, "XR24");
> +	}
> +
> +	plane->fourcc = util_format_fourcc(plane->format_str);
> +	if (plane->fourcc == 0) {
> +		fprintf(stderr, "unknown format %s\n", plane->format_str);
> +		return -EINVAL;
> +	}
> +
> +	return 0;
> +}
> +
> +static int parse_property(struct property_arg *p, const char *arg)
> +{
> +	if (sscanf(arg, "%d:%32[^:]:%" SCNu64, &p->obj_id, p->name, &p->value) != 3)
> +		return -1;
> +
> +	p->obj_type = 0;
> +	p->name[DRM_PROP_NAME_LEN] = '\0';
> +
> +	return 0;
> +}
> +
> +static void usage(char *name)
> +{
> +	fprintf(stderr, "usage: %s [-cDdefMPpsCvw]\n", name);
> +
> +	fprintf(stderr, "\n Query options:\n\n");
> +	fprintf(stderr, "\t-c\tlist connectors\n");
> +	fprintf(stderr, "\t-e\tlist encoders\n");
> +	fprintf(stderr, "\t-f\tlist framebuffers\n");
> +	fprintf(stderr, "\t-p\tlist CRTCs and planes (pipes)\n");
> +
> +	fprintf(stderr, "\n Test options:\n\n");
> +	fprintf(stderr, "\t-P <plane_id>@<crtc_id>:<w>x<h>[+<x>+<y>][*<scale>][@<format>]\tset a plane\n");
> +	fprintf(stderr, "\t-s <connector_id>[,<connector_id>][@<crtc_id>]:<mode>[-<vrefresh>][@<format>]\tset a mode\n");
> +	fprintf(stderr, "\t-v\ttest loop\n");
> +	fprintf(stderr, "\t-w <obj_id>:<prop_name>:<value>\tset property\n");
> +
> +	fprintf(stderr, "\n Generic options:\n\n");
> +	fprintf(stderr, "\t-d\tdrop master after mode set\n");
> +	fprintf(stderr, "\t-M module\tuse the given driver\n");
> +	fprintf(stderr, "\t-D device\tuse the given device\n");
> +
> +	fprintf(stderr, "\n\tDefault is to dump all info.\n");
> +	exit(0);
> +}
> +
> +static int pipe_resolve_connectors(struct device *dev, struct pipe_arg *pipe)
> +{
> +	drmModeConnector *connector;
> +	unsigned int i;
> +	uint32_t id;
> +	char *endp;
> +
> +	for (i = 0; i < pipe->num_cons; i++) {
> +		id = strtoul(pipe->cons[i], &endp, 10);
> +		if (endp == pipe->cons[i]) {
> +			connector = get_connector_by_name(dev, pipe->cons[i]);
> +			if (!connector) {
> +				fprintf(stderr, "no connector named '%s'\n",
> +					pipe->cons[i]);
> +				return -ENODEV;
> +			}
> +
> +			id = connector->connector_id;
> +		}
> +
> +		pipe->con_ids[i] = id;
> +	}
> +
> +	return 0;
> +}
> +
> +static char optstr[] = "cdD:efM:P:ps:vw:";
> +
> +int main(int argc, char **argv)
> +{
> +	struct device dev;
> +
> +	int c;
> +	int encoders = 0, connectors = 0, crtcs = 0, planes = 0, framebuffers = 0;
> +	int drop_master = 0;
> +	int test_loop = 0;
> +	char *device = NULL;
> +	char *module = NULL;
> +	unsigned int i;
> +	unsigned int count = 0, plane_count = 0;
> +	unsigned int prop_count = 0;
> +	struct pipe_arg *pipe_args = NULL;
> +	struct plane_arg *plane_args = NULL;
> +	struct property_arg *prop_args = NULL;
> +	unsigned int args = 0;
> +	int ret;
> +
> +	memset(&dev, 0, sizeof dev);
> +
> +	opterr = 0;
> +	while ((c = getopt(argc, argv, optstr)) != -1) {
> +		args++;
> +
> +		switch (c) {
> +		case 'c':
> +			connectors = 1;
> +			break;
> +		case 'D':
> +			device = optarg;
> +			args--;
> +			break;
> +		case 'd':
> +			drop_master = 1;
> +			break;
> +		case 'e':
> +			encoders = 1;
> +			break;
> +		case 'f':
> +			framebuffers = 1;
> +			break;
> +		case 'M':
> +			module = optarg;
> +			/* Preserve the default behaviour of dumping all information. */
> +			args--;
> +			break;
> +		case 'P':
> +			plane_args = realloc(plane_args,
> +					     (plane_count + 1) * sizeof *plane_args);
> +			if (plane_args == NULL) {
> +				fprintf(stderr, "memory allocation failed\n");
> +				return 1;
> +			}
> +			memset(&plane_args[plane_count], 0, sizeof(*plane_args));
> +
> +			if (parse_plane(&plane_args[plane_count], optarg) < 0)
> +				usage(argv[0]);
> +
> +			plane_count++;
> +			break;
> +		case 'p':
> +			crtcs = 1;
> +			planes = 1;
> +			break;
> +		case 's':
> +			pipe_args = realloc(pipe_args,
> +					    (count + 1) * sizeof *pipe_args);
> +			if (pipe_args == NULL) {
> +				fprintf(stderr, "memory allocation failed\n");
> +				return 1;
> +			}
> +			memset(&pipe_args[count], 0, sizeof(*pipe_args));
> +
> +			if (parse_connector(&pipe_args[count], optarg) < 0)
> +				usage(argv[0]);
> +
> +			count++;
> +			break;
> +		case 'v':
> +			test_loop = 1;
> +			break;
> +		case 'w':
> +			prop_args = realloc(prop_args,
> +					   (prop_count + 1) * sizeof *prop_args);
> +			if (prop_args == NULL) {
> +				fprintf(stderr, "memory allocation failed\n");
> +				return 1;
> +			}
> +			memset(&prop_args[prop_count], 0, sizeof(*prop_args));
> +
> +			if (parse_property(&prop_args[prop_count], optarg) < 0)
> +				usage(argv[0]);
> +
> +			prop_count++;
> +			break;
> +		default:
> +			usage(argv[0]);
> +			break;
> +		}
> +	}
> +
> +	if (!args)
> +		encoders = connectors = crtcs = planes = framebuffers = 1;
> +
> +	dev.fd = util_open(device, module);
> +	if (dev.fd < 0)
> +		return -1;
> +
> +	ret = drmSetClientCap(dev.fd, DRM_CLIENT_CAP_ATOMIC, 1);
> +	if (ret) {
> +		fprintf(stderr, "no atomic modesetting support: %s\n", strerror(errno));
> +		drmClose(dev.fd);
> +		return -1;
> +	}
> +
> +	dev.resources = get_resources(&dev);
> +	if (!dev.resources) {
> +		drmClose(dev.fd);
> +		return 1;
> +	}
> +
> +	for (i = 0; i < count; i++) {
> +		if (pipe_resolve_connectors(&dev, &pipe_args[i]) < 0) {
> +			free_resources(dev.resources);
> +			drmClose(dev.fd);
> +			return 1;
> +		}
> +	}
> +
> +#define dump_resource(dev, res) if (res) dump_##res(dev)
> +
> +	dump_resource(&dev, encoders);
> +	dump_resource(&dev, connectors);
> +	dump_resource(&dev, crtcs);
> +	dump_resource(&dev, planes);
> +	dump_resource(&dev, framebuffers);
> +
> +	dev.req = drmModeAtomicAlloc();
> +
> +	for (i = 0; i < prop_count; ++i)
> +		set_property(&dev, &prop_args[i]);
> +
> +	if (count && plane_count) {
> +		uint64_t cap = 0;
> +
> +		ret = drmGetCap(dev.fd, DRM_CAP_DUMB_BUFFER, &cap);
> +		if (ret || cap == 0) {
> +			fprintf(stderr, "driver doesn't support the dumb buffer API\n");
> +			drmModeAtomicFree(dev.req);
> +			return 1;
> +		}
> +
> +		set_mode(&dev, pipe_args, count);
> +		set_planes(&dev, plane_args, plane_count, false);
> +
> +		ret = drmModeAtomicCommit(dev.fd, dev.req, DRM_MODE_ATOMIC_ALLOW_MODESET, NULL);
> +		if (ret) {
> +			fprintf(stderr, "Atomic Commit failed [1]\n");
> +			return 1;
> +		}
> +
> +		gettimeofday(&pipe_args->start, NULL);
> +		pipe_args->swap_count = 0;
> +
> +		while (test_loop) {
> +			drmModeAtomicFree(dev.req);
> +			dev.req = drmModeAtomicAlloc();
> +			set_planes(&dev, plane_args, plane_count, true);
> +
> +			ret = drmModeAtomicCommit(dev.fd, dev.req, DRM_MODE_ATOMIC_ALLOW_MODESET, NULL);
> +			if (ret) {
> +				fprintf(stderr, "Atomic Commit failed [2]\n");
> +				return 1;
> +			}
> +
> +			pipe_args->swap_count++;
> +			if (pipe_args->swap_count == 60) {
> +				struct timeval end;
> +				double t;
> +
> +				gettimeofday(&end, NULL);
> +				t = end.tv_sec + end.tv_usec * 1e-6 -
> +				    (pipe_args->start.tv_sec + pipe_args->start.tv_usec * 1e-6);
> +				fprintf(stderr, "freq: %.02fHz\n", pipe_args->swap_count / t);
> +				pipe_args->swap_count = 0;
> +				pipe_args->start = end;
> +			}
> +		}
> +
> +		if (drop_master)
> +			drmDropMaster(dev.fd);
> +
> +		getchar();
> +
> +		drmModeAtomicFree(dev.req);
> +		dev.req = drmModeAtomicAlloc();
> +
> +		clear_mode(&dev, pipe_args, count);
> +		clear_planes(&dev, plane_args, plane_count);
> +		ret = drmModeAtomicCommit(dev.fd, dev.req, DRM_MODE_ATOMIC_ALLOW_MODESET, NULL);
> +		if (ret) {
> +			fprintf(stderr, "Atomic Commit failed\n");
> +			return 1;
> +		}
> +
> +		clear_FB(&dev, plane_args, plane_count);
> +	}
> +
> +	drmModeAtomicFree(dev.req);
> +	free_resources(dev.resources);
> +
> +	return 0;
> +}
> -- 
> 2.15.0
> 
> _______________________________________________
> dri-devel mailing list
> dri-devel at lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/dri-devel


More information about the dri-devel mailing list