#ifndef UTILITIES_H_INCLUDED
#define UTILITIES_H_INCLUDED

#include "header.h"

#include <sstream> // stringstream
#include <time.h> // for time(0) to generate different random number
#include <omp.h>
#include <queue>

#define ASSERT_RELEASE(cond, msg)                        \
do {                                                 \
if (!(cond)) {                                   \
std::cerr << "Assertion failed: " << msg     \
<< " (" << __FILE__ << ":"         \
<< __LINE__ << ")\n";              \
std::abort();                                \
}                                                \
} while (0)

/**
Convert an integer to string
**/
inline string int2str(int x)
{
    stringstream ss;
    ss << x;
    return ss.str();
}

/**
Get sign
**/
inline int sgn(float x)
{
    if (x >= 0) return 1;
    else return -1;
    // return 0;
}

void inline wht_bfly (float& a, float& b)
{
    float tmp = a;
    a += b;
    b = tmp - b;
}

void inline FWHT (Ref<VectorXf> data)
{
    int n = (int)data.size();
    int nlog2 = log2(n);

    int l, m;
    for (int i = 0; i < nlog2; ++i)
    {
        l = 1 << (i + 1);
        for (int j = 0; j < n; j += l)
        {
            m = 1 << i;
            for (int k = 0; k < m; ++k)
            {
                //cout << data (j + k) << endl;
                data (j + k) = data (j + k);
                //cout << data (j + k) << endl;

                //cout << data (j + k + m) << endl;
                data (j + k + m) = data (j + k + m);
                //cout << data (j + k + m) << endl;

                wht_bfly (data (j + k), data (j + k + m));
                //cout << data (j + k) << endl;
                //cout << data (j + k + m) << endl;

            }

        }
    }
}
//vector<int> samplingWOR(vector<int>, int);

float computeDist(const Ref<VectorXf> &, const Ref<VectorXf> &, const string& );
float computeChi2(const Ref<VectorXf>&, const Ref<VectorXf>&);

void embedChi2(const Ref<VectorXf> &, Ref<VectorXf>, int, int, float );
void embedJS(const Ref<VectorXf>&, Ref<VectorXf>, int , int , float );

/* Generate Hadamard matrix
*/
void bitHD3Generator(int, int, boost::dynamic_bitset<> &); // 1 HD for thousand points
void bitHD3Generator2(int, int, boost::dynamic_bitset<> &, boost::dynamic_bitset<> &); // 2 HD for million points

MatrixXf gaussGenerator(int, int, float, float, int);
MatrixXf cauchyGenerator(int, int, float, float, int);

// Saving
void outputLabels(const IVector &, const string&);
void outputIndices(const vector<IVector> & , const string& );
void outputNeighbors(const vector<FVector> & , const string&);

void outputOptics(const IVector &, const FVector &, const string&);

// Parsing input and param
void readParam(int , char**, sVDCParam & );
void loadtxtData(const string&, const string&, int , int, RowMajorMatrixXf & ); // load data fron filename
void loadbinData(const string& , const string& , int , int , RowMajorMatrixXf &);
void transformData(RowMajorMatrixXf & , const string& );

void sort_by_distance(Ref<VectorXf> , Ref<VectorXi>);

#endif // UTILITIES_H_INCLUDED
