#pragma once
#include <cstddef>
#include <cstdint>
#include <memory>


namespace npeff {

// Dense rank 3 tensor.
// 
// NOTE: Unlike the matrix types elsewhere, this tensor will have C-like arrangement
// of its values in memory.
template<typename T>
struct Dense3Tensor {
    const int64_t d1, d2, d3;

    std::unique_ptr<T> data;

    // Derived attributes.    
    const int64_t n_entries;
    const size_t size_bytes;
    const int64_t d2_d3;  // d2 times d3

    Dense3Tensor(int64_t d1, int64_t d2, int64_t d3) :
        Dense3Tensor(d1, d2, d3, std::unique_ptr<T>(new T[d1 * d2 * d3]))
    {}

    Dense3Tensor(int64_t d1, int64_t d2, int64_t d3, std::unique_ptr<T> data) :
        d1(d1), d2(d2), d3(d3),
        data(std::move(data)),
        n_entries(d1 * d2 * d3),
        d2_d3(d2 * d3),
        size_bytes(n_entries * sizeof(T))
    {}

    T get_entry(int64_t i1, int64_t i2, int64_t i3) const {
        return data.get()[i3 + d3 * i2 + d2_d3 * i1];
    }

    T& get_entry(int64_t i1, int64_t i2, int64_t i3) {
        return data.get()[i3 + d3 * i2 + d2_d3 * i1];
    }

};

// A contiguous view (i.e. a contiguous slice along the first dimension) on a dense rank 3 tensor.
// 
// NOTE: This class does NOT own the memory associated to it.
// NOTE: Unlike the matrix types elsewhere, this tensor will have C-like arrangement
// of its values in memory.
template<typename T>
struct Dense3TensorContiguousView {
    const int64_t d1, d2, d3;

    T* data;

    // Derived attributes.    
    const int64_t n_entries;
    const size_t size_bytes;
    const int64_t d2_d3;

    Dense3TensorContiguousView(int64_t d1, int64_t d2, int64_t d3, T* data) :
        d1(d1), d2(d2), d3(d3),
        data(data),
        n_entries(d1 * d2 * d3),
        d2_d3(d2 * d3),
        size_bytes(n_entries * sizeof(T))
    {}
};

}  // npeff
