| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup datatoc | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* for bool */ | 
					
						
							|  |  |  | #include "../blenlib/BLI_sys_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* for DIR */ | 
					
						
							| 
									
										
										
										
											2014-01-14 23:57:00 +02:00
										 |  |  | #if !defined(WIN32) || defined(FREEWINDOWS)
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | #  include <dirent.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "png.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-15 08:38:00 +11:00
										 |  |  | /* for Win32 DIR functions */ | 
					
						
							|  |  |  | #ifdef WIN32
 | 
					
						
							|  |  |  | #  include "../blenlib/BLI_winstuff.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | #ifdef WIN32
 | 
					
						
							|  |  |  | #  define SEP '\\'
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #  define SEP '/'
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* Utility functions */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int path_ensure_slash(char *string) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int len = strlen(string); | 
					
						
							|  |  |  |   if (len == 0 || string[len - 1] != SEP) { | 
					
						
							|  |  |  |     string[len] = SEP; | 
					
						
							|  |  |  |     string[len + 1] = '\0'; | 
					
						
							|  |  |  |     return len + 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool path_test_extension(const char *str, const char *ext) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const size_t a = strlen(str); | 
					
						
							|  |  |  |   const size_t b = strlen(ext); | 
					
						
							|  |  |  |   return !(a == 0 || b == 0 || b >= a) && (strcmp(ext, str + a - b) == 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  | static void endian_switch_uint32(uint *val) | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint tval = *val; | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   *val = ((tval >> 24)) | ((tval << 8) & 0x00ff0000) | ((tval >> 8) & 0x0000ff00) | ((tval << 24)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  | static const char *path_slash_rfind(const char *string) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const char *const lfslash = strrchr(string, '/'); | 
					
						
							|  |  |  |   const char *const lbslash = strrchr(string, '\\'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!lfslash) { | 
					
						
							|  |  |  |     return lbslash; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!lbslash) { | 
					
						
							|  |  |  |     return lfslash; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (lfslash > lbslash) ? lfslash : lbslash; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *path_basename(const char *path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const char *const filename = path_slash_rfind(path); | 
					
						
							|  |  |  |   return filename ? filename + 1 : path; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* Write a PNG from RGBA pixels */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  | static bool write_png(const char *name, const uint *pixels, const int width, const int height) | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | { | 
					
						
							|  |  |  |   png_structp png_ptr; | 
					
						
							|  |  |  |   png_infop info_ptr; | 
					
						
							|  |  |  |   png_bytepp row_pointers = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   FILE *fp; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   const int bytesperpixel = 4; | 
					
						
							|  |  |  |   const int compression = 9; | 
					
						
							|  |  |  |   int i; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   fp = fopen(name, "wb"); | 
					
						
							|  |  |  |   if (fp == NULL) { | 
					
						
							|  |  |  |     printf("%s: Cannot open file for writing '%s'\n", __func__, name); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); | 
					
						
							|  |  |  |   if (png_ptr == NULL) { | 
					
						
							|  |  |  |     printf("%s: Cannot png_create_write_struct for file: '%s'\n", __func__, name); | 
					
						
							|  |  |  |     fclose(fp); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   info_ptr = png_create_info_struct(png_ptr); | 
					
						
							|  |  |  |   if (info_ptr == NULL) { | 
					
						
							|  |  |  |     png_destroy_write_struct(&png_ptr, (png_infopp)NULL); | 
					
						
							|  |  |  |     printf("%s: Cannot png_create_info_struct for file: '%s'\n", __func__, name); | 
					
						
							|  |  |  |     fclose(fp); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   if (setjmp(png_jmpbuf(png_ptr))) { | 
					
						
							|  |  |  |     png_destroy_write_struct(&png_ptr, &info_ptr); | 
					
						
							|  |  |  |     printf("%s: Cannot setjmp for file: '%s'\n", __func__, name); | 
					
						
							|  |  |  |     fclose(fp); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* write the file */ | 
					
						
							|  |  |  |   png_init_io(png_ptr, fp); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   png_set_compression_level(png_ptr, compression); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* png image settings */ | 
					
						
							|  |  |  |   png_set_IHDR(png_ptr, | 
					
						
							|  |  |  |                info_ptr, | 
					
						
							|  |  |  |                width, | 
					
						
							|  |  |  |                height, | 
					
						
							|  |  |  |                8, | 
					
						
							|  |  |  |                PNG_COLOR_TYPE_RGBA, | 
					
						
							|  |  |  |                PNG_INTERLACE_NONE, | 
					
						
							|  |  |  |                PNG_COMPRESSION_TYPE_DEFAULT, | 
					
						
							|  |  |  |                PNG_FILTER_TYPE_DEFAULT); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* write the file header information */ | 
					
						
							|  |  |  |   png_write_info(png_ptr, info_ptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef __LITTLE_ENDIAN__
 | 
					
						
							|  |  |  |   png_set_swap(png_ptr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* allocate memory for an array of row-pointers */ | 
					
						
							|  |  |  |   row_pointers = (png_bytepp)malloc(height * sizeof(png_bytep)); | 
					
						
							|  |  |  |   if (row_pointers == NULL) { | 
					
						
							|  |  |  |     printf("%s: Cannot allocate row-pointers array for file '%s'\n", __func__, name); | 
					
						
							|  |  |  |     png_destroy_write_struct(&png_ptr, &info_ptr); | 
					
						
							|  |  |  |     if (fp) { | 
					
						
							|  |  |  |       fclose(fp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* set the individual row-pointers to point at the correct offsets */ | 
					
						
							|  |  |  |   for (i = 0; i < height; i++) { | 
					
						
							| 
									
										
										
										
											2021-07-30 16:19:19 +10:00
										 |  |  |     row_pointers[height - 1 - i] = (png_bytep)(((const unsigned char *)pixels) + | 
					
						
							|  |  |  |                                                (i * width) * bytesperpixel * | 
					
						
							|  |  |  |                                                    sizeof(unsigned char)); | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* write out the entire image data in one call */ | 
					
						
							|  |  |  |   png_write_image(png_ptr, row_pointers); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* write the additional chunks to the PNG file (not really needed) */ | 
					
						
							|  |  |  |   png_write_end(png_ptr, info_ptr); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   /* clean up */ | 
					
						
							|  |  |  |   free(row_pointers); | 
					
						
							|  |  |  |   png_destroy_write_struct(&png_ptr, &info_ptr); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   fflush(fp); | 
					
						
							|  |  |  |   fclose(fp); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* Merge icon-data from files */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct IconHead { | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint icon_w, icon_h; | 
					
						
							|  |  |  |   uint orig_x, orig_y; | 
					
						
							|  |  |  |   uint canvas_w, canvas_h; | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  | struct IconInfo { | 
					
						
							|  |  |  |   struct IconHead head; | 
					
						
							|  |  |  |   char *file_name; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct IconMergeContext { | 
					
						
							|  |  |  |   /* Information about all icons read from disk.
 | 
					
						
							|  |  |  |    * Is used for sanity checks like prevention of two files defining icon for | 
					
						
							|  |  |  |    * the same position on canvas. */ | 
					
						
							|  |  |  |   int num_read_icons; | 
					
						
							|  |  |  |   struct IconInfo *read_icons; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void icon_merge_context_init(struct IconMergeContext *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   context->num_read_icons = 0; | 
					
						
							|  |  |  |   context->read_icons = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Get icon information from the context which matches given icon head.
 | 
					
						
							|  |  |  |  * Is used to check whether icon is re-defined, and to provide useful information about which | 
					
						
							|  |  |  |  * files are conflicting. */ | 
					
						
							|  |  |  | static struct IconInfo *icon_merge_context_info_for_icon_head(struct IconMergeContext *context, | 
					
						
							|  |  |  |                                                               struct IconHead *icon_head) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (context->read_icons == NULL) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int i = 0; i < context->num_read_icons; i++) { | 
					
						
							|  |  |  |     struct IconInfo *read_icon_info = &context->read_icons[i]; | 
					
						
							|  |  |  |     const struct IconHead *read_icon_head = &read_icon_info->head; | 
					
						
							|  |  |  |     if (read_icon_head->orig_x == icon_head->orig_x && | 
					
						
							|  |  |  |         read_icon_head->orig_y == icon_head->orig_y) { | 
					
						
							|  |  |  |       return read_icon_info; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void icon_merge_context_register_icon(struct IconMergeContext *context, | 
					
						
							|  |  |  |                                              const char *file_name, | 
					
						
							|  |  |  |                                              struct IconHead *icon_head) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   context->read_icons = realloc(context->read_icons, | 
					
						
							|  |  |  |                                 sizeof(struct IconInfo) * (context->num_read_icons + 1)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   struct IconInfo *icon_info = &context->read_icons[context->num_read_icons]; | 
					
						
							|  |  |  |   icon_info->head = *icon_head; | 
					
						
							|  |  |  |   icon_info->file_name = strdup(path_basename(file_name)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   context->num_read_icons++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void icon_merge_context_free(struct IconMergeContext *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (context->read_icons != NULL) { | 
					
						
							|  |  |  |     for (int i = 0; i < context->num_read_icons; i++) { | 
					
						
							|  |  |  |       free(context->read_icons[i].file_name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     free(context->read_icons); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | static bool icon_decode_head(FILE *f_src, struct IconHead *r_head) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (fread(r_head, 1, sizeof(*r_head), f_src) == sizeof(*r_head)) { | 
					
						
							|  |  |  | #ifndef __LITTLE_ENDIAN__
 | 
					
						
							|  |  |  |     endian_switch_uint32(&r_head->icon_w); | 
					
						
							|  |  |  |     endian_switch_uint32(&r_head->icon_h); | 
					
						
							|  |  |  |     endian_switch_uint32(&r_head->orig_x); | 
					
						
							|  |  |  |     endian_switch_uint32(&r_head->orig_y); | 
					
						
							|  |  |  |     endian_switch_uint32(&r_head->canvas_w); | 
					
						
							|  |  |  |     endian_switch_uint32(&r_head->canvas_h); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* quiet warning */ | 
					
						
							|  |  |  |   (void)endian_switch_uint32; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  | static bool icon_decode(FILE *f_src, struct IconHead *r_head, uint **r_pixels) | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint *pixels; | 
					
						
							|  |  |  |   uint pixels_size; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   if (!icon_decode_head(f_src, r_head)) { | 
					
						
							|  |  |  |     printf("%s: failed to read header\n", __func__); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   pixels_size = sizeof(char[4]) * r_head->icon_w * r_head->icon_h; | 
					
						
							|  |  |  |   pixels = malloc(pixels_size); | 
					
						
							|  |  |  |   if (pixels == NULL) { | 
					
						
							|  |  |  |     printf("%s: failed to allocate pixels\n", __func__); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   if (fread(pixels, 1, pixels_size, f_src) != pixels_size) { | 
					
						
							|  |  |  |     printf("%s: failed to read pixels\n", __func__); | 
					
						
							|  |  |  |     free(pixels); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   *r_pixels = pixels; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  | static bool icon_read(const char *file_src, struct IconHead *r_head, uint **r_pixels) | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | { | 
					
						
							|  |  |  |   FILE *f_src; | 
					
						
							|  |  |  |   bool success; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   f_src = fopen(file_src, "rb"); | 
					
						
							|  |  |  |   if (f_src == NULL) { | 
					
						
							|  |  |  |     printf("%s: failed to open '%s'\n", __func__, file_src); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   success = icon_decode(f_src, r_head, r_pixels); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fclose(f_src); | 
					
						
							|  |  |  |   return success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  | static bool icon_merge(struct IconMergeContext *context, | 
					
						
							|  |  |  |                        const char *file_src, | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:28 +10:00
										 |  |  |                        uint32_t **r_pixels_canvas, | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |                        uint *r_canvas_w, | 
					
						
							|  |  |  |                        uint *r_canvas_h) | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | { | 
					
						
							|  |  |  |   struct IconHead head; | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint *pixels; | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint x, y; | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* canvas */ | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:28 +10:00
										 |  |  |   uint32_t *pixels_canvas; | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint canvas_w, canvas_h; | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!icon_read(file_src, &head, &pixels)) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  |   struct IconInfo *read_icon_info = icon_merge_context_info_for_icon_head(context, &head); | 
					
						
							|  |  |  |   if (read_icon_info != NULL) { | 
					
						
							|  |  |  |     printf( | 
					
						
							|  |  |  |         "Conflicting icon files %s and %s\n", path_basename(file_src), read_icon_info->file_name); | 
					
						
							|  |  |  |     free(pixels); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   icon_merge_context_register_icon(context, file_src, &head); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   if (*r_canvas_w == 0) { | 
					
						
							|  |  |  |     /* init once */ | 
					
						
							|  |  |  |     *r_canvas_w = head.canvas_w; | 
					
						
							|  |  |  |     *r_canvas_h = head.canvas_h; | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:28 +10:00
										 |  |  |     *r_pixels_canvas = calloc(1, (head.canvas_w * head.canvas_h) * sizeof(uint32_t)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   canvas_w = *r_canvas_w; | 
					
						
							|  |  |  |   canvas_h = *r_canvas_h; | 
					
						
							|  |  |  |   pixels_canvas = *r_pixels_canvas; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   assert(head.canvas_w == canvas_w); | 
					
						
							|  |  |  |   assert(head.canvas_h == canvas_h); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (x = 0; x < head.icon_w; x++) { | 
					
						
							|  |  |  |     for (y = 0; y < head.icon_h; y++) { | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |       uint pixel; | 
					
						
							|  |  |  |       uint dst_x, dst_y; | 
					
						
							|  |  |  |       uint pixel_xy_dst; | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* get pixel */ | 
					
						
							|  |  |  |       pixel = pixels[(y * head.icon_w) + x]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* set pixel */ | 
					
						
							|  |  |  |       dst_x = head.orig_x + x; | 
					
						
							|  |  |  |       dst_y = head.orig_y + y; | 
					
						
							|  |  |  |       pixel_xy_dst = (dst_y * canvas_w) + dst_x; | 
					
						
							|  |  |  |       assert(pixel_xy_dst < (canvas_w * canvas_h)); | 
					
						
							|  |  |  |       pixels_canvas[pixel_xy_dst] = pixel; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   free(pixels); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 15:48:22 +10:00
										 |  |  |   /* only for bounds check */ | 
					
						
							|  |  |  |   (void)canvas_h; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool icondir_to_png(const char *path_src, const char *file_dst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Takes a path full of 'dat' files and writes out */ | 
					
						
							|  |  |  |   DIR *dir; | 
					
						
							|  |  |  |   const struct dirent *fname; | 
					
						
							|  |  |  |   char filepath[1024]; | 
					
						
							|  |  |  |   char *filename; | 
					
						
							|  |  |  |   int path_str_len; | 
					
						
							|  |  |  |   int found = 0, fail = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  |   struct IconMergeContext context; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:28 +10:00
										 |  |  |   uint32_t *pixels_canvas = NULL; | 
					
						
							| 
									
										
										
										
											2021-06-13 14:47:25 +10:00
										 |  |  |   uint canvas_w = 0, canvas_h = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  |   icon_merge_context_init(&context); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   errno = 0; | 
					
						
							|  |  |  |   dir = opendir(path_src); | 
					
						
							|  |  |  |   if (dir == NULL) { | 
					
						
							|  |  |  |     printf( | 
					
						
							|  |  |  |         "%s: failed to dir '%s', (%s)\n", __func__, path_src, errno ? strerror(errno) : "unknown"); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   strcpy(filepath, path_src); | 
					
						
							|  |  |  |   path_str_len = path_ensure_slash(filepath); | 
					
						
							|  |  |  |   filename = &filepath[path_str_len]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   while ((fname = readdir(dir)) != NULL) { | 
					
						
							|  |  |  |     if (path_test_extension(fname->d_name, ".dat")) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |       strcpy(filename, fname->d_name); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  |       if (icon_merge(&context, filepath, &pixels_canvas, &canvas_w, &canvas_h)) { | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |         found++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         fail++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  |   icon_merge_context_free(&context); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   closedir(dir); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   if (found == 0) { | 
					
						
							|  |  |  |     printf("%s: dir '%s' has no icons\n", __func__, path_src); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   if (fail != 0) { | 
					
						
							|  |  |  |     printf("%s: dir '%s' failed %d icons\n", __func__, path_src, fail); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-24 15:56:58 +10:00
										 |  |  |   /* Write pixels. */ | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   write_png(file_dst, pixels_canvas, canvas_w, canvas_h); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  |   free(pixels_canvas); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 10:27:48 +01:00
										 |  |  |   return (fail == 0); | 
					
						
							| 
									
										
										
										
											2014-01-13 23:47:33 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* Main and parse args */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int main(int argc, char **argv) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const char *path_src; | 
					
						
							|  |  |  |   const char *file_dst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (argc < 3) { | 
					
						
							|  |  |  |     printf("Usage: datatoc_icon <dir_icons> <data_icon_to.png>\n"); | 
					
						
							|  |  |  |     exit(1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   path_src = argv[1]; | 
					
						
							|  |  |  |   file_dst = argv[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (icondir_to_png(path_src, file_dst) == true) ? 0 : 1; | 
					
						
							|  |  |  | } |