// Copyright (c) 2017-2018, Lawrence Livermore National Security, LLC. // Produced at the Lawrence Livermore National Laboratory. LLNL-CODE-734707. // All Rights reserved. See files LICENSE and NOTICE for details. // // This file is part of CEED, a collection of benchmarks, miniapps, software // libraries and APIs for efficient high-order finite element and spectral // element discretizations for exascale applications. For more information and // source code availability see http://github.com/ceed. // // The CEED research is supported by the Exascale Computing Project 17-SC-20-SC, // a collaborative effort of two U.S. Department of Energy organizations (Office // of Science and the National Nuclear Security Administration) responsible for // the planning and preparation of a capable exascale ecosystem, including // software, applications, hardware, advanced system engineering and early // testbed platforms, in support of the nation's exascale computing imperative. #include #include #include #include #include #include #include "ceed-ref.h" //------------------------------------------------------------------------------ // Setup Input/Output Fields //------------------------------------------------------------------------------ static int CeedOperatorSetupFields_Ref(CeedQFunction qf, CeedOperator op, bool inOrOut, CeedVector *full_evecs, CeedVector *e_vecs, CeedVector *q_vecs, CeedInt starte, CeedInt num_fields, CeedInt Q) { CeedInt dim, ierr, size, P; Ceed ceed; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); CeedBasis basis; CeedElemRestriction elem_restr; CeedOperatorField *op_fields; CeedQFunctionField *qf_fields; if (inOrOut) { ierr = CeedOperatorGetFields(op, NULL, &op_fields); CeedChkBackend(ierr); ierr = CeedQFunctionGetFields(qf, NULL, &qf_fields); CeedChkBackend(ierr); } else { ierr = CeedOperatorGetFields(op, &op_fields, NULL); CeedChkBackend(ierr); ierr = CeedQFunctionGetFields(qf, &qf_fields, NULL); CeedChkBackend(ierr); } // Loop over fields for (CeedInt i=0; iidentity_qf); CeedChkBackend(ierr); ierr = CeedQFunctionGetNumArgs(qf, &num_input_fields, &num_output_fields); CeedChkBackend(ierr); CeedOperatorField *op_input_fields, *op_output_fields; ierr = CeedOperatorGetFields(op, &op_input_fields, &op_output_fields); CeedChkBackend(ierr); CeedQFunctionField *qf_input_fields, *qf_output_fields; ierr = CeedQFunctionGetFields(qf, &qf_input_fields, &qf_output_fields); CeedChkBackend(ierr); // Allocate ierr = CeedCalloc(num_input_fields + num_output_fields, &impl->e_vecs); CeedChkBackend(ierr); ierr = CeedCalloc(num_input_fields + num_output_fields, &impl->e_data); CeedChkBackend(ierr); ierr = CeedCalloc(16, &impl->input_state); CeedChkBackend(ierr); ierr = CeedCalloc(16, &impl->e_vecs_in); CeedChkBackend(ierr); ierr = CeedCalloc(16, &impl->e_vecs_out); CeedChkBackend(ierr); ierr = CeedCalloc(16, &impl->q_vecs_in); CeedChkBackend(ierr); ierr = CeedCalloc(16, &impl->q_vecs_out); CeedChkBackend(ierr); impl->num_e_vecs_in = num_input_fields; impl->num_e_vecs_out = num_output_fields; // Set up infield and outfield e_vecs and q_vecs // Infields ierr = CeedOperatorSetupFields_Ref(qf, op, 0, impl->e_vecs, impl->e_vecs_in, impl->q_vecs_in, 0, num_input_fields, Q); CeedChkBackend(ierr); // Outfields ierr = CeedOperatorSetupFields_Ref(qf, op, 1, impl->e_vecs, impl->e_vecs_out, impl->q_vecs_out, num_input_fields, num_output_fields, Q); CeedChkBackend(ierr); // Identity QFunctions if (impl->identity_qf) { CeedEvalMode in_mode, out_mode; CeedQFunctionField *in_fields, *out_fields; ierr = CeedQFunctionGetFields(qf, &in_fields, &out_fields); CeedChkBackend(ierr); for (CeedInt i=0; iq_vecs_out[i]); CeedChkBackend(ierr); impl->q_vecs_out[i] = impl->q_vecs_in[i]; ierr = CeedVectorAddReference(impl->q_vecs_in[i]); CeedChkBackend(ierr); } } ierr = CeedOperatorSetSetupDone(op); CeedChkBackend(ierr); return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Setup Operator Inputs //------------------------------------------------------------------------------ static inline int CeedOperatorSetupInputs_Ref(CeedInt num_input_fields, CeedQFunctionField *qf_input_fields, CeedOperatorField *op_input_fields, CeedVector in_vec, const bool skip_active, CeedOperator_Ref *impl, CeedRequest *request) { CeedInt ierr; CeedEvalMode eval_mode; CeedVector vec; CeedElemRestriction elem_restr; uint64_t state; for (CeedInt i=0; iinput_state[i] || vec == in_vec) { ierr = CeedOperatorFieldGetElemRestriction(op_input_fields[i], &elem_restr); CeedChkBackend(ierr); ierr = CeedElemRestrictionApply(elem_restr, CEED_NOTRANSPOSE, vec, impl->e_vecs[i], request); CeedChkBackend(ierr); impl->input_state[i] = state; } // Get evec ierr = CeedVectorGetArrayRead(impl->e_vecs[i], CEED_MEM_HOST, (const CeedScalar **) &impl->e_data[i]); CeedChkBackend(ierr); } } return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Input Basis Action //------------------------------------------------------------------------------ static inline int CeedOperatorInputBasis_Ref(CeedInt e, CeedInt Q, CeedQFunctionField *qf_input_fields, CeedOperatorField *op_input_fields, CeedInt num_input_fields, const bool skip_active, CeedOperator_Ref *impl) { CeedInt ierr; CeedInt dim, elem_size, size; CeedElemRestriction elem_restr; CeedEvalMode eval_mode; CeedBasis basis; for (CeedInt i=0; iq_vecs_in[i], CEED_MEM_HOST, CEED_USE_POINTER, &impl->e_data[i][e*Q*size]); CeedChkBackend(ierr); break; case CEED_EVAL_INTERP: ierr = CeedOperatorFieldGetBasis(op_input_fields[i], &basis); CeedChkBackend(ierr); ierr = CeedVectorSetArray(impl->e_vecs_in[i], CEED_MEM_HOST, CEED_USE_POINTER, &impl->e_data[i][e*elem_size*size]); CeedChkBackend(ierr); ierr = CeedBasisApply(basis, 1, CEED_NOTRANSPOSE, CEED_EVAL_INTERP, impl->e_vecs_in[i], impl->q_vecs_in[i]); CeedChkBackend(ierr); break; case CEED_EVAL_GRAD: ierr = CeedOperatorFieldGetBasis(op_input_fields[i], &basis); CeedChkBackend(ierr); ierr = CeedBasisGetDimension(basis, &dim); CeedChkBackend(ierr); ierr = CeedVectorSetArray(impl->e_vecs_in[i], CEED_MEM_HOST, CEED_USE_POINTER, &impl->e_data[i][e*elem_size*size/dim]); CeedChkBackend(ierr); ierr = CeedBasisApply(basis, 1, CEED_NOTRANSPOSE, CEED_EVAL_GRAD, impl->e_vecs_in[i], impl->q_vecs_in[i]); CeedChkBackend(ierr); break; case CEED_EVAL_WEIGHT: break; // No action // LCOV_EXCL_START case CEED_EVAL_DIV: case CEED_EVAL_CURL: { ierr = CeedOperatorFieldGetBasis(op_input_fields[i], &basis); CeedChkBackend(ierr); Ceed ceed; ierr = CeedBasisGetCeed(basis, &ceed); CeedChkBackend(ierr); return CeedError(ceed, CEED_ERROR_BACKEND, "Ceed evaluation mode not implemented"); // LCOV_EXCL_STOP } } } return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Output Basis Action //------------------------------------------------------------------------------ static inline int CeedOperatorOutputBasis_Ref(CeedInt e, CeedInt Q, CeedQFunctionField *qf_output_fields, CeedOperatorField *op_output_fields, CeedInt num_input_fields, CeedInt num_output_fields, CeedOperator op, CeedOperator_Ref *impl) { CeedInt ierr; CeedInt dim, elem_size, size; CeedElemRestriction elem_restr; CeedEvalMode eval_mode; CeedBasis basis; for (CeedInt i=0; ie_vecs_out[i], CEED_MEM_HOST, CEED_USE_POINTER, &impl->e_data[i + num_input_fields][e*elem_size*size]); CeedChkBackend(ierr); ierr = CeedBasisApply(basis, 1, CEED_TRANSPOSE, CEED_EVAL_INTERP, impl->q_vecs_out[i], impl->e_vecs_out[i]); CeedChkBackend(ierr); break; case CEED_EVAL_GRAD: ierr = CeedOperatorFieldGetBasis(op_output_fields[i], &basis); CeedChkBackend(ierr); ierr = CeedBasisGetDimension(basis, &dim); CeedChkBackend(ierr); ierr = CeedVectorSetArray(impl->e_vecs_out[i], CEED_MEM_HOST, CEED_USE_POINTER, &impl->e_data[i + num_input_fields][e*elem_size*size/dim]); CeedChkBackend(ierr); ierr = CeedBasisApply(basis, 1, CEED_TRANSPOSE, CEED_EVAL_GRAD, impl->q_vecs_out[i], impl->e_vecs_out[i]); CeedChkBackend(ierr); break; // LCOV_EXCL_START case CEED_EVAL_WEIGHT: { Ceed ceed; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); return CeedError(ceed, CEED_ERROR_BACKEND, "CEED_EVAL_WEIGHT cannot be an output " "evaluation mode"); } case CEED_EVAL_DIV: case CEED_EVAL_CURL: { Ceed ceed; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); return CeedError(ceed, CEED_ERROR_BACKEND, "Ceed evaluation mode not implemented"); // LCOV_EXCL_STOP } } } return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Restore Input Vectors //------------------------------------------------------------------------------ static inline int CeedOperatorRestoreInputs_Ref(CeedInt num_input_fields, CeedQFunctionField *qf_input_fields, CeedOperatorField *op_input_fields, const bool skip_active, CeedOperator_Ref *impl) { CeedInt ierr; CeedEvalMode eval_mode; for (CeedInt i=0; ie_vecs[i], (const CeedScalar **) &impl->e_data[i]); CeedChkBackend(ierr); } } return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Operator Apply //------------------------------------------------------------------------------ static int CeedOperatorApplyAdd_Ref(CeedOperator op, CeedVector in_vec, CeedVector out_vec, CeedRequest *request) { int ierr; CeedOperator_Ref *impl; ierr = CeedOperatorGetData(op, &impl); CeedChkBackend(ierr); CeedQFunction qf; ierr = CeedOperatorGetQFunction(op, &qf); CeedChkBackend(ierr); CeedInt Q, num_elem, num_input_fields, num_output_fields, size; ierr = CeedOperatorGetNumQuadraturePoints(op, &Q); CeedChkBackend(ierr); ierr = CeedOperatorGetNumElements(op, &num_elem); CeedChkBackend(ierr); ierr= CeedQFunctionGetNumArgs(qf, &num_input_fields, &num_output_fields); CeedChkBackend(ierr); CeedOperatorField *op_input_fields, *op_output_fields; ierr = CeedOperatorGetFields(op, &op_input_fields, &op_output_fields); CeedChkBackend(ierr); CeedQFunctionField *qf_input_fields, *qf_output_fields; ierr = CeedQFunctionGetFields(qf, &qf_input_fields, &qf_output_fields); CeedChkBackend(ierr); CeedEvalMode eval_mode; CeedVector vec; CeedElemRestriction elem_restr; // Setup ierr = CeedOperatorSetup_Ref(op); CeedChkBackend(ierr); // Input Evecs and Restriction ierr = CeedOperatorSetupInputs_Ref(num_input_fields, qf_input_fields, op_input_fields, in_vec, false, impl, request); CeedChkBackend(ierr); // Output Evecs for (CeedInt i=0; ie_vecs[i+impl->num_e_vecs_in], CEED_MEM_HOST, &impl->e_data[i + num_input_fields]); CeedChkBackend(ierr); } // Loop through elements for (CeedInt e=0; eq_vecs_out[i], CEED_MEM_HOST, CEED_USE_POINTER, &impl->e_data[i + num_input_fields][e*Q*size]); CeedChkBackend(ierr); } } // Input basis apply ierr = CeedOperatorInputBasis_Ref(e, Q, qf_input_fields, op_input_fields, num_input_fields, false, impl); CeedChkBackend(ierr); // Q function if (!impl->identity_qf) { ierr = CeedQFunctionApply(qf, Q, impl->q_vecs_in, impl->q_vecs_out); CeedChkBackend(ierr); } // Output basis apply ierr = CeedOperatorOutputBasis_Ref(e, Q, qf_output_fields, op_output_fields, num_input_fields, num_output_fields, op, impl); CeedChkBackend(ierr); } // Output restriction for (CeedInt i=0; ie_vecs[i+impl->num_e_vecs_in], &impl->e_data[i + num_input_fields]); CeedChkBackend(ierr); // Get output vector ierr = CeedOperatorFieldGetVector(op_output_fields[i], &vec); CeedChkBackend(ierr); // Active if (vec == CEED_VECTOR_ACTIVE) vec = out_vec; // Restrict ierr = CeedOperatorFieldGetElemRestriction(op_output_fields[i], &elem_restr); CeedChkBackend(ierr); ierr = CeedElemRestrictionApply(elem_restr, CEED_TRANSPOSE, impl->e_vecs[i+impl->num_e_vecs_in], vec, request); CeedChkBackend(ierr); } // Restore input arrays ierr = CeedOperatorRestoreInputs_Ref(num_input_fields, qf_input_fields, op_input_fields, false, impl); CeedChkBackend(ierr); return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Assemble Linear QFunction //------------------------------------------------------------------------------ static int CeedOperatorLinearAssembleQFunction_Ref(CeedOperator op, CeedVector *assembled, CeedElemRestriction *rstr, CeedRequest *request) { int ierr; CeedOperator_Ref *impl; ierr = CeedOperatorGetData(op, &impl); CeedChkBackend(ierr); CeedQFunction qf; ierr = CeedOperatorGetQFunction(op, &qf); CeedChkBackend(ierr); CeedInt Q, num_elem, num_input_fields, num_output_fields, size; ierr = CeedOperatorGetNumQuadraturePoints(op, &Q); CeedChkBackend(ierr); ierr = CeedOperatorGetNumElements(op, &num_elem); CeedChkBackend(ierr); ierr= CeedQFunctionGetNumArgs(qf, &num_input_fields, &num_output_fields); CeedChkBackend(ierr); CeedOperatorField *op_input_fields, *op_output_fields; ierr = CeedOperatorGetFields(op, &op_input_fields, &op_output_fields); CeedChkBackend(ierr); CeedQFunctionField *qf_input_fields, *qf_output_fields; ierr = CeedQFunctionGetFields(qf, &qf_input_fields, &qf_output_fields); CeedChkBackend(ierr); CeedVector vec; CeedInt num_active_in = 0, num_active_out = 0; CeedVector *active_in = NULL; CeedScalar *a, *tmp; Ceed ceed, ceed_parent; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); ierr = CeedGetOperatorFallbackParentCeed(ceed, &ceed_parent); CeedChkBackend(ierr); ceed_parent = ceed_parent ? ceed_parent : ceed; // Setup ierr = CeedOperatorSetup_Ref(op); CeedChkBackend(ierr); // Check for identity if (impl->identity_qf) // LCOV_EXCL_START return CeedError(ceed, CEED_ERROR_BACKEND, "Assembling identity QFunctions not supported"); // LCOV_EXCL_STOP // Input Evecs and Restriction ierr = CeedOperatorSetupInputs_Ref(num_input_fields, qf_input_fields, op_input_fields, NULL, true, impl, request); CeedChkBackend(ierr); // Count number of active input fields for (CeedInt i=0; iq_vecs_in[i], 0.0); CeedChkBackend(ierr); ierr = CeedVectorGetArray(impl->q_vecs_in[i], CEED_MEM_HOST, &tmp); CeedChkBackend(ierr); ierr = CeedRealloc(num_active_in + size, &active_in); CeedChkBackend(ierr); for (CeedInt field=0; fieldq_vecs_in[i], &tmp); CeedChkBackend(ierr); } } // Count number of active output fields for (CeedInt i=0; i 1) { ierr = CeedVectorSetValue(active_in[(in+num_active_in-1)%num_active_in], 0.0); CeedChkBackend(ierr); } // Set Outputs for (CeedInt out=0; outq_vecs_out[out], CEED_MEM_HOST, CEED_USE_POINTER, a); CeedChkBackend(ierr); ierr = CeedQFunctionFieldGetSize(qf_output_fields[out], &size); CeedChkBackend(ierr); a += size*Q; // Advance the pointer by the size of the output } } // Apply QFunction ierr = CeedQFunctionApply(qf, Q, impl->q_vecs_in, impl->q_vecs_out); CeedChkBackend(ierr); } } // Un-set output Qvecs to prevent accidental overwrite of Assembled for (CeedInt out=0; outq_vecs_out[out], CEED_MEM_HOST, NULL); CeedChkBackend(ierr); } } // Restore input arrays ierr = CeedOperatorRestoreInputs_Ref(num_input_fields, qf_input_fields, op_input_fields, true, impl); CeedChkBackend(ierr); // Restore output ierr = CeedVectorRestoreArray(*assembled, &a); CeedChkBackend(ierr); // Cleanup for (CeedInt i=0; i max) max = pbOffsets[i]; } // Create new restriction ierr = CeedElemRestrictionCreate(ceed, num_elem, elem_size, num_comp*num_comp, 1, max + num_comp*num_comp, CEED_MEM_HOST, CEED_OWN_POINTER, pbOffsets, pb_rstr); CeedChkBackend(ierr); // Cleanup ierr = CeedElemRestrictionRestoreOffsets(rstr, &offsets); CeedChkBackend(ierr); return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Assemble diagonal common code //------------------------------------------------------------------------------ static inline int CeedOperatorAssembleAddDiagonalCore_Ref(CeedOperator op, CeedVector assembled, CeedRequest *request, const bool is_pointblock) { int ierr; Ceed ceed; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); // Assemble QFunction CeedQFunction qf; ierr = CeedOperatorGetQFunction(op, &qf); CeedChkBackend(ierr); CeedInt num_input_fields, num_output_fields; ierr= CeedQFunctionGetNumArgs(qf, &num_input_fields, &num_output_fields); CeedChkBackend(ierr); CeedVector assembled_qf; CeedElemRestriction rstr; ierr = CeedOperatorLinearAssembleQFunction(op, &assembled_qf, &rstr, request); CeedChkBackend(ierr); ierr = CeedElemRestrictionDestroy(&rstr); CeedChkBackend(ierr); CeedScalar max_norm = 0; ierr = CeedVectorNorm(assembled_qf, CEED_NORM_MAX, &max_norm); CeedChkBackend(ierr); // Determine active input basis CeedOperatorField *op_fields; CeedQFunctionField *qf_fields; ierr = CeedOperatorGetFields(op, &op_fields, NULL); CeedChkBackend(ierr); ierr = CeedQFunctionGetFields(qf, &qf_fields, NULL); CeedChkBackend(ierr); CeedInt num_eval_mode_in = 0, num_comp, dim = 1; CeedEvalMode *eval_mode_in = NULL; CeedBasis basis_in = NULL; CeedElemRestriction rstr_in = NULL; for (CeedInt i=0; i qf_value_bound) for (CeedInt n=0; n qf_value_bound) for (CeedInt n=0; n max_norm*1e-12) { elem_avg[e] += assembledarray[e*num_elem*num_qpts*num_comp*num_comp*num_fields + i*num_qpts + q] / q_weight_array[q]; count++; } if (count) elem_avg[e] /= count; } ierr = CeedVectorRestoreArrayRead(assembled, &assembledarray); CeedChkBackend(ierr); ierr = CeedVectorDestroy(&assembled); CeedChkBackend(ierr); ierr = CeedVectorRestoreArrayRead(q_weight, &q_weight_array); CeedChkBackend(ierr); ierr = CeedVectorDestroy(&q_weight); CeedChkBackend(ierr); // Build FDM diagonal CeedVector q_data; CeedScalar *q_data_array; ierr = CeedVectorCreate(ceed_parent, num_elem*num_comp*l_size, &q_data); CeedChkBackend(ierr); ierr = CeedVectorSetArray(q_data, CEED_MEM_HOST, CEED_COPY_VALUES, NULL); CeedChkBackend(ierr); ierr = CeedVectorGetArray(q_data, CEED_MEM_HOST, &q_data_array); CeedChkBackend(ierr); for (CeedInt e=0; enum_e_vecs_in+impl->num_e_vecs_out; i++) { ierr = CeedVectorDestroy(&impl->e_vecs[i]); CeedChkBackend(ierr); } ierr = CeedFree(&impl->e_vecs); CeedChkBackend(ierr); ierr = CeedFree(&impl->e_data); CeedChkBackend(ierr); ierr = CeedFree(&impl->input_state); CeedChkBackend(ierr); for (CeedInt i=0; inum_e_vecs_in; i++) { ierr = CeedVectorDestroy(&impl->e_vecs_in[i]); CeedChkBackend(ierr); ierr = CeedVectorDestroy(&impl->q_vecs_in[i]); CeedChkBackend(ierr); } ierr = CeedFree(&impl->e_vecs_in); CeedChkBackend(ierr); ierr = CeedFree(&impl->q_vecs_in); CeedChkBackend(ierr); for (CeedInt i=0; inum_e_vecs_out; i++) { ierr = CeedVectorDestroy(&impl->e_vecs_out[i]); CeedChkBackend(ierr); ierr = CeedVectorDestroy(&impl->q_vecs_out[i]); CeedChkBackend(ierr); } ierr = CeedFree(&impl->e_vecs_out); CeedChkBackend(ierr); ierr = CeedFree(&impl->q_vecs_out); CeedChkBackend(ierr); ierr = CeedFree(&impl); CeedChkBackend(ierr); return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Operator Create //------------------------------------------------------------------------------ int CeedOperatorCreate_Ref(CeedOperator op) { int ierr; Ceed ceed; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); CeedOperator_Ref *impl; ierr = CeedCalloc(1, &impl); CeedChkBackend(ierr); ierr = CeedOperatorSetData(op, impl); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "LinearAssembleQFunction", CeedOperatorLinearAssembleQFunction_Ref); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "LinearAssembleAddDiagonal", CeedOperatorLinearAssembleAddDiagonal_Ref); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "LinearAssembleAddPointBlockDiagonal", CeedOperatorLinearAssembleAddPointBlockDiagonal_Ref); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "CreateFDMElementInverse", CeedOperatorCreateFDMElementInverse_Ref); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "ApplyAdd", CeedOperatorApplyAdd_Ref); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "Destroy", CeedOperatorDestroy_Ref); CeedChkBackend(ierr); return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------ // Composite Operator Create //------------------------------------------------------------------------------ int CeedCompositeOperatorCreate_Ref(CeedOperator op) { int ierr; Ceed ceed; ierr = CeedOperatorGetCeed(op, &ceed); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "LinearAssembleAddDiagonal", CeedOperatorLinearAssembleAddDiagonal_Ref); CeedChkBackend(ierr); ierr = CeedSetBackendFunction(ceed, "Operator", op, "LinearAssembleAddPointBlockDiagonal", CeedOperatorLinearAssembleAddPointBlockDiagonal_Ref); CeedChkBackend(ierr); return CEED_ERROR_SUCCESS; } //------------------------------------------------------------------------------