fix recent png loader break with etc1 support that broke interlaced imgs
authorCarsten Haitzler (Rasterman) <raster@rasterman.com>
Tue, 15 Apr 2014 08:56:07 +0000 (17:56 +0900)
committerCarsten Haitzler (Rasterman) <raster@rasterman.com>
Tue, 15 Apr 2014 08:56:07 +0000 (17:56 +0900)
this fixes the png loader code to use png_read_row properly with the
number of passes needed to load aninterlaced image as well as handling
this right with scale ratio scaledown set.

src/modules/evas/loaders/png/evas_image_load_png.c

index d2e84c3537d2d226266a6ccd356afdf1febde43f..f155d8ba23d8c2f3a38a9c5fa5dfcb6e38cc1b93 100644 (file)
@@ -218,9 +218,9 @@ evas_image_load_file_data_png(void *loader_data,
    unsigned int pack_offset;
    int w, h;
    int bit_depth, color_type, interlace_type;
-   char hasa;
-   int i, j;
-   int scale_ratio = 1, image_w = 0;
+   char hasa, passes;
+   int i, j, p, k;
+   int scale_ratio = 1, image_w = 0, image_h = 0;
    Eina_Bool r = EINA_FALSE;
 
    opts = loader->opts;
@@ -278,6 +278,7 @@ evas_image_load_file_data_png(void *loader_data,
                (png_uint_32 *) (&h32), &bit_depth, &color_type,
                &interlace_type, NULL, NULL);
    image_w = w32;
+   image_h = h32;
    if (opts->scale_down_by > 1)
      {
         scale_ratio = opts->scale_down_by;
@@ -348,32 +349,65 @@ evas_image_load_file_data_png(void *loader_data,
       default: abort();
      }
 
+   passes = png_set_interlace_handling(png_ptr);
+   
    /* we read image line by line if scale down was set */
    if (scale_ratio == 1)
      {
-        for (i = 0; i < h; i++)
-          png_read_row(png_ptr, surface + (i * w * pack_offset), NULL);
+        for (p = 0; p < passes; p++)
+          {
+             for (i = 0; i < h; i++)
+               png_read_row(png_ptr, surface + (i * w * pack_offset), NULL);
+          }
         png_read_end(png_ptr, info_ptr);
      }
    else
      {
         unsigned char *src_ptr, *dst_ptr;
 
-        tmp_line = (unsigned char *) alloca(image_w * pack_offset);
         dst_ptr = surface;
-        for (i = 0; i < h; i++)
+        if (passes == 1)
           {
-             png_read_row(png_ptr, tmp_line, NULL);
-             src_ptr = tmp_line;
-             for (j = 0; j < w; j++)
+             tmp_line = (unsigned char *) alloca(image_w * pack_offset);
+             for (i = 0; i < h; i++)
                {
-                  *dst_ptr = *src_ptr;
-                  dst_ptr += pack_offset;
-                  src_ptr += scale_ratio * pack_offset;
+                  png_read_row(png_ptr, tmp_line, NULL);
+                  src_ptr = tmp_line;
+                  for (j = 0; j < w; j++)
+                    {
+                       for (k = 0; k < (int)pack_offset; k++)
+                         dst_ptr[k] = src_ptr[k];
+                       dst_ptr += pack_offset;
+                       src_ptr += scale_ratio * pack_offset;
+                    }
+                  for (j = 0; j < (scale_ratio - 1); j++)
+                    png_read_row(png_ptr, tmp_line, NULL);
                }
-             for (j = 0; j < (scale_ratio - 1); j++)
+          }
+        else
+          {
+             unsigned char *pixels2 = malloc(image_w * image_h * pack_offset);
+
+             if (pixels2)
                {
-                  png_read_row(png_ptr, tmp_line, NULL);
+                  for (p = 0; p < passes; p++)
+                    {
+                       for (i = 0; i < image_h; i++)
+                         png_read_row(png_ptr, pixels2 + (i * image_w * pack_offset), NULL);
+                    }
+                  
+                  for (i = 0; i < h; i++)
+                    {
+                       src_ptr = pixels2 + (i * scale_ratio * image_w * pack_offset);
+                       for (j = 0; j < w; j++)
+                         {
+                            for (k = 0; k < (int)pack_offset; k++)
+                              dst_ptr[k] = src_ptr[k];
+                            src_ptr += scale_ratio * pack_offset;
+                            dst_ptr += pack_offset;
+                         }
+                    }
+                  free(pixels2);
                }
           }
      }