/*
  Copyright 2017 SINTEF Digital, Mathematics and Cybernetics.
  Copyright 2017 Statoil ASA.
  Copyright 2016 - 2017 IRIS AS.
  This file is part of the Open Porous Media project (OPM).
  OPM is free software: you can redistribute it and/or modify
  it under the terms of the GNU General Public License as published by
  the Free Software Foundation, either version 3 of the License, or
  (at your option) any later version.
  OPM is distributed in the hope that it will be useful,
  but WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  GNU General Public License for more details.
  You should have received a copy of the GNU General Public License
  along with OPM.  If not, see .
*/
#ifndef OPM_STANDARDWELL_EQUATIONS_HEADER_INCLUDED
#define OPM_STANDARDWELL_EQUATIONS_HEADER_INCLUDED
#include 
#include 
#include 
#include 
#include 
#include 
namespace Opm
{
class ParallelWellInfo;
template class StandardWellEquationAccess;
class WellContributions;
class WellInterfaceGeneric;
class WellState;
template
class StandardWellEquations
{
public:
    // sparsity pattern for the matrices
    //[A C^T    [x       =  [ res
    // B  D ]   x_well]      res_well]
    // the vector type for the res_well and x_well
    using VectorBlockWellType = Dune::DynamicVector;
    using BVectorWell = Dune::BlockVector;
    // the matrix type for the diagonal matrix D
    using DiagMatrixBlockWellType = Dune::DynamicMatrix;
    using DiagMatWell = Dune::BCRSMatrix;
    // the matrix type for the non-diagonal matrix B and C^T
    using OffDiagMatrixBlockWellType = Dune::DynamicMatrix;
    using OffDiagMatWell = Dune::BCRSMatrix;
    // block vector type
    using BVector = Dune::BlockVector>;
    StandardWellEquations(const ParallelWellInfo& parallel_well_info);
    //! \brief Setup sparsity pattern for the matrices.
    //! \param num_cells Total number of cells
    //! \param numWellEq Number of well equations
    //! \param numPerfs Number of perforations
    //! \param cells Cell indices for perforations
    void init(const int num_cells,
              const int numWellEq,
              const int numPerfs,
              const std::vector& cells);
    //! \brief Set all coefficients to 0.
    void clear();
    //! \brief Apply linear operator to vector.
    void apply(const BVector& x, BVector& Ax) const;
    //! \brief Apply linear operator to vector.
    void apply(BVector& r) const;
    //! \brief Apply inverted D matrix to residual and store in vector.
    void solve(BVectorWell& dx_well) const;
    //! \brief Invert D matrix.
    void invert();
    //! \brief Recover well solution.
    //! \details xw = inv(D)*(rw - C*x)
    void recoverSolutionWell(const BVector& x, BVectorWell& xw) const;
    //! \brief Add the matrices of this well to the WellContributions object.
    void extract(const int numStaticWellEq,
                 WellContributions& wellContribs) const;
    //! \brief Add the matrices of this well to the sparse matrix adapter.
    template
    void extract(SparseMatrixAdapter& jacobian) const;
    //! \brief Extract CPR pressure matrix.
    template
    void extractCPRPressureMatrix(PressureMatrix& jacobian,
                                  const BVector& weights,
                                  const int pressureVarIndex,
                                  const bool use_well_weights,
                                  const WellInterfaceGeneric& well,
                                  const int bhp_var_index,
                                  const WellState& well_state) const;
    //! \brief Get the number of blocks of the C and B matrices.
    unsigned int getNumBlocks() const;
    //! \brief Sum with off-process contribution.
    void sumDistributed(Parallel::Communication comm);
    //! \brief Returns a const reference to the residual.
    const BVectorWell& residual() const
    {
        return resWell_;
    }
private:
    friend class StandardWellEquationAccess;
    // two off-diagonal matrices
    OffDiagMatWell duneB_;
    OffDiagMatWell duneC_;
    // diagonal matrix for the well
    DiagMatWell invDuneD_;
    DiagMatWell duneD_;
    // Wrapper for the parallel application of B for distributed wells
    wellhelpers::ParallelStandardWellB parallelB_;
    // residuals of the well equations
    BVectorWell resWell_;
    // several vector used in the matrix calculation
    mutable BVectorWell Bx_;
    mutable BVectorWell invDrw_;
};
}
#endif // OPM_STANDARDWELL_EQUATIONS_HEADER_INCLUDED