| /* |
| * Copyright © 2026 Behdad Esfahbod |
| * |
| * This is part of HarfBuzz, a text shaping library. |
| * |
| * Permission is hereby granted, without written agreement and without |
| * license or royalty fees, to use, copy, modify, and distribute this |
| * software and its documentation for any purpose, provided that the |
| * above copyright notice and the following two paragraphs appear in |
| * all copies of this software. |
| * |
| * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR |
| * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES |
| * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN |
| * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH |
| * DAMAGE. |
| * |
| * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, |
| * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND |
| * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS |
| * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO |
| * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. |
| * |
| * Author(s): Behdad Esfahbod |
| */ |
| |
| #include "hb.hh" |
| |
| #include "hb-raster-image.hh" |
| |
| #include <math.h> |
| |
| #ifdef HAVE_PNG |
| #include <png.h> |
| #endif |
| |
| #ifdef HAVE_PNG |
| struct hb_raster_png_read_blob_t |
| { |
| const uint8_t *data = nullptr; |
| size_t size = 0; |
| size_t offset = 0; |
| }; |
| |
| static void |
| hb_raster_png_error (png_structp png, |
| png_const_charp msg HB_UNUSED) |
| { |
| png_longjmp (png, 1); |
| } |
| |
| static void |
| hb_raster_png_warning (png_structp png HB_UNUSED, |
| png_const_charp msg HB_UNUSED) |
| {} |
| |
| static void |
| hb_raster_png_read_blob (png_structp png, png_bytep out, png_size_t length) |
| { |
| hb_raster_png_read_blob_t *r = (hb_raster_png_read_blob_t *) png_get_io_ptr (png); |
| |
| if (!r || !r->data || length > r->size - r->offset) |
| png_error (png, "read error"); |
| |
| hb_memcpy (out, r->data + r->offset, length); |
| r->offset += length; |
| } |
| |
| struct hb_raster_png_write_blob_t |
| { |
| hb_vector_t<char> data; |
| }; |
| |
| static void |
| hb_raster_png_write_blob (png_structp png, png_bytep in, png_size_t length) |
| { |
| hb_raster_png_write_blob_t *w = (hb_raster_png_write_blob_t *) png_get_io_ptr (png); |
| if (!w) |
| png_error (png, "write error"); |
| |
| unsigned old_length = w->data.length; |
| if (!w->data.resize_dirty ((int) (old_length + length))) |
| png_error (png, "write error"); |
| |
| hb_memcpy (w->data.arrayZ + old_length, in, length); |
| } |
| |
| static void |
| hb_raster_png_flush_blob (png_structp png HB_UNUSED) |
| {} |
| #endif |
| |
| |
| /* |
| * Image compositing |
| */ |
| |
| /* Unpack premultiplied pixel to float RGBA [0,1]. */ |
| static inline void |
| unpack_to_float (uint32_t px, float &r, float &g, float &b, float &a) |
| { |
| b = (px & 0xFF) / 255.f; |
| g = ((px >> 8) & 0xFF) / 255.f; |
| r = ((px >> 16) & 0xFF) / 255.f; |
| a = (px >> 24) / 255.f; |
| } |
| |
| /* Pack float RGBA [0,1] premultiplied back to uint32_t. */ |
| static inline uint32_t |
| pack_from_float (float r, float g, float b, float a) |
| { |
| return hb_raster_pack_pixel ((uint8_t) (hb_clamp (b, 0.f, 1.f) * 255.f + 0.5f), |
| (uint8_t) (hb_clamp (g, 0.f, 1.f) * 255.f + 0.5f), |
| (uint8_t) (hb_clamp (r, 0.f, 1.f) * 255.f + 0.5f), |
| (uint8_t) (hb_clamp (a, 0.f, 1.f) * 255.f + 0.5f)); |
| } |
| |
| /* Separable blend mode functions: operate on unpremultiplied [0,1] channels. */ |
| static inline float |
| blend_multiply (float sc, float dc) { return sc * dc; } |
| static inline float |
| blend_screen (float sc, float dc) { return sc + dc - sc * dc; } |
| static inline float |
| blend_overlay (float sc, float dc) |
| { return dc <= 0.5f ? 2.f * sc * dc : 1.f - 2.f * (1.f - sc) * (1.f - dc); } |
| static inline float |
| blend_darken (float sc, float dc) { return hb_min (sc, dc); } |
| static inline float |
| blend_lighten (float sc, float dc) { return hb_max (sc, dc); } |
| static inline float |
| blend_color_dodge (float sc, float dc) |
| { |
| if (dc <= 0.f) return 0.f; |
| if (sc >= 1.f) return 1.f; |
| return hb_min (1.f, dc / (1.f - sc)); |
| } |
| static inline float |
| blend_color_burn (float sc, float dc) |
| { |
| if (dc >= 1.f) return 1.f; |
| if (sc <= 0.f) return 0.f; |
| return 1.f - hb_min (1.f, (1.f - dc) / sc); |
| } |
| static inline float |
| blend_hard_light (float sc, float dc) |
| { return sc <= 0.5f ? 2.f * sc * dc : 1.f - 2.f * (1.f - sc) * (1.f - dc); } |
| static inline float |
| blend_soft_light (float sc, float dc) |
| { |
| if (sc <= 0.5f) |
| return dc - (1.f - 2.f * sc) * dc * (1.f - dc); |
| float d = (dc <= 0.25f) ? ((16.f * dc - 12.f) * dc + 4.f) * dc |
| : sqrtf (dc); |
| return dc + (2.f * sc - 1.f) * (d - dc); |
| } |
| static inline float |
| blend_difference (float sc, float dc) { return fabsf (sc - dc); } |
| static inline float |
| blend_exclusion (float sc, float dc) { return sc + dc - 2.f * sc * dc; } |
| |
| /* Apply a separable blend mode per-pixel. |
| * Both src and dst are premultiplied BGRA32. */ |
| static inline uint32_t |
| apply_separable_blend (uint32_t src, uint32_t dst, |
| float (*blend_fn)(float, float)) |
| { |
| float sr, sg, sb, sa; |
| float dr, dg, db, da; |
| unpack_to_float (src, sr, sg, sb, sa); |
| unpack_to_float (dst, dr, dg, db, da); |
| |
| float usr = sa > 0.f ? sr / sa : 0.f; |
| float usg = sa > 0.f ? sg / sa : 0.f; |
| float usb = sa > 0.f ? sb / sa : 0.f; |
| float udr = da > 0.f ? dr / da : 0.f; |
| float udg = da > 0.f ? dg / da : 0.f; |
| float udb = da > 0.f ? db / da : 0.f; |
| |
| float br = blend_fn (usr, udr); |
| float bg = blend_fn (usg, udg); |
| float bb = blend_fn (usb, udb); |
| |
| float ra = sa + da - sa * da; |
| float rr = sa * da * br + sa * (1.f - da) * usr + (1.f - sa) * da * udr; |
| float rg = sa * da * bg + sa * (1.f - da) * usg + (1.f - sa) * da * udg; |
| float rb = sa * da * bb + sa * (1.f - da) * usb + (1.f - sa) * da * udb; |
| |
| return pack_from_float (rr, rg, rb, ra); |
| } |
| |
| /* HSL helpers */ |
| static inline float |
| hsl_luminosity (float r, float g, float b) |
| { return 0.299f * r + 0.587f * g + 0.114f * b; } |
| |
| static inline float |
| hsl_saturation (float r, float g, float b) |
| { return hb_max (hb_max (r, g), b) - hb_min (hb_min (r, g), b); } |
| |
| static inline void |
| hsl_clip_color (float &r, float &g, float &b) |
| { |
| float l = hsl_luminosity (r, g, b); |
| float mn = hb_min (hb_min (r, g), b); |
| float mx = hb_max (hb_max (r, g), b); |
| if (mn < 0.f) |
| { |
| float d = l - mn; |
| if (d > 0.f) { r = l + (r - l) * l / d; g = l + (g - l) * l / d; b = l + (b - l) * l / d; } |
| } |
| if (mx > 1.f) |
| { |
| float d = mx - l; |
| if (d > 0.f) { r = l + (r - l) * (1.f - l) / d; g = l + (g - l) * (1.f - l) / d; b = l + (b - l) * (1.f - l) / d; } |
| } |
| } |
| |
| static inline void |
| hsl_set_luminosity (float &r, float &g, float &b, float l) |
| { |
| float d = l - hsl_luminosity (r, g, b); |
| r += d; g += d; b += d; |
| hsl_clip_color (r, g, b); |
| } |
| |
| static inline void |
| hsl_set_saturation_inner (float &mn, float &mid, float &mx, float s) |
| { |
| if (mx > mn) |
| { |
| mid = (mid - mn) * s / (mx - mn); |
| mx = s; |
| } |
| else |
| mid = mx = 0.f; |
| mn = 0.f; |
| } |
| |
| static inline void |
| hsl_set_saturation (float &r, float &g, float &b, float s) |
| { |
| if (r <= g) |
| { |
| if (g <= b) hsl_set_saturation_inner (r, g, b, s); |
| else if (r <= b) hsl_set_saturation_inner (r, b, g, s); |
| else hsl_set_saturation_inner (b, r, g, s); |
| } |
| else |
| { |
| if (r <= b) hsl_set_saturation_inner (g, r, b, s); |
| else if (g <= b) hsl_set_saturation_inner (g, b, r, s); |
| else hsl_set_saturation_inner (b, g, r, s); |
| } |
| } |
| |
| static inline uint32_t |
| apply_hsl_blend (uint32_t src, uint32_t dst, |
| hb_paint_composite_mode_t mode) |
| { |
| float sr, sg, sb, sa; |
| float dr, dg, db, da; |
| unpack_to_float (src, sr, sg, sb, sa); |
| unpack_to_float (dst, dr, dg, db, da); |
| |
| float usr = sa > 0.f ? sr / sa : 0.f; |
| float usg = sa > 0.f ? sg / sa : 0.f; |
| float usb = sa > 0.f ? sb / sa : 0.f; |
| float udr = da > 0.f ? dr / da : 0.f; |
| float udg = da > 0.f ? dg / da : 0.f; |
| float udb = da > 0.f ? db / da : 0.f; |
| |
| float br = udr, bg = udg, bb = udb; |
| |
| if (mode == HB_PAINT_COMPOSITE_MODE_HSL_HUE) |
| { |
| br = usr; bg = usg; bb = usb; |
| hsl_set_saturation (br, bg, bb, hsl_saturation (udr, udg, udb)); |
| hsl_set_luminosity (br, bg, bb, hsl_luminosity (udr, udg, udb)); |
| } |
| else if (mode == HB_PAINT_COMPOSITE_MODE_HSL_SATURATION) |
| { |
| br = udr; bg = udg; bb = udb; |
| hsl_set_saturation (br, bg, bb, hsl_saturation (usr, usg, usb)); |
| hsl_set_luminosity (br, bg, bb, hsl_luminosity (udr, udg, udb)); |
| } |
| else if (mode == HB_PAINT_COMPOSITE_MODE_HSL_COLOR) |
| { |
| br = usr; bg = usg; bb = usb; |
| hsl_set_luminosity (br, bg, bb, hsl_luminosity (udr, udg, udb)); |
| } |
| else /* HSL_LUMINOSITY */ |
| { |
| br = udr; bg = udg; bb = udb; |
| hsl_set_luminosity (br, bg, bb, hsl_luminosity (usr, usg, usb)); |
| } |
| |
| float ra = sa + da - sa * da; |
| float rr = sa * da * br + sa * (1.f - da) * usr + (1.f - sa) * da * udr; |
| float rg = sa * da * bg + sa * (1.f - da) * usg + (1.f - sa) * da * udg; |
| float rb = sa * da * bb + sa * (1.f - da) * usb + (1.f - sa) * da * udb; |
| |
| return pack_from_float (rr, rg, rb, ra); |
| } |
| |
| /* Composite per-pixel with full blend mode support. */ |
| static inline uint32_t |
| composite_pixel (uint32_t src, uint32_t dst, |
| hb_paint_composite_mode_t mode) |
| { |
| uint8_t sa = (uint8_t) (src >> 24); |
| uint8_t da = (uint8_t) (dst >> 24); |
| |
| switch (mode) |
| { |
| case HB_PAINT_COMPOSITE_MODE_CLEAR: |
| return 0; |
| case HB_PAINT_COMPOSITE_MODE_SRC: |
| return src; |
| case HB_PAINT_COMPOSITE_MODE_DEST: |
| return dst; |
| case HB_PAINT_COMPOSITE_MODE_SRC_OVER: |
| return hb_raster_src_over (src, dst); |
| case HB_PAINT_COMPOSITE_MODE_DEST_OVER: |
| return hb_raster_src_over (dst, src); |
| case HB_PAINT_COMPOSITE_MODE_SRC_IN: |
| return hb_raster_alpha_mul (src, da); |
| case HB_PAINT_COMPOSITE_MODE_DEST_IN: |
| return hb_raster_alpha_mul (dst, sa); |
| case HB_PAINT_COMPOSITE_MODE_SRC_OUT: |
| return hb_raster_alpha_mul (src, 255 - da); |
| case HB_PAINT_COMPOSITE_MODE_DEST_OUT: |
| return hb_raster_alpha_mul (dst, 255 - sa); |
| case HB_PAINT_COMPOSITE_MODE_SRC_ATOP: |
| { |
| /* Fa=Da, Fb=1-Sa */ |
| uint32_t a = hb_raster_alpha_mul (src, da); |
| uint32_t b = hb_raster_alpha_mul (dst, 255 - sa); |
| uint8_t rb = (uint8_t) hb_min (255u, (unsigned) (a & 0xFF) + (b & 0xFF)); |
| uint8_t rg = (uint8_t) hb_min (255u, (unsigned) ((a >> 8) & 0xFF) + ((b >> 8) & 0xFF)); |
| uint8_t rr = (uint8_t) hb_min (255u, (unsigned) ((a >> 16) & 0xFF) + ((b >> 16) & 0xFF)); |
| uint8_t ra = (uint8_t) hb_min (255u, (unsigned) (a >> 24) + (b >> 24)); |
| return hb_raster_pack_pixel (rb, rg, rr, ra); |
| } |
| case HB_PAINT_COMPOSITE_MODE_DEST_ATOP: |
| { |
| uint32_t a = hb_raster_alpha_mul (dst, sa); |
| uint32_t b = hb_raster_alpha_mul (src, 255 - da); |
| uint8_t rb = (uint8_t) hb_min (255u, (unsigned) (a & 0xFF) + (b & 0xFF)); |
| uint8_t rg = (uint8_t) hb_min (255u, (unsigned) ((a >> 8) & 0xFF) + ((b >> 8) & 0xFF)); |
| uint8_t rr = (uint8_t) hb_min (255u, (unsigned) ((a >> 16) & 0xFF) + ((b >> 16) & 0xFF)); |
| uint8_t ra = (uint8_t) hb_min (255u, (unsigned) (a >> 24) + (b >> 24)); |
| return hb_raster_pack_pixel (rb, rg, rr, ra); |
| } |
| case HB_PAINT_COMPOSITE_MODE_XOR: |
| { |
| uint32_t a = hb_raster_alpha_mul (src, 255 - da); |
| uint32_t b = hb_raster_alpha_mul (dst, 255 - sa); |
| uint8_t rb = (uint8_t) hb_min (255u, (unsigned) (a & 0xFF) + (b & 0xFF)); |
| uint8_t rg = (uint8_t) hb_min (255u, (unsigned) ((a >> 8) & 0xFF) + ((b >> 8) & 0xFF)); |
| uint8_t rr = (uint8_t) hb_min (255u, (unsigned) ((a >> 16) & 0xFF) + ((b >> 16) & 0xFF)); |
| uint8_t ra = (uint8_t) hb_min (255u, (unsigned) (a >> 24) + (b >> 24)); |
| return hb_raster_pack_pixel (rb, rg, rr, ra); |
| } |
| case HB_PAINT_COMPOSITE_MODE_PLUS: |
| { |
| uint8_t rb = (uint8_t) hb_min (255u, (unsigned) (src & 0xFF) + (dst & 0xFF)); |
| uint8_t rg = (uint8_t) hb_min (255u, (unsigned) ((src >> 8) & 0xFF) + ((dst >> 8) & 0xFF)); |
| uint8_t rr = (uint8_t) hb_min (255u, (unsigned) ((src >> 16) & 0xFF) + ((dst >> 16) & 0xFF)); |
| uint8_t ra = (uint8_t) hb_min (255u, (unsigned) (src >> 24) + (dst >> 24)); |
| return hb_raster_pack_pixel (rb, rg, rr, ra); |
| } |
| |
| case HB_PAINT_COMPOSITE_MODE_MULTIPLY: return apply_separable_blend (src, dst, blend_multiply); |
| case HB_PAINT_COMPOSITE_MODE_SCREEN: return apply_separable_blend (src, dst, blend_screen); |
| case HB_PAINT_COMPOSITE_MODE_OVERLAY: return apply_separable_blend (src, dst, blend_overlay); |
| case HB_PAINT_COMPOSITE_MODE_DARKEN: return apply_separable_blend (src, dst, blend_darken); |
| case HB_PAINT_COMPOSITE_MODE_LIGHTEN: return apply_separable_blend (src, dst, blend_lighten); |
| case HB_PAINT_COMPOSITE_MODE_COLOR_DODGE: return apply_separable_blend (src, dst, blend_color_dodge); |
| case HB_PAINT_COMPOSITE_MODE_COLOR_BURN: return apply_separable_blend (src, dst, blend_color_burn); |
| case HB_PAINT_COMPOSITE_MODE_HARD_LIGHT: return apply_separable_blend (src, dst, blend_hard_light); |
| case HB_PAINT_COMPOSITE_MODE_SOFT_LIGHT: return apply_separable_blend (src, dst, blend_soft_light); |
| case HB_PAINT_COMPOSITE_MODE_DIFFERENCE: return apply_separable_blend (src, dst, blend_difference); |
| case HB_PAINT_COMPOSITE_MODE_EXCLUSION: return apply_separable_blend (src, dst, blend_exclusion); |
| |
| case HB_PAINT_COMPOSITE_MODE_HSL_HUE: |
| case HB_PAINT_COMPOSITE_MODE_HSL_SATURATION: |
| case HB_PAINT_COMPOSITE_MODE_HSL_COLOR: |
| case HB_PAINT_COMPOSITE_MODE_HSL_LUMINOSITY: |
| return apply_hsl_blend (src, dst, mode); |
| |
| default: |
| return hb_raster_src_over (src, dst); |
| } |
| } |
| |
| /* hb_raster_image_t */ |
| |
| unsigned |
| hb_raster_image_t::bytes_per_pixel (hb_raster_format_t format) |
| { |
| return format == HB_RASTER_FORMAT_BGRA32 ? 4u : 1u; |
| } |
| |
| bool |
| hb_raster_image_t::configure (hb_raster_format_t format, |
| hb_raster_extents_t extents) |
| { |
| if (format != HB_RASTER_FORMAT_A8 && |
| format != HB_RASTER_FORMAT_BGRA32) |
| format = HB_RASTER_FORMAT_A8; |
| |
| unsigned bpp = bytes_per_pixel (format); |
| if (extents.width > UINT_MAX / bpp) |
| return false; |
| |
| unsigned min_stride = extents.width * bpp; |
| if (extents.stride == 0 || extents.stride < min_stride) |
| extents.stride = min_stride; |
| |
| if (extents.height && extents.stride > (size_t) -1 / extents.height) |
| return false; |
| |
| size_t buf_size = (size_t) extents.stride * extents.height; |
| if (buf_size > HB_RASTER_MAX_BUFFER_SIZE) |
| return false; |
| if (unlikely (!buffer.resize_dirty (buf_size))) |
| return false; |
| |
| this->format = format; |
| this->extents = extents; |
| return true; |
| } |
| |
| bool |
| hb_raster_image_t::deserialize_from_png (hb_blob_t *blob) |
| { |
| #ifndef HAVE_PNG |
| return false; |
| #else |
| if (!blob) |
| return false; |
| |
| unsigned blob_len = 0; |
| const uint8_t *blob_data = (const uint8_t *) hb_blob_get_data (blob, &blob_len); |
| if (!blob_data || !blob_len) |
| return false; |
| |
| png_structp png = png_create_read_struct (PNG_LIBPNG_VER_STRING, nullptr, |
| hb_raster_png_error, |
| hb_raster_png_warning); |
| if (!png) |
| return false; |
| |
| png_infop info = png_create_info_struct (png); |
| if (!info) |
| { |
| png_destroy_read_struct (&png, nullptr, nullptr); |
| return false; |
| } |
| |
| hb_raster_png_read_blob_t reader; |
| reader.data = blob_data; |
| reader.size = (size_t) blob_len; |
| reader.offset = 0; |
| hb_vector_t<uint8_t> rgba; |
| hb_vector_t<png_bytep> rows; |
| if (setjmp (png_jmpbuf (png))) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| rgba.fini (); |
| rows.fini (); |
| return false; |
| } |
| |
| png_set_read_fn (png, &reader, hb_raster_png_read_blob); |
| png_read_info (png, info); |
| |
| png_uint_32 w = 0, h = 0; |
| int bit_depth = 0, color_type = 0; |
| int interlace_type = 0, compression_type = 0, filter_method = 0; |
| png_get_IHDR (png, info, &w, &h, &bit_depth, &color_type, |
| &interlace_type, &compression_type, &filter_method); |
| |
| if (!w || !h || w > (png_uint_32) INT_MAX || h > (png_uint_32) INT_MAX) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| return false; |
| } |
| |
| bool has_trns = png_get_valid (png, info, PNG_INFO_tRNS); |
| |
| if (bit_depth == 16) |
| png_set_strip_16 (png); |
| if (color_type == PNG_COLOR_TYPE_PALETTE) |
| png_set_palette_to_rgb (png); |
| if (color_type == PNG_COLOR_TYPE_GRAY && bit_depth < 8) |
| png_set_expand_gray_1_2_4_to_8 (png); |
| if (has_trns) |
| png_set_tRNS_to_alpha (png); |
| if (color_type == PNG_COLOR_TYPE_GRAY || color_type == PNG_COLOR_TYPE_GRAY_ALPHA) |
| png_set_gray_to_rgb (png); |
| if (!(color_type & PNG_COLOR_MASK_ALPHA) && !has_trns) |
| png_set_add_alpha (png, 0xff, PNG_FILLER_AFTER); |
| if (interlace_type != PNG_INTERLACE_NONE) |
| png_set_interlace_handling (png); |
| |
| png_read_update_info (png, info); |
| |
| if (png_get_bit_depth (png, info) != 8 || png_get_channels (png, info) != 4) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| return false; |
| } |
| |
| png_size_t rowbytes = png_get_rowbytes (png, info); |
| if (rowbytes < (png_size_t) w * 4u) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| return false; |
| } |
| |
| size_t rgba_size = (size_t) rowbytes * (size_t) h; |
| if (h && rgba_size / (size_t) h != (size_t) rowbytes) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| return false; |
| } |
| |
| if (!rgba.resize (rgba_size)) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| return false; |
| } |
| |
| if (!rows.resize ((unsigned) h)) |
| { |
| png_destroy_read_struct (&png, &info, nullptr); |
| return false; |
| } |
| |
| for (unsigned y = 0; y < (unsigned) h; y++) |
| rows[y] = (png_bytep) (rgba.arrayZ + (size_t) y * (size_t) rowbytes); |
| |
| png_read_image (png, rows.arrayZ); |
| png_read_end (png, nullptr); |
| png_destroy_read_struct (&png, &info, nullptr); |
| |
| hb_raster_image_t decoded; |
| hb_raster_extents_t decoded_extents = {0, 0, (unsigned) w, (unsigned) h, 0}; |
| if (!decoded.configure (HB_RASTER_FORMAT_BGRA32, decoded_extents)) |
| return false; |
| |
| for (unsigned y = 0; y < (unsigned) h; y++) |
| { |
| hb_packed_t<uint32_t> *dst = (hb_packed_t<uint32_t> *) (decoded.buffer.arrayZ + (size_t) ((unsigned) h - 1 - y) * decoded.extents.stride); |
| const uint8_t *src = rgba.arrayZ + (size_t) y * (size_t) rowbytes; |
| for (unsigned x = 0; x < (unsigned) w; x++) |
| { |
| uint8_t r = src[4 * x + 0]; |
| uint8_t g = src[4 * x + 1]; |
| uint8_t b = src[4 * x + 2]; |
| uint8_t a = src[4 * x + 3]; |
| dst[x] = hb_packed_t<uint32_t> ((uint32_t) hb_raster_div255 (b * a) |
| | ((uint32_t) hb_raster_div255 (g * a) << 8) |
| | ((uint32_t) hb_raster_div255 (r * a) << 16) |
| | ((uint32_t) a << 24)); |
| } |
| } |
| |
| hb_swap (buffer, decoded.buffer); |
| hb_swap (this->extents, decoded.extents); |
| hb_swap (format, decoded.format); |
| return true; |
| #endif |
| } |
| |
| hb_blob_t * |
| hb_raster_image_t::serialize_to_png_or_fail () const |
| { |
| #ifndef HAVE_PNG |
| return nullptr; |
| #else |
| if (format != HB_RASTER_FORMAT_BGRA32 || !extents.width || !extents.height) |
| return nullptr; |
| |
| png_structp png = png_create_write_struct (PNG_LIBPNG_VER_STRING, nullptr, |
| hb_raster_png_error, |
| hb_raster_png_warning); |
| if (!png) |
| return nullptr; |
| |
| png_infop info = png_create_info_struct (png); |
| if (!info) |
| { |
| png_destroy_write_struct (&png, nullptr); |
| return nullptr; |
| } |
| |
| hb_raster_png_write_blob_t writer; |
| hb_vector_t<uint8_t> rgba; |
| hb_vector_t<png_bytep> rows; |
| if (setjmp (png_jmpbuf (png))) |
| { |
| png_destroy_write_struct (&png, &info); |
| writer.data.fini (); |
| rgba.fini (); |
| rows.fini (); |
| return nullptr; |
| } |
| |
| png_set_write_fn (png, &writer, hb_raster_png_write_blob, hb_raster_png_flush_blob); |
| png_set_IHDR (png, info, |
| extents.width, extents.height, |
| 8, PNG_COLOR_TYPE_RGBA, |
| PNG_INTERLACE_NONE, |
| PNG_COMPRESSION_TYPE_DEFAULT, |
| PNG_FILTER_TYPE_DEFAULT); |
| png_write_info (png, info); |
| |
| size_t rowbytes = (size_t) extents.width * 4u; |
| size_t rgba_size = rowbytes * (size_t) extents.height; |
| if (extents.height && rgba_size / (size_t) extents.height != rowbytes) |
| { |
| png_destroy_write_struct (&png, &info); |
| return nullptr; |
| } |
| |
| if (!rgba.resize (rgba_size)) |
| { |
| png_destroy_write_struct (&png, &info); |
| return nullptr; |
| } |
| |
| if (!rows.resize (extents.height)) |
| { |
| png_destroy_write_struct (&png, &info); |
| return nullptr; |
| } |
| |
| for (unsigned y = 0; y < extents.height; y++) |
| { |
| uint8_t *dst = rgba.arrayZ + (size_t) y * rowbytes; |
| const uint8_t *src = buffer.arrayZ + (size_t) (extents.height - 1 - y) * extents.stride; |
| |
| for (unsigned x = 0; x < extents.width; x++) |
| { |
| uint32_t px; |
| hb_memcpy (&px, src + x * 4, 4); |
| uint8_t b = (uint8_t) (px & 0xFF); |
| uint8_t g = (uint8_t) ((px >> 8) & 0xFF); |
| uint8_t r = (uint8_t) ((px >> 16) & 0xFF); |
| uint8_t a = (uint8_t) (px >> 24); |
| |
| dst[4 * x + 3] = a; |
| if (a) |
| { |
| dst[4 * x + 0] = (uint8_t) hb_min (255u, ((unsigned) r * 255u + a / 2u) / a); |
| dst[4 * x + 1] = (uint8_t) hb_min (255u, ((unsigned) g * 255u + a / 2u) / a); |
| dst[4 * x + 2] = (uint8_t) hb_min (255u, ((unsigned) b * 255u + a / 2u) / a); |
| } |
| else |
| dst[4 * x + 0] = dst[4 * x + 1] = dst[4 * x + 2] = 0; |
| } |
| |
| rows[y] = (png_bytep) dst; |
| } |
| |
| png_write_image (png, rows.arrayZ); |
| png_write_end (png, info); |
| png_destroy_write_struct (&png, &info); |
| |
| unsigned length = 0; |
| char *data = writer.data.steal (&length); |
| if (!data && length) |
| return nullptr; |
| |
| hb_blob_t *blob = hb_blob_create_or_fail (data, length, |
| HB_MEMORY_MODE_WRITABLE, |
| data, hb_free); |
| if (!blob) |
| hb_free (data); |
| return blob; |
| #endif |
| } |
| |
| void |
| hb_raster_image_t::clear () |
| { |
| size_t buf_size = (size_t) extents.stride * extents.height; |
| hb_memset (buffer.arrayZ, 0, buf_size); |
| } |
| |
| const uint8_t * |
| hb_raster_image_t::get_buffer () const |
| { |
| return buffer.arrayZ; |
| } |
| |
| void |
| hb_raster_image_t::composite_from (const hb_raster_image_t *src, |
| hb_paint_composite_mode_t mode) |
| { |
| unsigned w = extents.width; |
| unsigned h = extents.height; |
| unsigned stride = extents.stride; |
| |
| for (unsigned y = 0; y < h; y++) |
| { |
| hb_packed_t<uint32_t> *dp = (hb_packed_t<uint32_t> *) (buffer.arrayZ + y * stride); |
| const hb_packed_t<uint32_t> *sp = (const hb_packed_t<uint32_t> *) (src->buffer.arrayZ + y * stride); |
| for (unsigned x = 0; x < w; x++) |
| dp[x] = hb_packed_t<uint32_t> (composite_pixel ((uint32_t) sp[x], (uint32_t) dp[x], mode)); |
| } |
| } |
| |
| /* Composite src image onto dst image. |
| * Both images must have the same extents and BGRA32 format. */ |
| void |
| hb_raster_image_composite (hb_raster_image_t *dst, |
| const hb_raster_image_t *src, |
| hb_paint_composite_mode_t mode) |
| { |
| dst->composite_from (src, mode); |
| } |
| |
| /** |
| * hb_raster_image_create_or_fail: |
| * |
| * Creates a new raster image object. |
| * |
| * Return value: (transfer full): |
| * A newly allocated #hb_raster_image_t with a reference count of 1, |
| * or `NULL` on allocation failure. |
| * |
| * The returned image can be released with hb_raster_image_destroy(), or |
| * transferred for reuse with hb_raster_draw_recycle_image() or |
| * hb_raster_paint_recycle_image(). |
| * |
| * Since: 13.0.0 |
| **/ |
| hb_raster_image_t * |
| hb_raster_image_create_or_fail (void) |
| { |
| return hb_object_create<hb_raster_image_t> (); |
| } |
| |
| /** |
| * hb_raster_image_reference: (skip) |
| * @image: a raster image |
| * |
| * Increases the reference count on @image by one. |
| * |
| * This prevents @image from being destroyed until a matching |
| * call to hb_raster_image_destroy() is made. |
| * |
| * Return value: (transfer full): |
| * The referenced #hb_raster_image_t. |
| * |
| * Since: 13.0.0 |
| **/ |
| hb_raster_image_t * |
| hb_raster_image_reference (hb_raster_image_t *image) |
| { |
| return hb_object_reference (image); |
| } |
| |
| /** |
| * hb_raster_image_destroy: (skip) |
| * @image: a raster image |
| * |
| * Decreases the reference count on @image by one. When the |
| * reference count reaches zero, the image and its pixel buffer |
| * are freed. |
| * |
| * Since: 13.0.0 |
| **/ |
| void |
| hb_raster_image_destroy (hb_raster_image_t *image) |
| { |
| if (!hb_object_destroy (image)) return; |
| hb_free (image); |
| } |
| |
| /** |
| * hb_raster_image_set_user_data: (skip) |
| * @image: a raster image |
| * @key: the user-data key |
| * @data: a pointer to the user data |
| * @destroy: (nullable): a callback to call when @data is not needed anymore |
| * @replace: whether to replace an existing data with the same key |
| * |
| * Attaches a user-data key/data pair to the specified raster image. |
| * |
| * Return value: `true` if success, `false` otherwise |
| * |
| * Since: 13.0.0 |
| **/ |
| hb_bool_t |
| hb_raster_image_set_user_data (hb_raster_image_t *image, |
| hb_user_data_key_t *key, |
| void *data, |
| hb_destroy_func_t destroy, |
| hb_bool_t replace) |
| { |
| return hb_object_set_user_data (image, key, data, destroy, replace); |
| } |
| |
| /** |
| * hb_raster_image_get_user_data: (skip) |
| * @image: a raster image |
| * @key: the user-data key |
| * |
| * Fetches the user-data associated with the specified key, |
| * attached to the specified raster image. |
| * |
| * Return value: (transfer none): |
| * A pointer to the user data |
| * |
| * Since: 13.0.0 |
| **/ |
| void * |
| hb_raster_image_get_user_data (hb_raster_image_t *image, |
| hb_user_data_key_t *key) |
| { |
| return hb_object_get_user_data (image, key); |
| } |
| |
| /** |
| * hb_raster_image_configure: |
| * @image: a raster image |
| * @format: the pixel format |
| * @extents: (nullable): desired image extents |
| * |
| * Configures @image format and extents together, resizing backing storage |
| * at most once. This function does not clear pixel contents. |
| * |
| * Passing `NULL` for @extents clears extents and releases the backing |
| * allocation. |
| * |
| * Return value: `true` if configuration succeeds, `false` on allocation |
| * failure |
| * |
| * Since: 13.0.0 |
| **/ |
| hb_bool_t |
| hb_raster_image_configure (hb_raster_image_t *image, |
| hb_raster_format_t format, |
| const hb_raster_extents_t *extents) |
| { |
| if (unlikely (!extents)) |
| { |
| image->extents = {}; |
| image->buffer.resize_exact (0); |
| return true; |
| } |
| return image->configure (format, *extents); |
| } |
| |
| /** |
| * hb_raster_image_clear: |
| * @image: a raster image |
| * |
| * Clears @image pixels to zero while keeping current extents and format. |
| * |
| * Since: 13.0.0 |
| **/ |
| void |
| hb_raster_image_clear (hb_raster_image_t *image) |
| { |
| image->clear (); |
| } |
| |
| /** |
| * hb_raster_image_get_buffer: |
| * @image: a raster image |
| * |
| * Fetches the raw pixel buffer of @image. The buffer layout is |
| * described by the extents obtained from hb_raster_image_get_extents() |
| * and the format from hb_raster_image_get_format(). Rows are stored |
| * bottom-to-top. |
| * |
| * Return value: (transfer none) (array): |
| * The pixel buffer, or `NULL` |
| * |
| * Since: 13.0.0 |
| **/ |
| const uint8_t * |
| hb_raster_image_get_buffer (hb_raster_image_t *image) |
| { |
| return image->get_buffer (); |
| } |
| |
| /** |
| * hb_raster_image_get_extents: |
| * @image: a raster image |
| * @extents: (out) (nullable): the image extents |
| * |
| * Fetches the pixel-buffer extents of @image. |
| * |
| * Since: 13.0.0 |
| **/ |
| void |
| hb_raster_image_get_extents (hb_raster_image_t *image, |
| hb_raster_extents_t *extents) |
| { |
| if (extents) |
| *extents = image->extents; |
| } |
| |
| /** |
| * hb_raster_image_get_format: |
| * @image: a raster image |
| * |
| * Fetches the pixel format of @image. |
| * |
| * Return value: |
| * The #hb_raster_format_t of the image |
| * |
| * Since: 13.0.0 |
| **/ |
| hb_raster_format_t |
| hb_raster_image_get_format (hb_raster_image_t *image) |
| { |
| return image->format; |
| } |
| |
| /** |
| * hb_raster_image_deserialize_from_png_or_fail: |
| * @image: a raster image |
| * @png: PNG data |
| * |
| * Replaces @image contents by deserializing a PNG blob into a |
| * #HB_RASTER_FORMAT_BGRA32 raster image. |
| * |
| * On success, @image extents are reset to pixel extents with origin |
| * `(0, 0)`. Rows in the resulting image buffer are stored bottom-to-top. |
| * On failure, @image is left unchanged. |
| * |
| * Return value: `true` if deserialization succeeded, `false` otherwise |
| * |
| * Since: 13.1.0 |
| **/ |
| hb_bool_t |
| hb_raster_image_deserialize_from_png_or_fail (hb_raster_image_t *image, |
| hb_blob_t *png) |
| { |
| return image->deserialize_from_png (png); |
| } |
| |
| /** |
| * hb_raster_image_serialize_to_png_or_fail: |
| * @image: a raster image |
| * |
| * Serializes @image to a PNG blob. |
| * |
| * Currently only #HB_RASTER_FORMAT_BGRA32 images are supported. |
| * |
| * Return value: (transfer full): |
| * A newly allocated PNG #hb_blob_t, or `NULL` on failure |
| * |
| * Since: 13.1.0 |
| **/ |
| hb_blob_t * |
| hb_raster_image_serialize_to_png_or_fail (hb_raster_image_t *image) |
| { |
| return image->serialize_to_png_or_fail (); |
| } |