⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 rfftw_threads_test.c

📁 FFTW, a collection of fast C routines to compute the Discrete Fourier Transform in one or more dime
💻 C
📖 第 1 页 / 共 2 页
字号:
     fftw(validated_plan, 1, in2, 1, n, out2, 1, n);     if (dir == FFTW_REAL_TO_COMPLEX) {	  int n2 = (n + 1) / 2;	  out3[0] = c_re(out2[0]);	  for (i = 1; i < n2; ++i) {	       out3[i] = c_re(out2[i]);	       out3[n - i] = c_im(out2[i]);	  }	  if (n2 * 2 == n)	       out3[n2] = c_re(out2[n2]);     } else {	  for (i = 0; i < n; ++i)	       out3[i] = c_re(out2[i]);     }     for (j = 0; j < howmany; ++j)	  CHECK(compute_error(out1 + j * n * ostride, ostride, out3, 1, n)		< TOLERANCE,		"test_in_place: wrong answer");     WHEN_VERBOSE(2, printf("OK\n"));     fftw_free(in1);     fftw_free(in2);     fftw_free(out2);     fftw_free(out3);}void test_out_of_place_both(int n, int istride, int ostride,			    int howmany,			    fftw_plan validated_plan_forward,			    fftw_plan validated_plan_backward){     int specific;     for (specific = 0; specific <= 1; ++specific) {	  WHEN_VERBOSE(2,	       printf("TEST CORRECTNESS (out of place, FFTW_FORWARD, %s)"		   " n = %d  istride = %d  ostride = %d  howmany = %d\n",		      SPECIFICP(specific),		      n, istride, ostride, howmany));	  test_out_of_place(n, istride, ostride, howmany, FFTW_FORWARD,			    validated_plan_forward, specific);	  WHEN_VERBOSE(2,	      printf("TEST CORRECTNESS (out of place, FFTW_BACKWARD, %s)"		   " n = %d  istride = %d  ostride = %d  howmany = %d\n",		     SPECIFICP(specific),		     n, istride, ostride, howmany));	  test_out_of_place(n, istride, ostride, howmany, FFTW_BACKWARD,			    validated_plan_backward, specific);     }}void test_in_place_both(int n, int istride, int howmany,			fftw_plan validated_plan_forward,			fftw_plan validated_plan_backward){     int specific;     for (specific = 0; specific <= 1; ++specific) {	  WHEN_VERBOSE(2,		  printf("TEST CORRECTNESS (in place, FFTW_FORWARD, %s) "			 "n = %d  istride = %d  howmany = %d\n",			 SPECIFICP(specific),			 n, istride, howmany));	  test_in_place(n, istride, howmany, FFTW_FORWARD,			validated_plan_forward, specific);	  WHEN_VERBOSE(2,		 printf("TEST CORRECTNESS (in place, FFTW_BACKWARD, %s) "			"n = %d  istride = %d  howmany = %d\n",			SPECIFICP(specific),			n, istride, howmany));	  test_in_place(n, istride, howmany, FFTW_BACKWARD,			validated_plan_backward, specific);     }}void test_correctness(int n){     int istride, ostride, howmany;     fftw_plan validated_plan_forward, validated_plan_backward;     WHEN_VERBOSE(1,		  printf("Testing correctness for n = %d...", n);		  fflush(stdout));     /* produce a *good* plan (validated by Ergun's test procedure) */     validated_plan_forward =	 fftw_create_plan(n, FFTW_FORWARD, measure_flag | wisdom_flag);     validated_plan_backward =	 fftw_create_plan(n, FFTW_BACKWARD, measure_flag | wisdom_flag);     CHECK(validated_plan_forward != NULL, "can't create plan");     CHECK(validated_plan_backward != NULL, "can't create plan");     for (istride = 1; istride <= MAX_STRIDE; ++istride)	  for (ostride = 1; ostride <= MAX_STRIDE; ++ostride)	       for (howmany = 1; howmany <= MAX_HOWMANY; ++howmany)		    test_out_of_place_both(n, istride, ostride, howmany,					   validated_plan_forward,					   validated_plan_backward);     for (istride = 1; istride <= MAX_STRIDE; ++istride)	  for (howmany = 1; howmany <= MAX_HOWMANY; ++howmany)	       test_in_place_both(n, istride, howmany,				  validated_plan_forward,				  validated_plan_backward);     fftw_destroy_plan(validated_plan_forward);     fftw_destroy_plan(validated_plan_backward);     if (!(wisdom_flag & FFTW_USE_WISDOM) && chk_mem_leak)	  fftw_check_memory_leaks();     WHEN_VERBOSE(1, printf("OK\n"));}/************************************************* * multi-dimensional correctness tests *************************************************/void testnd_out_of_place(int rank, int *n, fftwnd_plan validated_plan){     int istride, ostride;     int N, dim, i, j, k;     int nc, nhc, nr;     fftw_real *in1, *out3;     fftw_complex *in2, *out1, *out2;     fftwnd_plan p, ip;     int flags = measure_flag | wisdom_flag;     if (coinflip())	  flags |= FFTW_THREADSAFE;     N = nc = nr = nhc = 1;     for (dim = 0; dim < rank; ++dim)	  N *= n[dim];     if (rank > 0) {	  nr = n[rank - 1];	  nc = N / nr;	  nhc = nr / 2 + 1;     }     in1 = (fftw_real *) fftw_malloc(N * MAX_STRIDE * sizeof(fftw_real));     out3 = (fftw_real *) fftw_malloc(N * MAX_STRIDE * sizeof(fftw_real));     out1 = (fftw_complex *) fftw_malloc(nhc * nc * MAX_STRIDE					 * sizeof(fftw_complex));     in2 = (fftw_complex *) fftw_malloc(N * sizeof(fftw_complex));     out2 = (fftw_complex *) fftw_malloc(N * sizeof(fftw_complex));     p = rfftwnd_create_plan(rank, n, FFTW_REAL_TO_COMPLEX, flags);     ip = rfftwnd_create_plan(rank, n, FFTW_COMPLEX_TO_REAL, flags);     CHECK(p != NULL && ip != NULL, "can't create plan");     for (istride = 1; istride <= MAX_STRIDE; ++istride) {	  /* generate random inputs */	  for (i = 0; i < nc; ++i)	       for (j = 0; j < nr; ++j) {		    c_re(in2[i * nr + j]) = DRAND();		    c_im(in2[i * nr + j]) = 0.0;		    for (k = 0; k < istride; ++k)			 in1[(i * nr + j) * istride + k]			     = c_re(in2[i * nr + j]);	       }	  for (i = 0; i < N * istride; ++i)	       out3[i] = 0.0;	  fftwnd(validated_plan, 1, in2, 1, 1, out2, 1, 1);	  for (ostride = 1; ostride <= MAX_STRIDE; ++ostride) {	       int howmany = (istride < ostride) ? istride : ostride;	       WHEN_VERBOSE(2, printf("\n    testing stride %d/%d...",				      istride, ostride));	       if (howmany != 1 || istride != 1 || ostride != 1 || coinflip())		    rfftwnd_threads_real_to_complex(nthreads, p, howmany, in1, istride, 1,					    out1, ostride, 1);	       else		    rfftwnd_threads_one_real_to_complex(nthreads, p, in1, out1);	       for (i = 0; i < nc; ++i)		    for (k = 0; k < howmany; ++k)			 CHECK(compute_error_complex(out1 + i * nhc * ostride + k,						     ostride,						     out2 + i * nr, 1,						     nhc) < TOLERANCE,			       "out-of-place (r2c): wrong answer");	       if (howmany != 1 || istride != 1 || ostride != 1 || coinflip())		    rfftwnd_threads_complex_to_real(nthreads, ip, howmany, out1, ostride, 1,					    out3, istride, 1);	       else		    rfftwnd_threads_one_complex_to_real(nthreads, ip, out1, out3);	       for (i = 0; i < N * istride; ++i)		    out3[i] *= 1.0 / N;	       if (istride == howmany)		    CHECK(compute_error(out3, 1, in1, 1, N * istride)			< TOLERANCE, "out-of-place (c2r): wrong answer");	       for (i = 0; i < nc; ++i)		    for (k = 0; k < howmany; ++k)			 CHECK(compute_error(out3 + i * nr * istride + k,					     istride,					 (fftw_real *) (in2 + i * nr), 2,					     nr) < TOLERANCE,			   "out-of-place (c2r): wrong answer (check 2)");	  }     }     rfftwnd_destroy_plan(p);     rfftwnd_destroy_plan(ip);     fftw_free(out3);     fftw_free(out2);     fftw_free(in2);     fftw_free(out1);     fftw_free(in1);}void testnd_in_place(int rank, int *n, fftwnd_plan validated_plan,		     int alternate_api, int specific){     int istride, ostride, howmany;     int N, dim, i, j, k;     int nc, nhc, nr;     fftw_real *in1, *out3;     fftw_complex *in2, *out1, *out2;     fftwnd_plan p, ip;     int flags = measure_flag | wisdom_flag | FFTW_IN_PLACE;     if (coinflip())	  flags |= FFTW_THREADSAFE;     N = nc = nr = nhc = 1;     for (dim = 0; dim < rank; ++dim)	  N *= n[dim];     if (rank > 0) {	  nr = n[rank - 1];	  nc = N / nr;	  nhc = nr / 2 + 1;     }     in1 = (fftw_real *) fftw_malloc(2 * nhc * nc * MAX_STRIDE * sizeof(fftw_real));     out3 = in1;     out1 = (fftw_complex *) in1;     in2 = (fftw_complex *) fftw_malloc(N * sizeof(fftw_complex));     out2 = (fftw_complex *) fftw_malloc(N * sizeof(fftw_complex));     if (alternate_api && specific && (rank == 2 || rank == 3)) {	  if (rank == 2) {	       p = rfftw2d_create_plan_specific(n[0], n[1],					     FFTW_REAL_TO_COMPLEX, flags,						in1, MAX_STRIDE, 0, 0);	       ip = rfftw2d_create_plan_specific(n[0], n[1],					     FFTW_COMPLEX_TO_REAL, flags,						 in1, MAX_STRIDE, 0, 0);	  } else {	       p = rfftw3d_create_plan_specific(n[0], n[1], n[2],					     FFTW_REAL_TO_COMPLEX, flags,						in1, MAX_STRIDE, 0, 0);	       ip = rfftw3d_create_plan_specific(n[0], n[1], n[2],					     FFTW_COMPLEX_TO_REAL, flags,						 in1, MAX_STRIDE, 0, 0);	  }     } else if (specific) {	  p = rfftwnd_create_plan_specific(rank, n, FFTW_REAL_TO_COMPLEX,					   flags,				       in1, MAX_STRIDE, in1, MAX_STRIDE);	  ip = rfftwnd_create_plan_specific(rank, n, FFTW_COMPLEX_TO_REAL,					    flags,				       in1, MAX_STRIDE, in1, MAX_STRIDE);     } else if (alternate_api && (rank == 2 || rank == 3)) {	  if (rank == 2) {	       p = rfftw2d_create_plan(n[0], n[1], FFTW_REAL_TO_COMPLEX,				       flags);	       ip = rfftw2d_create_plan(n[0], n[1], FFTW_COMPLEX_TO_REAL,					flags);	  } else {	       p = rfftw3d_create_plan(n[0], n[1], n[2], FFTW_REAL_TO_COMPLEX,				       flags);	       ip = rfftw3d_create_plan(n[0], n[1], n[2], FFTW_COMPLEX_TO_REAL,					flags);	  }     } else {	  p = rfftwnd_create_plan(rank, n, FFTW_REAL_TO_COMPLEX, flags);	  ip = rfftwnd_create_plan(rank, n, FFTW_COMPLEX_TO_REAL, flags);     }     CHECK(p != NULL && ip != NULL, "can't create plan");     for (i = 0; i < nc * nhc * 2 * MAX_STRIDE; ++i)	  out3[i] = 0;     for (istride = 1; istride <= MAX_STRIDE; ++istride) {	  /* generate random inputs */	  for (i = 0; i < nc; ++i)	       for (j = 0; j < nr; ++j) {		    c_re(in2[i * nr + j]) = DRAND();		    c_im(in2[i * nr + j]) = 0.0;		    for (k = 0; k < istride; ++k)			 in1[(i * nhc * 2 + j) * istride + k]			     = c_re(in2[i * nr + j]);	       }	  fftwnd(validated_plan, 1, in2, 1, 1, out2, 1, 1);	  howmany = ostride = istride;	  WHEN_VERBOSE(2, printf("\n    testing in-place stride %d...",				 istride); fflush(stdout););	  if (howmany != 1 || istride != 1 || ostride != 1 || coinflip())	       rfftwnd_threads_real_to_complex(nthreads, p, howmany, in1, istride, 1,				       out1, ostride, 1);	  else	       rfftwnd_threads_one_real_to_complex(nthreads, p, in1, NULL);	  for (i = 0; i < nc; ++i)	       for (k = 0; k < howmany; ++k)		    CHECK(compute_error_complex(out1 + i * nhc * ostride + k,						ostride,						out2 + i * nr, 1,						nhc) < TOLERANCE,			  "in-place (r2c): wrong answer");	  if (howmany != 1 || istride != 1 || ostride != 1 || coinflip())	       rfftwnd_threads_complex_to_real(nthreads, ip, howmany, out1, ostride, 1,				       out3, istride, 1);	  else	       rfftwnd_threads_one_complex_to_real(nthreads, ip, out1, NULL);	  for (i = 0; i < nc * nhc * 2 * istride; ++i)	       out3[i] *= 1.0 / N;	  for (i = 0; i < nc; ++i)	       for (k = 0; k < howmany; ++k)		    CHECK(compute_error(out3 + i * nhc * 2 * istride + k,					istride,					(fftw_real *) (in2 + i * nr), 2,					nr) < TOLERANCE,			  "in-place (c2r): wrong answer (check 2)");     }     rfftwnd_destroy_plan(p);     rfftwnd_destroy_plan(ip);     fftw_free(out2);     fftw_free(in2);     fftw_free(in1);}void testnd_correctness(struct size sz, fftw_direction dir,			int alt_api, int specific, int force_buf){     fftwnd_plan validated_plan;     if (dir != FFTW_FORWARD)	  return;     if (force_buf)	  return;     validated_plan = fftwnd_create_plan(sz.rank, sz.narray, 					 dir, measure_flag | wisdom_flag);     CHECK(validated_plan != NULL, "can't create plan");     testnd_out_of_place(sz.rank, sz.narray, validated_plan);     testnd_in_place(sz.rank, sz.narray,		     validated_plan, alt_api, specific);     fftwnd_destroy_plan(validated_plan);}/************************************************* * planner tests *************************************************/void test_planner(int rank){     WHEN_VERBOSE(1, printf("Use rfftw_test to test the planner.\n"););}/************************************************* * test initialization *************************************************/void test_init(int *argc, char ***argv){     int i;     if (*argc >= 2)          nthreads = atoi((*argv)[1]);     if (nthreads <= 0) {          fprintf(stderr, "Usage: fftw_threads_test <nthreads> [ options ]\n");          exit(EXIT_FAILURE);     }     for (i = 2; i < *argc; ++i)          (*argv)[i - 1] = (*argv)[i];     *argc -= 1;     if (fftw_threads_init()) {          fprintf(stderr, "Error initializing threads!");          exit(EXIT_FAILURE);     }}void test_finish(void){}void enter_paranoid_mode(void){}int get_option(int argc, char **argv, char *argval, int argval_maxlen){     return default_get_option(argc,argv,argval,argval_maxlen);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -