/** * crt-calibrator – Calibration utility for CRT monitors * Copyright © 2014 Mattias Andrée (maandree@member.fsf.org) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "calibrator.h" #include "gamma.h" #include /** * Draw bars in different shades of grey, red, green and blue * used for calibrating the contrast and brightness * * @return Zero on success, -1 on error */ int draw_contrast_brightness(void) { const int CONTRAST_BRIGHTNESS_LEVELS[21] = { 0, 17, 27, 38, 48, 59, 70, 82, 94, 106, 119, 131, 144, 158, 171, 185, 198, 212, 226, 241, 255 }; size_t f, y, x, fn = fb_count(); for (f = 0; f < fn; f++) { framebuffer_t fb; if (fb_open(f, &fb) < 0) return -1; for (y = 0; y < 4; y++) for (x = 0; x < 21; x++) { int v = CONTRAST_BRIGHTNESS_LEVELS[x]; uint32_t colour = fb_colour(v * ((y == 1) | (y == 0)), v * ((y == 2) | (y == 0)), v * ((y == 3) | (y == 0))); fb_fill_rectangle(&fb, colour, x * fb.width / 21, y * fb.height / 4, (x + 1) * fb.width / 21 - x * fb.width / 21, (y + 1) * fb.height / 4 - y * fb.height / 4); } fb_close(&fb); } return 0; } /** * Draw a seven segment display * * @param fb The framebuffer to draw on * @param colour The intensity of the least intense colour to use * @param x The X component of the top left corner of the seven segment display * @param y The Y component of the top left corner of the seven segment display */ void draw_digit(framebuffer_t* restrict fb, int colour, uint32_t x, uint32_t y) { uint32_t c; c = fb_colour(colour + 0, colour + 0, colour + 0); fb_fill_rectangle(fb, c, x + 20, y, 80, 20); c = fb_colour(colour + 1, colour + 1, colour + 1); fb_fill_rectangle(fb, c, x, y + 20, 20, 80); c = fb_colour(colour + 2, colour + 2, colour + 2); fb_fill_rectangle(fb, c, x + 100, y + 20, 20, 80); c = fb_colour(colour + 3, colour + 3, colour + 3); fb_fill_rectangle(fb, c, x + 20, y + 100, 80, 20); c = fb_colour(colour + 4, colour + 4, colour + 4); fb_fill_rectangle(fb, c, x, y + 120, 20, 80); c = fb_colour(colour + 5, colour + 5, colour + 5); fb_fill_rectangle(fb, c, x + 100, y + 120, 20, 80); c = fb_colour(colour + 6, colour + 6, colour + 6); fb_fill_rectangle(fb, c, x + 20, y + 200, 80, 20); } /** * Manipulate a CRT controllers gamma ramps to display a specific digit * for one of the seven segment display on only that CRT controller's * monitors * * @param crtc The CRT controller information * @param colour The intensity of the least intense colour in the seven segment display * @param value The valud of the digit to display * @return Zero on success, -1 on error */ int gamma_digit(drm_crtc_t* restrict crtc, int colour, size_t value) { #define __ 0 const int DIGITS[11] = { 1 | 2 | 4 | __ | 16 | 32 | 64, /* (0) */ __ | __ | 4 | __ | __ | 32 | __, /* (1) */ 1 | __ | 4 | 8 | 16 | __ | 64, /* (2) */ 1 | __ | 4 | 8 | __ | 32 | 64, /* (3) */ __ | 2 | 4 | 8 | __ | 32 | __, /* (4) */ 1 | 2 | __ | 8 | __ | 32 | 64, /* (5) */ 1 | 2 | __ | 8 | 16 | 32 | 64, /* (6) */ 1 | __ | 4 | __ | __ | 32 | __, /* (7) */ 1 | 2 | 4 | 8 | 16 | 32 | 64, /* (8) */ 1 | 2 | 4 | 8 | __ | 32 | 64, /* (9) */ __ | __ | __ | __ | __ | __ | __}; /* not visible */ int i, digit = DIGITS[value]; for (i = 0; i < 7; i++) { uint16_t value = (digit & (1 << i)) ? 0xFFFF : 0; int j = i + colour; crtc->red[j] = crtc->green[j] = crtc->blue[j] = value; } #undef __ } /** * Draw an unique index on each monitor * * @return Zero on success, -1 on error */ int draw_id(void) { size_t f, c, i, id = 0, fn = fb_count(), cn = drm_card_count(); for (f = 0; f < fn; f++) { framebuffer_t fb; if (fb_open(f, &fb) < 0) return -1; fb_fill_rectangle(&fb, fb_colour(0, 0, 0), 0, 0, fb.width, fb.height); draw_digit(&fb, 1, 40, 40); draw_digit(&fb, 8, 180, 40); fb_close(&fb); } for (c = 0; c < cn; c++) { drm_card_t card; if (drm_card_open(c, &card) < 0) return -1; for (i = 0; i < card.crtc_count; i++) { drm_crtc_t crtc; if (drm_crtc_open(i, &card, &crtc) < 0) { drm_card_close(&card); return -1; } if (crtc.connected == 0) goto not_connected; if (drm_get_gamma(&crtc) < 0) goto crtc_fail; gamma_digit(&crtc, 1, id < 10 ? 10 : (id / 10) % 10); gamma_digit(&crtc, 8, (id / 1) % 10); id++; if (drm_set_gamma(&crtc) < 0) goto crtc_fail; not_connected: drm_crtc_close(&crtc); continue; crtc_fail: drm_crtc_close(&crtc); drm_card_close(&card); return -1; } drm_card_close(&card); } return 0; } /** * Draw squares used as reference when tweeking the gamma correction * * @return Zero on success, -1 on error */ int draw_gamma(void) { size_t f, x, y, fn = fb_count(); for (f = 0; f < fn; f++) { framebuffer_t fb; if (fb_open(f, &fb) < 0) return -1; for (x = 0; x < 4; x++) { int r = (x == 1) || (x == 0); int g = (x == 2) || (x == 0); int b = (x == 3) || (x == 0); uint32_t background = fb_colour(128 * r, 128 * g, 128 * b); uint32_t average = fb_colour(188 * r, 188 * g, 188 * b); uint32_t high = fb_colour(255 * r, 255 * g, 255 * b); uint32_t low = fb_colour(0, 0, 0); uint32_t xoff = x * fb.width / 4; fb_fill_rectangle(&fb, background, xoff, 0, fb.width / 4, fb.height); xoff += (fb.width / 4 - 200) / 2; fb_fill_rectangle(&fb, high, xoff, 40, 200, 200); fb_fill_rectangle(&fb, average, xoff + 50, 40, 100, 200); fb_fill_rectangle(&fb, average, xoff, 280, 200, 200); for (y = 0; y < 200; y += 2) { fb_draw_horizontal_line(&fb, high, xoff + 50, 280 + y + 0, 100); fb_draw_horizontal_line(&fb, low , xoff + 50, 280 + y + 1, 100); } fb_fill_rectangle(&fb, average, xoff, 520, 200, 200); fb_fill_rectangle(&fb, high, xoff + 50, 520, 100, 200); } fb_close(&fb); } return 0; } int main(int argc __attribute__((unused)), char* argv[]) { if (draw_gamma() < 0) goto fail; return 0; fail: perror(*argv); return 1; }