/* * SPDX-License-Identifier: MIT * * Tiny subset of PIXMAN API commonly used by QEMU. * * Copyright 1987, 1988, 1989, 1998 The Open Group * Copyright 1987, 1988, 1989 Digital Equipment Corporation * Copyright 1999, 2004, 2008 Keith Packard * Copyright 2000 SuSE, Inc. * Copyright 2000 Keith Packard, member of The XFree86 Project, Inc. * Copyright 2004, 2005, 2007, 2008, 2009, 2010 Red Hat, Inc. * Copyright 2004 Nicholas Miell * Copyright 2005 Lars Knoll & Zack Rusin, Trolltech * Copyright 2005 Trolltech AS * Copyright 2007 Luca Barbato * Copyright 2008 Aaron Plattner, NVIDIA Corporation * Copyright 2008 Rodrigo Kumpera * Copyright 2008 André Tupinambá * Copyright 2008 Mozilla Corporation * Copyright 2008 Frederic Plourde * Copyright 2009, Oracle and/or its affiliates. All rights reserved. * Copyright 2009, 2010 Nokia Corporation * * 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 (including the next * paragraph) 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. */ #ifndef PIXMAN_MINIMAL_H #define PIXMAN_MINIMAL_H #define PIXMAN_TYPE_OTHER 0 #define PIXMAN_TYPE_ARGB 2 #define PIXMAN_TYPE_ABGR 3 #define PIXMAN_TYPE_BGRA 8 #define PIXMAN_TYPE_RGBA 9 #define PIXMAN_FORMAT(bpp, type, a, r, g, b) (((bpp) << 24) | \ ((type) << 16) | \ ((a) << 12) | \ ((r) << 8) | \ ((g) << 4) | \ ((b))) #define PIXMAN_FORMAT_RESHIFT(val, ofs, num) \ (((val >> (ofs)) & ((1 << (num)) - 1)) << ((val >> 22) & 3)) #define PIXMAN_FORMAT_BPP(f) PIXMAN_FORMAT_RESHIFT(f, 24, 8) #define PIXMAN_FORMAT_TYPE(f) (((f) >> 16) & 0x3f) #define PIXMAN_FORMAT_A(f) PIXMAN_FORMAT_RESHIFT(f, 12, 4) #define PIXMAN_FORMAT_R(f) PIXMAN_FORMAT_RESHIFT(f, 8, 4) #define PIXMAN_FORMAT_G(f) PIXMAN_FORMAT_RESHIFT(f, 4, 4) #define PIXMAN_FORMAT_B(f) PIXMAN_FORMAT_RESHIFT(f, 0, 4) #define PIXMAN_FORMAT_DEPTH(f) (PIXMAN_FORMAT_A(f) + \ PIXMAN_FORMAT_R(f) + \ PIXMAN_FORMAT_G(f) + \ PIXMAN_FORMAT_B(f)) typedef enum { /* 32bpp formats */ PIXMAN_a8r8g8b8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_ARGB, 8, 8, 8, 8), PIXMAN_x8r8g8b8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_ARGB, 0, 8, 8, 8), PIXMAN_a8b8g8r8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_ABGR, 8, 8, 8, 8), PIXMAN_x8b8g8r8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_ABGR, 0, 8, 8, 8), PIXMAN_b8g8r8a8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_BGRA, 8, 8, 8, 8), PIXMAN_b8g8r8x8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_BGRA, 0, 8, 8, 8), PIXMAN_r8g8b8a8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_RGBA, 8, 8, 8, 8), PIXMAN_r8g8b8x8 = PIXMAN_FORMAT(32, PIXMAN_TYPE_RGBA, 0, 8, 8, 8), /* 24bpp formats */ PIXMAN_r8g8b8 = PIXMAN_FORMAT(24, PIXMAN_TYPE_ARGB, 0, 8, 8, 8), PIXMAN_b8g8r8 = PIXMAN_FORMAT(24, PIXMAN_TYPE_ABGR, 0, 8, 8, 8), /* 16bpp formats */ PIXMAN_r5g6b5 = PIXMAN_FORMAT(16, PIXMAN_TYPE_ARGB, 0, 5, 6, 5), PIXMAN_a1r5g5b5 = PIXMAN_FORMAT(16, PIXMAN_TYPE_ARGB, 1, 5, 5, 5), PIXMAN_x1r5g5b5 = PIXMAN_FORMAT(16, PIXMAN_TYPE_ARGB, 0, 5, 5, 5), } pixman_format_code_t; typedef struct pixman_image pixman_image_t; typedef void (*pixman_image_destroy_func_t)(pixman_image_t *image, void *data); struct pixman_image { int ref_count; pixman_format_code_t format; int width; int height; int stride; uint32_t *data; uint32_t *free_me; pixman_image_destroy_func_t destroy_func; void *destroy_data; }; typedef struct pixman_color { uint16_t red; uint16_t green; uint16_t blue; uint16_t alpha; } pixman_color_t; static inline uint32_t *create_bits(pixman_format_code_t format, int width, int height, int *rowstride_bytes) { int stride = 0; size_t buf_size = 0; int bpp = PIXMAN_FORMAT_BPP(format); /* * Calculate the following while checking for overflow truncation: * stride = ((width * bpp + 0x1f) >> 5) * sizeof(uint32_t); */ if (unlikely(__builtin_mul_overflow(width, bpp, &stride))) { return NULL; } if (unlikely(__builtin_add_overflow(stride, 0x1f, &stride))) { return NULL; } stride >>= 5; stride *= sizeof(uint32_t); if (unlikely(__builtin_mul_overflow((size_t) height, (size_t) stride, &buf_size))) { return NULL; } if (rowstride_bytes) { *rowstride_bytes = stride; } return g_malloc0(buf_size); } static inline pixman_image_t *pixman_image_create_bits(pixman_format_code_t format, int width, int height, uint32_t *bits, int rowstride_bytes) { pixman_image_t *i = g_new0(pixman_image_t, 1); i->width = width; i->height = height; i->format = format; if (bits) { i->data = bits; } else { i->free_me = i->data = create_bits(format, width, height, &rowstride_bytes); if (width && height) { assert(i->data); } } i->stride = rowstride_bytes ? rowstride_bytes : width * DIV_ROUND_UP(PIXMAN_FORMAT_BPP(format), 8); i->ref_count = 1; return i; } static inline pixman_image_t *pixman_image_ref(pixman_image_t *i) { i->ref_count++; return i; } static inline bool pixman_image_unref(pixman_image_t *i) { i->ref_count--; if (i->ref_count == 0) { if (i->destroy_func) { i->destroy_func(i, i->destroy_data); } g_free(i->free_me); g_free(i); return true; } return false; } static inline void pixman_image_set_destroy_function(pixman_image_t *i, pixman_image_destroy_func_t func, void *data) { i->destroy_func = func; i->destroy_data = data; } static inline uint32_t *pixman_image_get_data(pixman_image_t *i) { return i->data; } static inline int pixman_image_get_height(pixman_image_t *i) { return i->height; } static inline int pixman_image_get_width(pixman_image_t *i) { return i->width; } static inline int pixman_image_get_stride(pixman_image_t *i) { return i->stride; } static inline pixman_format_code_t pixman_image_get_format(pixman_image_t *i) { return i->format; } #endif /* PIXMAN_MINIMAL_H */