static const char help[] = "1D multiphysics prototype with analytic Jacobians to solve individual problems and a coupled problem.\n\n"; /* Solve a PDE coupled to an algebraic system in 1D * * PDE (U): * -(k u_x)_x = 1 on (0,1), subject to u(0) = 0, u(1) = 1 * Algebraic (K): * exp(k-1) + k = 1/(1/(1+u) + 1/(1+u_x^2)) * * The discretization places k at staggered points, and a separate DMDA is used for each "physics". * * This example is a prototype for coupling in multi-physics problems, therefore residual evaluation and assembly for * each problem (referred to as U and K) are written separately. This permits the same "physics" code to be used for * solving each uncoupled problem as well as the coupled system. In particular, run with -problem_type 0 to solve only * problem U (with K fixed), -problem_type 1 to solve only K (with U fixed), and -problem_type 2 to solve both at once. * * In all cases, a fully-assembled analytic Jacobian is available, so the systems can be solved with a direct solve or * any other standard method. Additionally, by running with * * -pack_dm_mat_type nest * * The same code assembles a coupled matrix where each block is stored separately, which allows the use of PCFieldSplit * without copying values to extract submatrices. */ #include #include #include #include typedef struct _UserCtx *User; struct _UserCtx { PetscInt ptype; DM pack; Vec Uloc, Kloc; }; static PetscErrorCode FormFunctionLocal_U(User user, DMDALocalInfo *info, const PetscScalar u[], const PetscScalar k[], PetscScalar f[]) { PetscReal hx = 1. / info->mx; PetscInt i; PetscFunctionBeginUser; for (i = info->xs; i < info->xs + info->xm; i++) { if (i == 0) f[i] = 1. / hx * u[i]; else if (i == info->mx - 1) f[i] = 1. / hx * (u[i] - 1.0); else f[i] = hx * ((k[i - 1] * (u[i] - u[i - 1]) - k[i] * (u[i + 1] - u[i])) / (hx * hx) - 1.0); } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormFunctionLocal_K(User user, DMDALocalInfo *info, const PetscScalar u[], const PetscScalar k[], PetscScalar f[]) { PetscReal hx = 1. / info->mx; PetscInt i; PetscFunctionBeginUser; for (i = info->xs; i < info->xs + info->xm; i++) { const PetscScalar ubar = 0.5 * (u[i + 1] + u[i]), gradu = (u[i + 1] - u[i]) / hx, g = 1. + gradu * gradu, w = 1. / (1. + ubar) + 1. / g; f[i] = hx * (PetscExpScalar(k[i] - 1.0) + k[i] - 1. / w); } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormFunction_All(SNES snes, Vec X, Vec F, PetscCtx ctx) { User user = (User)ctx; DM dau, dak; DMDALocalInfo infou, infok; PetscScalar *u, *k; PetscScalar *fu, *fk; Vec Uloc, Kloc, Fu, Fk; PetscFunctionBeginUser; PetscCall(DMCompositeGetEntries(user->pack, &dau, &dak)); PetscCall(DMDAGetLocalInfo(dau, &infou)); PetscCall(DMDAGetLocalInfo(dak, &infok)); PetscCall(DMCompositeGetLocalVectors(user->pack, &Uloc, &Kloc)); switch (user->ptype) { case 0: PetscCall(DMGlobalToLocalBegin(dau, X, INSERT_VALUES, Uloc)); PetscCall(DMGlobalToLocalEnd(dau, X, INSERT_VALUES, Uloc)); PetscCall(DMDAVecGetArray(dau, Uloc, &u)); PetscCall(DMDAVecGetArray(dak, user->Kloc, &k)); PetscCall(DMDAVecGetArray(dau, F, &fu)); PetscCall(FormFunctionLocal_U(user, &infou, u, k, fu)); PetscCall(DMDAVecRestoreArray(dau, F, &fu)); PetscCall(DMDAVecRestoreArray(dau, Uloc, &u)); PetscCall(DMDAVecRestoreArray(dak, user->Kloc, &k)); break; case 1: PetscCall(DMGlobalToLocalBegin(dak, X, INSERT_VALUES, Kloc)); PetscCall(DMGlobalToLocalEnd(dak, X, INSERT_VALUES, Kloc)); PetscCall(DMDAVecGetArray(dau, user->Uloc, &u)); PetscCall(DMDAVecGetArray(dak, Kloc, &k)); PetscCall(DMDAVecGetArray(dak, F, &fk)); PetscCall(FormFunctionLocal_K(user, &infok, u, k, fk)); PetscCall(DMDAVecRestoreArray(dak, F, &fk)); PetscCall(DMDAVecRestoreArray(dau, user->Uloc, &u)); PetscCall(DMDAVecRestoreArray(dak, Kloc, &k)); break; case 2: PetscCall(DMCompositeScatter(user->pack, X, Uloc, Kloc)); PetscCall(DMDAVecGetArray(dau, Uloc, &u)); PetscCall(DMDAVecGetArray(dak, Kloc, &k)); PetscCall(DMCompositeGetAccess(user->pack, F, &Fu, &Fk)); PetscCall(DMDAVecGetArray(dau, Fu, &fu)); PetscCall(DMDAVecGetArray(dak, Fk, &fk)); PetscCall(FormFunctionLocal_U(user, &infou, u, k, fu)); PetscCall(FormFunctionLocal_K(user, &infok, u, k, fk)); PetscCall(DMDAVecRestoreArray(dau, Fu, &fu)); PetscCall(DMDAVecRestoreArray(dak, Fk, &fk)); PetscCall(DMCompositeRestoreAccess(user->pack, F, &Fu, &Fk)); PetscCall(DMDAVecRestoreArray(dau, Uloc, &u)); PetscCall(DMDAVecRestoreArray(dak, Kloc, &k)); break; } PetscCall(DMCompositeRestoreLocalVectors(user->pack, &Uloc, &Kloc)); PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormJacobianLocal_U(User user, DMDALocalInfo *info, const PetscScalar u[], const PetscScalar k[], Mat Buu) { PetscReal hx = 1. / info->mx; PetscInt i; PetscFunctionBeginUser; for (i = info->xs; i < info->xs + info->xm; i++) { PetscInt row = i - info->gxs, cols[] = {row - 1, row, row + 1}; PetscScalar val = 1. / hx; if (i == 0) { PetscCall(MatSetValuesLocal(Buu, 1, &row, 1, &row, &val, INSERT_VALUES)); } else if (i == info->mx - 1) { PetscCall(MatSetValuesLocal(Buu, 1, &row, 1, &row, &val, INSERT_VALUES)); } else { PetscScalar vals[] = {-k[i - 1] / hx, (k[i - 1] + k[i]) / hx, -k[i] / hx}; PetscCall(MatSetValuesLocal(Buu, 1, &row, 3, cols, vals, INSERT_VALUES)); } } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormJacobianLocal_K(User user, DMDALocalInfo *info, const PetscScalar u[], const PetscScalar k[], Mat Bkk) { PetscReal hx = 1. / info->mx; PetscInt i; PetscFunctionBeginUser; for (i = info->xs; i < info->xs + info->xm; i++) { PetscInt row = i - info->gxs; PetscScalar vals[] = {hx * (PetscExpScalar(k[i] - 1.) + (PetscScalar)1.)}; PetscCall(MatSetValuesLocal(Bkk, 1, &row, 1, &row, vals, INSERT_VALUES)); } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormJacobianLocal_UK(User user, DMDALocalInfo *info, DMDALocalInfo *infok, const PetscScalar u[], const PetscScalar k[], Mat Buk) { PetscReal hx = 1. / info->mx; PetscInt i; PetscInt row, cols[2]; PetscScalar vals[2]; PetscFunctionBeginUser; if (!Buk) PetscFunctionReturn(PETSC_SUCCESS); /* Not assembling this block */ for (i = info->xs; i < info->xs + info->xm; i++) { if (i == 0 || i == info->mx - 1) continue; row = i - info->gxs; cols[0] = i - 1 - infok->gxs; vals[0] = (u[i] - u[i - 1]) / hx; cols[1] = i - infok->gxs; vals[1] = (u[i] - u[i + 1]) / hx; PetscCall(MatSetValuesLocal(Buk, 1, &row, 2, cols, vals, INSERT_VALUES)); } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormJacobianLocal_KU(User user, DMDALocalInfo *info, DMDALocalInfo *infok, const PetscScalar u[], const PetscScalar k[], Mat Bku) { PetscInt i; PetscReal hx = 1. / (info->mx - 1); PetscFunctionBeginUser; if (!Bku) PetscFunctionReturn(PETSC_SUCCESS); /* Not assembling this block */ for (i = infok->xs; i < infok->xs + infok->xm; i++) { PetscInt row = i - infok->gxs, cols[2]; PetscScalar vals[2]; const PetscScalar ubar = 0.5 * (u[i] + u[i + 1]), ubar_L = 0.5, ubar_R = 0.5, gradu = (u[i + 1] - u[i]) / hx, gradu_L = -1. / hx, gradu_R = 1. / hx, g = 1. + PetscSqr(gradu), g_gradu = 2. * gradu, w = 1. / (1. + ubar) + 1. / g, w_ubar = -1. / PetscSqr(1. + ubar), w_gradu = -g_gradu / PetscSqr(g), iw = 1. / w, iw_ubar = -w_ubar * PetscSqr(iw), iw_gradu = -w_gradu * PetscSqr(iw); cols[0] = i - info->gxs; vals[0] = -hx * (iw_ubar * ubar_L + iw_gradu * gradu_L); cols[1] = i + 1 - info->gxs; vals[1] = -hx * (iw_ubar * ubar_R + iw_gradu * gradu_R); PetscCall(MatSetValuesLocal(Bku, 1, &row, 2, cols, vals, INSERT_VALUES)); } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormJacobian_All(SNES snes, Vec X, Mat J, Mat B, PetscCtx ctx) { User user = (User)ctx; DM dau, dak; DMDALocalInfo infou, infok; PetscScalar *u, *k; Vec Uloc, Kloc; PetscFunctionBeginUser; PetscCall(DMCompositeGetEntries(user->pack, &dau, &dak)); PetscCall(DMDAGetLocalInfo(dau, &infou)); PetscCall(DMDAGetLocalInfo(dak, &infok)); PetscCall(DMCompositeGetLocalVectors(user->pack, &Uloc, &Kloc)); switch (user->ptype) { case 0: PetscCall(DMGlobalToLocalBegin(dau, X, INSERT_VALUES, Uloc)); PetscCall(DMGlobalToLocalEnd(dau, X, INSERT_VALUES, Uloc)); PetscCall(DMDAVecGetArray(dau, Uloc, &u)); PetscCall(DMDAVecGetArray(dak, user->Kloc, &k)); PetscCall(FormJacobianLocal_U(user, &infou, u, k, B)); PetscCall(DMDAVecRestoreArray(dau, Uloc, &u)); PetscCall(DMDAVecRestoreArray(dak, user->Kloc, &k)); break; case 1: PetscCall(DMGlobalToLocalBegin(dak, X, INSERT_VALUES, Kloc)); PetscCall(DMGlobalToLocalEnd(dak, X, INSERT_VALUES, Kloc)); PetscCall(DMDAVecGetArray(dau, user->Uloc, &u)); PetscCall(DMDAVecGetArray(dak, Kloc, &k)); PetscCall(FormJacobianLocal_K(user, &infok, u, k, B)); PetscCall(DMDAVecRestoreArray(dau, user->Uloc, &u)); PetscCall(DMDAVecRestoreArray(dak, Kloc, &k)); break; case 2: { Mat Buu, Buk, Bku, Bkk; PetscBool nest; IS *is; PetscCall(DMCompositeScatter(user->pack, X, Uloc, Kloc)); PetscCall(DMDAVecGetArray(dau, Uloc, &u)); PetscCall(DMDAVecGetArray(dak, Kloc, &k)); PetscCall(DMCompositeGetLocalISs(user->pack, &is)); PetscCall(MatGetLocalSubMatrix(B, is[0], is[0], &Buu)); PetscCall(MatGetLocalSubMatrix(B, is[0], is[1], &Buk)); PetscCall(MatGetLocalSubMatrix(B, is[1], is[0], &Bku)); PetscCall(MatGetLocalSubMatrix(B, is[1], is[1], &Bkk)); PetscCall(FormJacobianLocal_U(user, &infou, u, k, Buu)); PetscCall(PetscObjectTypeCompare((PetscObject)B, MATNEST, &nest)); if (!nest) { /* DMCreateMatrix_Composite() for a nested matrix does not generate off-block matrices that one can call MatSetValuesLocal() on, it just creates dummy matrices with no entries; there cannot insert values into them. Commit b6480e041dd2293a65f96222772d68cdb4ed6306 changed Mat_Nest() from returning NULL pointers for these submatrices to dummy matrices because PCFIELDSPLIT could not handle the returned null matrices. */ PetscCall(FormJacobianLocal_UK(user, &infou, &infok, u, k, Buk)); PetscCall(FormJacobianLocal_KU(user, &infou, &infok, u, k, Bku)); } PetscCall(FormJacobianLocal_K(user, &infok, u, k, Bkk)); PetscCall(MatRestoreLocalSubMatrix(B, is[0], is[0], &Buu)); PetscCall(MatRestoreLocalSubMatrix(B, is[0], is[1], &Buk)); PetscCall(MatRestoreLocalSubMatrix(B, is[1], is[0], &Bku)); PetscCall(MatRestoreLocalSubMatrix(B, is[1], is[1], &Bkk)); PetscCall(DMDAVecRestoreArray(dau, Uloc, &u)); PetscCall(DMDAVecRestoreArray(dak, Kloc, &k)); PetscCall(ISDestroy(&is[0])); PetscCall(ISDestroy(&is[1])); PetscCall(PetscFree(is)); } break; } PetscCall(DMCompositeRestoreLocalVectors(user->pack, &Uloc, &Kloc)); PetscCall(MatAssemblyBegin(B, MAT_FINAL_ASSEMBLY)); PetscCall(MatAssemblyEnd(B, MAT_FINAL_ASSEMBLY)); if (J != B) { PetscCall(MatAssemblyBegin(J, MAT_FINAL_ASSEMBLY)); PetscCall(MatAssemblyEnd(J, MAT_FINAL_ASSEMBLY)); } PetscFunctionReturn(PETSC_SUCCESS); } static PetscErrorCode FormInitial_Coupled(User user, Vec X) { DM dau, dak; DMDALocalInfo infou, infok; Vec Xu, Xk; PetscScalar *u, *k, hx; PetscInt i; PetscFunctionBeginUser; PetscCall(DMCompositeGetEntries(user->pack, &dau, &dak)); PetscCall(DMCompositeGetAccess(user->pack, X, &Xu, &Xk)); PetscCall(DMDAVecGetArray(dau, Xu, &u)); PetscCall(DMDAVecGetArray(dak, Xk, &k)); PetscCall(DMDAGetLocalInfo(dau, &infou)); PetscCall(DMDAGetLocalInfo(dak, &infok)); hx = 1. / (infok.mx); for (i = infou.xs; i < infou.xs + infou.xm; i++) u[i] = (PetscScalar)i * hx * (1. - (PetscScalar)i * hx); for (i = infok.xs; i < infok.xs + infok.xm; i++) k[i] = 1.0 + 0.5 * PetscSinScalar(2 * PETSC_PI * i * hx); PetscCall(DMDAVecRestoreArray(dau, Xu, &u)); PetscCall(DMDAVecRestoreArray(dak, Xk, &k)); PetscCall(DMCompositeRestoreAccess(user->pack, X, &Xu, &Xk)); PetscCall(DMCompositeScatter(user->pack, X, user->Uloc, user->Kloc)); PetscFunctionReturn(PETSC_SUCCESS); } int main(int argc, char *argv[]) { DM dau, dak, pack; const PetscInt *lxu; PetscInt *lxk, m, sizes; User user; SNES snes; Vec X, F, Xu, Xk, Fu, Fk; Mat B; IS *isg; PetscBool pass_dm; VecType vtype; PetscFunctionBeginUser; PetscCall(PetscInitialize(&argc, &argv, 0, help)); PetscCall(DMDACreate1d(PETSC_COMM_WORLD, DM_BOUNDARY_NONE, 10, 1, 1, NULL, &dau)); PetscCall(DMSetOptionsPrefix(dau, "u_")); PetscCall(DMSetFromOptions(dau)); PetscCall(DMSetUp(dau)); PetscCall(DMDAGetOwnershipRanges(dau, &lxu, 0, 0)); PetscCall(DMDAGetInfo(dau, 0, &m, 0, 0, &sizes, 0, 0, 0, 0, 0, 0, 0, 0)); PetscCall(PetscMalloc1(sizes, &lxk)); PetscCall(PetscArraycpy(lxk, lxu, sizes)); lxk[0]--; PetscCall(DMDACreate1d(PETSC_COMM_WORLD, DM_BOUNDARY_NONE, m - 1, 1, 1, lxk, &dak)); PetscCall(DMSetOptionsPrefix(dak, "k_")); PetscCall(DMSetFromOptions(dak)); PetscCall(DMSetUp(dak)); PetscCall(PetscFree(lxk)); PetscCall(DMCompositeCreate(PETSC_COMM_WORLD, &pack)); PetscCall(DMSetOptionsPrefix(pack, "pack_")); PetscCall(DMCompositeAddDM(pack, dau)); PetscCall(DMCompositeAddDM(pack, dak)); PetscCall(DMDASetFieldName(dau, 0, "u")); PetscCall(DMDASetFieldName(dak, 0, "k")); PetscCall(DMSetFromOptions(pack)); /* if we call -pack_dm_vec_type xxx it does not get propagated to the subDMs */ PetscCall(DMGetVecType(pack, &vtype)); PetscCall(DMSetVecType(dau, vtype)); PetscCall(DMSetVecType(dak, vtype)); PetscCall(DMCreateGlobalVector(pack, &X)); PetscCall(VecDuplicate(X, &F)); PetscCall(PetscNew(&user)); user->pack = pack; PetscCall(DMCompositeGetGlobalISs(pack, &isg)); PetscCall(DMCompositeGetLocalVectors(pack, &user->Uloc, &user->Kloc)); PetscCall(DMCompositeScatter(pack, X, user->Uloc, user->Kloc)); PetscOptionsBegin(PETSC_COMM_WORLD, NULL, "Coupled problem options", "SNES"); { user->ptype = 0; pass_dm = PETSC_TRUE; PetscCall(PetscOptionsInt("-problem_type", "0: solve for u only, 1: solve for k only, 2: solve for both", 0, user->ptype, &user->ptype, NULL)); PetscCall(PetscOptionsBool("-pass_dm", "Pass the packed DM to SNES to use when determining splits and forward into splits", 0, pass_dm, &pass_dm, NULL)); } PetscOptionsEnd(); PetscCall(FormInitial_Coupled(user, X)); PetscCall(SNESCreate(PETSC_COMM_WORLD, &snes)); switch (user->ptype) { case 0: PetscCall(DMCompositeGetAccess(pack, X, &Xu, PETSC_NULLPTR)); PetscCall(DMCompositeGetAccess(pack, F, &Fu, PETSC_NULLPTR)); PetscCall(DMCreateMatrix(dau, &B)); PetscCall(SNESSetFunction(snes, Fu, FormFunction_All, user)); PetscCall(SNESSetJacobian(snes, B, B, FormJacobian_All, user)); PetscCall(SNESSetFromOptions(snes)); PetscCall(SNESSetDM(snes, dau)); PetscCall(SNESSolve(snes, NULL, Xu)); PetscCall(DMCompositeRestoreAccess(pack, X, &Xu, PETSC_NULLPTR)); PetscCall(DMCompositeRestoreAccess(pack, F, &Fu, PETSC_NULLPTR)); break; case 1: PetscCall(DMCompositeGetAccess(pack, X, PETSC_NULLPTR, &Xk)); PetscCall(DMCompositeGetAccess(pack, F, PETSC_NULLPTR, &Fk)); PetscCall(DMCreateMatrix(dak, &B)); PetscCall(SNESSetFunction(snes, Fk, FormFunction_All, user)); PetscCall(SNESSetJacobian(snes, B, B, FormJacobian_All, user)); PetscCall(SNESSetFromOptions(snes)); PetscCall(SNESSetDM(snes, dak)); PetscCall(SNESSolve(snes, NULL, Xk)); PetscCall(DMCompositeRestoreAccess(pack, X, PETSC_NULLPTR, &Xk)); PetscCall(DMCompositeRestoreAccess(pack, F, PETSC_NULLPTR, &Fk)); break; case 2: PetscCall(DMCreateMatrix(pack, &B)); /* This example does not correctly allocate off-diagonal blocks. These options allows new nonzeros (slow). */ PetscCall(MatSetOption(B, MAT_NEW_NONZERO_LOCATION_ERR, PETSC_FALSE)); PetscCall(MatSetOption(B, MAT_NEW_NONZERO_ALLOCATION_ERR, PETSC_FALSE)); PetscCall(SNESSetFunction(snes, F, FormFunction_All, user)); PetscCall(SNESSetJacobian(snes, B, B, FormJacobian_All, user)); PetscCall(SNESSetFromOptions(snes)); if (!pass_dm) { /* Manually provide index sets and names for the splits */ KSP ksp; PC pc; PetscCall(SNESGetKSP(snes, &ksp)); PetscCall(KSPGetPC(ksp, &pc)); PetscCall(PCFieldSplitSetIS(pc, "u", isg[0])); PetscCall(PCFieldSplitSetIS(pc, "k", isg[1])); } else { /* The same names come from the options prefix for dau and dak. This option can support geometric multigrid inside * of splits, but it requires using a DM (perhaps your own implementation). */ PetscCall(SNESSetDM(snes, pack)); } PetscCall(SNESSolve(snes, NULL, X)); break; } PetscCall(VecViewFromOptions(X, NULL, "-view_sol")); if (0) { PetscInt col = 0; PetscBool mult_dup = PETSC_FALSE, view_dup = PETSC_FALSE; Mat D; Vec Y; PetscCall(PetscOptionsGetInt(NULL, 0, "-col", &col, 0)); PetscCall(PetscOptionsGetBool(NULL, 0, "-mult_dup", &mult_dup, 0)); PetscCall(PetscOptionsGetBool(NULL, 0, "-view_dup", &view_dup, 0)); PetscCall(VecDuplicate(X, &Y)); /* PetscCall(MatAssemblyBegin(B,MAT_FINAL_ASSEMBLY)); */ /* PetscCall(MatAssemblyEnd(B,MAT_FINAL_ASSEMBLY)); */ PetscCall(MatConvert(B, MATAIJ, MAT_INITIAL_MATRIX, &D)); PetscCall(VecZeroEntries(X)); PetscCall(VecSetValue(X, col, 1.0, INSERT_VALUES)); PetscCall(VecAssemblyBegin(X)); PetscCall(VecAssemblyEnd(X)); PetscCall(MatMult(mult_dup ? D : B, X, Y)); PetscCall(MatView(view_dup ? D : B, PETSC_VIEWER_STDOUT_WORLD)); /* PetscCall(VecView(X,PETSC_VIEWER_STDOUT_WORLD)); */ PetscCall(VecView(Y, PETSC_VIEWER_STDOUT_WORLD)); PetscCall(MatDestroy(&D)); PetscCall(VecDestroy(&Y)); } PetscCall(DMCompositeRestoreLocalVectors(pack, &user->Uloc, &user->Kloc)); PetscCall(PetscFree(user)); PetscCall(ISDestroy(&isg[0])); PetscCall(ISDestroy(&isg[1])); PetscCall(PetscFree(isg)); PetscCall(VecDestroy(&X)); PetscCall(VecDestroy(&F)); PetscCall(MatDestroy(&B)); PetscCall(DMDestroy(&dau)); PetscCall(DMDestroy(&dak)); PetscCall(DMDestroy(&pack)); PetscCall(SNESDestroy(&snes)); PetscCall(PetscFinalize()); return 0; } /*TEST test: suffix: 0 nsize: 3 args: -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -problem_type 0 test: suffix: 1 nsize: 3 args: -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -problem_type 1 test: suffix: 2 nsize: 3 args: -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -problem_type 2 testset: suffix: 3 nsize: 3 output_file: output/ex28_3.out args: -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -ksp_monitor_short -problem_type 2 -snes_mf_operator -pc_type fieldsplit -pc_fieldsplit_dm_splits -pc_fieldsplit_type additive -fieldsplit_u_ksp_type gmres -fieldsplit_k_pc_type jacobi test: suffix: std args: -pack_dm_mat_type {{aij nest}} test: suffix: cuda requires: cuda args: -pack_dm_mat_type aijcusparse -pack_dm_vec_type cuda test: suffix: hip requires: hip args: -pack_dm_mat_type aijhipsparse -pack_dm_vec_type hip test: suffix: kok requires: kokkos_kernels args: -pack_dm_mat_type aijkokkos -pack_dm_vec_type kokkos test: requires: mumps suffix: 3_nest_lu nsize: {{1 3}} output_file: output/ex28_3.out args: -pack_dm_mat_type nest -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -ksp_monitor_short -problem_type 2 -snes_mf_operator -pc_type lu -pc_factor_mat_solver_type mumps test: suffix: 4 nsize: 6 args: -u_da_grid_x 257 -snes_converged_reason -snes_monitor_short -ksp_monitor_short -problem_type 2 -snes_mf_operator -pack_dm_mat_type aij -pc_type fieldsplit -pc_fieldsplit_type multiplicative -fieldsplit_u_ksp_type gmres -fieldsplit_u_ksp_pc_side right -fieldsplit_u_pc_type mg -fieldsplit_u_pc_mg_levels 4 -fieldsplit_u_mg_levels_ksp_type richardson -fieldsplit_u_mg_levels_ksp_max_it 1 -fieldsplit_u_mg_levels_pc_type sor -fieldsplit_u_pc_mg_galerkin pmat -fieldsplit_u_ksp_converged_reason -fieldsplit_k_pc_type jacobi requires: !single test: suffix: glvis_composite_da_1d args: -u_da_grid_x 20 -problem_type 0 -snes_monitor_solution glvis: test: suffix: cuda nsize: 1 requires: cuda args: -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -problem_type 2 -pack_dm_mat_type aijcusparse -pack_dm_vec_type cuda test: suffix: viennacl nsize: 1 requires: viennacl args: -u_da_grid_x 20 -snes_converged_reason -snes_monitor_short -problem_type 2 -pack_dm_mat_type aijviennacl -pack_dm_vec_type viennacl TEST*/