Path: blob/master/scripts/deps/libpng-1.6.54-apng.patch
7197 views
diff -Naru libpng-1.6.54.org/png.h libpng-1.6.54/png.h1--- libpng-1.6.54.org/png.h 2026-01-15 09:54:33.410881877 +09002+++ libpng-1.6.54/png.h 2026-01-15 14:49:49.345302877 +09003@@ -328,6 +328,10 @@4# include "pnglibconf.h"5#endif67+#define PNG_APNG_SUPPORTED8+#define PNG_READ_APNG_SUPPORTED9+#define PNG_WRITE_APNG_SUPPORTED10+11#ifndef PNG_VERSION_INFO_ONLY12/* Machine specific configuration. */13# include "pngconf.h"14@@ -423,6 +427,17 @@15* See pngconf.h for base types that vary by machine/system16*/1718+#ifdef PNG_APNG_SUPPORTED19+/* dispose_op flags from inside fcTL */20+#define PNG_DISPOSE_OP_NONE 0x00U21+#define PNG_DISPOSE_OP_BACKGROUND 0x01U22+#define PNG_DISPOSE_OP_PREVIOUS 0x02U23+24+/* blend_op flags from inside fcTL */25+#define PNG_BLEND_OP_SOURCE 0x00U26+#define PNG_BLEND_OP_OVER 0x01U27+#endif /* PNG_APNG_SUPPORTED */28+29/* This triggers a compiler error in png.c, if png.c and png.h30* do not agree upon the version number.31*/32@@ -801,6 +816,10 @@33(png_structp, png_infop));34typedef PNG_CALLBACK(void, *png_progressive_end_ptr,35(png_structp, png_infop));36+#ifdef PNG_APNG_SUPPORTED37+typedef PNG_CALLBACK(void, *png_progressive_frame_ptr,38+ (png_structp, png_uint_32));39+#endif4041/* The following callback receives png_uint_32 row_number, int pass for the42* png_bytep data of the row. When transforming an interlaced image the43@@ -3506,6 +3525,80 @@44* END OF HARDWARE AND SOFTWARE OPTIONS45******************************************************************************/4647+#ifdef PNG_APNG_SUPPORTED48+PNG_EXPORT(260, png_uint_32, png_get_acTL,49+ (png_structp png_ptr, png_infop info_ptr,50+ png_uint_32 *num_frames, png_uint_32 *num_plays));51+52+PNG_EXPORT(261, png_uint_32, png_set_acTL,53+ (png_structp png_ptr, png_infop info_ptr,54+ png_uint_32 num_frames, png_uint_32 num_plays));55+56+PNG_EXPORT(262, png_uint_32, png_get_num_frames,57+ (png_structp png_ptr, png_infop info_ptr));58+59+PNG_EXPORT(263, png_uint_32, png_get_num_plays,60+ (png_structp png_ptr, png_infop info_ptr));61+62+PNG_EXPORT(264, png_uint_32, png_get_next_frame_fcTL,63+ (png_structp png_ptr, png_infop info_ptr,64+ png_uint_32 *width, png_uint_32 *height,65+ png_uint_32 *x_offset, png_uint_32 *y_offset,66+ png_uint_16 *delay_num, png_uint_16 *delay_den,67+ png_byte *dispose_op, png_byte *blend_op));68+69+PNG_EXPORT(265, png_uint_32, png_set_next_frame_fcTL,70+ (png_structp png_ptr, png_infop info_ptr,71+ png_uint_32 width, png_uint_32 height,72+ png_uint_32 x_offset, png_uint_32 y_offset,73+ png_uint_16 delay_num, png_uint_16 delay_den,74+ png_byte dispose_op, png_byte blend_op));75+76+PNG_EXPORT(266, png_uint_32, png_get_next_frame_width,77+ (png_structp png_ptr, png_infop info_ptr));78+PNG_EXPORT(267, png_uint_32, png_get_next_frame_height,79+ (png_structp png_ptr, png_infop info_ptr));80+PNG_EXPORT(268, png_uint_32, png_get_next_frame_x_offset,81+ (png_structp png_ptr, png_infop info_ptr));82+PNG_EXPORT(269, png_uint_32, png_get_next_frame_y_offset,83+ (png_structp png_ptr, png_infop info_ptr));84+PNG_EXPORT(270, png_uint_16, png_get_next_frame_delay_num,85+ (png_structp png_ptr, png_infop info_ptr));86+PNG_EXPORT(271, png_uint_16, png_get_next_frame_delay_den,87+ (png_structp png_ptr, png_infop info_ptr));88+PNG_EXPORT(272, png_byte, png_get_next_frame_dispose_op,89+ (png_structp png_ptr, png_infop info_ptr));90+PNG_EXPORT(273, png_byte, png_get_next_frame_blend_op,91+ (png_structp png_ptr, png_infop info_ptr));92+PNG_EXPORT(274, png_byte, png_get_first_frame_is_hidden,93+ (png_structp png_ptr, png_infop info_ptr));94+PNG_EXPORT(275, png_uint_32, png_set_first_frame_is_hidden,95+ (png_structp png_ptr, png_infop info_ptr, png_byte is_hidden));96+97+#ifdef PNG_READ_APNG_SUPPORTED98+PNG_EXPORT(276, void, png_read_frame_head,99+ (png_structp png_ptr, png_infop info_ptr));100+#ifdef PNG_PROGRESSIVE_READ_SUPPORTED101+PNG_EXPORT(277, void, png_set_progressive_frame_fn,102+ (png_structp png_ptr,103+ png_progressive_frame_ptr frame_info_fn,104+ png_progressive_frame_ptr frame_end_fn));105+#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */106+#endif /* PNG_READ_APNG_SUPPORTED */107+108+#ifdef PNG_WRITE_APNG_SUPPORTED109+PNG_EXPORT(278, void, png_write_frame_head,110+ (png_structp png_ptr, png_infop info_ptr,111+ png_uint_32 width, png_uint_32 height,112+ png_uint_32 x_offset, png_uint_32 y_offset,113+ png_uint_16 delay_num, png_uint_16 delay_den,114+ png_byte dispose_op, png_byte blend_op));115+116+PNG_EXPORT(279, void, png_write_frame_tail,117+ (png_structp png_ptr, png_infop info_ptr));118+#endif /* PNG_WRITE_APNG_SUPPORTED */119+#endif /* PNG_APNG_SUPPORTED */120+121/* Maintainer: Put new public prototypes here ^, in libpng.3, in project122* defs, and in scripts/symbols.def.123*/124@@ -3514,7 +3608,11 @@125* one to use is one more than this.)126*/127#ifdef PNG_EXPORT_LAST_ORDINAL128+#ifdef PNG_APNG_SUPPORTED129+ PNG_EXPORT_LAST_ORDINAL(279);130+#else131PNG_EXPORT_LAST_ORDINAL(259);132+#endif /* PNG_APNG_SUPPORTED */133#endif134135#ifdef __cplusplus136diff -Naru libpng-1.6.54.org/pngget.c libpng-1.6.54/pngget.c137--- libpng-1.6.54.org/pngget.c 2026-01-15 09:54:33.410881877 +0900138+++ libpng-1.6.54/pngget.c 2026-01-16 15:37:23.199617980 +0900139@@ -1366,4 +1366,166 @@140# endif141#endif142143+#ifdef PNG_APNG_SUPPORTED144+png_uint_32 PNGAPI145+png_get_acTL(png_structp png_ptr, png_infop info_ptr,146+ png_uint_32 *num_frames, png_uint_32 *num_plays)147+{148+ png_debug1(1, "in %s retrieval function", "acTL");149+150+ if (png_ptr != NULL && info_ptr != NULL &&151+ (info_ptr->valid & PNG_INFO_acTL) &&152+ num_frames != NULL && num_plays != NULL)153+ {154+ *num_frames = info_ptr->num_frames;155+ *num_plays = info_ptr->num_plays;156+ return 1;157+ }158+159+ return 0;160+}161+162+png_uint_32 PNGAPI163+png_get_num_frames(png_structp png_ptr, png_infop info_ptr)164+{165+ png_debug(1, "in png_get_num_frames");166+167+ if (png_ptr != NULL && info_ptr != NULL)168+ return info_ptr->num_frames;169+ return 0;170+}171+172+png_uint_32 PNGAPI173+png_get_num_plays(png_structp png_ptr, png_infop info_ptr)174+{175+ png_debug(1, "in png_get_num_plays");176+177+ if (png_ptr != NULL && info_ptr != NULL)178+ return info_ptr->num_plays;179+ return 0;180+}181+182+png_uint_32 PNGAPI183+png_get_next_frame_fcTL(png_structp png_ptr, png_infop info_ptr,184+ png_uint_32 *width, png_uint_32 *height,185+ png_uint_32 *x_offset, png_uint_32 *y_offset,186+ png_uint_16 *delay_num, png_uint_16 *delay_den,187+ png_byte *dispose_op, png_byte *blend_op)188+{189+ png_debug1(1, "in %s retrieval function", "fcTL");190+191+ if (png_ptr != NULL && info_ptr != NULL &&192+ (info_ptr->valid & PNG_INFO_fcTL) &&193+ width != NULL && height != NULL &&194+ x_offset != NULL && y_offset != NULL &&195+ delay_num != NULL && delay_den != NULL &&196+ dispose_op != NULL && blend_op != NULL)197+ {198+ *width = info_ptr->next_frame_width;199+ *height = info_ptr->next_frame_height;200+ *x_offset = info_ptr->next_frame_x_offset;201+ *y_offset = info_ptr->next_frame_y_offset;202+ *delay_num = info_ptr->next_frame_delay_num;203+ *delay_den = info_ptr->next_frame_delay_den;204+ *dispose_op = info_ptr->next_frame_dispose_op;205+ *blend_op = info_ptr->next_frame_blend_op;206+ return 1;207+ }208+209+ return 0;210+}211+212+png_uint_32 PNGAPI213+png_get_next_frame_width(png_structp png_ptr, png_infop info_ptr)214+{215+ png_debug(1, "in png_get_next_frame_width");216+217+ if (png_ptr != NULL && info_ptr != NULL)218+ return info_ptr->next_frame_width;219+ return 0;220+}221+222+png_uint_32 PNGAPI223+png_get_next_frame_height(png_structp png_ptr, png_infop info_ptr)224+{225+ png_debug(1, "in png_get_next_frame_height");226+227+ if (png_ptr != NULL && info_ptr != NULL)228+ return info_ptr->next_frame_height;229+ return 0;230+}231+232+png_uint_32 PNGAPI233+png_get_next_frame_x_offset(png_structp png_ptr, png_infop info_ptr)234+{235+ png_debug(1, "in png_get_next_frame_x_offset");236+237+ if (png_ptr != NULL && info_ptr != NULL)238+ return info_ptr->next_frame_x_offset;239+ return 0;240+}241+242+png_uint_32 PNGAPI243+png_get_next_frame_y_offset(png_structp png_ptr, png_infop info_ptr)244+{245+ png_debug(1, "in png_get_next_frame_y_offset");246+247+ if (png_ptr != NULL && info_ptr != NULL)248+ return info_ptr->next_frame_y_offset;249+ return 0;250+}251+252+png_uint_16 PNGAPI253+png_get_next_frame_delay_num(png_structp png_ptr, png_infop info_ptr)254+{255+ png_debug(1, "in png_get_next_frame_delay_num");256+257+ if (png_ptr != NULL && info_ptr != NULL)258+ return info_ptr->next_frame_delay_num;259+ return 0;260+}261+262+png_uint_16 PNGAPI263+png_get_next_frame_delay_den(png_structp png_ptr, png_infop info_ptr)264+{265+ png_debug(1, "in png_get_next_frame_delay_den");266+267+ if (png_ptr != NULL && info_ptr != NULL)268+ return info_ptr->next_frame_delay_den;269+ return 0;270+}271+272+png_byte PNGAPI273+png_get_next_frame_dispose_op(png_structp png_ptr, png_infop info_ptr)274+{275+ png_debug(1, "in png_get_next_frame_dispose_op");276+277+ if (png_ptr != NULL && info_ptr != NULL)278+ return info_ptr->next_frame_dispose_op;279+ return 0;280+}281+282+png_byte PNGAPI283+png_get_next_frame_blend_op(png_structp png_ptr, png_infop info_ptr)284+{285+ png_debug(1, "in png_get_next_frame_blend_op");286+287+ if (png_ptr != NULL && info_ptr != NULL)288+ return info_ptr->next_frame_blend_op;289+ return 0;290+}291+292+png_byte PNGAPI293+png_get_first_frame_is_hidden(png_structp png_ptr, png_infop info_ptr)294+{295+ png_debug(1, "in png_first_frame_is_hidden");296+297+ if (png_ptr != NULL)298+ return (png_byte)(png_ptr->apng_flags & PNG_FIRST_FRAME_HIDDEN);299+300+ PNG_UNUSED(info_ptr)301+302+ return 0;303+}304+#endif /* PNG_APNG_SUPPORTED */305#endif /* READ || WRITE */306diff -Naru libpng-1.6.54.org/pnginfo.h libpng-1.6.54/pnginfo.h307--- libpng-1.6.54.org/pnginfo.h 2025-05-11 18:44:02.085040902 +0900308+++ libpng-1.6.54/pnginfo.h 2026-01-15 13:50:32.228564907 +0900309@@ -259,5 +259,18 @@310#ifdef PNG_sRGB_SUPPORTED311int rendering_intent;312#endif313+314+#ifdef PNG_APNG_SUPPORTED315+ png_uint_32 num_frames; /* including default image */316+ png_uint_32 num_plays;317+ png_uint_32 next_frame_width;318+ png_uint_32 next_frame_height;319+ png_uint_32 next_frame_x_offset;320+ png_uint_32 next_frame_y_offset;321+ png_uint_16 next_frame_delay_num;322+ png_uint_16 next_frame_delay_den;323+ png_byte next_frame_dispose_op;324+ png_byte next_frame_blend_op;325+#endif326};327#endif /* PNGINFO_H */328diff -Naru libpng-1.6.54.org/pngpread.c libpng-1.6.54/pngpread.c329--- libpng-1.6.54.org/pngpread.c 2025-07-06 11:06:43.933735454 +0900330+++ libpng-1.6.54/pngpread.c 2026-01-16 16:44:17.565675229 +0900331@@ -200,6 +200,105 @@332333chunk_name = png_ptr->chunk_name;334335+#ifdef PNG_READ_APNG_SUPPORTED336+ if (png_ptr->num_frames_read > 0 &&337+ png_ptr->num_frames_read < info_ptr->num_frames)338+ {339+ if (chunk_name == png_IDAT)340+ {341+ /* Discard trailing IDATs for the first frame. */342+ if (png_ptr->mode & PNG_HAVE_fcTL || png_ptr->num_frames_read > 1)343+ png_error(png_ptr, "Misplaced IDAT in APNG stream");344+345+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)346+ {347+ png_push_save_buffer(png_ptr);348+ return;349+ }350+351+ png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER;352+ return;353+ }354+ else if (chunk_name == png_fdAT)355+ {356+ if (png_ptr->buffer_size < 4)357+ {358+ png_push_save_buffer(png_ptr);359+ return;360+ }361+362+ png_ensure_sequence_number(png_ptr, 4);363+364+ if (!(png_ptr->mode & PNG_HAVE_fcTL))365+ {366+ /* Discard trailing fdATs for frames other than the first. */367+ if (png_ptr->num_frames_read < 2)368+ png_error(png_ptr, "Misplaced fdAT in APNG stream");369+370+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)371+ {372+ png_push_save_buffer(png_ptr);373+ return;374+ }375+376+ png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER;377+ return;378+ }379+380+ else381+ {382+ /* Frame data follows. */383+ png_ptr->idat_size = png_ptr->push_length - 4;384+ png_ptr->mode |= PNG_HAVE_IDAT;385+ png_ptr->process_mode = PNG_READ_IDAT_MODE;386+387+ return;388+ }389+ }390+391+ else if (chunk_name == png_fcTL)392+ {393+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)394+ {395+ png_push_save_buffer(png_ptr);396+ return;397+ }398+399+ png_read_reset(png_ptr);400+ png_ptr->mode &= ~PNG_HAVE_fcTL;401+402+ png_handle_fcTL(png_ptr, info_ptr, png_ptr->push_length);403+404+ if (!(png_ptr->mode & PNG_HAVE_fcTL))405+ png_error(png_ptr, "Missing required fcTL chunk in APNG stream");406+407+ png_read_reinit(png_ptr, info_ptr);408+ png_progressive_read_reset(png_ptr);409+410+ if (png_ptr->frame_info_fn != NULL)411+ (*(png_ptr->frame_info_fn))(png_ptr, png_ptr->num_frames_read);412+413+ png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER;414+415+ return;416+ }417+418+ else419+ {420+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)421+ {422+ png_push_save_buffer(png_ptr);423+ return;424+ }425+ png_warning(png_ptr, "Ignoring unexpected chunk in APNG sequence");426+ png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER;427+ return;428+ }429+430+ return;431+ }432+#endif /* PNG_READ_APNG_SUPPORTED */433+434if (chunk_name == png_IDAT)435{436if ((png_ptr->mode & PNG_AFTER_IDAT) != 0)437@@ -268,6 +367,9 @@438439else if (chunk_name == png_IDAT)440{441+#ifdef PNG_READ_APNG_SUPPORTED442+ png_have_info(png_ptr, info_ptr);443+#endif444png_ptr->idat_size = png_ptr->push_length;445png_ptr->process_mode = PNG_READ_IDAT_MODE;446png_push_have_info(png_ptr, info_ptr);447@@ -278,6 +380,31 @@448return;449}450451+#ifdef PNG_READ_APNG_SUPPORTED452+ else if (chunk_name == png_acTL)453+ {454+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)455+ {456+ png_push_save_buffer(png_ptr);457+ return;458+ }459+460+ png_handle_acTL(png_ptr, info_ptr, png_ptr->push_length);461+ }462+463+ else if (chunk_name == png_fcTL)464+ {465+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)466+ {467+ png_push_save_buffer(png_ptr);468+ return;469+ }470+471+ png_handle_fcTL(png_ptr, info_ptr, png_ptr->push_length);472+ }473+474+#endif /* PNG_READ_APNG_SUPPORTED */475+476else477{478PNG_PUSH_SAVE_BUFFER_IF_FULL479@@ -409,7 +536,11 @@480png_byte chunk_tag[4];481482/* TODO: this code can be commoned up with the same code in push_read */483+#ifdef PNG_READ_APNG_SUPPORTED484+ PNG_PUSH_SAVE_BUFFER_IF_LT(12)485+#else486PNG_PUSH_SAVE_BUFFER_IF_LT(8)487+#endif488png_push_fill_buffer(png_ptr, chunk_length, 4);489png_ptr->push_length = png_get_uint_31(png_ptr, chunk_length);490png_reset_crc(png_ptr);491@@ -417,17 +548,63 @@492png_ptr->chunk_name = PNG_CHUNK_FROM_STRING(chunk_tag);493png_ptr->mode |= PNG_HAVE_CHUNK_HEADER;494495+#ifdef PNG_READ_APNG_SUPPORTED496+ if (png_ptr->chunk_name != png_fdAT && png_ptr->num_frames_read > 0)497+ {498+ if (png_ptr->flags & PNG_FLAG_ZSTREAM_ENDED)499+ {500+ png_ptr->process_mode = PNG_READ_CHUNK_MODE;501+ if (png_ptr->frame_end_fn != NULL)502+ (*(png_ptr->frame_end_fn))(png_ptr, png_ptr->num_frames_read);503+ png_ptr->num_frames_read++;504+ return;505+ }506+ else507+ {508+ if (png_ptr->chunk_name == png_IEND)509+ png_error(png_ptr, "Not enough image data");510+ if (png_ptr->push_length + 4 > png_ptr->buffer_size)511+ {512+ png_push_save_buffer(png_ptr);513+ return;514+ }515+ png_warning(png_ptr, "Ignoring unexpected chunk in APNG sequence");516+ png_crc_finish(png_ptr, png_ptr->push_length);517+ png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER;518+ return;519+ }520+ }521+ else522+#endif523+#ifdef PNG_READ_APNG_SUPPORTED524+ if (png_ptr->chunk_name != png_IDAT && png_ptr->num_frames_read == 0)525+#else526if (png_ptr->chunk_name != png_IDAT)527+#endif528{529png_ptr->process_mode = PNG_READ_CHUNK_MODE;530531if ((png_ptr->flags & PNG_FLAG_ZSTREAM_ENDED) == 0)532png_error(png_ptr, "Not enough compressed data");533534+#ifdef PNG_READ_APNG_SUPPORTED535+ if (png_ptr->frame_end_fn != NULL)536+ (*(png_ptr->frame_end_fn))(png_ptr, png_ptr->num_frames_read);537+ png_ptr->num_frames_read++;538+#endif539+540return;541}542543png_ptr->idat_size = png_ptr->push_length;544+545+#ifdef PNG_READ_APNG_SUPPORTED546+ if (png_ptr->num_frames_read > 0)547+ {548+ png_ensure_sequence_number(png_ptr, 4);549+ png_ptr->idat_size -= 4;550+ }551+#endif552}553554if (png_ptr->idat_size != 0 && png_ptr->save_buffer_size != 0)555@@ -501,6 +678,15 @@556if (!(buffer_length > 0) || buffer == NULL)557png_error(png_ptr, "No IDAT data (internal error)");558559+#ifdef PNG_READ_APNG_SUPPORTED560+ /* If the app is not APNG-aware, decode only the first frame. */561+ if (!(png_ptr->apng_flags & PNG_APNG_APP) && png_ptr->num_frames_read > 0)562+ {563+ png_ptr->flags |= PNG_FLAG_ZSTREAM_ENDED;564+ return;565+ }566+#endif567+568/* This routine must process all the data it has been given569* before returning, calling the row callback as required to570* handle the uncompressed results.571@@ -934,6 +1120,18 @@572png_set_read_fn(png_ptr, progressive_ptr, png_push_fill_buffer);573}574575+#ifdef PNG_READ_APNG_SUPPORTED576+void PNGAPI577+png_set_progressive_frame_fn(png_structp png_ptr,578+ png_progressive_frame_ptr frame_info_fn,579+ png_progressive_frame_ptr frame_end_fn)580+{581+ png_ptr->frame_info_fn = frame_info_fn;582+ png_ptr->frame_end_fn = frame_end_fn;583+ png_ptr->apng_flags |= PNG_APNG_APP;584+}585+#endif586+587png_voidp PNGAPI588png_get_progressive_ptr(png_const_structrp png_ptr)589{590diff -Naru libpng-1.6.54.org/pngpriv.h libpng-1.6.54/pngpriv.h591--- libpng-1.6.54.org/pngpriv.h 2026-01-15 09:54:33.411881875 +0900592+++ libpng-1.6.54/pngpriv.h 2026-01-16 15:15:06.727223750 +0900593@@ -653,6 +653,10 @@594#define PNG_HAVE_CHUNK_AFTER_IDAT 0x2000U /* Have another chunk after IDAT */595#define PNG_WROTE_eXIf 0x4000U596#define PNG_IS_READ_STRUCT 0x8000U /* Else is a write struct */597+#ifdef PNG_APNG_SUPPORTED598+#define PNG_HAVE_acTL 0x10000U599+#define PNG_HAVE_fcTL 0x20000U600+#endif601602/* Flags for the transformations the PNG library does on the image data */603#define PNG_BGR 0x0001U604@@ -917,6 +921,13 @@605#define png_tRNS PNG_U32(116, 82, 78, 83)606#define png_zTXt PNG_U32(122, 84, 88, 116)607608+#ifdef PNG_APNG_SUPPORTED609+610+/* For png_struct.apng_flags: */611+#define PNG_FIRST_FRAME_HIDDEN 0x0001U612+#define PNG_APNG_APP 0x0002U613+#endif614+615/* The following will work on (signed char*) strings, whereas the get_uint_32616* macro will fail on top-bit-set values because of the sign extension.617*/618@@ -1880,6 +1891,68 @@619PNG_EMPTY);620#endif /* PROGRESSIVE_READ */621622+#ifdef PNG_APNG_SUPPORTED623+PNG_INTERNAL_FUNCTION(void, png_ensure_fcTL_is_valid,624+ (png_structp png_ptr,625+ png_uint_32 width, png_uint_32 height,626+ png_uint_32 x_offset, png_uint_32 y_offset,627+ png_uint_16 delay_num, png_uint_16 delay_den,628+ png_byte dispose_op, png_byte blend_op),629+ PNG_EMPTY);630+631+#ifdef PNG_READ_APNG_SUPPORTED632+PNG_INTERNAL_FUNCTION(void, png_handle_acTL,633+ (png_structp png_ptr, png_infop info_ptr, png_uint_32 length),634+ PNG_EMPTY);635+PNG_INTERNAL_FUNCTION(void, png_handle_fcTL,636+ (png_structp png_ptr, png_infop info_ptr, png_uint_32 length),637+ PNG_EMPTY);638+PNG_INTERNAL_FUNCTION(void, png_handle_fdAT,639+ (png_structp png_ptr, png_infop info_ptr, png_uint_32 length),640+ PNG_EMPTY);641+PNG_INTERNAL_FUNCTION(void, png_have_info,642+ (png_structp png_ptr, png_infop info_ptr),643+ PNG_EMPTY);644+PNG_INTERNAL_FUNCTION(void, png_ensure_sequence_number,645+ (png_structp png_ptr, png_uint_32 length),646+ PNG_EMPTY);647+PNG_INTERNAL_FUNCTION(void, png_read_reset,648+ (png_structp png_ptr),649+ PNG_EMPTY);650+PNG_INTERNAL_FUNCTION(void, png_read_reinit,651+ (png_structp png_ptr, png_infop info_ptr),652+ PNG_EMPTY);653+#ifdef PNG_PROGRESSIVE_READ_SUPPORTED654+PNG_INTERNAL_FUNCTION(void, png_progressive_read_reset,655+ (png_structp png_ptr),656+ PNG_EMPTY);657+#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */658+#endif /* PNG_READ_APNG_SUPPORTED */659+660+#ifdef PNG_WRITE_APNG_SUPPORTED661+PNG_INTERNAL_FUNCTION(void, png_write_acTL,662+ (png_structp png_ptr, png_uint_32 num_frames, png_uint_32 num_plays),663+ PNG_EMPTY);664+PNG_INTERNAL_FUNCTION(void, png_write_fcTL,665+ (png_structp png_ptr,666+ png_uint_32 width, png_uint_32 height,667+ png_uint_32 x_offset, png_uint_32 y_offset,668+ png_uint_16 delay_num, png_uint_16 delay_den,669+ png_byte dispose_op, png_byte blend_op),670+ PNG_EMPTY);671+PNG_INTERNAL_FUNCTION(void, png_write_fdAT,672+ (png_structp png_ptr, png_const_bytep data, png_size_t length),673+ PNG_EMPTY);674+PNG_INTERNAL_FUNCTION(void, png_write_reset,675+ (png_structp png_ptr),676+ PNG_EMPTY);677+PNG_INTERNAL_FUNCTION(void, png_write_reinit,678+ (png_structp png_ptr, png_infop info_ptr,679+ png_uint_32 width, png_uint_32 height),680+ PNG_EMPTY);681+#endif /* PNG_WRITE_APNG_SUPPORTED */682+#endif /* PNG_APNG_SUPPORTED */683+684#ifdef PNG_iCCP_SUPPORTED685/* Routines for checking parts of an ICC profile. */686#ifdef PNG_READ_iCCP_SUPPORTED687diff -Naru libpng-1.6.54.org/pngread.c libpng-1.6.54/pngread.c688--- libpng-1.6.54.org/pngread.c 2026-01-15 09:54:33.412881873 +0900689+++ libpng-1.6.54/pngread.c 2026-01-16 19:10:26.174024695 +0900690@@ -157,16 +157,95 @@691692else if (chunk_name == png_IDAT)693{694+#ifdef PNG_READ_APNG_SUPPORTED695+ png_have_info(png_ptr, info_ptr);696+#endif697png_ptr->idat_size = length;698break;699}700701+#ifdef PNG_READ_APNG_SUPPORTED702+ else if (chunk_name == png_acTL)703+ png_handle_acTL(png_ptr, info_ptr, length);704+705+ else if (chunk_name == png_fcTL)706+ png_handle_fcTL(png_ptr, info_ptr, length);707+708+ else if (chunk_name == png_fdAT)709+ png_handle_fdAT(png_ptr, info_ptr, length);710+#endif711+712else713png_handle_chunk(png_ptr, info_ptr, length);714}715}716#endif /* SEQUENTIAL_READ */717718+#ifdef PNG_READ_APNG_SUPPORTED719+void PNGAPI720+png_read_frame_head(png_structp png_ptr, png_infop info_ptr)721+{722+ png_byte have_chunk_after_DAT; /* after IDAT or after fdAT */723+724+ png_debug(1, "Reading frame head");725+726+ if (!(png_ptr->mode & PNG_HAVE_acTL))727+ png_error(png_ptr, "Cannot read APNG frame: missing acTL");728+729+ /* Do nothing for the main IDAT. */730+ if (png_ptr->num_frames_read == 0)731+ return;732+733+ png_read_reset(png_ptr);734+ png_ptr->flags &= ~PNG_FLAG_ROW_INIT;735+ png_ptr->mode &= ~PNG_HAVE_fcTL;736+737+ have_chunk_after_DAT = 0;738+ for (;;)739+ {740+ png_uint_32 length = png_read_chunk_header(png_ptr);741+742+ if (png_ptr->chunk_name == png_IDAT)743+ {744+ /* Discard trailing IDATs for the first frame. */745+ if (have_chunk_after_DAT || png_ptr->num_frames_read > 1)746+ png_error(png_ptr, "Misplaced IDAT in APNG stream");747+ png_crc_finish(png_ptr, length);748+ }749+ else if (png_ptr->chunk_name == png_fcTL)750+ {751+ png_handle_fcTL(png_ptr, info_ptr, length);752+ have_chunk_after_DAT = 1;753+ }754+ else if (png_ptr->chunk_name == png_fdAT)755+ {756+ png_ensure_sequence_number(png_ptr, length);757+758+ /* Discard trailing fdATs for all frames except the first. */759+ if (!have_chunk_after_DAT && png_ptr->num_frames_read > 1)760+ {761+ png_crc_finish(png_ptr, length - 4);762+ }763+ else if (png_ptr->mode & PNG_HAVE_fcTL)764+ {765+ png_ptr->idat_size = length - 4;766+ png_ptr->mode |= PNG_HAVE_IDAT;767+ break;768+ }769+ else770+ {771+ png_error(png_ptr, "Misplaced fdAT in APNG stream");772+ }773+ }774+ else775+ {776+ png_warning(png_ptr, "Ignoring unexpected chunk in APNG sequence");777+ png_crc_finish(png_ptr, length);778+ }779+ }780+}781+#endif /* PNG_READ_APNG_SUPPORTED */782+783/* Optional call to update the users info_ptr structure */784void PNGAPI785png_read_update_info(png_structrp png_ptr, png_inforp info_ptr)786diff -Naru libpng-1.6.54.org/pngrutil.c libpng-1.6.54/pngrutil.c787--- libpng-1.6.54.org/pngrutil.c 2026-01-15 09:54:33.413881871 +0900788+++ libpng-1.6.54/pngrutil.c 2026-01-17 10:41:44.972759729 +0900789@@ -922,6 +922,11 @@790filter_type = buf[11];791interlace_type = buf[12];792793+#ifdef PNG_READ_APNG_SUPPORTED794+ png_ptr->first_frame_width = width;795+ png_ptr->first_frame_height = height;796+#endif797+798/* Set internal variables */799png_ptr->width = width;800png_ptr->height = height;801@@ -2718,6 +2723,184 @@802# define png_handle_iTXt NULL803#endif804805+#ifdef PNG_READ_APNG_SUPPORTED806+void /* PRIVATE */807+png_handle_acTL(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)808+{809+ png_byte data[8];810+ png_uint_32 num_frames;811+ png_uint_32 num_plays;812+813+ png_debug(1, "in png_handle_acTL");814+815+ if (!(png_ptr->mode & PNG_HAVE_IHDR))816+ {817+ png_error(png_ptr, "Missing IHDR before acTL");818+ }819+ else if (png_ptr->mode & PNG_HAVE_IDAT)820+ {821+ png_warning(png_ptr, "Ignoring misplaced acTL after IDAT");822+ png_crc_finish(png_ptr, length);823+ return;824+ }825+ else if (png_ptr->mode & PNG_HAVE_acTL)826+ {827+ png_warning(png_ptr, "Ignoring duplicate acTL");828+ png_crc_finish(png_ptr, length);829+ return;830+ }831+ else if (length != 8)832+ {833+ png_warning(png_ptr, "Ignoring acTL with incorrect length");834+ png_crc_finish(png_ptr, length);835+ return;836+ }837+838+ png_crc_read(png_ptr, data, 8);839+ png_crc_finish(png_ptr, 0);840+841+ num_frames = png_get_uint_31(png_ptr, data);842+ num_plays = png_get_uint_31(png_ptr, data + 4);843+844+ /* The set function will do error checking on num_frames. */845+ if (png_set_acTL(png_ptr, info_ptr, num_frames, num_plays))846+ png_ptr->mode |= PNG_HAVE_acTL;847+}848+849+void /* PRIVATE */850+png_handle_fcTL(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)851+{852+ png_byte data[22];853+ png_uint_32 width;854+ png_uint_32 height;855+ png_uint_32 x_offset;856+ png_uint_32 y_offset;857+ png_uint_16 delay_num;858+ png_uint_16 delay_den;859+ png_byte dispose_op;860+ png_byte blend_op;861+862+ png_debug(1, "in png_handle_fcTL");863+864+ png_ensure_sequence_number(png_ptr, length);865+866+ if (!(png_ptr->mode & PNG_HAVE_IHDR))867+ {868+ png_error(png_ptr, "Missing IHDR before fcTL");869+ }870+ else if (png_ptr->mode & PNG_HAVE_IDAT)871+ {872+ /* For any frames other then the first this message may be misleading,873+ * but correct. PNG_HAVE_IDAT is unset before the frame head is read.874+ * I can't think of a better message.875+ */876+ png_warning(png_ptr, "Ignoring invalid fcTL after IDAT");877+ png_crc_finish(png_ptr, length-4);878+ return;879+ }880+ else if (png_ptr->mode & PNG_HAVE_fcTL)881+ {882+ png_warning(png_ptr, "Ignoring duplicate fcTL within one frame");883+ png_crc_finish(png_ptr, length-4);884+ return;885+ }886+ else if (length != 26)887+ {888+ png_warning(png_ptr, "Ignoring fcTL with incorrect length");889+ png_crc_finish(png_ptr, length-4);890+ return;891+ }892+893+ png_crc_read(png_ptr, data, 22);894+ png_crc_finish(png_ptr, 0);895+896+ width = png_get_uint_31(png_ptr, data);897+ height = png_get_uint_31(png_ptr, data + 4);898+ x_offset = png_get_uint_31(png_ptr, data + 8);899+ y_offset = png_get_uint_31(png_ptr, data + 12);900+ delay_num = png_get_uint_16(data + 16);901+ delay_den = png_get_uint_16(data + 18);902+ dispose_op = data[20];903+ blend_op = data[21];904+905+ if (png_ptr->num_frames_read == 0 && (x_offset != 0 || y_offset != 0))906+ {907+ png_warning(png_ptr, "Ignoring leading fcTL with non-zero frame offset");908+ return;909+ }910+911+ if (info_ptr != NULL)912+ {913+ if (png_ptr->num_frames_read == 0 &&914+ (width != info_ptr->width || height != info_ptr->height))915+ {916+ png_warning(png_ptr,917+ "Ignoring leading fcTL with incorrect frame size");918+ return;919+ }920+921+ /* The set function will do more error checking. */922+ png_set_next_frame_fcTL(png_ptr, info_ptr, width, height,923+ x_offset, y_offset, delay_num, delay_den,924+ dispose_op, blend_op);925+926+ png_read_reinit(png_ptr, info_ptr);927+928+ png_ptr->mode |= PNG_HAVE_fcTL;929+ }930+}931+932+void /* PRIVATE */933+png_have_info(png_structp png_ptr, png_infop info_ptr)934+{935+ if ((info_ptr->valid & PNG_INFO_acTL) && !(info_ptr->valid & PNG_INFO_fcTL))936+ {937+ png_ptr->apng_flags |= PNG_FIRST_FRAME_HIDDEN;938+ info_ptr->num_frames++;939+ }940+}941+942+void /* PRIVATE */943+png_handle_fdAT(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)944+{945+ png_ensure_sequence_number(png_ptr, length);946+947+ /* This function is called only from png_read_end(), png_read_info(),948+ * and png_push_read_chunk(). This means one of the following:949+ * - The user doesn't want to read this frame.950+ * - This is an out-of-place fdAT.951+ * In either case, it is safe to ignore the chunk with a warning.952+ */953+ png_warning(png_ptr, "Ignoring fdAT chunk");954+ png_crc_finish(png_ptr, length - 4);955+ PNG_UNUSED(info_ptr)956+}957+958+void /* PRIVATE */959+png_ensure_sequence_number(png_structp png_ptr, png_uint_32 length)960+{961+ png_byte data[4];962+ png_uint_32 sequence_number;963+964+ if (length < 4)965+ {966+ /* TODO: Write a more precise message. */967+ png_error(png_ptr, "Invalid fcTL or fdAT chunk");968+ }969+970+ png_crc_read(png_ptr, data, 4);971+ sequence_number = png_get_uint_31(png_ptr, data);972+973+ if (sequence_number != png_ptr->next_seq_num)974+ {975+ /* TODO: Write a more precise message. */976+ png_error(png_ptr, "Out-of-order sequence number in fcTL or fdAT");977+ }978+979+ png_ptr->next_seq_num++;980+}981+#endif /* PNG_READ_APNG_SUPPORTED */982+983#ifdef PNG_READ_UNKNOWN_CHUNKS_SUPPORTED984/* Utility function for png_handle_unknown; set up png_ptr::unknown_chunk */985static int986@@ -4200,7 +4383,38 @@987{988uInt avail_in;989png_bytep buffer;990+#ifdef PNG_READ_APNG_SUPPORTED991+ png_uint_32 bytes_to_skip = 0;992+993+ while (png_ptr->idat_size == 0 || bytes_to_skip != 0)994+ {995+ png_crc_finish(png_ptr, bytes_to_skip);996+ bytes_to_skip = 0;997+998+ png_ptr->idat_size = png_read_chunk_header(png_ptr);999+ if (png_ptr->num_frames_read == 0)1000+ {1001+ if (png_ptr->chunk_name != png_IDAT)1002+ png_error(png_ptr, "Not enough image data");1003+ }1004+ else1005+ {1006+ if (png_ptr->chunk_name == png_IEND)1007+ png_error(png_ptr, "Not enough image data");1008+ if (png_ptr->chunk_name != png_fdAT)1009+ {1010+ png_warning(png_ptr,1011+ "Ignoring unexpected chunk in APNG sequence");1012+ bytes_to_skip = png_ptr->idat_size;1013+ continue;1014+ }1015+1016+ png_ensure_sequence_number(png_ptr, png_ptr->idat_size);10171018+ png_ptr->idat_size -= 4;1019+ }1020+ }1021+#else1022while (png_ptr->idat_size == 0)1023{1024png_crc_finish(png_ptr, 0);1025@@ -4212,7 +4426,7 @@1026if (png_ptr->chunk_name != png_IDAT)1027png_error(png_ptr, "Not enough image data");1028}1029-1030+#endif /* PNG_READ_APNG_SUPPORTED */1031avail_in = png_ptr->IDAT_read_size;10321033if (avail_in > png_chunk_max(png_ptr))1034@@ -4283,6 +4497,9 @@10351036png_ptr->mode |= PNG_AFTER_IDAT;1037png_ptr->flags |= PNG_FLAG_ZSTREAM_ENDED;1038+#ifdef PNG_READ_APNG_SUPPORTED1039+ png_ptr->num_frames_read++;1040+#endif10411042if (png_ptr->zstream.avail_in > 0 || png_ptr->idat_size > 0)1043png_chunk_benign_error(png_ptr, "Extra compressed data");1044@@ -4692,4 +4909,82 @@10451046png_ptr->flags |= PNG_FLAG_ROW_INIT;1047}1048+1049+#ifdef PNG_READ_APNG_SUPPORTED1050+/* This function should be called after the main IDAT sequence has been read1051+ * and before a new fdAT is about to be read. It resets some parts of png_ptr1052+ * to make them usable by the read functions again.1053+ */1054+void /* PRIVATE */1055+png_read_reset(png_structp png_ptr)1056+{1057+ png_ptr->mode &= ~PNG_HAVE_IDAT;1058+ png_ptr->mode &= ~PNG_AFTER_IDAT;1059+ png_ptr->row_number = 0;1060+ png_ptr->pass = 0;1061+}1062+1063+void /* PRIVATE */1064+png_read_reinit(png_structp png_ptr, png_infop info_ptr)1065+{1066+ png_ptr->width = info_ptr->next_frame_width;1067+ png_ptr->height = info_ptr->next_frame_height;1068+ png_ptr->rowbytes = PNG_ROWBYTES(png_ptr->pixel_depth,png_ptr->width);1069+ png_ptr->info_rowbytes = PNG_ROWBYTES(info_ptr->pixel_depth,1070+ png_ptr->width);1071+ if (png_ptr->prev_row)1072+ memset(png_ptr->prev_row, 0, png_ptr->rowbytes + 1);1073+}1074+1075+#ifdef PNG_PROGRESSIVE_READ_SUPPORTED1076+/* Same as png_read_reset(), but for the progressive reader. */1077+void /* PRIVATE */1078+png_progressive_read_reset(png_structp png_ptr)1079+{1080+#ifdef PNG_READ_INTERLACING_SUPPORTED1081+ /* Arrays to facilitate easy interlacing - use pass (0 - 6) as index */1082+1083+ /* Start of interlace block */1084+ const int png_pass_start[] = {0, 4, 0, 2, 0, 1, 0};1085+1086+ /* Offset to next interlace block */1087+ const int png_pass_inc[] = {8, 8, 4, 4, 2, 2, 1};1088+1089+ /* Start of interlace block in the y direction */1090+ const int png_pass_ystart[] = {0, 0, 4, 0, 2, 0, 1};1091+1092+ /* Offset to next interlace block in the y direction */1093+ const int png_pass_yinc[] = {8, 8, 8, 4, 4, 2, 2};1094+1095+ if (png_ptr->interlaced)1096+ {1097+ if (!(png_ptr->transformations & PNG_INTERLACE))1098+ png_ptr->num_rows = (png_ptr->height + png_pass_yinc[0] - 1 -1099+ png_pass_ystart[0]) /1100+ png_pass_yinc[0];1101+ else1102+ png_ptr->num_rows = png_ptr->height;1103+1104+ png_ptr->iwidth = (png_ptr->width +1105+ png_pass_inc[png_ptr->pass] - 1 -1106+ png_pass_start[png_ptr->pass]) /1107+ png_pass_inc[png_ptr->pass];1108+ }1109+ else1110+#endif /* PNG_READ_INTERLACING_SUPPORTED */1111+ {1112+ png_ptr->num_rows = png_ptr->height;1113+ png_ptr->iwidth = png_ptr->width;1114+ }1115+ png_ptr->flags &= ~PNG_FLAG_ZSTREAM_ENDED;1116+ if (inflateReset(&(png_ptr->zstream)) != Z_OK)1117+ png_error(png_ptr, "inflateReset failed");1118+ png_ptr->zstream.avail_in = 0;1119+ png_ptr->zstream.next_in = 0;1120+ png_ptr->zstream.next_out = png_ptr->row_buf;1121+ png_ptr->zstream.avail_out =1122+ (uInt)PNG_ROWBYTES(png_ptr->pixel_depth, png_ptr->iwidth) + 1;1123+}1124+#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */1125+#endif /* PNG_READ_APNG_SUPPORTED */1126#endif /* READ */1127diff -Naru libpng-1.6.54.org/pngset.c libpng-1.6.54/pngset.c1128--- libpng-1.6.54.org/pngset.c 2025-05-11 18:44:02.087040907 +09001129+++ libpng-1.6.54/pngset.c 2026-01-16 19:22:09.948006831 +09001130@@ -460,6 +460,13 @@1131info_ptr->pixel_depth = (png_byte)(info_ptr->channels * info_ptr->bit_depth);11321133info_ptr->rowbytes = PNG_ROWBYTES(info_ptr->pixel_depth, width);1134+1135+#ifdef PNG_APNG_SUPPORTED1136+ /* Assume a non-animated PNG in the beginning. This may be overridden after1137+ * seeing an acTL chunk later.1138+ */1139+ info_ptr->num_frames = 1;1140+#endif1141}11421143#ifdef PNG_oFFs_SUPPORTED1144@@ -1315,6 +1322,145 @@1145}1146#endif /* sPLT */11471148+#ifdef PNG_APNG_SUPPORTED1149+png_uint_32 PNGAPI1150+png_set_acTL(png_structp png_ptr, png_infop info_ptr,1151+ png_uint_32 num_frames, png_uint_32 num_plays)1152+{1153+ png_debug1(1, "in %s storage function", "acTL");1154+1155+ if (png_ptr == NULL || info_ptr == NULL)1156+ {1157+ png_warning(png_ptr,1158+ "Ignoring call to png_set_acTL with NULL libpng object args");1159+ return 0;1160+ }1161+ if (num_frames == 0)1162+ {1163+ png_warning(png_ptr,1164+ "Ignoring attempt to set acTL with num_frames zero");1165+ return 0;1166+ }1167+ if (num_frames > PNG_UINT_31_MAX)1168+ {1169+ png_warning(png_ptr,1170+ "Ignoring attempt to set acTL with num_frames > 2^31-1");1171+ return 0;1172+ }1173+ if (num_plays > PNG_UINT_31_MAX)1174+ {1175+ png_warning(png_ptr,1176+ "Ignoring attempt to set acTL with num_plays > 2^31-1");1177+ return 0;1178+ }1179+1180+ info_ptr->num_frames = num_frames;1181+ info_ptr->num_plays = num_plays;1182+1183+ info_ptr->valid |= PNG_INFO_acTL;1184+1185+ return 1;1186+}1187+1188+/* delay_num and delay_den can hold any 16-bit values including zero */1189+png_uint_32 PNGAPI1190+png_set_next_frame_fcTL(png_structp png_ptr, png_infop info_ptr,1191+ png_uint_32 width, png_uint_32 height,1192+ png_uint_32 x_offset, png_uint_32 y_offset,1193+ png_uint_16 delay_num, png_uint_16 delay_den,1194+ png_byte dispose_op, png_byte blend_op)1195+{1196+ png_debug1(1, "in %s storage function", "fcTL");1197+1198+ if (png_ptr == NULL || info_ptr == NULL)1199+ {1200+ png_warning(png_ptr,1201+ "Ignoring call to png_set_fcTL with NULL libpng object args");1202+ return 0;1203+ }1204+1205+ png_ensure_fcTL_is_valid(png_ptr, width, height, x_offset, y_offset,1206+ delay_num, delay_den, dispose_op, blend_op);1207+1208+ /* No checking is required for delay_num and delay_den.1209+ * They can hold any 16-bit value, including zero.1210+ */1211+1212+ if (blend_op == PNG_BLEND_OP_OVER)1213+ {1214+ if (!(png_ptr->color_type & PNG_COLOR_MASK_ALPHA) &&1215+ !(png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS)))1216+ {1217+ png_warning(png_ptr,1218+ "Ignoring wasteful fcTL BLEND_OP_OVER in opaque images");1219+ blend_op = PNG_BLEND_OP_SOURCE;1220+ }1221+ }1222+1223+ info_ptr->next_frame_width = width;1224+ info_ptr->next_frame_height = height;1225+ info_ptr->next_frame_x_offset = x_offset;1226+ info_ptr->next_frame_y_offset = y_offset;1227+ info_ptr->next_frame_delay_num = delay_num;1228+ info_ptr->next_frame_delay_den = delay_den;1229+ info_ptr->next_frame_dispose_op = dispose_op;1230+ info_ptr->next_frame_blend_op = blend_op;1231+1232+ info_ptr->valid |= PNG_INFO_fcTL;1233+1234+ return 1;1235+}1236+1237+void /* PRIVATE */1238+png_ensure_fcTL_is_valid(png_structp png_ptr,1239+ png_uint_32 width, png_uint_32 height,1240+ png_uint_32 x_offset, png_uint_32 y_offset,1241+ png_uint_16 delay_num, png_uint_16 delay_den,1242+ png_byte dispose_op, png_byte blend_op)1243+{1244+ if (width == 0 || width > PNG_UINT_31_MAX)1245+ png_error(png_ptr, "Invalid frame width in fcTL");1246+ if (height == 0 || height > PNG_UINT_31_MAX)1247+ png_error(png_ptr, "Invalid frame height in fcTL");1248+ if (x_offset > PNG_UINT_31_MAX || y_offset > PNG_UINT_31_MAX)1249+ png_error(png_ptr, "Invalid frame offset in fcTL");1250+ if (width + x_offset > png_ptr->first_frame_width ||1251+ height + y_offset > png_ptr->first_frame_height)1252+ png_error(png_ptr, "Oversized frame in fcTL");1253+1254+ if (dispose_op != PNG_DISPOSE_OP_NONE &&1255+ dispose_op != PNG_DISPOSE_OP_BACKGROUND &&1256+ dispose_op != PNG_DISPOSE_OP_PREVIOUS)1257+ png_error(png_ptr, "Invalid dispose_op in fcTL");1258+1259+ if (blend_op != PNG_BLEND_OP_SOURCE &&1260+ blend_op != PNG_BLEND_OP_OVER)1261+ png_error(png_ptr, "Invalid blend_op in fcTL");1262+1263+ PNG_UNUSED(delay_num)1264+ PNG_UNUSED(delay_den)1265+}1266+1267+png_uint_32 PNGAPI1268+png_set_first_frame_is_hidden(png_structp png_ptr, png_infop info_ptr,1269+ png_byte is_hidden)1270+{1271+ png_debug(1, "in png_first_frame_is_hidden");1272+1273+ if (png_ptr == NULL)1274+ return 0;1275+1276+ if (is_hidden)1277+ png_ptr->apng_flags |= PNG_FIRST_FRAME_HIDDEN;1278+ else1279+ png_ptr->apng_flags &= ~PNG_FIRST_FRAME_HIDDEN;1280+1281+ PNG_UNUSED(info_ptr)1282+1283+ return 1;1284+}1285+#endif /* PNG_APNG_SUPPORTED */1286+1287#ifdef PNG_STORE_UNKNOWN_CHUNKS_SUPPORTED1288static png_byte1289check_location(png_const_structrp png_ptr, int location)1290diff -Naru libpng-1.6.54.org/pngstruct.h libpng-1.6.54/pngstruct.h1291--- libpng-1.6.54.org/pngstruct.h 2025-12-06 20:00:54.866030732 +09001292+++ libpng-1.6.54/pngstruct.h 2026-01-15 13:50:32.229564906 +09001293@@ -391,6 +391,27 @@1294png_byte filter_type;1295#endif12961297+#ifdef PNG_APNG_SUPPORTED1298+ png_uint_32 apng_flags;1299+ png_uint_32 next_seq_num; /* next fcTL/fdAT chunk sequence number */1300+ png_uint_32 first_frame_width;1301+ png_uint_32 first_frame_height;1302+1303+#ifdef PNG_READ_APNG_SUPPORTED1304+ png_uint_32 num_frames_read; /* incremented after all image data of */1305+ /* a frame is read */1306+#ifdef PNG_PROGRESSIVE_READ_SUPPORTED1307+ png_progressive_frame_ptr frame_info_fn; /* frame info read callback */1308+ png_progressive_frame_ptr frame_end_fn; /* frame data read callback */1309+#endif1310+#endif1311+1312+#ifdef PNG_WRITE_APNG_SUPPORTED1313+ png_uint_32 num_frames_to_write;1314+ png_uint_32 num_frames_written;1315+#endif1316+#endif /* PNG_APNG_SUPPORTED */1317+1318/* New members added in libpng-1.2.0 */13191320/* New members added in libpng-1.0.2 but first enabled by default in 1.2.0 */1321diff -Naru libpng-1.6.54.org/pngtest.c libpng-1.6.54/pngtest.c1322--- libpng-1.6.54.org/pngtest.c 2026-01-15 09:54:33.413881871 +09001323+++ libpng-1.6.54/pngtest.c 2026-01-16 15:31:15.016058178 +09001324@@ -877,6 +877,10 @@1325int bit_depth, color_type;1326user_chunk_info my_user_chunk_data;1327int pass, num_passes;1328+#ifdef PNG_APNG_SUPPORTED1329+ png_uint_32 num_frames;1330+ png_uint_32 num_plays;1331+#endif13321333row_buf = NULL;1334error_parameters.file_name = inname;1335@@ -1437,6 +1441,22 @@1336}1337}1338#endif1339+1340+#ifdef PNG_APNG_SUPPORTED1341+ if (png_get_valid(read_ptr, read_info_ptr, PNG_INFO_acTL))1342+ {1343+ if (png_get_acTL(read_ptr, read_info_ptr, &num_frames, &num_plays))1344+ {1345+ png_byte is_hidden;1346+ pngtest_debug2("Handling acTL chunks (frames %ld, plays %ld)",1347+ num_frames, num_plays);1348+ png_set_acTL(write_ptr, write_info_ptr, num_frames, num_plays);1349+ is_hidden = png_get_first_frame_is_hidden(read_ptr, read_info_ptr);1350+ png_set_first_frame_is_hidden(write_ptr, write_info_ptr, is_hidden);1351+ }1352+ }1353+#endif1354+1355#ifdef PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED1356{1357png_unknown_chunkp unknowns;1358@@ -1496,6 +1516,111 @@1359t_misc += (t_stop - t_start);1360t_start = t_stop;1361#endif1362+#ifdef PNG_APNG_SUPPORTED1363+ if (png_get_valid(read_ptr, read_info_ptr, PNG_INFO_acTL))1364+ {1365+ png_uint_32 frame;1366+ for (frame = 0; frame < num_frames; frame++)1367+ {1368+ png_uint_32 frame_width;1369+ png_uint_32 frame_height;1370+ png_uint_32 x_offset;1371+ png_uint_32 y_offset;1372+ png_uint_16 delay_num;1373+ png_uint_16 delay_den;1374+ png_byte dispose_op;1375+ png_byte blend_op;1376+ png_read_frame_head(read_ptr, read_info_ptr);1377+ if (png_get_valid(read_ptr, read_info_ptr, PNG_INFO_fcTL))1378+ {1379+ png_get_next_frame_fcTL(read_ptr, read_info_ptr,1380+ &frame_width, &frame_height,1381+ &x_offset, &y_offset,1382+ &delay_num, &delay_den,1383+ &dispose_op, &blend_op);1384+ }1385+ else1386+ {1387+ frame_width = width;1388+ frame_height = height;1389+ x_offset = 0;1390+ y_offset = 0;1391+ delay_num = 1;1392+ delay_den = 1;1393+ dispose_op = PNG_DISPOSE_OP_NONE;1394+ blend_op = PNG_BLEND_OP_SOURCE;1395+ }1396+#ifdef PNG_WRITE_APNG_SUPPORTED1397+ png_write_frame_head(write_ptr, write_info_ptr,1398+ frame_width, frame_height,1399+ x_offset, y_offset,1400+ delay_num, delay_den,1401+ dispose_op, blend_op);1402+#endif1403+ for (pass = 0; pass < num_passes; pass++)1404+ {1405+# ifdef calc_pass_height1406+ png_uint_32 pass_height;1407+1408+ if (num_passes == 7) /* interlaced */1409+ {1410+ if (PNG_PASS_COLS(frame_width, pass) > 0)1411+ pass_height = PNG_PASS_ROWS(frame_height, pass);1412+1413+ else1414+ pass_height = 0;1415+ }1416+1417+ else /* not interlaced */1418+ pass_height = frame_height;1419+# else1420+# define pass_height frame_height1421+# endif1422+1423+ pngtest_debug1("Writing row data for pass %d", pass);1424+ for (y = 0; y < pass_height; y++)1425+ {1426+#ifndef SINGLE_ROWBUF_ALLOC1427+ pngtest_debug2("Allocating row buffer (pass %d, y = %u)...",1428+ pass, y);1429+1430+ row_buf = (png_bytep)png_malloc(read_ptr,1431+ png_get_rowbytes(read_ptr, read_info_ptr));1432+1433+ pngtest_debug2("\t0x%08lx (%lu bytes)", (unsigned long)row_buf,1434+ (unsigned long)png_get_rowbytes(read_ptr, read_info_ptr));1435+1436+#endif /* !SINGLE_ROWBUF_ALLOC */1437+ png_read_rows(read_ptr, (png_bytepp)&row_buf, NULL, 1);1438+1439+#ifdef PNG_WRITE_SUPPORTED1440+#ifdef PNGTEST_TIMING1441+ t_stop = (float)clock();1442+ t_decode += (t_stop - t_start);1443+ t_start = t_stop;1444+#endif1445+ png_write_rows(write_ptr, (png_bytepp)&row_buf, 1);1446+#ifdef PNGTEST_TIMING1447+ t_stop = (float)clock();1448+ t_encode += (t_stop - t_start);1449+ t_start = t_stop;1450+#endif1451+#endif /* PNG_WRITE_SUPPORTED */1452+1453+#ifndef SINGLE_ROWBUF_ALLOC1454+ pngtest_debug2("Freeing row buffer (pass %d, y = %u)", pass, y);1455+ png_free(read_ptr, row_buf);1456+ row_buf = NULL;1457+#endif /* !SINGLE_ROWBUF_ALLOC */1458+ }1459+ }1460+#ifdef PNG_WRITE_APNG_SUPPORTED1461+ png_write_frame_tail(write_ptr, write_info_ptr);1462+#endif1463+ }1464+ }1465+ else1466+#endif1467for (pass = 0; pass < num_passes; pass++)1468{1469# ifdef calc_pass_height1470diff -Naru libpng-1.6.54.org/pngwrite.c libpng-1.6.54/pngwrite.c1471--- libpng-1.6.54.org/pngwrite.c 2026-01-15 09:54:33.413881871 +09001472+++ libpng-1.6.54/pngwrite.c 2026-01-16 16:49:59.704764231 +09001473@@ -127,6 +127,11 @@1474* the application continues writing the PNG. So check the 'invalid'1475* flag here too.1476*/1477+#ifdef PNG_WRITE_APNG_SUPPORTED1478+ if ((info_ptr->valid & PNG_INFO_acTL) != 0)1479+ png_write_acTL(png_ptr, info_ptr->num_frames, info_ptr->num_plays);1480+#endif1481+1482#ifdef PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED1483/* Write unknown chunks first; PNG v3 establishes a precedence order1484* for colourspace chunks. It is certain therefore that new1485@@ -405,6 +410,11 @@1486png_benign_error(png_ptr, "Wrote palette index exceeding num_palette");1487#endif14881489+#ifdef PNG_WRITE_APNG_SUPPORTED1490+ if (png_ptr->num_frames_written != png_ptr->num_frames_to_write)1491+ png_error(png_ptr, "Not enough frames written");1492+#endif1493+1494/* See if user wants us to write information chunks */1495if (info_ptr != NULL)1496{1497@@ -1517,6 +1527,42 @@1498}1499#endif15001501+#ifdef PNG_WRITE_APNG_SUPPORTED1502+void PNGAPI1503+png_write_frame_head(png_structp png_ptr, png_infop info_ptr,1504+ png_uint_32 width, png_uint_32 height,1505+ png_uint_32 x_offset, png_uint_32 y_offset,1506+ png_uint_16 delay_num, png_uint_16 delay_den,1507+ png_byte dispose_op, png_byte blend_op)1508+{1509+ png_debug(1, "in png_write_frame_head");1510+1511+ /* There is a chance this has been set after png_write_info was called,1512+ * so it would be set but not written. Is there a way to be sure?1513+ */1514+ if (!(info_ptr->valid & PNG_INFO_acTL))1515+ png_error(png_ptr, "Cannot write APNG frame: missing acTL");1516+1517+ png_write_reset(png_ptr);1518+1519+ png_write_reinit(png_ptr, info_ptr, width, height);1520+1521+ if (!(png_ptr->num_frames_written == 0 &&1522+ (png_ptr->apng_flags & PNG_FIRST_FRAME_HIDDEN)))1523+ png_write_fcTL(png_ptr, width, height, x_offset, y_offset,1524+ delay_num, delay_den, dispose_op, blend_op);1525+}1526+1527+void PNGAPI1528+png_write_frame_tail(png_structp png_ptr, png_infop info_ptr)1529+{1530+ png_debug(1, "in png_write_frame_tail");1531+1532+ png_ptr->num_frames_written++;1533+1534+ PNG_UNUSED(info_ptr)1535+}1536+#endif /* PNG_WRITE_APNG_SUPPORTED */15371538#ifdef PNG_SIMPLIFIED_WRITE_SUPPORTED1539/* Initialize the write structure - general purpose utility. */1540diff -Naru libpng-1.6.54.org/pngwutil.c libpng-1.6.54/pngwutil.c1541--- libpng-1.6.54.org/pngwutil.c 2026-01-15 09:54:33.414881869 +09001542+++ libpng-1.6.54/pngwutil.c 2026-01-16 21:11:48.539248758 +09001543@@ -838,6 +838,11 @@1544/* Write the chunk */1545png_write_complete_chunk(png_ptr, png_IHDR, buf, 13);15461547+#ifdef PNG_WRITE_APNG_SUPPORTED1548+ png_ptr->first_frame_width = width;1549+ png_ptr->first_frame_height = height;1550+#endif1551+1552if ((png_ptr->do_filter) == PNG_NO_FILTERS)1553{1554if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE ||1555@@ -1020,7 +1025,17 @@1556#endif15571558if (size > 0)1559+ {1560+#ifdef PNG_WRITE_APNG_SUPPORTED1561+ if (png_ptr->num_frames_written == 0)1562+ png_write_complete_chunk(png_ptr, png_IDAT, data, size);1563+ else1564+ png_write_fdAT(png_ptr, data, size);1565+#else1566png_write_complete_chunk(png_ptr, png_IDAT, data, size);1567+#endif /* PNG_WRITE_APNG_SUPPORTED */1568+ }1569+1570png_ptr->mode |= PNG_HAVE_IDAT;15711572png_ptr->zstream.next_out = data;1573@@ -1067,7 +1082,17 @@1574#endif15751576if (size > 0)1577+ {1578+#ifdef PNG_WRITE_APNG_SUPPORTED1579+ if (png_ptr->num_frames_written == 0)1580+ png_write_complete_chunk(png_ptr, png_IDAT, data, size);1581+ else1582+ png_write_fdAT(png_ptr, data, size);1583+#else1584png_write_complete_chunk(png_ptr, png_IDAT, data, size);1585+#endif /* PNG_WRITE_APNG_SUPPORTED */1586+ }1587+1588png_ptr->zstream.avail_out = 0;1589png_ptr->zstream.next_out = NULL;1590png_ptr->mode |= PNG_HAVE_IDAT | PNG_AFTER_IDAT;1591@@ -1969,6 +1994,82 @@1592}1593#endif15941595+#ifdef PNG_WRITE_APNG_SUPPORTED1596+void /* PRIVATE */1597+png_write_acTL(png_structp png_ptr,1598+ png_uint_32 num_frames, png_uint_32 num_plays)1599+{1600+ png_byte buf[8];1601+1602+ png_debug(1, "in png_write_acTL");1603+1604+ png_ptr->num_frames_to_write = num_frames;1605+1606+ if (png_ptr->apng_flags & PNG_FIRST_FRAME_HIDDEN)1607+ num_frames--;1608+1609+ png_save_uint_32(buf, num_frames);1610+ png_save_uint_32(buf + 4, num_plays);1611+1612+ png_write_complete_chunk(png_ptr, png_acTL, buf, (png_size_t)8);1613+}1614+1615+void /* PRIVATE */1616+png_write_fcTL(png_structp png_ptr,1617+ png_uint_32 width, png_uint_32 height,1618+ png_uint_32 x_offset, png_uint_32 y_offset,1619+ png_uint_16 delay_num, png_uint_16 delay_den,1620+ png_byte dispose_op, png_byte blend_op)1621+{1622+ png_byte buf[26];1623+1624+ png_debug(1, "in png_write_fcTL");1625+1626+ if (png_ptr->num_frames_written == 0 && (x_offset != 0 || y_offset != 0))1627+ png_error(png_ptr, "Non-zero frame offset in leading fcTL");1628+ if (png_ptr->num_frames_written == 0 &&1629+ (width != png_ptr->first_frame_width ||1630+ height != png_ptr->first_frame_height))1631+ png_error(png_ptr, "Incorrect frame size in leading fcTL");1632+1633+ /* More error checking. */1634+ png_ensure_fcTL_is_valid(png_ptr, width, height, x_offset, y_offset,1635+ delay_num, delay_den, dispose_op, blend_op);1636+1637+ png_save_uint_32(buf, png_ptr->next_seq_num);1638+ png_save_uint_32(buf + 4, width);1639+ png_save_uint_32(buf + 8, height);1640+ png_save_uint_32(buf + 12, x_offset);1641+ png_save_uint_32(buf + 16, y_offset);1642+ png_save_uint_16(buf + 20, delay_num);1643+ png_save_uint_16(buf + 22, delay_den);1644+ buf[24] = dispose_op;1645+ buf[25] = blend_op;1646+1647+ png_write_complete_chunk(png_ptr, png_fcTL, buf, (png_size_t)26);1648+1649+ png_ptr->next_seq_num++;1650+}1651+1652+void /* PRIVATE */1653+png_write_fdAT(png_structp png_ptr,1654+ png_const_bytep data, png_size_t length)1655+{1656+ png_byte buf[4];1657+1658+ png_write_chunk_header(png_ptr, png_fdAT, (png_uint_32)(4 + length));1659+1660+ png_save_uint_32(buf, png_ptr->next_seq_num);1661+ png_write_chunk_data(png_ptr, buf, 4);1662+1663+ png_write_chunk_data(png_ptr, data, length);1664+1665+ png_write_chunk_end(png_ptr);1666+1667+ png_ptr->next_seq_num++;1668+}1669+#endif /* PNG_WRITE_APNG_SUPPORTED */1670+1671/* Initializes the row writing capability of libpng */1672void /* PRIVATE */1673png_write_start_row(png_structrp png_ptr)1674@@ -2822,4 +2923,37 @@1675}1676#endif /* WRITE_FLUSH */1677}1678+1679+#ifdef PNG_WRITE_APNG_SUPPORTED1680+void /* PRIVATE */1681+png_write_reset(png_structp png_ptr)1682+{1683+ png_ptr->row_number = 0;1684+ png_ptr->pass = 0;1685+ png_ptr->mode &= ~PNG_HAVE_IDAT;1686+}1687+1688+void /* PRIVATE */1689+png_write_reinit(png_structp png_ptr, png_infop info_ptr,1690+ png_uint_32 width, png_uint_32 height)1691+{1692+ if (png_ptr->num_frames_written == 0 &&1693+ (width != png_ptr->first_frame_width ||1694+ height != png_ptr->first_frame_height))1695+ png_error(png_ptr, "Incorrect frame size in leading fcTL");1696+ if (width > png_ptr->first_frame_width ||1697+ height > png_ptr->first_frame_height)1698+ png_error(png_ptr, "Oversized frame in fcTL");1699+1700+ png_set_IHDR(png_ptr, info_ptr, width, height,1701+ info_ptr->bit_depth, info_ptr->color_type,1702+ info_ptr->interlace_type, info_ptr->compression_type,1703+ info_ptr->filter_type);1704+1705+ png_ptr->width = width;1706+ png_ptr->height = height;1707+ png_ptr->rowbytes = PNG_ROWBYTES(png_ptr->pixel_depth, width);1708+ png_ptr->usr_width = png_ptr->width;1709+}1710+#endif /* PNG_WRITE_APNG_SUPPORTED */1711#endif /* WRITE */1712diff -Naru libpng-1.6.54.org/scripts/symbols.def libpng-1.6.54/scripts/symbols.def1713--- libpng-1.6.54.org/scripts/symbols.def 2025-01-26 08:07:01.594606745 +09001714+++ libpng-1.6.54/scripts/symbols.def 2026-01-15 13:50:32.230564905 +09001715@@ -263,3 +263,23 @@1716png_get_mDCV_fixed @2571717png_set_mDCV @2581718png_set_mDCV_fixed @2591719+ png_get_acTL @2601720+ png_set_acTL @2611721+ png_get_num_frames @2621722+ png_get_num_plays @2631723+ png_get_next_frame_fcTL @2641724+ png_set_next_frame_fcTL @2651725+ png_get_next_frame_width @2661726+ png_get_next_frame_height @2671727+ png_get_next_frame_x_offset @2681728+ png_get_next_frame_y_offset @2691729+ png_get_next_frame_delay_num @2701730+ png_get_next_frame_delay_den @2711731+ png_get_next_frame_dispose_op @2721732+ png_get_next_frame_blend_op @2731733+ png_get_first_frame_is_hidden @2741734+ png_set_first_frame_is_hidden @2751735+ png_read_frame_head @2761736+ png_set_progressive_frame_fn @2771737+ png_write_frame_head @2781738+ png_write_frame_tail @279173917401741