1*7688df22SAndroid Build Coastguard Worker /*
2*7688df22SAndroid Build Coastguard Worker * Copyright 2008 Tungsten Graphics
3*7688df22SAndroid Build Coastguard Worker * Jakob Bornecrantz <[email protected]>
4*7688df22SAndroid Build Coastguard Worker * Copyright 2008 Intel Corporation
5*7688df22SAndroid Build Coastguard Worker * Jesse Barnes <[email protected]>
6*7688df22SAndroid Build Coastguard Worker *
7*7688df22SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
8*7688df22SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
9*7688df22SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
10*7688df22SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11*7688df22SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
12*7688df22SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
13*7688df22SAndroid Build Coastguard Worker *
14*7688df22SAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included in
15*7688df22SAndroid Build Coastguard Worker * all copies or substantial portions of the Software.
16*7688df22SAndroid Build Coastguard Worker *
17*7688df22SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18*7688df22SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19*7688df22SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20*7688df22SAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21*7688df22SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22*7688df22SAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
23*7688df22SAndroid Build Coastguard Worker * IN THE SOFTWARE.
24*7688df22SAndroid Build Coastguard Worker */
25*7688df22SAndroid Build Coastguard Worker
26*7688df22SAndroid Build Coastguard Worker /*
27*7688df22SAndroid Build Coastguard Worker * This fairly simple test program dumps output in a similar format to the
28*7688df22SAndroid Build Coastguard Worker * "xrandr" tool everyone knows & loves. It's necessarily slightly different
29*7688df22SAndroid Build Coastguard Worker * since the kernel separates outputs into encoder and connector structures,
30*7688df22SAndroid Build Coastguard Worker * each with their own unique ID. The program also allows test testing of the
31*7688df22SAndroid Build Coastguard Worker * memory management and mode setting APIs by allowing the user to specify a
32*7688df22SAndroid Build Coastguard Worker * connector and mode to use for mode setting. If all works as expected, a
33*7688df22SAndroid Build Coastguard Worker * blue background should be painted on the monitor attached to the specified
34*7688df22SAndroid Build Coastguard Worker * connector after the selected mode is set.
35*7688df22SAndroid Build Coastguard Worker *
36*7688df22SAndroid Build Coastguard Worker * TODO: use cairo to write the mode info on the selected output once
37*7688df22SAndroid Build Coastguard Worker * the mode has been programmed, along with possible test patterns.
38*7688df22SAndroid Build Coastguard Worker */
39*7688df22SAndroid Build Coastguard Worker
40*7688df22SAndroid Build Coastguard Worker #include <errno.h>
41*7688df22SAndroid Build Coastguard Worker #include <stdint.h>
42*7688df22SAndroid Build Coastguard Worker #include <stdio.h>
43*7688df22SAndroid Build Coastguard Worker #include <stdlib.h>
44*7688df22SAndroid Build Coastguard Worker #include <string.h>
45*7688df22SAndroid Build Coastguard Worker #include <fcntl.h>
46*7688df22SAndroid Build Coastguard Worker #include <unistd.h>
47*7688df22SAndroid Build Coastguard Worker
48*7688df22SAndroid Build Coastguard Worker #include "xf86drm.h"
49*7688df22SAndroid Build Coastguard Worker #include "xf86drmMode.h"
50*7688df22SAndroid Build Coastguard Worker
51*7688df22SAndroid Build Coastguard Worker #include "common.h"
52*7688df22SAndroid Build Coastguard Worker
53*7688df22SAndroid Build Coastguard Worker struct type_name {
54*7688df22SAndroid Build Coastguard Worker unsigned int type;
55*7688df22SAndroid Build Coastguard Worker const char *name;
56*7688df22SAndroid Build Coastguard Worker };
57*7688df22SAndroid Build Coastguard Worker
util_lookup_type_name(unsigned int type,const struct type_name * table,unsigned int count)58*7688df22SAndroid Build Coastguard Worker static const char *util_lookup_type_name(unsigned int type,
59*7688df22SAndroid Build Coastguard Worker const struct type_name *table,
60*7688df22SAndroid Build Coastguard Worker unsigned int count)
61*7688df22SAndroid Build Coastguard Worker {
62*7688df22SAndroid Build Coastguard Worker unsigned int i;
63*7688df22SAndroid Build Coastguard Worker
64*7688df22SAndroid Build Coastguard Worker for (i = 0; i < count; i++)
65*7688df22SAndroid Build Coastguard Worker if (table[i].type == type)
66*7688df22SAndroid Build Coastguard Worker return table[i].name;
67*7688df22SAndroid Build Coastguard Worker
68*7688df22SAndroid Build Coastguard Worker return NULL;
69*7688df22SAndroid Build Coastguard Worker }
70*7688df22SAndroid Build Coastguard Worker
71*7688df22SAndroid Build Coastguard Worker static const struct type_name encoder_type_names[] = {
72*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_NONE, "none" },
73*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_DAC, "DAC" },
74*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_TMDS, "TMDS" },
75*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_LVDS, "LVDS" },
76*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_TVDAC, "TVDAC" },
77*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_VIRTUAL, "Virtual" },
78*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_DSI, "DSI" },
79*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_DPMST, "DPMST" },
80*7688df22SAndroid Build Coastguard Worker { DRM_MODE_ENCODER_DPI, "DPI" },
81*7688df22SAndroid Build Coastguard Worker };
82*7688df22SAndroid Build Coastguard Worker
util_lookup_encoder_type_name(unsigned int type)83*7688df22SAndroid Build Coastguard Worker const char *util_lookup_encoder_type_name(unsigned int type)
84*7688df22SAndroid Build Coastguard Worker {
85*7688df22SAndroid Build Coastguard Worker return util_lookup_type_name(type, encoder_type_names,
86*7688df22SAndroid Build Coastguard Worker ARRAY_SIZE(encoder_type_names));
87*7688df22SAndroid Build Coastguard Worker }
88*7688df22SAndroid Build Coastguard Worker
89*7688df22SAndroid Build Coastguard Worker static const struct type_name connector_status_names[] = {
90*7688df22SAndroid Build Coastguard Worker { DRM_MODE_CONNECTED, "connected" },
91*7688df22SAndroid Build Coastguard Worker { DRM_MODE_DISCONNECTED, "disconnected" },
92*7688df22SAndroid Build Coastguard Worker { DRM_MODE_UNKNOWNCONNECTION, "unknown" },
93*7688df22SAndroid Build Coastguard Worker };
94*7688df22SAndroid Build Coastguard Worker
util_lookup_connector_status_name(unsigned int status)95*7688df22SAndroid Build Coastguard Worker const char *util_lookup_connector_status_name(unsigned int status)
96*7688df22SAndroid Build Coastguard Worker {
97*7688df22SAndroid Build Coastguard Worker return util_lookup_type_name(status, connector_status_names,
98*7688df22SAndroid Build Coastguard Worker ARRAY_SIZE(connector_status_names));
99*7688df22SAndroid Build Coastguard Worker }
100*7688df22SAndroid Build Coastguard Worker
util_open(const char * device,const char * module)101*7688df22SAndroid Build Coastguard Worker int util_open(const char *device, const char *module)
102*7688df22SAndroid Build Coastguard Worker {
103*7688df22SAndroid Build Coastguard Worker int fd = -1;
104*7688df22SAndroid Build Coastguard Worker drmVersionPtr version;
105*7688df22SAndroid Build Coastguard Worker
106*7688df22SAndroid Build Coastguard Worker if (module || device) {
107*7688df22SAndroid Build Coastguard Worker fd = drmOpen(module, device);
108*7688df22SAndroid Build Coastguard Worker if (fd < 0) {
109*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "failed to open device '%s' with busid '%s': %s\n",
110*7688df22SAndroid Build Coastguard Worker module, device, strerror(errno));
111*7688df22SAndroid Build Coastguard Worker return -errno;
112*7688df22SAndroid Build Coastguard Worker }
113*7688df22SAndroid Build Coastguard Worker } else {
114*7688df22SAndroid Build Coastguard Worker unsigned int i;
115*7688df22SAndroid Build Coastguard Worker drmDevicePtr devices[64];
116*7688df22SAndroid Build Coastguard Worker int num_devices = drmGetDevices2(0, devices, ARRAY_SIZE(devices));
117*7688df22SAndroid Build Coastguard Worker if (num_devices < 0) {
118*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "drmGetDevices2() failed with %s\n", strerror(num_devices));
119*7688df22SAndroid Build Coastguard Worker return num_devices;
120*7688df22SAndroid Build Coastguard Worker }
121*7688df22SAndroid Build Coastguard Worker
122*7688df22SAndroid Build Coastguard Worker for (i = 0; i < num_devices; i++) {
123*7688df22SAndroid Build Coastguard Worker drmDevicePtr device = devices[i];
124*7688df22SAndroid Build Coastguard Worker // Select only primary nodes
125*7688df22SAndroid Build Coastguard Worker if ((device->available_nodes & 1 << DRM_NODE_PRIMARY) == 0)
126*7688df22SAndroid Build Coastguard Worker continue;
127*7688df22SAndroid Build Coastguard Worker
128*7688df22SAndroid Build Coastguard Worker printf("trying to open device '%s'... ", device->nodes[DRM_NODE_PRIMARY]);
129*7688df22SAndroid Build Coastguard Worker fd = open(device->nodes[DRM_NODE_PRIMARY], O_RDWR | O_CLOEXEC);
130*7688df22SAndroid Build Coastguard Worker
131*7688df22SAndroid Build Coastguard Worker if (fd < 0) {
132*7688df22SAndroid Build Coastguard Worker printf("failed\n");
133*7688df22SAndroid Build Coastguard Worker } else if (!drmIsKMS(fd)) {
134*7688df22SAndroid Build Coastguard Worker printf("is not a KMS device\n");
135*7688df22SAndroid Build Coastguard Worker close(fd);
136*7688df22SAndroid Build Coastguard Worker fd = -1;
137*7688df22SAndroid Build Coastguard Worker } else {
138*7688df22SAndroid Build Coastguard Worker printf("done\n");
139*7688df22SAndroid Build Coastguard Worker break;
140*7688df22SAndroid Build Coastguard Worker }
141*7688df22SAndroid Build Coastguard Worker }
142*7688df22SAndroid Build Coastguard Worker
143*7688df22SAndroid Build Coastguard Worker if (fd < 0) {
144*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "no device found\n");
145*7688df22SAndroid Build Coastguard Worker return -ENODEV;
146*7688df22SAndroid Build Coastguard Worker }
147*7688df22SAndroid Build Coastguard Worker }
148*7688df22SAndroid Build Coastguard Worker
149*7688df22SAndroid Build Coastguard Worker version = drmGetVersion(fd);
150*7688df22SAndroid Build Coastguard Worker printf("opened device `%s` on driver `%s` (version %d.%d.%d at %s)\n",
151*7688df22SAndroid Build Coastguard Worker version->desc,
152*7688df22SAndroid Build Coastguard Worker version->name,
153*7688df22SAndroid Build Coastguard Worker version->version_major,
154*7688df22SAndroid Build Coastguard Worker version->version_minor,
155*7688df22SAndroid Build Coastguard Worker version->version_patchlevel,
156*7688df22SAndroid Build Coastguard Worker version->date);
157*7688df22SAndroid Build Coastguard Worker drmFreeVersion(version);
158*7688df22SAndroid Build Coastguard Worker
159*7688df22SAndroid Build Coastguard Worker return fd;
160*7688df22SAndroid Build Coastguard Worker }
161