source: mainline/uspace/lib/fb/imgmap.c@ 12b4a7f

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 12b4a7f was 7c014d1, checked in by Martin Decky <martin@…>, 14 years ago

console and framebuffer server rewrite

  • Property mode set to 100644
File size: 13.2 KB
Line 
1/*
2 * Copyright (c) 2011 Martin Decky
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup imgmap
30 * @{
31 */
32/**
33 * @file
34 */
35
36#include <stdlib.h>
37#include <byteorder.h>
38#include <align.h>
39#include <bool.h>
40#include <mem.h>
41#include <as.h>
42#include "imgmap.h"
43
44struct imgmap {
45 size_t size;
46 imgmap_flags_t flags;
47 sysarg_t width;
48 sysarg_t height;
49 visual_t visual;
50 uint8_t data[];
51};
52
53/** RGB conversion and mask functions.
54 *
55 * These functions write an RGB pixel value to a memory location
56 * in a predefined format. The naming convention corresponds to
57 * the names of the visuals and the format created by these functions.
58 * The functions use the so called network bit order (i.e. big endian)
59 * with respect to their names.
60 */
61
62#define RED(pixel, bits) (((pixel) >> (8 + 8 + 8 - (bits))) & ((1 << (bits)) - 1))
63#define GREEN(pixel, bits) (((pixel) >> (8 + 8 - (bits))) & ((1 << (bits)) - 1))
64#define BLUE(pixel, bits) (((pixel) >> (8 - (bits))) & ((1 << (bits)) - 1))
65
66void pixel2rgb_0888(void *dst, pixel_t pixel)
67{
68 *((uint32_t *) dst) = host2uint32_t_be(
69 (RED(pixel, 8) << 16) | (GREEN(pixel, 8) << 8) | (BLUE(pixel, 8)));
70}
71
72void pixel2bgr_0888(void *dst, pixel_t pixel)
73{
74 *((uint32_t *) dst) = host2uint32_t_be(
75 (BLUE(pixel, 8) << 16) | (GREEN(pixel, 8) << 8) | (RED(pixel, 8)));
76}
77
78void pixel2rgb_8880(void *dst, pixel_t pixel)
79{
80 *((uint32_t *) dst) = host2uint32_t_be(
81 (RED(pixel, 8) << 24) | (GREEN(pixel, 8) << 16) | (BLUE(pixel, 8) << 8));
82}
83
84void pixel2bgr_8880(void *dst, pixel_t pixel)
85{
86 *((uint32_t *) dst) = host2uint32_t_be(
87 (BLUE(pixel, 8) << 24) | (GREEN(pixel, 8) << 16) | (RED(pixel, 8) << 8));
88}
89
90void pixel2rgb_888(void *dst, pixel_t pixel)
91{
92 ((uint8_t *) dst)[0] = RED(pixel, 8);
93 ((uint8_t *) dst)[1] = GREEN(pixel, 8);
94 ((uint8_t *) dst)[2] = BLUE(pixel, 8);
95}
96
97void pixel2bgr_888(void *dst, pixel_t pixel)
98{
99 ((uint8_t *) dst)[0] = BLUE(pixel, 8);
100 ((uint8_t *) dst)[1] = GREEN(pixel, 8);
101 ((uint8_t *) dst)[2] = RED(pixel, 8);
102}
103
104void pixel2rgb_555_be(void *dst, pixel_t pixel)
105{
106 *((uint16_t *) dst) = host2uint16_t_be(
107 (RED(pixel, 5) << 10) | (GREEN(pixel, 5) << 5) | (BLUE(pixel, 5)));
108}
109
110void pixel2rgb_555_le(void *dst, pixel_t pixel)
111{
112 *((uint16_t *) dst) = host2uint16_t_le(
113 (RED(pixel, 5) << 10) | (GREEN(pixel, 5) << 5) | (BLUE(pixel, 5)));
114}
115
116void pixel2rgb_565_be(void *dst, pixel_t pixel)
117{
118 *((uint16_t *) dst) = host2uint16_t_be(
119 (RED(pixel, 5) << 11) | (GREEN(pixel, 6) << 5) | (BLUE(pixel, 5)));
120}
121
122void pixel2rgb_565_le(void *dst, pixel_t pixel)
123{
124 *((uint16_t *) dst) = host2uint16_t_le(
125 (RED(pixel, 5) << 11) | (GREEN(pixel, 6) << 5) | (BLUE(pixel, 5)));
126}
127
128void pixel2bgr_323(void *dst, pixel_t pixel)
129{
130 *((uint8_t *) dst) =
131 ~((RED(pixel, 3) << 5) | (GREEN(pixel, 2) << 3) | BLUE(pixel, 3));
132}
133
134void pixel2gray_8(void *dst, pixel_t pixel)
135{
136 uint32_t red = RED(pixel, 8) * 5034375;
137 uint32_t green = GREEN(pixel, 8) * 9886846;
138 uint32_t blue = BLUE(pixel, 8) * 1920103;
139
140 *((uint8_t *) dst) = (red + green + blue) >> 24;
141}
142
143void visual_mask_0888(void *dst, bool mask)
144{
145 pixel2bgr_0888(dst, mask ? 0xffffff : 0);
146}
147
148void visual_mask_8880(void *dst, bool mask)
149{
150 pixel2bgr_8880(dst, mask ? 0xffffff : 0);
151}
152
153void visual_mask_888(void *dst, bool mask)
154{
155 pixel2bgr_888(dst, mask ? 0xffffff : 0);
156}
157
158void visual_mask_555(void *dst, bool mask)
159{
160 pixel2rgb_555_be(dst, mask ? 0xffffff : 0);
161}
162
163void visual_mask_565(void *dst, bool mask)
164{
165 pixel2rgb_565_be(dst, mask ? 0xffffff : 0);
166}
167
168void visual_mask_323(void *dst, bool mask)
169{
170 pixel2bgr_323(dst, mask ? 0x0 : ~0x0);
171}
172
173void visual_mask_8(void *dst, bool mask)
174{
175 pixel2gray_8(dst, mask ? 0xffffff : 0);
176}
177
178pixel_t rgb_0888_2pixel(void *src)
179{
180 return (uint32_t_be2host(*((uint32_t *) src)) & 0xffffff);
181}
182
183pixel_t bgr_0888_2pixel(void *src)
184{
185 uint32_t val = uint32_t_be2host(*((uint32_t *) src));
186 return (((val & 0xff0000) >> 16) | (val & 0xff00) | ((val & 0xff) << 16));
187}
188
189pixel_t rgb_8880_2pixel(void *src)
190{
191 return (uint32_t_be2host(*((uint32_t *) src)) >> 8);
192}
193
194pixel_t bgr_8880_2pixel(void *src)
195{
196 uint32_t val = uint32_t_be2host(*((uint32_t *) src));
197 return (((val & 0xff000000) >> 24) | ((val & 0xff0000) >> 8) | ((val & 0xff00) << 8));
198}
199
200pixel_t rgb_888_2pixel(void *src)
201{
202 uint8_t red = ((uint8_t *) src)[0];
203 uint8_t green = ((uint8_t *) src)[1];
204 uint8_t blue = ((uint8_t *) src)[2];
205
206 return ((red << 16) | (green << 8) | (blue));
207}
208
209pixel_t bgr_888_2pixel(void *src)
210{
211 uint8_t blue = ((uint8_t *) src)[0];
212 uint8_t green = ((uint8_t *) src)[1];
213 uint8_t red = ((uint8_t *) src)[2];
214
215 return ((red << 16) | (green << 8) | (blue));
216}
217
218pixel_t rgb_555_be_2pixel(void *src)
219{
220 uint16_t val = uint16_t_be2host(*((uint16_t *) src));
221 return (((val & 0x7c00) << 9) | ((val & 0x3e0) << 6) | ((val & 0x1f) << 3));
222}
223
224pixel_t rgb_555_le_2pixel(void *src)
225{
226 uint16_t val = uint16_t_le2host(*((uint16_t *) src));
227 return (((val & 0x7c00) << 9) | ((val & 0x3e0) << 6) | ((val & 0x1f) << 3));
228}
229
230pixel_t rgb_565_be_2pixel(void *src)
231{
232 uint16_t val = uint16_t_be2host(*((uint16_t *) src));
233 return (((val & 0xf800) << 8) | ((val & 0x7e0) << 5) | ((val & 0x1f) << 3));
234}
235
236pixel_t rgb_565_le_2pixel(void *src)
237{
238 uint16_t val = uint16_t_le2host(*((uint16_t *) src));
239 return (((val & 0xf800) << 8) | ((val & 0x7e0) << 5) | ((val & 0x1f) << 3));
240}
241
242pixel_t bgr_323_2pixel(void *src)
243{
244 uint8_t val = ~(*((uint8_t *) src));
245 return (((val & 0xe0) << 16) | ((val & 0x18) << 11) | ((val & 0x7) << 5));
246}
247
248pixel_t gray_8_2pixel(void *src)
249{
250 uint8_t val = *((uint8_t *) src);
251 return ((val << 16) | (val << 8) | (val));
252}
253
254typedef struct {
255 uint8_t id_length;
256 uint8_t cmap_type;
257 uint8_t img_type;
258
259 uint16_t cmap_first_entry;
260 uint16_t cmap_entries;
261 uint8_t cmap_bpp;
262
263 uint16_t startx;
264 uint16_t starty;
265 uint16_t width;
266 uint16_t height;
267 uint8_t img_bpp;
268 uint8_t img_descr;
269} __attribute__((packed)) tga_header_t;
270
271typedef enum {
272 CMAP_NOT_PRESENT = 0,
273 CMAP_PRESENT = 1,
274 CMAP_RESERVED_START = 2,
275 CMAP_PRIVATE_START = 128
276} cmap_type_t;
277
278typedef enum {
279 IMG_EMTPY = 0,
280 IMG_CMAP = 1,
281 IMG_BGRA = 2,
282 IMG_GRAY = 3,
283 IMG_CMAP_RLE = 9,
284 IMG_BGRA_RLE = 10,
285 IMG_GRAY_RLE = 11
286} img_type_t;
287
288typedef struct {
289 cmap_type_t cmap_type;
290 img_type_t img_type;
291
292 uint16_t cmap_first_entry;
293 uint16_t cmap_entries;
294 uint8_t cmap_bpp;
295
296 uint16_t startx;
297 uint16_t starty;
298 uint16_t width;
299 uint16_t height;
300 uint8_t img_bpp;
301 uint8_t img_alpha_bpp;
302 uint8_t img_alpha_dir;
303
304 void *id_data;
305 size_t id_length;
306
307 void *cmap_data;
308 size_t cmap_length;
309
310 void *img_data;
311 size_t img_length;
312} tga_t;
313
314/** Decode Truevision TGA header
315 *
316 * @param[in] data Memory representation of TGA.
317 * @param[in] size Size of the representation (in bytes).
318 * @param[out] tga Decoded TGA.
319 *
320 * @return True on succesful decoding.
321 * @return False on failure.
322 *
323 */
324static bool decode_tga_header(void *data, size_t size, tga_t *tga)
325{
326 /* Header sanity check */
327 if (size < sizeof(tga_header_t))
328 return false;
329
330 tga_header_t *head = (tga_header_t *) data;
331
332 /* Image ID field */
333 tga->id_data = data + sizeof(tga_header_t);
334 tga->id_length = head->id_length;
335
336 if (size < sizeof(tga_header_t) + tga->id_length)
337 return false;
338
339 /* Color map type */
340 tga->cmap_type = head->cmap_type;
341
342 /* Image type */
343 tga->img_type = head->img_type;
344
345 /* Color map specification */
346 tga->cmap_first_entry = uint16_t_le2host(head->cmap_first_entry);
347 tga->cmap_entries = uint16_t_le2host(head->cmap_entries);
348 tga->cmap_bpp = head->cmap_bpp;
349 tga->cmap_data = tga->id_data + tga->id_length;
350 tga->cmap_length = ALIGN_UP(tga->cmap_entries * tga->cmap_bpp, 8) >> 3;
351
352 if (size < sizeof(tga_header_t) + tga->id_length +
353 tga->cmap_length)
354 return false;
355
356 /* Image specification */
357 tga->startx = uint16_t_le2host(head->startx);
358 tga->starty = uint16_t_le2host(head->starty);
359 tga->width = uint16_t_le2host(head->width);
360 tga->height = uint16_t_le2host(head->height);
361 tga->img_bpp = head->img_bpp;
362 tga->img_alpha_bpp = head->img_descr & 0x0f;
363 tga->img_alpha_dir = (head->img_descr & 0xf0) >> 4;
364 tga->img_data = tga->cmap_data + tga->cmap_length;
365 tga->img_length = ALIGN_UP(tga->width * tga->height * tga->img_bpp, 8) >> 3;
366
367 if (size < sizeof(tga_header_t) + tga->id_length +
368 tga->cmap_length + tga->img_length)
369 return false;
370
371 return true;
372}
373
374void imgmap_put_pixel(imgmap_t *imgmap, sysarg_t x, sysarg_t y, pixel_t pixel)
375{
376 if ((x >= imgmap->width) || (y >= imgmap->height))
377 return;
378
379 size_t offset = y * imgmap->width + x;
380
381 switch (imgmap->visual) {
382 case VISUAL_RGB_0_8_8_8:
383 pixel2rgb_0888(((uint32_t *) imgmap->data) + offset, pixel);
384 break;
385 default:
386 break;
387 }
388}
389
390pixel_t imgmap_get_pixel(imgmap_t *imgmap, sysarg_t x, sysarg_t y)
391{
392 if ((x >= imgmap->width) || (y >= imgmap->height))
393 return 0;
394
395 size_t offset = y * imgmap->width + x;
396
397 switch (imgmap->visual) {
398 case VISUAL_RGB_0_8_8_8:
399 return rgb_0888_2pixel(((uint32_t *) imgmap->data) + offset);
400 default:
401 return 0;
402 }
403}
404
405imgmap_t *imgmap_create(sysarg_t width, sysarg_t height, visual_t visual,
406 imgmap_flags_t flags)
407{
408 size_t bsize;
409
410 switch (visual) {
411 case VISUAL_RGB_0_8_8_8:
412 bsize = (width * height) << 2;
413 break;
414 default:
415 return NULL;
416 }
417
418 size_t size = sizeof(imgmap_t) + bsize;
419 imgmap_t *imgmap;
420
421 if ((flags & IMGMAP_FLAG_SHARED) == IMGMAP_FLAG_SHARED) {
422 imgmap = (imgmap_t *) as_get_mappable_page(size);
423
424 if (as_area_create((void *) imgmap, size, AS_AREA_READ |
425 AS_AREA_WRITE | AS_AREA_CACHEABLE) != (void *) imgmap)
426 return NULL;
427 } else {
428 imgmap = (imgmap_t *) malloc(size);
429 if (imgmap == NULL)
430 return NULL;
431 }
432
433 imgmap->size = size;
434 imgmap->flags = flags;
435 imgmap->width = width;
436 imgmap->height = height;
437 imgmap->visual = visual;
438
439 memset(imgmap->data, 0, bsize);
440
441 return imgmap;
442}
443
444/** Decode Truevision TGA format
445 *
446 * Decode Truevision TGA format and create an image map
447 * from it. The supported variants of TGA are currently
448 * limited to uncompressed 24 bit true-color images without
449 * alpha channel.
450 *
451 * @param[in] data Memory representation of TGA.
452 * @param[in] size Size of the representation (in bytes).
453 * @param[in] flags Image map creation flags.
454 *
455 * @return Newly allocated image map.
456 * @return NULL on error or unsupported format.
457 *
458 */
459imgmap_t *imgmap_decode_tga(void *data, size_t size, imgmap_flags_t flags)
460{
461 tga_t tga;
462 if (!decode_tga_header(data, size, &tga))
463 return NULL;
464
465 /*
466 * Check for unsupported features.
467 */
468
469 switch (tga.cmap_type) {
470 case CMAP_NOT_PRESENT:
471 break;
472 default:
473 /* Unsupported */
474 return NULL;
475 }
476
477 switch (tga.img_type) {
478 case IMG_BGRA:
479 if (tga.img_bpp != 24)
480 return NULL;
481 break;
482 case IMG_GRAY:
483 if (tga.img_bpp != 8)
484 return NULL;
485 break;
486 default:
487 /* Unsupported */
488 return NULL;
489 }
490
491 if (tga.img_alpha_bpp != 0)
492 return NULL;
493
494 sysarg_t twidth = tga.startx + tga.width;
495 sysarg_t theight = tga.starty + tga.height;
496
497 imgmap_t *imgmap = imgmap_create(twidth, theight, VISUAL_RGB_0_8_8_8,
498 flags);
499 if (imgmap == NULL)
500 return NULL;
501
502 /*
503 * TGA is encoded in a bottom-up manner, the true-color
504 * variant is in BGR 8:8:8 encoding.
505 */
506
507 switch (tga.img_type) {
508 case IMG_BGRA:
509 for (sysarg_t y = tga.starty; y < theight; y++) {
510 for (sysarg_t x = tga.startx; x < twidth; x++) {
511 size_t offset =
512 ((y - tga.starty) * tga.width + (x - tga.startx)) * 3;
513
514 pixel_t pixel =
515 bgr_888_2pixel(((uint8_t *) tga.img_data) + offset);
516 imgmap_put_pixel(imgmap, x, theight - y - 1, pixel);
517 }
518 }
519 break;
520 case IMG_GRAY:
521 for (sysarg_t y = tga.starty; y < theight; y++) {
522 for (sysarg_t x = tga.startx; x < twidth; x++) {
523 size_t offset =
524 (y - tga.starty) * tga.width + (x - tga.startx);
525
526 pixel_t pixel =
527 gray_8_2pixel(((uint8_t *) tga.img_data) + offset);
528 imgmap_put_pixel(imgmap, x, theight - y - 1, pixel);
529 }
530 }
531 break;
532 default:
533 break;
534 }
535
536 return imgmap;
537}
538
539void imgmap_get_resolution(imgmap_t *imgmap, sysarg_t *width, sysarg_t *height)
540{
541 assert(width);
542 assert(height);
543
544 *width = imgmap->width;
545 *height = imgmap->height;
546}
547
548/** @}
549 */
Note: See TracBrowser for help on using the repository browser.