-
Notifications
You must be signed in to change notification settings - Fork 198
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[WIP] AIR example #933
Draft
victorapm
wants to merge
8
commits into
master
Choose a base branch
from
air-example
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Draft
[WIP] AIR example #933
Changes from all commits
Commits
Show all changes
8 commits
Select commit
Hold shift + click to select a range
1e3ae70
Add doxygen string for HYPRE_ParCSRMatrixMatvec
lindsayad af0ca63
Make an example demonstrating AIR for advection
lindsayad 405f766
Use classical interpolation if not doing AIR
lindsayad 18abe9c
Merge pull request #932 from lindsayad/air-example
victorapm cf014b7
Remove comment from protos.h file
victorapm 7e1bfe7
Remove HYPRE_ data types
victorapm afd4ee5
Style changes and update comment about grid_relax_points
victorapm 0cf65b0
Replace hypre allocs with malloc
victorapm File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,363 @@ | ||
/****************************************************************************** | ||
* Copyright (c) 1998 Lawrence Livermore National Security, LLC and other | ||
* HYPRE Project Developers. See the top-level COPYRIGHT file for details. | ||
* | ||
* SPDX-License-Identifier: (Apache-2.0 OR MIT) | ||
******************************************************************************/ | ||
|
||
/* | ||
Example 19 | ||
|
||
Interface: Linear-Algebraic (IJ) | ||
|
||
Compile with: make ex19 | ||
|
||
Sample run: mpirun -np 4 ex19 | ||
|
||
Description: This example solves a 1D steady upwinded advection problem using | ||
AIR. With one up sweep and one coarse sweep, AIR converges in one | ||
iteration. If the transpose of the interpolation operator is used | ||
instead (-restriction 0), the multigrid solve fails to converge | ||
*/ | ||
|
||
|
||
#include <stdio.h> | ||
#include <stdlib.h> | ||
#include <string.h> | ||
#include <math.h> | ||
#include "_hypre_utilities.h" | ||
#include "HYPRE_krylov.h" | ||
#include "HYPRE.h" | ||
#include "HYPRE_parcsr_ls.h" | ||
#include "ex.h" | ||
|
||
#define my_min(a,b) (((a)<(b)) ? (a) : (b)) | ||
|
||
int main (int argc, char *argv[]) | ||
{ | ||
int i; | ||
int myid, num_procs; | ||
int N, n = 33; | ||
|
||
int ilower, iupper; | ||
int local_size, extra; | ||
|
||
int print_system = 0; | ||
|
||
HYPRE_IJMatrix A; | ||
HYPRE_ParCSRMatrix parcsr_A; | ||
HYPRE_IJVector b; | ||
HYPRE_ParVector par_b; | ||
HYPRE_IJVector x; | ||
HYPRE_ParVector par_x; | ||
HYPRE_IJVector work; | ||
HYPRE_ParVector par_work; | ||
|
||
HYPRE_Solver solver; | ||
|
||
/* AMG */ | ||
int num_iterations; | ||
double final_res_norm; | ||
int interp_type = 100; /* 1-pt Interp */ | ||
int ns_coarse = 1, ns_down = 0, ns_up = 1; | ||
int **grid_relax_points = NULL; | ||
int coarse_threshold = 20; | ||
int agg_num_levels = 0; /* AIR does not support aggressive coarsening */ | ||
int restriction = 1; | ||
|
||
/* Initialize MPI */ | ||
MPI_Init(&argc, &argv); | ||
MPI_Comm_rank(MPI_COMM_WORLD, &myid); | ||
MPI_Comm_size(MPI_COMM_WORLD, &num_procs); | ||
|
||
/* Initialize HYPRE */ | ||
HYPRE_Initialize(); | ||
|
||
/* Print GPU info */ | ||
/* HYPRE_PrintDeviceInfo(); */ | ||
#if defined(HYPRE_USING_GPU) | ||
/* use vendor implementation for SpGEMM */ | ||
HYPRE_SetSpGemmUseVendor(0); | ||
#endif | ||
|
||
/* Parse command line */ | ||
{ | ||
int arg_index = 0; | ||
int print_usage = 0; | ||
|
||
while (arg_index < argc) | ||
{ | ||
if ( strcmp(argv[arg_index], "-n") == 0 ) | ||
{ | ||
arg_index++; | ||
n = atoi(argv[arg_index++]); | ||
} | ||
else if ( strcmp(argv[arg_index], "-print_system") == 0 ) | ||
{ | ||
arg_index++; | ||
print_system = 1; | ||
} | ||
else if ( strcmp(argv[arg_index], "-help") == 0 ) | ||
{ | ||
print_usage = 1; | ||
break; | ||
} | ||
else if ( strcmp(argv[arg_index], "-ns_coarse") == 0) | ||
{ | ||
arg_index++; | ||
ns_coarse = atoi(argv[arg_index++]); | ||
} | ||
else if ( strcmp(argv[arg_index], "-ns_down") == 0) | ||
{ | ||
arg_index++; | ||
ns_down = atoi(argv[arg_index++]); | ||
} | ||
else if ( strcmp(argv[arg_index], "-ns_up") == 0) | ||
{ | ||
arg_index++; | ||
ns_up = atoi(argv[arg_index++]); | ||
} | ||
else if ( strcmp(argv[arg_index], "-restriction") == 0) | ||
{ | ||
arg_index++; | ||
restriction = atoi(argv[arg_index++]); | ||
} | ||
else | ||
{ | ||
arg_index++; | ||
} | ||
} | ||
|
||
/* Do classical modified interpolation if not doing AIR */ | ||
if (!restriction) | ||
{ | ||
interp_type = 0; | ||
} | ||
|
||
if ((print_usage) && (myid == 0)) | ||
{ | ||
printf("\n"); | ||
printf("Usage: %s [<options>]\n", argv[0]); | ||
printf("\n"); | ||
printf(" -n <n> : problem size (default: 33)\n"); | ||
printf(" -print_system : print the matrix, rhs, expected solution, and solution\n"); | ||
printf(" -ns_coarse : the number of sweeps performed for the coarse cycle " | ||
"(default: 1)\n"); | ||
printf(" -ns_down : the number of sweeps performed for the down cycle " | ||
"(default: 0)\n"); | ||
printf(" -ns_up : the number of sweeps performed for the up cycle " | ||
"(default: 1)\n"); | ||
printf(" -restriction : defines the restriction operator, 0 for transpose of " | ||
"interpolation, 1 for AIR-1, 2 for AIR-2 (default: 1)\n"); | ||
printf("\n"); | ||
} | ||
|
||
if (print_usage) | ||
{ | ||
MPI_Finalize(); | ||
return (0); | ||
} | ||
} | ||
|
||
/* Preliminaries: want at least one processor per row */ | ||
N = n; /* global number of rows */ | ||
|
||
/* Each processor knows only of its own rows - the range is denoted by ilower | ||
and upper. Here we partition the rows. We account for the fact that | ||
N may not divide evenly by the number of processors. */ | ||
local_size = N / num_procs; | ||
extra = N - local_size * num_procs; | ||
|
||
ilower = local_size * myid; | ||
ilower += my_min(myid, extra); | ||
|
||
iupper = local_size * (myid + 1); | ||
iupper += my_min(myid + 1, extra); | ||
iupper = iupper - 1; | ||
|
||
/* How many rows do I have? */ | ||
local_size = iupper - ilower + 1; | ||
|
||
/* Create the matrix. | ||
Note that this is a square matrix, so we indicate the row partition | ||
size twice (since number of rows = number of cols) */ | ||
HYPRE_IJMatrixCreate(MPI_COMM_WORLD, ilower, iupper, ilower, iupper, &A); | ||
|
||
/* Choose a parallel csr format storage (see the User's Manual) */ | ||
HYPRE_IJMatrixSetObjectType(A, HYPRE_PARCSR); | ||
|
||
/* Initialize before setting coefficients */ | ||
HYPRE_IJMatrixInitialize(A); | ||
|
||
/* Set matrix entries corresponding to 1D upwind advection */ | ||
{ | ||
int nnz; | ||
/* OK to use constant-length arrays for CPUs | ||
double values[2]; | ||
int cols[2]; | ||
*/ | ||
double *values = (double *) malloc(2 * sizeof(double)); | ||
int *cols = (int *) malloc(2 * sizeof(int)); | ||
int *tmp = (int *) malloc(2 * sizeof(int)); | ||
|
||
for (i = ilower; i <= iupper; i++) | ||
{ | ||
if (i == 0) | ||
{ | ||
nnz = 1; | ||
cols[0] = 0; | ||
values[0] = 1; | ||
} | ||
else | ||
{ | ||
nnz = 2; | ||
cols[0] = i - 1; | ||
cols[1] = i; | ||
values[0] = -1; | ||
values[1] = 1; | ||
} | ||
|
||
/* Set the values for row i */ | ||
tmp[0] = nnz; | ||
tmp[1] = i; | ||
HYPRE_IJMatrixSetValues(A, 1, &tmp[0], &tmp[1], cols, values); | ||
} | ||
|
||
free(values); | ||
free(cols); | ||
free(tmp); | ||
} | ||
|
||
/* Assemble after setting the coefficients */ | ||
HYPRE_IJMatrixAssemble(A); | ||
|
||
/* Note: for the testing of small problems, one may wish to read | ||
in a matrix in IJ format (for the format, see the output files | ||
from the -print_system option). | ||
In this case, one would use the following routine: | ||
HYPRE_IJMatrixRead( <filename>, MPI_COMM_WORLD, | ||
HYPRE_PARCSR, &A ); | ||
<filename> = IJ.A.out to read in what has been printed out | ||
by -print_system (processor numbers are omitted). | ||
A call to HYPRE_IJMatrixRead is an *alternative* to the | ||
following sequence of HYPRE_IJMatrix calls: | ||
Create, SetObjectType, Initialize, SetValues, and Assemble | ||
*/ | ||
|
||
|
||
/* Get the parcsr matrix object to use */ | ||
HYPRE_IJMatrixGetObject(A, (void**) &parcsr_A); | ||
|
||
|
||
/* Create the solution, work vector, and b */ | ||
HYPRE_IJVectorCreate(MPI_COMM_WORLD, ilower, iupper, &x); | ||
HYPRE_IJVectorSetObjectType(x, HYPRE_PARCSR); | ||
HYPRE_IJVectorInitialize(x); | ||
|
||
HYPRE_IJVectorCreate(MPI_COMM_WORLD, ilower, iupper, &work); | ||
HYPRE_IJVectorSetObjectType(work, HYPRE_PARCSR); | ||
HYPRE_IJVectorInitialize(work); | ||
|
||
HYPRE_IJVectorCreate(MPI_COMM_WORLD, ilower, iupper, &b); | ||
HYPRE_IJVectorSetObjectType(b, HYPRE_PARCSR); | ||
HYPRE_IJVectorInitialize(b); | ||
|
||
HYPRE_IJVectorGetObject(x, (void **) &par_x); | ||
HYPRE_ParVectorSetRandomValues(par_x, 1); | ||
|
||
HYPRE_IJVectorGetObject(work, (void **) &par_work); | ||
HYPRE_ParVectorSetRandomValues(par_work, 2); | ||
|
||
HYPRE_IJVectorGetObject(b, (void **) &par_b); | ||
HYPRE_ParVectorSetConstantValues(par_b, 0); | ||
HYPRE_ParCSRMatrixMatvec(1, parcsr_A, par_work, 0, par_b); | ||
|
||
/* Print out the system - files names will be IJ.out.A.XXXXX | ||
and IJ.out.b.XXXXX, where XXXXX = processor id */ | ||
if (print_system) | ||
{ | ||
HYPRE_IJMatrixPrint(A, "IJ.out.A"); | ||
HYPRE_IJVectorPrint(work, "IJ.out.work"); | ||
HYPRE_IJVectorPrint(b, "IJ.out.b"); | ||
} | ||
|
||
/* This is a 2-D 4-by-k array. Hypre takes ownership of this pointer, | ||
so users do not need to free it */ | ||
grid_relax_points = (int**) malloc(4 * sizeof(int*)); | ||
grid_relax_points[0] = NULL; | ||
grid_relax_points[1] = (int*) malloc(ns_down * sizeof(int)); | ||
grid_relax_points[2] = (int*) malloc(ns_up * sizeof(int)); | ||
grid_relax_points[3] = (int*) malloc(ns_coarse * sizeof(int)); | ||
/* down cycle: C */ | ||
for (i = 0; i < ns_down; i++) | ||
{ | ||
grid_relax_points[1][i] = 1; // C | ||
} | ||
/* up cycle: F */ | ||
for (i = 0; i < ns_up; i++) | ||
{ | ||
grid_relax_points[2][0] = -1; // F | ||
} | ||
/* coarse: all */ | ||
for (i = 0; i < ns_coarse; i++) | ||
{ | ||
grid_relax_points[3][i] = 0; // A(ll) | ||
} | ||
|
||
/* Create solver */ | ||
HYPRE_BoomerAMGCreate(&solver); | ||
|
||
/* Set some parameters (See Reference Manual for more parameters) */ | ||
HYPRE_BoomerAMGSetPrintLevel(solver, 3); /* print solve info + parameters */ | ||
HYPRE_BoomerAMGSetRelaxType(solver, 0); /* Jacobi */ | ||
HYPRE_BoomerAMGSetRelaxOrder(solver, 1); /* uses C/F relaxation */ | ||
HYPRE_BoomerAMGSetCycleNumSweeps(solver, ns_down, 1); | ||
HYPRE_BoomerAMGSetCycleNumSweeps(solver, ns_up, 2); | ||
HYPRE_BoomerAMGSetCycleNumSweeps(solver, ns_coarse, 3); | ||
HYPRE_BoomerAMGSetGridRelaxPoints(solver, grid_relax_points); | ||
HYPRE_BoomerAMGSetMaxLevels(solver, 2); /* maximum number of levels */ | ||
HYPRE_BoomerAMGSetTol(solver, 1e-7); /* conv. tolerance */ | ||
HYPRE_BoomerAMGSetRestriction(solver, restriction); /* AIR */ | ||
HYPRE_BoomerAMGSetAggNumLevels(solver, agg_num_levels); | ||
HYPRE_BoomerAMGSetMaxCoarseSize(solver, coarse_threshold); | ||
HYPRE_BoomerAMGSetInterpType(solver, interp_type); | ||
HYPRE_BoomerAMGSetMaxIter(solver, 100); | ||
|
||
/* Now setup and solve! */ | ||
HYPRE_BoomerAMGSetup(solver, parcsr_A, par_b, par_x); | ||
HYPRE_BoomerAMGSolve(solver, parcsr_A, par_b, par_x); | ||
|
||
/* Run info - needed logging turned on */ | ||
HYPRE_BoomerAMGGetNumIterations(solver, &num_iterations); | ||
HYPRE_BoomerAMGGetFinalRelativeResidualNorm(solver, &final_res_norm); | ||
if (myid == 0) | ||
{ | ||
printf("\n"); | ||
printf("Iterations = %d\n", num_iterations); | ||
printf("Final Relative Residual Norm = %e\n", final_res_norm); | ||
printf("\n"); | ||
} | ||
|
||
/* Destroy solver */ | ||
HYPRE_BoomerAMGDestroy(solver); | ||
|
||
/* Print solution */ | ||
if (print_system) | ||
{ | ||
HYPRE_IJVectorPrint(x, "IJ.out.x"); | ||
} | ||
|
||
/* Clean up */ | ||
HYPRE_IJMatrixDestroy(A); | ||
HYPRE_IJVectorDestroy(b); | ||
HYPRE_IJVectorDestroy(x); | ||
HYPRE_IJVectorDestroy(work); | ||
|
||
/* Finalize HYPRE */ | ||
HYPRE_Finalize(); | ||
|
||
/* Finalize MPI */ | ||
MPI_Finalize(); | ||
|
||
return (0); | ||
} |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can't
int
andHYPRE_Int
differ?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
They can and I see your point (the APIs expect
HYPRE_Int
, so it's more correct to use it). But for the examples, the idea is to make them as simple as possible and use standard C types. The team might re-evaluate that...