/* timepng.c * * Copyright (c) 2013 John Cunningham Bowler * * Last changed in libpng 1.6.1 [March 28, 2013] * * This code is released under the libpng license. * For conditions of distribution and use, see the disclaimer * and license in png.h * * Load an arbitrary number of PNG files (from the command line, or, if there * are no arguments on the command line, from stdin) then run a time test by * reading each file by row. The test does nothing with the read result and * does no transforms. The only output is a time as a floating point number of * seconds with 9 decimal digits. */ #define _POSIX_C_SOURCE 199309L /* for clock_gettime */ #include <stdlib.h> #include <stdio.h> #include <string.h> #include <time.h> #if defined(HAVE_CONFIG_H) && !defined(PNG_NO_CONFIG_H) # include <config.h> #endif /* Define the following to use this test against your installed libpng, rather * than the one being built here: */ #ifdef PNG_FREESTANDING_TESTS # include <png.h> #else # include "../../png.h" #endif static int read_png(FILE *fp) { png_structp png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING,0,0,0); png_infop info_ptr = NULL; png_bytep row = NULL, display = NULL; if (png_ptr == NULL) return 0; if (setjmp(png_jmpbuf(png_ptr))) { png_destroy_read_struct(&png_ptr, &info_ptr, NULL); if (row != NULL) free(row); if (display != NULL) free(display); return 0; } png_init_io(png_ptr, fp); info_ptr = png_create_info_struct(png_ptr); if (info_ptr == NULL) png_error(png_ptr, "OOM allocating info structure"); png_read_info(png_ptr, info_ptr); { png_size_t rowbytes = png_get_rowbytes(png_ptr, info_ptr); row = malloc(rowbytes); display = malloc(rowbytes); if (row == NULL || display == NULL) png_error(png_ptr, "OOM allocating row buffers"); { png_uint_32 height = png_get_image_height(png_ptr, info_ptr); int passes = png_set_interlace_handling(png_ptr); int pass; png_start_read_image(png_ptr); for (pass = 0; pass < passes; ++pass) { png_uint_32 y = height; /* NOTE: this trashes the row each time; interlace handling won't * work, but this avoids memory thrashing for speed testing. */ while (y-- > 0) png_read_row(png_ptr, row, display); } } } /* Make sure to read to the end of the file: */ png_read_end(png_ptr, info_ptr); png_destroy_read_struct(&png_ptr, &info_ptr, NULL); free(row); free(display); return 1; } static int mytime(struct timespec *t) { /* Do the timing using clock_gettime and the per-process timer. */ if (!clock_gettime(CLOCK_PROCESS_CPUTIME_ID, t)) return 1; perror("CLOCK_PROCESS_CPUTIME_ID"); fprintf(stderr, "timepng: could not get the time\n"); return 0; } static int perform_one_test(FILE *fp, int nfiles) { int i; struct timespec before, after; /* Clear out all errors: */ rewind(fp); if (mytime(&before)) { for (i=0; i<nfiles; ++i) { if (read_png(fp)) { if (ferror(fp)) { perror("temporary file"); fprintf(stderr, "file %d: error reading PNG data\n", i); return 0; } } else { perror("temporary file"); fprintf(stderr, "file %d: error from libpng\n", i); return 0; } } } else return 0; if (mytime(&after)) { /* Work out the time difference and print it - this is the only output, * so flush it immediately. */ unsigned long s = after.tv_sec - before.tv_sec; long ns = after.tv_nsec - before.tv_nsec; if (ns < 0) { --s; ns += 1000000000; if (ns < 0) { fprintf(stderr, "timepng: bad clock from kernel\n"); return 0; } } printf("%lu.%.9ld\n", s, ns); fflush(stdout); if (ferror(stdout)) { fprintf(stderr, "timepng: error writing output\n"); return 0; } /* Successful return */ return 1; } else return 0; } static int add_one_file(FILE *fp, char *name) { FILE *ip = fopen(name, "rb"); if (ip != NULL) { int ch; for (;;) { ch = getc(ip); if (ch == EOF) break; putc(ch, fp); } if (ferror(ip)) { perror(name); fprintf(stderr, "%s: read error\n", name); return 0; } (void)fclose(ip); if (ferror(fp)) { perror("temporary file"); fprintf(stderr, "temporary file write error\n"); return 0; } } else { perror(name); fprintf(stderr, "%s: open failed\n", name); return 0; } return 1; } int main(int argc, char **argv) { int ok = 0; FILE *fp = tmpfile(); if (fp != NULL) { int err = 0; int nfiles = 0; if (argc > 1) { int i; for (i=1; i<argc; ++i) { if (add_one_file(fp, argv[i])) ++nfiles; else { err = 1; break; } } } else { char filename[FILENAME_MAX+1]; while (fgets(filename, FILENAME_MAX+1, stdin)) { size_t len = strlen(filename); if (filename[len-1] == '\n') { filename[len-1] = 0; if (add_one_file(fp, filename)) ++nfiles; else { err = 1; break; } } else { fprintf(stderr, "timepng: truncated file name ...%s\n", filename+len-32); err = 1; break; } } if (ferror(stdin)) { fprintf(stderr, "timepng: stdin: read error\n"); err = 1; } } if (!err) { if (nfiles > 0) ok = perform_one_test(fp, nfiles); else fprintf(stderr, "usage: timepng {files} or ls files | timepng\n"); } (void)fclose(fp); } else fprintf(stderr, "timepng: could not open temporary file\n"); /* Exit code 0 on success. */ return ok == 0; }