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

📄 fftw_mpi_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, howmany, in2, howmany, 1, out2, howmany, 1);     CHECK(compute_error_complex(in1, 1,				 out2 + local_start_after_transform*howmany, 1,				 howmany*local_n_after_transform) < TOLERANCE,	   "test_in_place: wrong answer");     WHEN_VERBOSE(2, my_printf("OK\n"));     fftw_free(in1);     fftw_free(work);     fftw_free(in2);     fftw_free(out2);}void test_out_of_place_both(int n, int istride, int ostride,			    int howmany,			    fftw_plan validated_plan_forward,			    fftw_plan validated_plan_backward){}void test_in_place_both(int n, int istride, int howmany,			fftw_plan validated_plan_forward,			fftw_plan validated_plan_backward){     WHEN_VERBOSE(2,		  my_printf("TEST CORRECTNESS (in place, FFTW_FORWARD, %s) "			 "n = %d  istride = %d  howmany = %d\n",			 SPECIFICP(0),			 n, istride, howmany));     test_in_place(n, istride, howmany, FFTW_FORWARD,		   validated_plan_forward, 0);          WHEN_VERBOSE(2,		  my_printf("TEST CORRECTNESS (in place, FFTW_BACKWARD, %s) "			 "n = %d  istride = %d  howmany = %d\n",			 SPECIFICP(0),			 n, istride, howmany));     test_in_place(n, istride, howmany, FFTW_BACKWARD,		   validated_plan_backward, 0);}void test_correctness(int n){     int howmany;     fftw_plan validated_plan_forward, validated_plan_backward;     WHEN_VERBOSE(1,		  my_printf("Testing correctness for n = %d...", n);		  my_fflush(stdout));     /* produce a good plan */     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);     for (howmany = 1; howmany <= MAX_HOWMANY; ++howmany)	  test_in_place_both(n, howmany, 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, my_printf("OK\n"));}/************************************************* * multi-dimensional correctness tests *************************************************/void testnd_out_of_place(int rank, int *n, fftw_direction dir,			 fftwnd_plan validated_plan){}void testnd_in_place(int rank, int *n, fftw_direction dir,		     fftwnd_plan validated_plan,		     int alternate_api, int specific, int force_buffered){     int local_nx, local_x_start, local_ny_after_transpose,          local_y_start_after_transpose, total_local_size;     int istride;     int N, dim, i;     fftw_complex *in1, *work = 0, *in2;     fftwnd_mpi_plan p = 0;     int flags = measure_flag | wisdom_flag | FFTW_IN_PLACE;     if (specific || rank < 2)	  return;     if (coinflip())	  flags |= FFTW_THREADSAFE;     if (force_buffered)	  flags |= FFTWND_FORCE_BUFFERED;     N = 1;     for (dim = 0; dim < rank; ++dim)	  N *= n[dim];     if (alternate_api && (rank == 2 || rank == 3)) {	  if (rank == 2)	       p = fftw2d_mpi_create_plan(MPI_COMM_WORLD,					  n[0], n[1], dir, flags);	  else	       p = fftw3d_mpi_create_plan(MPI_COMM_WORLD,					  n[0], n[1], n[2], dir, flags);     }     else		/* standard api */	  p = fftwnd_mpi_create_plan(MPI_COMM_WORLD, rank, n, dir, flags);     fftwnd_mpi_local_sizes(p, &local_nx, &local_x_start,                            &local_ny_after_transpose,                            &local_y_start_after_transpose,                            &total_local_size);     in1 = (fftw_complex *) fftw_malloc(total_local_size * MAX_STRIDE					* sizeof(fftw_complex));     if (coinflip()) {	  WHEN_VERBOSE(1, my_printf("w/work..."));	  work = (fftw_complex *) fftw_malloc(total_local_size * MAX_STRIDE					      * sizeof(fftw_complex));     }     in2 = (fftw_complex *) fftw_malloc(N * sizeof(fftw_complex));     for (istride = 1; istride <= MAX_STRIDE; ++istride) {	  /* generate random inputs */	  for (i = 0; i < N; ++i) {	       c_re(in2[i]) = DRAND();	       c_im(in2[i]) = DRAND();	  }	  for (i = 0; i < local_nx * (N/n[0]); ++i) {	       int j;	       for (j = 0; j < istride; ++j) {		    c_re(in1[i * istride + j]) = c_re((in2 + local_x_start 						       * (N/n[0])) [i]);		    c_im(in1[i * istride + j]) = c_im((in2 + local_x_start                                                       * (N/n[0])) [i]);	       }	  }	  fftwnd_mpi(p, istride, in1, work, FFTW_NORMAL_ORDER);	  fftwnd(validated_plan, 1, in2, 1, 1, NULL, 0, 0);	  for (i = 0; i < istride; ++i)	       CHECK(compute_error_complex(in1 + i, istride,					   in2 + local_x_start * (N/n[0]),					   1, local_nx * (N/n[0])) < TOLERANCE,		     "testnd_in_place: wrong answer");     }     fftwnd_mpi_destroy_plan(p);     fftw_free(in2);     fftw_free(work);     fftw_free(in1);}void testnd_correctness(struct size sz, fftw_direction dir,			int alt_api, int specific, int force_buf){     fftwnd_plan validated_plan;     validated_plan = fftwnd_create_plan(sz.rank, sz.narray,					 dir, measure_flag | wisdom_flag |					 FFTW_IN_PLACE);     testnd_in_place(sz.rank, sz.narray, dir, validated_plan, alt_api,		     specific, force_buf);     fftwnd_destroy_plan(validated_plan);}/************************************************* * planner tests *************************************************/void test_planner(int rank){     /*      * create and destroy many plans, at random.  Check the      * garbage-collecting allocator of twiddle factors      */     int i, dim;     int r, s;     fftw_mpi_plan p[PLANNER_TEST_SIZE];     fftwnd_mpi_plan pnd[PLANNER_TEST_SIZE];     int *narr, maxdim;     chk_mem_leak = 0;     verbose--;     please_wait();     if (rank < 1)          rank = 1;     narr = (int *) fftw_malloc(rank * sizeof(int));     for (i = 0; i < PLANNER_TEST_SIZE; ++i) {          p[i] = (fftw_mpi_plan) 0;          pnd[i] = (fftwnd_mpi_plan) 0;     }     if (PLANNER_TEST_SIZE >= 8) {	  p[0] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_FORWARD, 0);	  p[1] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_FORWARD, 0);	  p[2] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_BACKWARD, 0);	  p[3] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_BACKWARD, 0);	  p[4] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_FORWARD, 0);	  p[5] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_FORWARD, 0);	  p[6] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_BACKWARD, 0);	  p[7] = fftw_mpi_create_plan(MPI_COMM_WORLD, 1024, FFTW_BACKWARD, 0);     }     maxdim = (int) pow(8192, 1.0/rank);     for (i = 0; i < PLANNER_TEST_SIZE * PLANNER_TEST_SIZE; ++i) {          r = rand();          if (r < 0)               r = -r;          r = r % PLANNER_TEST_SIZE;          for (dim = 0; dim < rank; ++dim) {               do {                    s = rand();                    if (s < 0)                         s = -s;                    s = s % maxdim + 1;               } while (s == 0);               narr[dim] = s;          }          if (rank == 1) {               if (p[r])                    fftw_mpi_destroy_plan(p[r]);               p[r] = fftw_mpi_create_plan(MPI_COMM_WORLD,					   narr[0], random_dir(), 					   measure_flag | wisdom_flag);          }	  if (rank > 1) {	       if (pnd[r])		    fftwnd_mpi_destroy_plan(pnd[r]);	       	       pnd[r] = fftwnd_mpi_create_plan(MPI_COMM_WORLD, rank, narr,					       random_dir(), measure_flag |					       wisdom_flag);	  }          if (i % (PLANNER_TEST_SIZE * PLANNER_TEST_SIZE / 20) == 0) {               WHEN_VERBOSE(0, my_printf("test planner: so far so good\n"));               WHEN_VERBOSE(0, my_printf("test planner: iteration %d"					 " out of %d\n",                              i, PLANNER_TEST_SIZE * PLANNER_TEST_SIZE));          }     }     for (i = 0; i < PLANNER_TEST_SIZE; ++i) {          if (p[i])               fftw_mpi_destroy_plan(p[i]);          if (pnd[i])               fftwnd_mpi_destroy_plan(pnd[i]);     }     fftw_free(narr);     verbose++;     chk_mem_leak = 1;}/************************************************* * test initialization *************************************************/void test_init(int *argc, char ***argv){     int i;     unsigned int seed;     MPI_Init(argc,argv);     MPI_Comm_size(MPI_COMM_WORLD,&ncpus);     MPI_Comm_rank(MPI_COMM_WORLD,&my_cpu);     /* Only process 0 gets to do I/O: */     io_okay = my_cpu == 0;     if (io_okay)	  for (i = 1; i < *argc; ++i)	       if (!strcmp((*argv)[i], "--only-parallel")) {		    only_parallel = 1;		    strcpy((*argv)[i], "");	       }     MPI_Bcast(&only_parallel, 1, MPI_INT, 0, MPI_COMM_WORLD);     /* Make sure all processes use the same seed for random numbers: */     seed = time(NULL);     MPI_Bcast(&seed, 1, MPI_INT, 0, MPI_COMM_WORLD);     srand(seed);     fftw_die_hook = fftw_mpi_die; /* call MPI_Abort on failure */}void test_finish(void){     MPI_Finalize();}void enter_paranoid_mode(void){}/* in MPI, only process 0 is guaranteed to have access to the argument list */int get_option(int argc, char **argv, char *argval, int argval_maxlen){     int c;     int arglen;     if (io_okay) {	  c = default_get_option(argc,argv,argval,argval_maxlen);	  arglen = strlen(argval) + 1;     }     MPI_Bcast(&c, 1, MPI_INT, 0, MPI_COMM_WORLD);     MPI_Bcast(&arglen, 1, MPI_INT, 0, MPI_COMM_WORLD);     MPI_Bcast(argval, arglen, MPI_CHAR, 0, MPI_COMM_WORLD);     return c;}

⌨️ 快捷键说明

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