2
0

qemu-pixman.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  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. #include "qemu/osdep.h"
  6. #include "qemu-common.h"
  7. #include "ui/console.h"
  8. PixelFormat qemu_pixelformat_from_pixman(pixman_format_code_t format)
  9. {
  10. PixelFormat pf;
  11. uint8_t bpp;
  12. bpp = pf.bits_per_pixel = PIXMAN_FORMAT_BPP(format);
  13. pf.bytes_per_pixel = PIXMAN_FORMAT_BPP(format) / 8;
  14. pf.depth = PIXMAN_FORMAT_DEPTH(format);
  15. pf.abits = PIXMAN_FORMAT_A(format);
  16. pf.rbits = PIXMAN_FORMAT_R(format);
  17. pf.gbits = PIXMAN_FORMAT_G(format);
  18. pf.bbits = PIXMAN_FORMAT_B(format);
  19. switch (PIXMAN_FORMAT_TYPE(format)) {
  20. case PIXMAN_TYPE_ARGB:
  21. pf.ashift = pf.bbits + pf.gbits + pf.rbits;
  22. pf.rshift = pf.bbits + pf.gbits;
  23. pf.gshift = pf.bbits;
  24. pf.bshift = 0;
  25. break;
  26. case PIXMAN_TYPE_ABGR:
  27. pf.ashift = pf.rbits + pf.gbits + pf.bbits;
  28. pf.bshift = pf.rbits + pf.gbits;
  29. pf.gshift = pf.rbits;
  30. pf.rshift = 0;
  31. break;
  32. case PIXMAN_TYPE_BGRA:
  33. pf.bshift = bpp - pf.bbits;
  34. pf.gshift = bpp - (pf.bbits + pf.gbits);
  35. pf.rshift = bpp - (pf.bbits + pf.gbits + pf.rbits);
  36. pf.ashift = 0;
  37. break;
  38. case PIXMAN_TYPE_RGBA:
  39. pf.rshift = bpp - pf.rbits;
  40. pf.gshift = bpp - (pf.rbits + pf.gbits);
  41. pf.bshift = bpp - (pf.rbits + pf.gbits + pf.bbits);
  42. pf.ashift = 0;
  43. break;
  44. default:
  45. g_assert_not_reached();
  46. break;
  47. }
  48. pf.amax = (1 << pf.abits) - 1;
  49. pf.rmax = (1 << pf.rbits) - 1;
  50. pf.gmax = (1 << pf.gbits) - 1;
  51. pf.bmax = (1 << pf.bbits) - 1;
  52. pf.amask = pf.amax << pf.ashift;
  53. pf.rmask = pf.rmax << pf.rshift;
  54. pf.gmask = pf.gmax << pf.gshift;
  55. pf.bmask = pf.bmax << pf.bshift;
  56. return pf;
  57. }
  58. pixman_format_code_t qemu_default_pixman_format(int bpp, bool native_endian)
  59. {
  60. if (native_endian) {
  61. switch (bpp) {
  62. case 15:
  63. return PIXMAN_x1r5g5b5;
  64. case 16:
  65. return PIXMAN_r5g6b5;
  66. case 24:
  67. return PIXMAN_r8g8b8;
  68. case 32:
  69. return PIXMAN_x8r8g8b8;
  70. }
  71. } else {
  72. switch (bpp) {
  73. case 24:
  74. return PIXMAN_b8g8r8;
  75. case 32:
  76. return PIXMAN_b8g8r8x8;
  77. break;
  78. }
  79. }
  80. return 0;
  81. }
  82. int qemu_pixman_get_type(int rshift, int gshift, int bshift)
  83. {
  84. int type = PIXMAN_TYPE_OTHER;
  85. if (rshift > gshift && gshift > bshift) {
  86. if (bshift == 0) {
  87. type = PIXMAN_TYPE_ARGB;
  88. } else {
  89. #if PIXMAN_VERSION >= PIXMAN_VERSION_ENCODE(0, 21, 8)
  90. type = PIXMAN_TYPE_RGBA;
  91. #endif
  92. }
  93. } else if (rshift < gshift && gshift < bshift) {
  94. if (rshift == 0) {
  95. type = PIXMAN_TYPE_ABGR;
  96. } else {
  97. #if PIXMAN_VERSION >= PIXMAN_VERSION_ENCODE(0, 16, 0)
  98. type = PIXMAN_TYPE_BGRA;
  99. #endif
  100. }
  101. }
  102. return type;
  103. }
  104. pixman_format_code_t qemu_pixman_get_format(PixelFormat *pf)
  105. {
  106. pixman_format_code_t format;
  107. int type;
  108. type = qemu_pixman_get_type(pf->rshift, pf->gshift, pf->bshift);
  109. format = PIXMAN_FORMAT(pf->bits_per_pixel, type,
  110. pf->abits, pf->rbits, pf->gbits, pf->bbits);
  111. if (!pixman_format_supported_source(format)) {
  112. return 0;
  113. }
  114. return format;
  115. }
  116. /*
  117. * Return true for known-good pixman conversions.
  118. *
  119. * UIs using pixman for format conversion can hook this into
  120. * DisplayChangeListenerOps->dpy_gfx_check_format
  121. */
  122. bool qemu_pixman_check_format(DisplayChangeListener *dcl,
  123. pixman_format_code_t format)
  124. {
  125. switch (format) {
  126. /* 32 bpp */
  127. case PIXMAN_x8r8g8b8:
  128. case PIXMAN_a8r8g8b8:
  129. case PIXMAN_b8g8r8x8:
  130. case PIXMAN_b8g8r8a8:
  131. /* 24 bpp */
  132. case PIXMAN_r8g8b8:
  133. case PIXMAN_b8g8r8:
  134. /* 16 bpp */
  135. case PIXMAN_x1r5g5b5:
  136. case PIXMAN_r5g6b5:
  137. return true;
  138. default:
  139. return false;
  140. }
  141. }
  142. pixman_image_t *qemu_pixman_linebuf_create(pixman_format_code_t format,
  143. int width)
  144. {
  145. pixman_image_t *image = pixman_image_create_bits(format, width, 1, NULL, 0);
  146. assert(image != NULL);
  147. return image;
  148. }
  149. /* fill linebuf from framebuffer */
  150. void qemu_pixman_linebuf_fill(pixman_image_t *linebuf, pixman_image_t *fb,
  151. int width, int x, int y)
  152. {
  153. pixman_image_composite(PIXMAN_OP_SRC, fb, NULL, linebuf,
  154. x, y, 0, 0, 0, 0, width, 1);
  155. }
  156. /* copy linebuf to framebuffer */
  157. void qemu_pixman_linebuf_copy(pixman_image_t *fb, int width, int x, int y,
  158. pixman_image_t *linebuf)
  159. {
  160. pixman_image_composite(PIXMAN_OP_SRC, linebuf, NULL, fb,
  161. 0, 0, 0, 0, x, y, width, 1);
  162. }
  163. pixman_image_t *qemu_pixman_mirror_create(pixman_format_code_t format,
  164. pixman_image_t *image)
  165. {
  166. return pixman_image_create_bits(format,
  167. pixman_image_get_width(image),
  168. pixman_image_get_height(image),
  169. NULL,
  170. pixman_image_get_stride(image));
  171. }
  172. void qemu_pixman_image_unref(pixman_image_t *image)
  173. {
  174. if (image == NULL) {
  175. return;
  176. }
  177. pixman_image_unref(image);
  178. }
  179. pixman_color_t qemu_pixman_color(PixelFormat *pf, uint32_t color)
  180. {
  181. pixman_color_t c;
  182. c.red = ((color & pf->rmask) >> pf->rshift) << (16 - pf->rbits);
  183. c.green = ((color & pf->gmask) >> pf->gshift) << (16 - pf->gbits);
  184. c.blue = ((color & pf->bmask) >> pf->bshift) << (16 - pf->bbits);
  185. c.alpha = ((color & pf->amask) >> pf->ashift) << (16 - pf->abits);
  186. return c;
  187. }
  188. pixman_image_t *qemu_pixman_glyph_from_vgafont(int height, const uint8_t *font,
  189. unsigned int ch)
  190. {
  191. pixman_image_t *glyph;
  192. uint8_t *data;
  193. bool bit;
  194. int x, y;
  195. glyph = pixman_image_create_bits(PIXMAN_a8, 8, height,
  196. NULL, 0);
  197. data = (uint8_t *)pixman_image_get_data(glyph);
  198. font += height * ch;
  199. for (y = 0; y < height; y++, font++) {
  200. for (x = 0; x < 8; x++, data++) {
  201. bit = (*font) & (1 << (7-x));
  202. *data = bit ? 0xff : 0x00;
  203. }
  204. }
  205. return glyph;
  206. }
  207. void qemu_pixman_glyph_render(pixman_image_t *glyph,
  208. pixman_image_t *surface,
  209. pixman_color_t *fgcol,
  210. pixman_color_t *bgcol,
  211. int x, int y, int cw, int ch)
  212. {
  213. pixman_image_t *ifg = pixman_image_create_solid_fill(fgcol);
  214. pixman_image_t *ibg = pixman_image_create_solid_fill(bgcol);
  215. pixman_image_composite(PIXMAN_OP_SRC, ibg, NULL, surface,
  216. 0, 0, 0, 0,
  217. cw * x, ch * y,
  218. cw, ch);
  219. pixman_image_composite(PIXMAN_OP_OVER, ifg, glyph, surface,
  220. 0, 0, 0, 0,
  221. cw * x, ch * y,
  222. cw, ch);
  223. pixman_image_unref(ifg);
  224. pixman_image_unref(ibg);
  225. }