void process (struct dt_iop_module_t *self, dt_dev_pixelpipe_iop_t *piece, void *ivoid, void *ovoid, const dt_iop_roi_t *roi_in, const dt_iop_roi_t *roi_out) { // FIXME: this returns nan!! dt_iop_colortransfer_data_t *data = (dt_iop_colortransfer_data_t *)piece->data; float *in = (float *)ivoid; float *out = (float *)ovoid; const int ch = piece->colors; if(data->flag == ACQUIRE) { if(piece->pipe->type == DT_DEV_PIXELPIPE_PREVIEW) { // only get stuff from the preview pipe, rest stays untouched. int hist[HISTN]; // get histogram of L capture_histogram(in, roi_in, hist); // invert histogram of L invert_histogram(hist, data->hist); // get n clusters kmeans(in, roi_in, data->n, data->mean, data->var); // notify gui that commit_params should let stuff flow back! data->flag = ACQUIRED; dt_iop_colortransfer_params_t *p = (dt_iop_colortransfer_params_t *)self->params; p->flag = ACQUIRE2; } memcpy(out, in, sizeof(float)*ch*roi_out->width*roi_out->height); } else if(data->flag == APPLY) { // apply histogram of L and clustering of (a,b) int hist[HISTN]; capture_histogram(in, roi_in, hist); #ifdef _OPENMP #pragma omp parallel for default(none) schedule(static) shared(roi_out,data,in,out,hist) #endif for(int k=0; k<roi_out->height; k++) { int j = ch*roi_out->width*k; for(int i=0; i<roi_out->width; i++) { // L: match histogram out[j] = data->hist[hist[(int)CLAMP(HISTN*in[j]/100.0, 0, HISTN-1)]]; out[j] = CLAMP(out[j], 0, 100); j+=ch; } } // cluster input buffer float mean[data->n][2], var[data->n][2]; kmeans(in, roi_in, data->n, mean, var); // get mapping from input clusters to target clusters int mapio[data->n]; get_cluster_mapping(data->n, mean, data->mean, mapio); // for all pixels: find input cluster, transfer to mapped target cluster #ifdef _OPENMP #pragma omp parallel for default(none) schedule(static) shared(roi_out,data,mean,var,mapio,in,out) #endif for(int k=0; k<roi_out->height; k++) { float weight[MAXN]; int j = ch*roi_out->width*k; for(int i=0; i<roi_out->width; i++) { const float L = in[j]; const float Lab[3] = {L, in[j+1], in[j+2]}; // a, b: subtract mean, scale nvar/var, add nmean #if 0 // single cluster, gives color banding const int ki = get_cluster(in + j, data->n, mean); out[j+1] = 100.0/out[j] * ((Lab[1] - mean[ki][0])*data->var[mapio[ki]][0]/var[ki][0] + data->mean[mapio[ki]][0]); out[j+2] = 100.0/out[j] * ((Lab[2] - mean[ki][1])*data->var[mapio[ki]][1]/var[ki][1] + data->mean[mapio[ki]][1]); #else // fuzzy weighting get_clusters(in+j, data->n, mean, weight); out[j+1] = out[j+2] = 0.0f; for(int c=0; c<data->n; c++) { out[j+1] += weight[c] * ((Lab[1] - mean[c][0])*data->var[mapio[c]][0]/var[c][0] + data->mean[mapio[c]][0]); out[j+2] += weight[c] * ((Lab[2] - mean[c][1])*data->var[mapio[c]][1]/var[c][1] + data->mean[mapio[c]][1]); } #endif out[j+3] = in[j+3]; j+=ch; } } } else { memcpy(out, in, sizeof(float)*ch*roi_out->width*roi_out->height); } }
int main(int argc, char *arg[]) { if(argc < 2) { fprintf(stderr, "usage: %s input.pfm [-c a1 b1]\n", arg[0]); exit(1); } int wd, ht; float *input = read_pfm(arg[1], &wd, &ht); float max = 0.0f; // sanity checks: // for(int k=0;k<3*wd*ht;k++) input[k] = clamp(input[k], 0.0f, 1.0f); // correction requested? if(argc >= 9 && !strcmp(arg[2], "-c")) { const float a[3] = {atof(arg[3]), atof(arg[4]), atof(arg[5])}, b[3] = {atof(arg[6]), atof(arg[7]), atof(arg[8])}; // const float m[3] = {1, 1, 1}; // 2.0f*sqrt(a[0]*1.0f+b[0])/a[0], // 2.0f*sqrt(a[1]*1.0f+b[1])/a[1], // 2.0f*sqrt(a[2]*1.0f+b[2])/a[2]}; #if 1 // dump curves: for(int k=0;k<N;k++) { for(int c=0;c<3;c++) { // const float y = k/(N-1.0f); // const float x = m[c]*m[c]*a[c]*y*y/4.0f - b[c]/a[c]; float x = k/(N-1.0f)/a[c]; const float d = fmaxf(0.0f, x + 3./8. + (b[c]/a[c])*(b[c]/a[c])); x = 2.0f*sqrtf(d); fprintf(stderr, "%f ", x); } fprintf(stderr, "\n"); } #endif for(int k=0;k<wd*ht;k++) { for(int c=0;c<3;c++) { // input[3*k+c] = 2.0f*sqrtf(a[c]*input[3*k+c]+b[c])/(a[c]*m[c]); input[3*k+c] = input[3*k+c] / a[c]; const float d = fmaxf(0.0f, input[3*k+c] + 3./8. + (b[c]/a[c])*(b[c]/a[c])); input[3*k+c] = 2.0f*sqrtf(d); max = fmaxf(max, input[3*k+c]); } } for(int k=0;k<3*wd*ht;k++) input[k] /= max; } else if(argc >= 4 && !strcmp(arg[2], "-h")) { int bins = 0; float *hist = read_histogram(arg[3], &bins); float *inv_hist = (float *)malloc(3*sizeof(float)*bins); invert_histogram(hist, inv_hist, bins); #if 1 // output curves and their inverse: for(int k=0;k<bins;k++) // fprintf(stderr, "%f %f %f %f %f %f %f\n", k/(float)bins, hist[3*k], hist[3*k+1], hist[3*k+2], inv_hist[3*k], inv_hist[3*k+1], inv_hist[3*k+2]); fprintf(stderr, "%f %f %f\n", inv_hist[3*k], inv_hist[3*k+1], inv_hist[3*k+2]); // fprintf(stderr,"scanned %d bins\n", bins); #endif for(int k=0;k<wd*ht;k++) { for(int c=0;c<3;c++) { float f = clamp(input[3*k+c]*bins, 0, bins-2); const int bin = (int)f; f -= bin; input[3*k+c] = (1.0f-f)*inv_hist[3*bin+c] + f*inv_hist[3*(bin+1)+c]; } } } float std[N][3] = {{0.0f}}; float cnt[N][3] = {{0.0f}}; // one level haar decomposition, separable, decimated, lifting scheme for(int j=0;j<ht;j++) { for(int i=0;i<wd-1;i+=2) { float *buf = input + 3*(wd*j + i); for(int c=0;c<3;c++) { buf[c] += buf[3+c]; buf[c] *= .5f; buf[3+c] -= buf[c]; } // buf += 3; } } for(int i=0;i<wd;i++) { for(int j=0;j<ht-1;j+=2) { float *buf = input + 3*(wd*j + i); for(int c=0;c<3;c++) { buf[c] += buf[3*wd+c]; buf[c] *= .5f; buf[3*wd+c] -= buf[c]; } // buf += 3*wd; } } #if 0 // debug: write full wavelet transform: write_pfm("wt.pfm", input, wd, ht); // debug: write LL float *out = (float *)malloc(sizeof(float)*3*wd/2*ht/2); for(int j=0;j<ht-1;j+=2) { for(int i=0;i<wd-1;i+=2) { for(int c=0;c<3;c++) { out[3*((wd/2)*(j/2)+(i/2))+c] = input[3*(wd*j+i)+c]; } } } write_pfm("LL.pfm", out, wd/2, ht/2); free(out); #endif // sort pairs (LL,HH) for each color channel: float *llhh = (float *)malloc(sizeof(float)*wd*ht/2); for(int c=0;c<3;c++) { int k = 0; for(int j=0;j<ht-1;j+=2) { for(int i=0;i<wd-1;i+=2) { llhh[2*k] = input[3*(wd*j+i)+c]; llhh[2*k+1] = fabsf(input[3*(wd*(j+1)+(i+1))+c]); k++; } } qsort(llhh, k, 2*sizeof(float), compare_llhh); // estimate std deviation for every bin we've got: for(int begin=0;begin<k;) { // LL is used to estimate brightness: const int bin = (int)clamp(llhh[2*begin]*N, 0, N-1); int end = begin+1; while((end < k) && ((int)clamp(llhh[2*end]*N, 0, N-1) == bin)) end++; assert(end >= k || bin <= (int)clamp(llhh[2*end]*N, 0, N-1)); // fprintf(stderr, "from %d (%d) -- %d (%d)\n", begin, bin, end, (int)clamp(llhh[2*end]*N, 0, N-1)); // estimate noise by robust statistic (assumes zero mean of HH band): // MAD: median(|Y - med(Y)|) = 0.6745 sigma // if(end - begin > 10) // fprintf(stdout, "%d %f %d\n", bin, median(llhh+2*begin, end-begin)/0.6745, end - begin); std[bin][c] += median(llhh+2*begin, end-begin)/0.6745; cnt[bin][c] = end - begin; begin = end; } } #if 0 // recover noise curve: for(int k=0;k<wd*ht;k++) { for(int c=0;c<3;c++) { const int i = clamp(ref[3*k+c]*N, 0, N-1); cnt[i][c] ++; const float diff = input[3*k+c] - ref[3*k+c]; // assume zero mean: var[i][c] += diff*diff; // - E(X^2) } } #endif #if 0 // normalize for(int i=0;i<N;i++) for(int c=0;c<3;c++) if(cnt[i][c] > 0.0f) std[i][c] /= cnt[i][c]; else std[i][c] = 0.0f; #endif // scale back in case we needed to bin it down: if(max > 0.0f) for(int i=0;i<N;i++) for(int k=0;k<3;k++) std[i][k] *= max; // output variance per brightness level: // fprintf(stdout, "# bin std_r std_g std_b hist_r hist_g hist_b cdf_r cdf_g cdf_b\n"); float sum[3] = {0.0f}; for(int i=0;i<N;i++) for(int k=0;k<3;k++) sum[k] += std[i][k]; float cdf[3] = {0.0f}; for(int i=0;i<N;i++) { fprintf(stdout, "%f %f %f %f %f %f %f %f %f %f\n", i/(float)N, std[i][0], std[i][1], std[i][2], cnt[i][0], cnt[i][1], cnt[i][2], cdf[0]/sum[0], cdf[1]/sum[1], cdf[2]/sum[2]); // cdf[0], cdf[1], cdf[2]); for(int k=0;k<3;k++) cdf[k] += std[i][k]; } free(llhh); free(input); exit(0); }