/* FLAM3 - cosmic recursive fractal flames Copyright (C) 1992-2009 Spotworks LLC 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 3 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, see . */ #include #include "private.h" #include "img.h" int print_progress(void *foo, double fraction, int stage, double eta) { fprintf(stderr, "stage=%s progress=%g eta=%g\n", stage?"filtering":"chaos", fraction, eta); return 0; } int main(int argc, char **argv) { flam3_frame f; char *ai; flam3_genome *cps; int ncps; int i; void *image=NULL; FILE *fp; char fname[256]; size_t this_size; char *prefix = args("prefix", ""); char *out = args("out", NULL); const char *format = "png"; int verbose = argi("verbose", 1); int bits = argi("bits", 33); int bpc = argi("bpc",8); int transparency = argi("transparency", 0); char *inf = getenv("in"); double qs = argf("qs", 1.0); double ss = argf("ss", 1.0); double pixel_aspect = argf("pixel_aspect", 1.0); int sub_batch_size = argi("sub_batch_size",10000); int name_enable = argi("name_enable",0); int num_threads = argi("nthreads",0); int earlyclip = argi("earlyclip",0); FILE *in; unsigned int channels; long start_time = (long)time(0); flam3_img_comments fpc; stat_struct stats; char numiter_string[64]; char badval_string[64]; char rtime_string[64]; randctx rc; if (1 != argc) { docstring(); exit(0); } /* Init random number generators */ rand_seed(&rc); /* Set the number of threads */ if (num_threads==0) { num_threads = flam3_count_nthreads(); if (verbose > 1) fprintf(stderr,"Automatically detected %d core(s)...\n",num_threads); } else{ if (verbose) fprintf(stderr,"Manually specified %d thread(s)...\n",num_threads); } channels = strcmp(format, "png") ? 3 : 4; /* Check for 16-bit-per-channel processing */ if ( (16 == bpc) && (strcmp(format,"png") != 0)) { fprintf(stderr,"Support for 16 bpc images is only present for the png format.\n"); exit(1); } else if (bpc != 8 && bpc != 16) { fprintf(stderr,"Unexpected bpc specified (%d)\n",bpc); exit(1); } if (pixel_aspect <= 0.0) { fprintf(stderr, "pixel aspect ratio must be positive, not %g.\n", pixel_aspect); exit(1); } if (inf) in = fopen(inf, "rb"); else in = stdin; if (NULL == in) { perror(inf); exit(1); } cps = flam3_parse_from_file(in, inf, flam3_defaults_on, &ncps, &rc); if (NULL == cps) { fprintf(stderr,"error reading genomes from file\n"); exit(1); } if (inf) fclose(in); for (i = 0; i < ncps; i++) { /* Force ntemporal_samples to 1 for -render */ cps[i].ntemporal_samples = 1; cps[i].sample_density *= qs; cps[i].height = (int)(cps[i].height * ss); cps[i].width = (int)(cps[i].width * ss); cps[i].pixels_per_unit *= ss; if (cps[i].height<=0 || cps[i].width<=0) { fprintf(stderr,"output image has dimension <=0, aborting.\n"); exit(1); } } if (out && (ncps > 1)) { fprintf(stderr, "hqi-flame: warning: writing multiple images " "to one file. all but last will be lost.\n"); } for (i = 0; i < ncps; i++) { if (verbose && ncps > 1) { fprintf(stderr, "flame = %d/%d ", i+1, ncps); } // f.temporal_filter_radius = 0.0; f.genomes = &cps[i]; f.ngenomes = 1; f.verbose = verbose; f.bits = bits; f.time = 0.0; f.pixel_aspect_ratio = pixel_aspect; f.progress = 0;//print_progress; f.nthreads = num_threads; f.earlyclip = earlyclip; f.sub_batch_size = sub_batch_size; if (16==bpc) f.bytes_per_channel = 2; else f.bytes_per_channel = 1; this_size = (size_t)channels * (size_t)cps[i].width * (size_t)cps[i].height * f.bytes_per_channel; image = (void *) calloc(this_size, sizeof(char)); if (verbose && ncps > 1) { fprintf(stderr, "\n"); } cps[i].ntemporal_samples = 1; if (flam3_render(&f, image, flam3_field_both, channels, transparency, &stats)) { fprintf(stderr,"error rendering image: aborting.\n"); exit(1); } if (NULL != out) { strcpy(fname,out); } else if (name_enable && cps[i].flame_name[0]>0) { sprintf(fname, "%s.%s",cps[i].flame_name,format); } else { sprintf(fname, "%s%05d.%s", prefix, i, format); } if (verbose) { fprintf(stderr, "writing %s...", fname); } fp = fopen(fname, "wb"); if (NULL == fp) { perror(fname); exit(1); } /* Generate temp file with genome */ fpc.genome = flam3_print_to_string(f.genomes); sprintf(badval_string,"%g",stats.badvals/(double)stats.num_iters); fpc.badvals = badval_string; sprintf(numiter_string,"%g",(double)stats.num_iters); fpc.numiters = numiter_string; sprintf(rtime_string,"%d",stats.render_seconds); fpc.rtime = rtime_string; write_png(fp, image, cps[i].width, cps[i].height, &fpc, f.bytes_per_channel); /* Free string */ free(fpc.genome); fclose(fp); if (verbose) { fprintf(stderr, "done.\n"); } } if (verbose && (ncps > 1)) { long total_time = (long)time(0) - start_time; if (total_time > 100) fprintf(stderr, "total time = %.1f minutes\n", total_time / 60.0); else fprintf(stderr, "total time = %ld seconds\n", total_time); } for (i=0;i