/***************************************************************************** Major portions of this software are copyrighted by the Medical College of Wisconsin, 1994-2000, and are released under the Gnu General Public License, Version 2. See the file README.Copyright for details. ******************************************************************************/ #include "mrilib.h" #define NBIN 32768 static int histo[NBIN] ; void init_histo(void) { int ii ; for( ii=0 ; ii < NBIN ; ii++ ) histo[ii] = 0 ; return ; } int load_histo( THD_3dim_dataset * dset , int iv ) { int ii , nvox ; MRI_IMAGE * bim ; short * bar ; if( !ISVALID_DSET(dset) ) return 0 ; if( iv < 0 || iv >= DSET_NVALS(dset) ) return 0 ; if( DSET_BRICK_TYPE(dset,iv) != MRI_short ) return 0 ; DSET_load(dset) ; bar = DSET_ARRAY(dset,iv) ; if( bar == NULL ) return 0 ; nvox = DSET_NVOX(dset) ; for( ii=0 ; ii < nvox ; ii++ ) if( bar[ii] < 0 ) return 0 ; for( ii=0 ; ii < nvox ; ii++ ) histo[bar[ii]]++ ; return nvox ; } double chfit( double mu ) { int ntop = (2.5 * mu + 0.5) ; double ftop , ntot , ccc , eee ; int ii ; if( ntop <= 1 || ntop >= NBIN/2 ) return -1.0 ; ntot = 0 ; for( ii=0 ; ii <= ntop ; ii++ ) ntot += histo[ii] ; ftop = 1.0 - exp(-(ntop/mu)*(ntop/mu)) ; ntot *= ftop ; ccc = 0.0 ; for( ii=0 ; ii <= ntop ; ii++ ){ eee = ntot * ( exp(-(ii/mu)*(ii/mu)) - exp(-((ii+1)/mu)*((ii+1)/mu)) ) ; if( eee < 1.0 ) eee = 1.0 ; ccc += (histo[ii] - eee) * (histo[ii] - eee) / eee ; } ccc /= (ntop+1) ; return ccc ; } int main( int argc , char * argv[] ) { THD_3dim_dataset * dset ; double mu , ccc , mbest,cbest , perc , snr=2.5 , nlxx=0.0 ; int ii , narg=1 , blast=0 , iv , ncut , nnn , nvox , nl=0 ; int vmax ; short * bar ; if( argc < 2 || strcmp(argv[1],"-help") == 0 ){ printf("Usage: 3dnoise [-blast] [-snr fac] [-nl x ] datasets ...\n" "Estimates noise level in 3D datasets, and optionally\n" "set voxels below the noise threshold to zero.\n" "This only works on datasets that are stored as shorts,\n" "and whose elements are all nonnegative.\n" " -blast = Set values at or below the cutoff to zero.\n" " In 3D+time datasets, a spatial location\n" " is set to zero only if a majority of time\n" " points fall below the cutoff; in that case\n" " all the values at that location are zeroed.\n" " -snr fac = Set cutoff to 'fac' times the estimated\n" " noise level. Default fac = 2.5. What to\n" " use for this depends strongly on your MRI\n" " system -- I often use 5, but our true SNR\n" " is about 100 for EPI.\n" " -nl x = Set the noise level to 'x', skipping the\n" " estimation procedure. Also sets fac=1.0.\n" " You can use program 3dClipLevel to get an\n" " estimate of a value for 'x'.\n" "Author -- RW Cox\n" ) ; PRINT_COMPILE_DATE ; exit(0) ; } while( narg < argc && argv[narg][0] == '-' ){ if( strcmp(argv[narg],"-blast") == 0 ){ blast = 1 ; narg++ ; continue ; } if( strcmp(argv[narg],"-snr") == 0 ){ narg++ ; snr = strtod( argv[narg] , NULL ) ; if( snr <= 0.0 ){fprintf(stderr,"Illegal snr value!\n");exit(1);} narg++ ; continue ; } if( strcmp(argv[narg],"-nl") == 0 ){ narg++ ; nl = 1 ; nlxx = strtod( argv[narg] , NULL ) ; snr = 1.0 ; if( nlxx <= 0.0 ){fprintf(stderr,"Illegal nl value!\n");exit(1);} narg++ ; continue ; } fprintf(stderr,"Unknown option: %s\n",argv[narg]) ; exit(1) ; } if( narg >= argc ){fprintf(stderr,"No datasets?\n");exit(1);} for( ; narg < argc ; narg++ ){ dset = THD_open_one_dataset(argv[narg]) ; /** NOT THD_open_dataset()! **/ if( dset == NULL ){ ERROR_message("Can't open dataset '%s'",argv[narg]) ; continue ; } fprintf(stderr,"dataset %s",argv[narg]) ; init_histo() ; if( blast ) THD_force_malloc_type( dset->dblk , DATABLOCK_MEM_MALLOC ) ; nvox = DSET_NVOX(dset) ; vmax = 0 ; for( iv=0 ; iv < DSET_NVALS(dset) ; iv++ ){ ii = load_histo(dset,iv) ; if( ii <= 0 ){ fprintf(stderr,": Can't load data, or illegal data!\n"); break; } bar = DSET_ARRAY(dset,iv) ; for( ii=0 ; ii < nvox ; ii++ ) if( vmax < bar[ii] ) vmax = bar[ii] ; } if( iv < DSET_NVALS(dset) ) continue ; if( vmax < 40 ){ fprintf(stderr,": Didn't fit noise model!\n"); continue; } printf(":") ; fflush(stdout) ; #define DMU 0.5 if( !nl ){ mu = 1.0 ; mbest = mu ; cbest = chfit(mu) ; ii = 0 ; do { mu += DMU ; ccc = chfit(mu) ; if( ccc > cbest ) break ; cbest = ccc ; mbest = mu ; ii++ ; if( mu > 0.05 * vmax ){ ii=0 ; break; } } while( 1 ) ; if( ii <= 0 ){ fprintf(stderr," Didn't fit noise model!\n"); continue; } } else { mbest = nlxx ; } ncut = (int) (snr * mbest) ; nnn = 0 ; for( ii=0 ; ii <= ncut ; ii++ ) nnn += histo[ii] ; perc = (100.0*nnn) / (double)(DSET_NVOX(dset)*DSET_NVALS(dset)) ; fprintf(stderr," Cutoff=%d Count=%d [%4.1f%%]",ncut,nnn,perc) ; if( blast && nnn > 0 ){ int nkilled ; if( DSET_NUM_TIMES(dset) <= 1 ){ for( iv=0 ; iv < DSET_NVALS(dset) ; iv++ ){ bar = DSET_ARRAY(dset,iv) ; for( ii=0 ; ii < nvox ; ii++ ) if( bar[ii] <= ncut ) bar[ii]=0; } nkilled = nnn ; } else { /* 24 Mar 1998 */ int nvh = DSET_NVALS(dset)/2 , qq ; for( ii=0,nkilled=0 ; ii < nvox ; ii++ ){ for( iv=0,qq=0 ; iv < DSET_NVALS(dset) ; iv++ ){ bar = DSET_ARRAY(dset,iv) ; if( bar[ii] <= ncut ) qq++ ; } if( qq >= nvh ){ for( iv=0 ; iv < DSET_NVALS(dset) ; iv++ ){ bar = DSET_ARRAY(dset,iv) ; bar[ii]=0; } nkilled += DSET_NVALS(dset) ; } } } fprintf(stderr,"--blasted %d voxels\n",nkilled) ; { char str[128] ; sprintf(str,"3dnoise -- blasted %d voxels",nkilled) ; tross_Append_History( dset , str ) ; } DSET_write(dset) ; } THD_delete_3dim_dataset( dset , False ) ; } exit(0) ; }