Program Listing for File LevelHierarchy.hpp

Return to documentation for file (include/networkit/numerics/LAMG/LevelHierarchy.hpp)

/*
 * LevelHierarchy.hpp
 *
 *  Created on: 10.01.2015
 *      Author: Michael
 */

#ifndef NETWORKIT_NUMERICS_LAMG_LEVEL_HIERARCHY_HPP_
#define NETWORKIT_NUMERICS_LAMG_LEVEL_HIERARCHY_HPP_

#include <networkit/algebraic/CSRMatrix.hpp>
#include <networkit/algebraic/DenseMatrix.hpp>
#include <networkit/algebraic/DynamicMatrix.hpp>
#include <networkit/numerics/LAMG/LAMGSettings.hpp>
#include <networkit/numerics/LAMG/Level/Level.hpp>
#include <networkit/numerics/LAMG/Level/LevelAggregation.hpp>
#include <networkit/numerics/LAMG/Level/LevelElimination.hpp>
#include <networkit/numerics/LAMG/Level/LevelFinest.hpp>

namespace NetworKit {

template <class Matrix>
class LevelHierarchy {
private:
    std::vector<LevelType> levelType;
    std::vector<index> levelIndex;
    std::vector<LevelElimination<Matrix>> eliminationLevels;
    std::vector<LevelAggregation<Matrix>> aggregationLevels;
    LevelFinest<Matrix> finestLevel;
    DenseMatrix coarseLUMatrix;

    void createCoarseMatrix();

public:
    LevelHierarchy() = default;

    void addFinestLevel(const Matrix &A);
    void addEliminationLevel(const Matrix &A,
                             const std::vector<EliminationStage<Matrix>> &coarseningStages);
    void addAggregationLevel(const Matrix &A, const Matrix &P, const Matrix &R);
    void setLastAsCoarsest();
    inline const DenseMatrix &getCoarseMatrix() const { return coarseLUMatrix; }

    inline count size() const {
        return levelType.size() + 1; // elimination + aggregation levels + finestLevel
    }

    LevelType getType(index levelIdx) const;
    Level<Matrix> &at(index levelIdx);
    const Level<Matrix> &at(index levelIdx) const;
    double cycleIndex(index levelIdx) const;
};

template <class Matrix>
void LevelHierarchy<Matrix>::addFinestLevel(const Matrix &A) {
    finestLevel = LevelFinest<Matrix>(A);
}

template <class Matrix>
void LevelHierarchy<Matrix>::addEliminationLevel(
    const Matrix &A, const std::vector<EliminationStage<Matrix>> &coarseningStages) {
    levelType.push_back(ELIMINATION);
    levelIndex.push_back(eliminationLevels.size());
    eliminationLevels.push_back(LevelElimination<Matrix>(A, coarseningStages));
}

template <class Matrix>
void LevelHierarchy<Matrix>::addAggregationLevel(const Matrix &A, const Matrix &P,
                                                 const Matrix &R) {
    levelType.push_back(AGGREGATION);
    levelIndex.push_back(aggregationLevels.size());
    aggregationLevels.push_back(LevelAggregation<Matrix>(A, P, R));
}

template <class Matrix>
void LevelHierarchy<Matrix>::setLastAsCoarsest() {
    const Matrix &A = this->at(size() - 1).getLaplacian();
    count n = A.numberOfRows() + 1;
    std::vector<double> entries(n * n, 0.0);
    A.parallelForNonZeroElementsInRowOrder(
        [&](index i, index j, double value) { entries[i * n + j] = value; });

    for (index i = 0; i < n - 1; ++i) {
        entries[i * n + n - 1] = 1;
        entries[(n - 1) * n + i] = 1;
    }

    coarseLUMatrix = DenseMatrix(n, n, entries);
    DenseMatrix::LUDecomposition(coarseLUMatrix);
}

template <class Matrix>
LevelType LevelHierarchy<Matrix>::getType(index levelIdx) const {
    assert(levelIdx < this->size());

    if (levelIdx == 0) {
        return FINEST;
    } else {
        return levelType[levelIdx - 1];
    }
}

template <class Matrix>
Level<Matrix> &LevelHierarchy<Matrix>::at(index levelIdx) {
    assert(levelIdx < this->size());

    if (levelIdx == 0) { // finest level
        return finestLevel;
    } else {
        levelIdx--;
    }

    if (levelType[levelIdx] == ELIMINATION) {
        return eliminationLevels[levelIndex[levelIdx]];
    } else {
        return aggregationLevels[levelIndex[levelIdx]];
    }
}

template <class Matrix>
const Level<Matrix> &LevelHierarchy<Matrix>::at(index levelIdx) const {
    assert(levelIdx < this->size());

    if (levelIdx == 0) { // finest level
        return finestLevel;
    } else {
        levelIdx--;
    }

    if (levelType[levelIdx] == ELIMINATION) {
        return eliminationLevels[levelIndex[levelIdx]];
    } else {
        return aggregationLevels[levelIndex[levelIdx]];
    }
}

template <class Matrix>
double LevelHierarchy<Matrix>::cycleIndex(index levelIdx) const {
    double gamma = 1.0;
    if (getType(levelIdx + 1) != ELIMINATION) {
        const double finestNumEdges = finestLevel.getLaplacian().nnz();
        const double numFineEdges = this->at(levelIdx).getLaplacian().nnz();

        if (numFineEdges > 0.1 * finestNumEdges) {
            gamma = SETUP_CYCLE_INDEX;
        } else {
            double numCoarseEdges = this->at(levelIdx + 1).getLaplacian().nnz();
            gamma = std::max(
                1.0, std::min(1.5, SETUP_COARSENING_WORK_GUARD / (numCoarseEdges / numFineEdges)));
        }
    }

    return gamma;
}

extern template class LevelHierarchy<CSRMatrix>;
extern template class LevelHierarchy<DenseMatrix>;
extern template class LevelHierarchy<DynamicMatrix>;

} /* namespace NetworKit */

#endif // NETWORKIT_NUMERICS_LAMG_LEVEL_HIERARCHY_HPP_