| /* |
| * \file modedemo.c |
| * Test program to dump DRM kernel mode setting related information. |
| * Queries the kernel for all available information and dumps it to stdout. |
| * |
| * \author Jakob Bornecrantz <wallbraker@gmail.com> |
| */ |
| |
| /* |
| * Copyright (c) 2007-2008 Tungsten Graphics, Inc., Cedar Park, Texas. |
| * Copyright (c) 2007-2008 Jakob Bornecrantz <wallbraker@gmail.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. |
| * |
| */ |
| |
| #include <assert.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <stdint.h> |
| #include <unistd.h> |
| #include <string.h> |
| #include <inttypes.h> |
| |
| #include "xf86drm.h" |
| #include "xf86drmMode.h" |
| |
| #include "util/common.h" |
| |
| int current; |
| int connectors; |
| int full_props; |
| int edid; |
| int modes; |
| int full_modes; |
| int encoders; |
| int crtcs; |
| int fbs; |
| char *module_name; |
| |
| static const char* getConnectionText(drmModeConnection conn) |
| { |
| switch (conn) { |
| case DRM_MODE_CONNECTED: |
| return "connected"; |
| case DRM_MODE_DISCONNECTED: |
| return "disconnected"; |
| case DRM_MODE_UNKNOWNCONNECTION: |
| default: |
| return "unknown"; |
| } |
| |
| } |
| |
| static int printMode(struct drm_mode_modeinfo *mode) |
| { |
| if (full_modes) { |
| printf("Mode: %s\n", mode->name); |
| printf("\tclock : %i\n", mode->clock); |
| printf("\thdisplay : %i\n", mode->hdisplay); |
| printf("\thsync_start : %i\n", mode->hsync_start); |
| printf("\thsync_end : %i\n", mode->hsync_end); |
| printf("\thtotal : %i\n", mode->htotal); |
| printf("\thskew : %i\n", mode->hskew); |
| printf("\tvdisplay : %i\n", mode->vdisplay); |
| printf("\tvsync_start : %i\n", mode->vsync_start); |
| printf("\tvsync_end : %i\n", mode->vsync_end); |
| printf("\tvtotal : %i\n", mode->vtotal); |
| printf("\tvscan : %i\n", mode->vscan); |
| printf("\tvrefresh : %i\n", mode->vrefresh); |
| printf("\tflags : %i\n", mode->flags); |
| } else { |
| printf("Mode: \"%s\" %ix%i %i\n", mode->name, |
| mode->hdisplay, mode->vdisplay, mode->vrefresh); |
| } |
| return 0; |
| } |
| |
| static int printProperty(int fd, drmModeResPtr res, drmModePropertyPtr props, uint64_t value) |
| { |
| const char *name = NULL; |
| int j; |
| |
| printf("Property: %s\n", props->name); |
| printf("\tid : %i\n", props->prop_id); |
| printf("\tflags : %i\n", props->flags); |
| printf("\tcount_values : %d\n", props->count_values); |
| |
| |
| if (props->count_values) { |
| printf("\tvalues :"); |
| for (j = 0; j < props->count_values; j++) |
| printf(" %" PRIu64, props->values[j]); |
| printf("\n"); |
| } |
| |
| |
| printf("\tcount_enums : %d\n", props->count_enums); |
| |
| if (props->flags & DRM_MODE_PROP_BLOB) { |
| drmModePropertyBlobPtr blob; |
| |
| blob = drmModeGetPropertyBlob(fd, value); |
| if (blob) { |
| printf("blob is %d length, %08X\n", blob->length, *(uint32_t *)blob->data); |
| drmModeFreePropertyBlob(blob); |
| } else { |
| printf("error getting blob %" PRIu64 "\n", value); |
| } |
| |
| } else { |
| for (j = 0; j < props->count_enums; j++) { |
| printf("\t\t%lld = %s\n", props->enums[j].value, props->enums[j].name); |
| if (props->enums[j].value == value) |
| name = props->enums[j].name; |
| } |
| |
| if (props->count_enums && name) { |
| printf("\tcon_value : %s\n", name); |
| } else { |
| printf("\tcon_value : %" PRIu64 "\n", value); |
| } |
| } |
| |
| return 0; |
| } |
| |
| static const char * const output_names[] = { "None", |
| "VGA", |
| "DVI-I", |
| "DVI-D", |
| "DVI-A", |
| "Composite", |
| "SVIDEO", |
| "LVDS", |
| "Component", |
| "DIN", |
| "DP", |
| "HDMI-A", |
| "HDMI-B", |
| "TV", |
| "eDP", |
| "Virtual", |
| "DSI", |
| }; |
| |
| static int printConnector(int fd, drmModeResPtr res, drmModeConnectorPtr connector, uint32_t id) |
| { |
| int i = 0; |
| struct drm_mode_modeinfo *mode = NULL; |
| drmModePropertyPtr props; |
| |
| if (connector->connector_type < ARRAY_SIZE(output_names)) |
| printf("Connector: %s-%d\n", output_names[connector->connector_type], |
| connector->connector_type_id); |
| else |
| printf("Connector: %d-%d\n", connector->connector_type, |
| connector->connector_type_id); |
| printf("\tid : %i\n", id); |
| printf("\tencoder id : %i\n", connector->encoder_id); |
| printf("\tconn : %s\n", getConnectionText(connector->connection)); |
| printf("\tsize : %ix%i (mm)\n", connector->mmWidth, connector->mmHeight); |
| printf("\tcount_modes : %i\n", connector->count_modes); |
| printf("\tcount_props : %i\n", connector->count_props); |
| if (connector->count_props) { |
| printf("\tprops :"); |
| for (i = 0; i < connector->count_props; i++) |
| printf(" %i", connector->props[i]); |
| printf("\n"); |
| } |
| |
| printf("\tcount_encoders : %i\n", connector->count_encoders); |
| if (connector->count_encoders) { |
| printf("\tencoders :"); |
| for (i = 0; i < connector->count_encoders; i++) |
| printf(" %i", connector->encoders[i]); |
| printf("\n"); |
| } |
| |
| if (modes) { |
| for (i = 0; i < connector->count_modes; i++) { |
| mode = (struct drm_mode_modeinfo *)&connector->modes[i]; |
| printMode(mode); |
| } |
| } |
| |
| if (full_props) { |
| for (i = 0; i < connector->count_props; i++) { |
| props = drmModeGetProperty(fd, connector->props[i]); |
| if (props) { |
| printProperty(fd, res, props, connector->prop_values[i]); |
| drmModeFreeProperty(props); |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| static int printEncoder(int fd, drmModeResPtr res, drmModeEncoderPtr encoder, uint32_t id) |
| { |
| printf("Encoder\n"); |
| printf("\tid :%i\n", id); |
| printf("\tcrtc_id :%d\n", encoder->crtc_id); |
| printf("\ttype :%d\n", encoder->encoder_type); |
| printf("\tpossible_crtcs :0x%x\n", encoder->possible_crtcs); |
| printf("\tpossible_clones :0x%x\n", encoder->possible_clones); |
| return 0; |
| } |
| |
| static int printCrtc(int fd, drmModeResPtr res, drmModeCrtcPtr crtc, uint32_t id) |
| { |
| printf("Crtc\n"); |
| printf("\tid : %i\n", id); |
| printf("\tx : %i\n", crtc->x); |
| printf("\ty : %i\n", crtc->y); |
| printf("\twidth : %i\n", crtc->width); |
| printf("\theight : %i\n", crtc->height); |
| printf("\tmode : %p\n", &crtc->mode); |
| printf("\tgamma size : %d\n", crtc->gamma_size); |
| |
| return 0; |
| } |
| |
| static int printFrameBuffer(int fd, drmModeResPtr res, drmModeFBPtr fb) |
| { |
| printf("Framebuffer\n"); |
| printf("\thandle : %i\n", fb->handle); |
| printf("\twidth : %i\n", fb->width); |
| printf("\theight : %i\n", fb->height); |
| printf("\tpitch : %i\n", fb->pitch);; |
| printf("\tbpp : %i\n", fb->bpp); |
| printf("\tdepth : %i\n", fb->depth); |
| printf("\tbuffer_id : %i\n", fb->handle); |
| |
| return 0; |
| } |
| |
| static int printRes(int fd, drmModeResPtr res) |
| { |
| int i; |
| drmModeFBPtr fb; |
| drmModeCrtcPtr crtc; |
| drmModeEncoderPtr encoder; |
| drmModeConnectorPtr connector; |
| |
| printf("Resources\n\n"); |
| |
| printf("count_connectors : %i\n", res->count_connectors); |
| printf("count_encoders : %i\n", res->count_encoders); |
| printf("count_crtcs : %i\n", res->count_crtcs); |
| printf("count_fbs : %i\n", res->count_fbs); |
| |
| printf("\n"); |
| |
| if (connectors) { |
| for (i = 0; i < res->count_connectors; i++) { |
| connector = (current ? drmModeGetConnectorCurrent : drmModeGetConnector) (fd, res->connectors[i]); |
| |
| if (!connector) |
| printf("Could not get connector %i\n", res->connectors[i]); |
| else { |
| printConnector(fd, res, connector, res->connectors[i]); |
| drmModeFreeConnector(connector); |
| } |
| } |
| printf("\n"); |
| } |
| |
| |
| if (encoders) { |
| for (i = 0; i < res->count_encoders; i++) { |
| encoder = drmModeGetEncoder(fd, res->encoders[i]); |
| |
| if (!encoder) |
| printf("Could not get encoder %i\n", res->encoders[i]); |
| else { |
| printEncoder(fd, res, encoder, res->encoders[i]); |
| drmModeFreeEncoder(encoder); |
| } |
| } |
| printf("\n"); |
| } |
| |
| if (crtcs) { |
| for (i = 0; i < res->count_crtcs; i++) { |
| crtc = drmModeGetCrtc(fd, res->crtcs[i]); |
| |
| if (!crtc) |
| printf("Could not get crtc %i\n", res->crtcs[i]); |
| else { |
| printCrtc(fd, res, crtc, res->crtcs[i]); |
| drmModeFreeCrtc(crtc); |
| } |
| } |
| printf("\n"); |
| } |
| |
| if (fbs) { |
| for (i = 0; i < res->count_fbs; i++) { |
| fb = drmModeGetFB(fd, res->fbs[i]); |
| |
| if (!fb) |
| printf("Could not get fb %i\n", res->fbs[i]); |
| else { |
| printFrameBuffer(fd, res, fb); |
| drmModeFreeFB(fb); |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| static void args(int argc, char **argv) |
| { |
| int defaults = 1; |
| int i; |
| |
| fbs = 0; |
| edid = 0; |
| crtcs = 0; |
| modes = 0; |
| encoders = 0; |
| full_modes = 0; |
| full_props = 0; |
| connectors = 0; |
| current = 0; |
| |
| module_name = argv[1]; |
| |
| for (i = 2; i < argc; i++) { |
| if (strcmp(argv[i], "-fb") == 0) { |
| fbs = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-crtcs") == 0) { |
| crtcs = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-cons") == 0) { |
| connectors = 1; |
| modes = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-modes") == 0) { |
| connectors = 1; |
| modes = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-full") == 0) { |
| connectors = 1; |
| modes = 1; |
| full_modes = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-props") == 0) { |
| connectors = 1; |
| full_props = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-edids") == 0) { |
| connectors = 1; |
| edid = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-encoders") == 0) { |
| encoders = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-v") == 0) { |
| fbs = 1; |
| edid = 1; |
| crtcs = 1; |
| modes = 1; |
| encoders = 1; |
| full_modes = 1; |
| full_props = 1; |
| connectors = 1; |
| defaults = 0; |
| } else if (strcmp(argv[i], "-current") == 0) { |
| current = 1; |
| } |
| } |
| |
| if (defaults) { |
| fbs = 1; |
| edid = 1; |
| crtcs = 1; |
| modes = 1; |
| encoders = 1; |
| full_modes = 0; |
| full_props = 0; |
| connectors = 1; |
| } |
| } |
| |
| int main(int argc, char **argv) |
| { |
| int fd; |
| drmModeResPtr res; |
| |
| if (argc == 1) { |
| printf("Please add modulename as first argument\n"); |
| return 1; |
| } |
| |
| args(argc, argv); |
| |
| printf("Starting test\n"); |
| |
| fd = drmOpen(module_name, NULL); |
| |
| if (fd < 0) { |
| printf("Failed to open the card fd (%d)\n",fd); |
| return 1; |
| } |
| |
| res = drmModeGetResources(fd); |
| if (res == 0) { |
| printf("Failed to get resources from card\n"); |
| drmClose(fd); |
| return 1; |
| } |
| |
| printRes(fd, res); |
| |
| drmModeFreeResources(res); |
| |
| printf("Ok\n"); |
| |
| return 0; |
| } |