// Ceres Solver - A fast non-linear least squares minimizer
// Copyright 2023 Google Inc. All rights reserved.
// http://ceres-solver.org/
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
//   this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright notice,
//   this list of conditions and the following disclaimer in the documentation
//   and/or other materials provided with the distribution.
// * Neither the name of Google Inc. nor the names of its contributors may be
//   used to endorse or promote products derived from this software without
//   specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
//
// Authors: dmitriy.korchemkin@gmail.com (Dmitriy Korchemkin)

#include "ceres/internal/config.h"

#ifndef CERES_NO_CUDA

#include <glog/logging.h>
#include <gtest/gtest.h>

#include <numeric>

#include "ceres/block_sparse_matrix.h"
#include "ceres/cuda_block_structure.h"

namespace ceres::internal {

class CudaBlockStructureTest : public ::testing::Test {
 protected:
  void SetUp() final {
    std::string message;
    CHECK(context_.InitCuda(&message))
        << "InitCuda() failed because: " << message;

    BlockSparseMatrix::RandomMatrixOptions options;
    options.num_row_blocks = 1234;
    options.min_row_block_size = 1;
    options.max_row_block_size = 10;
    options.num_col_blocks = 567;
    options.min_col_block_size = 1;
    options.max_col_block_size = 10;
    options.block_density = 0.2;
    std::mt19937 rng;
    A_ = BlockSparseMatrix::CreateRandomMatrix(options, rng);
    std::iota(
        A_->mutable_values(), A_->mutable_values() + A_->num_nonzeros(), 1);
  }

  std::vector<Cell> GetCells(const CudaBlockSparseStructure& structure) {
    const auto& cuda_buffer = structure.cells_;
    std::vector<Cell> cells(cuda_buffer.size());
    cuda_buffer.CopyToCpu(cells.data(), cells.size());
    return cells;
  }
  std::vector<Block> GetRowBlocks(const CudaBlockSparseStructure& structure) {
    const auto& cuda_buffer = structure.row_blocks_;
    std::vector<Block> blocks(cuda_buffer.size());
    cuda_buffer.CopyToCpu(blocks.data(), blocks.size());
    return blocks;
  }
  std::vector<Block> GetColBlocks(const CudaBlockSparseStructure& structure) {
    const auto& cuda_buffer = structure.col_blocks_;
    std::vector<Block> blocks(cuda_buffer.size());
    cuda_buffer.CopyToCpu(blocks.data(), blocks.size());
    return blocks;
  }
  std::vector<int> GetRowBlockOffsets(
      const CudaBlockSparseStructure& structure) {
    const auto& cuda_buffer = structure.first_cell_in_row_block_;
    std::vector<int> first_cell_in_row_block(cuda_buffer.size());
    cuda_buffer.CopyToCpu(first_cell_in_row_block.data(), first_cell_in_row_block.size());
    return first_cell_in_row_block;
  }

  std::unique_ptr<BlockSparseMatrix> A_;
  ContextImpl context_;
};

TEST_F(CudaBlockStructureTest, StructureIdentity) {
  auto block_structure = A_->block_structure();
  const int num_row_blocks = block_structure->rows.size();
  const int num_col_blocks = block_structure->cols.size();

  CudaBlockSparseStructure cuda_block_structure(*block_structure, &context_);

  ASSERT_EQ(cuda_block_structure.num_rows(), A_->num_rows());
  ASSERT_EQ(cuda_block_structure.num_cols(), A_->num_cols());
  ASSERT_EQ(cuda_block_structure.num_nonzeros(), A_->num_nonzeros());
  ASSERT_EQ(cuda_block_structure.num_row_blocks(), num_row_blocks);
  ASSERT_EQ(cuda_block_structure.num_col_blocks(), num_col_blocks);

  std::vector<Block> blocks = GetColBlocks(cuda_block_structure);
  ASSERT_EQ(blocks.size(), num_col_blocks);
  for (int i = 0; i < num_col_blocks; ++i) {
    EXPECT_EQ(block_structure->cols[i].position, blocks[i].position);
    EXPECT_EQ(block_structure->cols[i].size, blocks[i].size);
  }

  std::vector<Cell> cells = GetCells(cuda_block_structure);
  std::vector<int> first_cell_in_row_block = GetRowBlockOffsets(cuda_block_structure);
  blocks = GetRowBlocks(cuda_block_structure);

  ASSERT_EQ(blocks.size(), num_row_blocks);
  ASSERT_EQ(first_cell_in_row_block.size(), num_row_blocks + 1);
  ASSERT_EQ(first_cell_in_row_block.back(), cells.size());

  for (int i = 0; i < num_row_blocks; ++i) {
    const int num_cells = block_structure->rows[i].cells.size();
    EXPECT_EQ(blocks[i].position, block_structure->rows[i].block.position);
    EXPECT_EQ(blocks[i].size, block_structure->rows[i].block.size);
    const int first_cell = first_cell_in_row_block[i];
    const int last_cell = first_cell_in_row_block[i + 1];
    ASSERT_EQ(last_cell - first_cell, num_cells);
    for (int j = 0; j < num_cells; ++j) {
      EXPECT_EQ(cells[first_cell + j].block_id,
                block_structure->rows[i].cells[j].block_id);
      EXPECT_EQ(cells[first_cell + j].position,
                block_structure->rows[i].cells[j].position);
    }
  }
}

}  // namespace ceres::internal

#endif  // CERES_NO_CUDA
