Gerd Hoffmann | daa8e5a | 2012-11-13 09:38:06 +0100 | [diff] [blame] | 1 | /* |
| 2 | * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| 3 | * See the COPYING file in the top-level directory. |
| 4 | */ |
| 5 | |
Peter Maydell | 0b08786 | 2013-01-21 12:03:47 +0000 | [diff] [blame] | 6 | #include "qemu-common.h" |
| 7 | #include "ui/console.h" |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 8 | |
Gerd Hoffmann | a93a3af | 2014-06-18 09:00:00 +0200 | [diff] [blame] | 9 | PixelFormat qemu_pixelformat_from_pixman(pixman_format_code_t format) |
| 10 | { |
| 11 | PixelFormat pf; |
| 12 | uint8_t bpp; |
| 13 | |
| 14 | bpp = pf.bits_per_pixel = PIXMAN_FORMAT_BPP(format); |
| 15 | pf.bytes_per_pixel = PIXMAN_FORMAT_BPP(format) / 8; |
| 16 | pf.depth = PIXMAN_FORMAT_DEPTH(format); |
| 17 | |
| 18 | pf.abits = PIXMAN_FORMAT_A(format); |
| 19 | pf.rbits = PIXMAN_FORMAT_R(format); |
| 20 | pf.gbits = PIXMAN_FORMAT_G(format); |
| 21 | pf.bbits = PIXMAN_FORMAT_B(format); |
| 22 | |
| 23 | switch (PIXMAN_FORMAT_TYPE(format)) { |
| 24 | case PIXMAN_TYPE_ARGB: |
| 25 | pf.ashift = pf.bbits + pf.gbits + pf.rbits; |
| 26 | pf.rshift = pf.bbits + pf.gbits; |
| 27 | pf.gshift = pf.bbits; |
| 28 | pf.bshift = 0; |
| 29 | break; |
| 30 | case PIXMAN_TYPE_ABGR: |
| 31 | pf.ashift = pf.rbits + pf.gbits + pf.bbits; |
| 32 | pf.bshift = pf.rbits + pf.gbits; |
| 33 | pf.gshift = pf.rbits; |
| 34 | pf.rshift = 0; |
| 35 | break; |
| 36 | case PIXMAN_TYPE_BGRA: |
| 37 | pf.bshift = bpp - pf.bbits; |
| 38 | pf.gshift = bpp - (pf.bbits + pf.gbits); |
| 39 | pf.rshift = bpp - (pf.bbits + pf.gbits + pf.rbits); |
| 40 | pf.ashift = 0; |
| 41 | break; |
| 42 | case PIXMAN_TYPE_RGBA: |
| 43 | pf.rshift = bpp - pf.rbits; |
| 44 | pf.gshift = bpp - (pf.rbits + pf.gbits); |
| 45 | pf.bshift = bpp - (pf.rbits + pf.gbits + pf.bbits); |
| 46 | pf.ashift = 0; |
| 47 | break; |
| 48 | default: |
| 49 | g_assert_not_reached(); |
| 50 | break; |
| 51 | } |
| 52 | |
| 53 | pf.amax = (1 << pf.abits) - 1; |
| 54 | pf.rmax = (1 << pf.rbits) - 1; |
| 55 | pf.gmax = (1 << pf.gbits) - 1; |
| 56 | pf.bmax = (1 << pf.bbits) - 1; |
| 57 | pf.amask = pf.amax << pf.ashift; |
| 58 | pf.rmask = pf.rmax << pf.rshift; |
| 59 | pf.gmask = pf.gmax << pf.gshift; |
| 60 | pf.bmask = pf.bmax << pf.bshift; |
| 61 | |
| 62 | return pf; |
| 63 | } |
| 64 | |
Gerd Hoffmann | 1527a25 | 2014-06-18 11:31:42 +0200 | [diff] [blame^] | 65 | pixman_format_code_t qemu_default_pixman_format(int bpp, bool native_endian) |
| 66 | { |
| 67 | if (native_endian) { |
| 68 | switch (bpp) { |
| 69 | case 15: |
| 70 | return PIXMAN_x1r5g5b5; |
| 71 | case 16: |
| 72 | return PIXMAN_r5g6b5; |
| 73 | case 24: |
| 74 | return PIXMAN_r8g8b8; |
| 75 | case 32: |
| 76 | return PIXMAN_x8r8g8b8; |
| 77 | } |
| 78 | } else { |
| 79 | switch (bpp) { |
| 80 | case 24: |
| 81 | return PIXMAN_b8g8r8; |
| 82 | case 32: |
| 83 | return PIXMAN_b8g8r8a8; |
| 84 | break; |
| 85 | } |
| 86 | } |
| 87 | g_assert_not_reached(); |
| 88 | } |
| 89 | |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 90 | int qemu_pixman_get_type(int rshift, int gshift, int bshift) |
| 91 | { |
| 92 | int type = PIXMAN_TYPE_OTHER; |
| 93 | |
| 94 | if (rshift > gshift && gshift > bshift) { |
| 95 | if (bshift == 0) { |
| 96 | type = PIXMAN_TYPE_ARGB; |
| 97 | } else { |
| 98 | #if PIXMAN_VERSION >= PIXMAN_VERSION_ENCODE(0, 21, 8) |
| 99 | type = PIXMAN_TYPE_RGBA; |
| 100 | #endif |
| 101 | } |
| 102 | } else if (rshift < gshift && gshift < bshift) { |
| 103 | if (rshift == 0) { |
| 104 | type = PIXMAN_TYPE_ABGR; |
| 105 | } else { |
Gerd Hoffmann | fbddfc7 | 2012-12-14 08:54:21 +0100 | [diff] [blame] | 106 | #if PIXMAN_VERSION >= PIXMAN_VERSION_ENCODE(0, 16, 0) |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 107 | type = PIXMAN_TYPE_BGRA; |
Alexander Graf | 6e72719 | 2012-11-26 19:49:58 +0100 | [diff] [blame] | 108 | #endif |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 109 | } |
| 110 | } |
| 111 | return type; |
| 112 | } |
| 113 | |
| 114 | pixman_format_code_t qemu_pixman_get_format(PixelFormat *pf) |
| 115 | { |
| 116 | pixman_format_code_t format; |
| 117 | int type; |
| 118 | |
| 119 | type = qemu_pixman_get_type(pf->rshift, pf->gshift, pf->bshift); |
| 120 | format = PIXMAN_FORMAT(pf->bits_per_pixel, type, |
| 121 | pf->abits, pf->rbits, pf->gbits, pf->bbits); |
| 122 | if (!pixman_format_supported_source(format)) { |
| 123 | return 0; |
| 124 | } |
| 125 | return format; |
| 126 | } |
| 127 | |
| 128 | pixman_image_t *qemu_pixman_linebuf_create(pixman_format_code_t format, |
| 129 | int width) |
| 130 | { |
| 131 | pixman_image_t *image = pixman_image_create_bits(format, width, 1, NULL, 0); |
| 132 | assert(image != NULL); |
| 133 | return image; |
| 134 | } |
| 135 | |
| 136 | void qemu_pixman_linebuf_fill(pixman_image_t *linebuf, pixman_image_t *fb, |
Gerd Hoffmann | bc210eb | 2012-12-14 07:54:24 +0000 | [diff] [blame] | 137 | int width, int x, int y) |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 138 | { |
| 139 | pixman_image_composite(PIXMAN_OP_SRC, fb, NULL, linebuf, |
Gerd Hoffmann | bc210eb | 2012-12-14 07:54:24 +0000 | [diff] [blame] | 140 | x, y, 0, 0, 0, 0, width, 1); |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 141 | } |
| 142 | |
Gerd Hoffmann | d9a8656 | 2012-11-02 09:12:49 +0100 | [diff] [blame] | 143 | pixman_image_t *qemu_pixman_mirror_create(pixman_format_code_t format, |
| 144 | pixman_image_t *image) |
| 145 | { |
| 146 | pixman_image_t *mirror; |
| 147 | |
| 148 | mirror = pixman_image_create_bits(format, |
| 149 | pixman_image_get_width(image), |
| 150 | pixman_image_get_height(image), |
| 151 | NULL, |
| 152 | pixman_image_get_stride(image)); |
| 153 | return mirror; |
| 154 | } |
| 155 | |
Gerd Hoffmann | d2ec7e2 | 2012-09-25 16:23:24 +0200 | [diff] [blame] | 156 | void qemu_pixman_image_unref(pixman_image_t *image) |
| 157 | { |
| 158 | if (image == NULL) { |
| 159 | return; |
| 160 | } |
| 161 | pixman_image_unref(image); |
| 162 | } |
Gerd Hoffmann | 867c538 | 2013-03-06 14:14:17 +0100 | [diff] [blame] | 163 | |
| 164 | pixman_color_t qemu_pixman_color(PixelFormat *pf, uint32_t color) |
| 165 | { |
| 166 | pixman_color_t c; |
| 167 | |
| 168 | c.red = ((color & pf->rmask) >> pf->rshift) << (16 - pf->rbits); |
| 169 | c.green = ((color & pf->gmask) >> pf->gshift) << (16 - pf->gbits); |
| 170 | c.blue = ((color & pf->bmask) >> pf->bshift) << (16 - pf->bbits); |
| 171 | c.alpha = ((color & pf->amask) >> pf->ashift) << (16 - pf->abits); |
| 172 | return c; |
| 173 | } |
Gerd Hoffmann | b762795 | 2013-03-07 15:23:48 +0100 | [diff] [blame] | 174 | |
| 175 | pixman_image_t *qemu_pixman_glyph_from_vgafont(int height, const uint8_t *font, |
| 176 | unsigned int ch) |
| 177 | { |
| 178 | pixman_image_t *glyph; |
| 179 | uint8_t *data; |
| 180 | bool bit; |
| 181 | int x, y; |
| 182 | |
| 183 | glyph = pixman_image_create_bits(PIXMAN_a8, 8, height, |
| 184 | NULL, 0); |
| 185 | data = (uint8_t *)pixman_image_get_data(glyph); |
| 186 | |
| 187 | font += height * ch; |
| 188 | for (y = 0; y < height; y++, font++) { |
| 189 | for (x = 0; x < 8; x++, data++) { |
| 190 | bit = (*font) & (1 << (7-x)); |
| 191 | *data = bit ? 0xff : 0x00; |
| 192 | } |
| 193 | } |
| 194 | return glyph; |
| 195 | } |
| 196 | |
| 197 | void qemu_pixman_glyph_render(pixman_image_t *glyph, |
| 198 | pixman_image_t *surface, |
| 199 | pixman_color_t *fgcol, |
| 200 | pixman_color_t *bgcol, |
| 201 | int x, int y, int cw, int ch) |
| 202 | { |
| 203 | pixman_image_t *ifg = pixman_image_create_solid_fill(fgcol); |
| 204 | pixman_image_t *ibg = pixman_image_create_solid_fill(bgcol); |
| 205 | |
| 206 | pixman_image_composite(PIXMAN_OP_SRC, ibg, NULL, surface, |
| 207 | 0, 0, 0, 0, |
| 208 | cw * x, ch * y, |
| 209 | cw, ch); |
| 210 | pixman_image_composite(PIXMAN_OP_OVER, ifg, glyph, surface, |
| 211 | 0, 0, 0, 0, |
| 212 | cw * x, ch * y, |
| 213 | cw, ch); |
| 214 | pixman_image_unref(ifg); |
| 215 | pixman_image_unref(ibg); |
| 216 | } |