/*
  Copyright (c) 2003-2021 Tommi Junttila
  Released under the GNU Lesser General Public License version 3.
  
  This file is part of bliss.
  
  bliss is free software: you can redistribute it and/or modify
  it under the terms of the GNU Lesser General Public License as published by
  the Free Software Foundation, version 3 of the License.

  bliss 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 Lesser General Public License for more details.

  You should have received a copy of the GNU Lesser General Public License
  along with bliss.  If not, see <http://www.gnu.org/licenses/>.
*/

#include <cassert>
#include <vector>
#include "utils.hh"

namespace pddl_bliss {

size_t
print_permutation(FILE* const fp,
		  const unsigned int N,
		  const unsigned int* perm,
		  const unsigned int offset)
{
  assert(N > 0);
  assert(perm);
  size_t r = 0;
  unsigned int nof_cycles = 0;
  std::vector<bool> seen(N, false);
  for(unsigned int first = 0; first < N; first++) {
    if(seen[first] or perm[first] == first) continue;
    nof_cycles++;
    r += fprintf(fp, "(%u", first+offset);
    for(unsigned int i = perm[first]; i != first; i = perm[i]) {
      seen[i] = true;
      r += fprintf(fp, ",%u", i+offset);
    }
    r += fprintf(fp, ")");
  }
  if(nof_cycles == 0)
    r += fprintf(fp, "()");
  return r;
}

size_t
print_permutation(FILE* const fp,
		  const std::vector<unsigned int>& perm,
		  const unsigned int offset)
{
  const unsigned int N = perm.size();
  size_t r = 0;
  unsigned int nof_cycles = 0;
  std::vector<bool> seen(N, false);
  for(unsigned int first = 0; first < N; first++) {
    if(seen[first] or perm[first] == first) continue;
    nof_cycles++;
    r += fprintf(fp, "(%u", first+offset);
    for(unsigned int i = perm[first]; i != first; i = perm[i]) {
      seen[i] = true;
      r += fprintf(fp, ",%u", i+offset);
    }
    r += fprintf(fp, ")");
  }
  if(nof_cycles == 0)
    r += fprintf(fp, "()");
  return r;
}

bool
is_permutation(const unsigned int N, const unsigned int* perm)
{
  if(N == 0)
    return true;
  std::vector<bool> m(N, false);
  for(unsigned int i = 0; i < N; i++) {
    if(perm[i] >= N) return false;
    if(m[perm[i]]) return false;
    m[perm[i]] = true;
  }
  return true;
}

bool
is_permutation(const std::vector<unsigned int>& perm)
{
  const unsigned int N = perm.size();
  if(N == 0)
    return true;
  std::vector<bool> m(N, false);
  for(unsigned int i = 0; i < N; i++) {
    if(perm[i] >= N) return false;
    if(m[perm[i]]) return false;
    m[perm[i]] = true;
  }
  return true;
}


} // namespace pddl_bliss
