/*============================================================================
*
*                    Code_Saturne version 1.3
*                    ------------------------
*
*
*     This file is part of the Code_Saturne Kernel, element of the
*     Code_Saturne CFD tool.
*
*     Copyright (C) 1998-2007 EDF S.A., France
*
*     contact: saturne-support@edf.fr
*
*     The Code_Saturne Kernel is free software; you can redistribute it
*     and/or modify it under the terms of the GNU General Public License
*     as published by the Free Software Foundation; either version 2 of
*     the License, or (at your option) any later version.
*
*     The Code_Saturne Kernel 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 General Public License for more details.
*
*     You should have received a copy of the GNU General Public License
*     along with the Code_Saturne Kernel; if not, write to the
*     Free Software Foundation, Inc.,
*     51 Franklin St, Fifth Floor,
*     Boston, MA  02110-1301  USA
*
*============================================================================*/

/*============================================================================
 * Passage d'une connectivité noyau à une connecitvité nodale d'une
 * structure principale associée à ou extraite d'un maillage
 *============================================================================*/

/* includes système */

#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>


/*----------------------------------------------------------------------------
 *  Fichiers `include' librairie standard C ou BFT
 *----------------------------------------------------------------------------*/

#include <bft_error.h>
#include <bft_mem.h>

/*----------------------------------------------------------------------------
 *  Fichiers `include' locaux
 *----------------------------------------------------------------------------*/

#include "cs_base.h"
#include "cs_maillage.h"
#include "cs_maillage_grd.h"

#include <fvm_defs.h>
#include <fvm_nodal.h>
#include <fvm_nodal_from_desc.h>
#include <fvm_nodal_order.h>


/*----------------------------------------------------------------------------
 *  Fichiers `include' associés au fichier courant
 *----------------------------------------------------------------------------*/

#include "cs_maillage_connect.h"


#ifdef __cplusplus
extern "C" {
#endif /* __cplusplus */


/*============================================================================
 *  Définitions d'énumerations
 *============================================================================*/


/*============================================================================
 *  Définition de macros
 *============================================================================*/


/*============================================================================
 *  Variables globales statiques
 *============================================================================*/


/*============================================================================
 * Prototypes de fonctions privées
 *============================================================================*/

/*============================================================================
 *  Fonctions publiques pour API Fortran
 *============================================================================*/


/*============================================================================
 * Fonctions publiques
 *============================================================================*/

/*----------------------------------------------------------------------------
 * Extraction de la connectivité "cellules -> faces" d'un maillage.
 *
 * On considère une numérotation commune des faces internes et des
 * faces de bord, dans laquelle les faces de bord sont définies en
 * premier. L'indice commun de la i-ème face de bord est donc égal à i,
 * et celui de la j-ième face interne à nbr_fbr + j.
 *
 * Si ind_cel_extr != NULL, alors :
 * --- ind_cel_extr[icel] = indice dans la liste à extraire (0 à n-1)
 *     si icel correspond à une cellule à extraire
 * --- ind_cel_extr[icel] = -1 si la cellule icel est à ignorer
 *----------------------------------------------------------------------------*/

void cs_maillage_ret_cel_fac
(
 const cs_maillage_t   *const maillage,       /* --> Maillage */
 const cs_int_t               nbr_cel_extr,   /* --> Taille de ind_cel_extr[] */
 const cs_int_t               ind_cel_extr[], /* --> ind_cel_extr[cellule]
                                               *     = indice cellule extraite
                                               *       ou -1 */
 cs_int_t            * *const p_pos_cel_fac,  /* <-- idx cellule -> face */
 cs_int_t            * *const p_val_cel_fac   /* <-- val cellule -> face */
)
{

  cs_int_t    icel, icel1, icel2, ifac, nbr_cel_loc;

  cs_int_t  * cpt_cel_fac = NULL;
  cs_int_t  * pos_cel_fac = NULL;
  cs_int_t  * val_cel_fac = NULL;

  /* Allocation et initialisation de l'indice des positions */

  nbr_cel_loc = maillage->nbr_cel;
  if (ind_cel_extr != NULL)
    nbr_cel_loc = nbr_cel_extr;

  BFT_MALLOC(pos_cel_fac, nbr_cel_loc + 1, cs_int_t);

  for (icel = 0 ; icel < nbr_cel_loc + 1 ; icel++)
    pos_cel_fac[icel] = 0;

  /* Comptage du nombre de faces par cellule
   * (on affecte le compteur temporaire correspondant à icel à
   * pos_cel_fac[icel + 1] et non pas à pos_cel_fac[icel] pour
   * faciliter l'étape suivante) */

  /* Remarque : test si icel < maillage->nbr_cel sur faces internes
     pour ignorer les cellules fantômes parallèles et/ou périodiques */

  for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++) {
    icel = maillage->fbr_cel[ifac] - 1;
    if (ind_cel_extr != NULL)
      icel = ind_cel_extr[icel];
    if (icel > -1)
      pos_cel_fac[icel + 1] += 1;
  }

  for (ifac = 0 ; ifac < maillage->nbr_fac ; ifac++) {
    icel1 = maillage->fac_cel[ifac*2    ] - 1;
    icel2 = maillage->fac_cel[ifac*2 + 1] - 1;
    if (ind_cel_extr != NULL) {
      if (icel1 < maillage->nbr_cel)
        icel1 = ind_cel_extr[icel1];
      else
        icel1 = -1;
      if (icel2 < maillage->nbr_cel)
        icel2 = ind_cel_extr[icel2];
      else
        icel2 = -1;
    }
    if (icel1 > -1 && icel1 < maillage->nbr_cel)
      pos_cel_fac[icel1 + 1] += 1;
    if (icel2 > -1 && icel2 < maillage->nbr_cel)
      pos_cel_fac[icel2 + 1] += 1;
  }

  /* Construction de l'indice des positions */

  pos_cel_fac[0] = 1;
  for (icel = 0 ; icel < nbr_cel_loc ; icel++)
    pos_cel_fac[icel + 1] = pos_cel_fac[icel] + pos_cel_fac[icel + 1];

  /* Construction du tableau des valeurs */

  BFT_MALLOC(val_cel_fac, pos_cel_fac[nbr_cel_loc] - 1, cs_int_t);
  BFT_MALLOC(cpt_cel_fac, nbr_cel_loc, cs_int_t);

  for (icel = 0 ; icel < nbr_cel_loc ; icel++)
    cpt_cel_fac[icel] = 0;

  for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++) {
    icel = maillage->fbr_cel[ifac] - 1;
    if (ind_cel_extr != NULL)
      icel = ind_cel_extr[icel];
    if (icel > -1) {
      val_cel_fac[pos_cel_fac[icel] + cpt_cel_fac[icel] - 1] = ifac + 1;
      cpt_cel_fac[icel] += 1;
    }
  }

  for (ifac = 0 ; ifac < maillage->nbr_fac ; ifac++) {
    icel1 = maillage->fac_cel[ifac*2    ] - 1;
    icel2 = maillage->fac_cel[ifac*2 + 1] - 1;
    if (ind_cel_extr != NULL) {
      if (icel1 < maillage->nbr_cel)
        icel1 = ind_cel_extr[icel1];
      else
        icel1 = -1;
      if (icel2 < maillage->nbr_cel)
        icel2 = ind_cel_extr[icel2];
      else
        icel2 = -1;
    }
    if (icel1 > -1 && icel1 < maillage->nbr_cel) {
      val_cel_fac[pos_cel_fac[icel1] + cpt_cel_fac[icel1] - 1]
        =   ifac + maillage->nbr_fbr + 1;
      cpt_cel_fac[icel1] += 1;
    }
    if (icel2 > -1 && icel2 < maillage->nbr_cel) {
      val_cel_fac[pos_cel_fac[icel2] + cpt_cel_fac[icel2] - 1]
        = -(ifac + maillage->nbr_fbr + 1);
      cpt_cel_fac[icel2] += 1;
    }
  }

  BFT_FREE(cpt_cel_fac);

  /* Valeurs de retour */

  *p_pos_cel_fac = pos_cel_fac;
  *p_val_cel_fac = val_cel_fac;

#if 0 && defined(DEBUG) && !defined(NDEBUG)
 {
   cs_int_t ipos, ival;
   /* Impression des tableaux */
   bft_printf("dbg : cs_maillage_ret_cel_fac\n"
              "nombre de cellules extraites = %d\n", nbr_cel_extr);
   for (ipos = 0 ; ipos < nbr_cel_extr ; ipos++) {
     bft_printf("  cellule %d\n", ipos);
     bft_printf("    pos_cel_fac[%d] = %d\n", ipos, pos_cel_fac[ipos]);
     for (ival = pos_cel_fac[ipos]     - 1;
          ival < pos_cel_fac[ipos + 1] - 1;
          ival++)
       bft_printf("      val_cel_fac[%d] = %d\n", ival, val_cel_fac[ival]);
   }
   bft_printf("  pos_cel_fac[%d] = %d\n", ipos, pos_cel_fac[ipos]);
 }
#endif

}


/*----------------------------------------------------------------------------
 * Extraction et conversion en connectivité nodale externe d'un sous-ensemble
 * des cellules d'un maillage.
 *
 * La liste des cellules à traiter est optionnelle ; elle peut ne pas
 * être ordonnée en entrée, elle le sera toujours en sortie (les cellules
 * étant extraites au cours d'un parcours en ordre croissant, la liste
 * est réordonnée pour assurer la cohérence des liens des cellules extraites
 * vers leurs cellules parentes, construits à partir de cette liste).
 *----------------------------------------------------------------------------*/

fvm_nodal_t  * cs_maillage_extrait_cel_nodal
(
 const cs_maillage_t  *const maillage,      /* --> maillage                   */
 const char           *const nom,           /* --> nom à affecter             */
 const cs_int_t              nbr_liste_cel, /* --> taille de liste_cel[]      */
       cs_int_t              liste_cel[]    /* <-> liste optionnelle des
                                             *     cellules à traiter (1 à n) */
)
{

  cs_int_t    icel ;

  cs_int_t    nbr_cel_extr = 0 ;
  cs_int_t  * ind_cel_extr = NULL ;

  cs_int_t  * pos_cel_fac = NULL;
  cs_int_t  * val_cel_fac = NULL;

  fvm_lnum_t  dec_num_faces[3];
  fvm_lnum_t  *pos_faces_som[2];
  fvm_lnum_t  *val_faces_som[2];
  fvm_lnum_t  *faces_polyedres = NULL;

  fvm_nodal_t  *maillage_ext;

  /* Vérification que le maillage contient bien les connectivités
     faces->sommets */

  if (maillage->pos_fbr_som == NULL || maillage->pos_fac_som == NULL)
    bft_error(__FILE__, __LINE__, 0,
              _("Le maillage principal ne contient pas de connectivité\n"
                "faces->sommets, indispensable à la reconstruction\n"
                "de la connectivité nodale (cs_maillage_extrait_cel_nodal)."));

  /* Comptage du nombre de cellules à convertir */

  if (liste_cel != NULL) {

    BFT_MALLOC(ind_cel_extr, maillage->nbr_cel, cs_int_t);

    /* Initialisation sous forme de marqueur */
    for (icel = 0 ; icel < maillage->nbr_cel ; icel++)
      ind_cel_extr[icel] = -1;
    for (icel = 0 ; icel < nbr_liste_cel ; icel++) {
      if (liste_cel[icel] <= maillage->nbr_cel)
        ind_cel_extr[liste_cel[icel] - 1] = 1;
    }

    /* conversion indices marqués comme utilisés en pointeurs (1 à n)
       et reconstruction des valeurs de liste_cel[] de manière à
       s'assurer qu'elle soit triée */
    nbr_cel_extr = 0;
    for (icel = 0 ; icel < maillage->nbr_cel ; icel++) {
      if (ind_cel_extr[icel] == 1) {
        liste_cel[nbr_cel_extr] = icel + 1;
        ind_cel_extr[icel] = nbr_cel_extr++;
      }
    }

    assert(nbr_cel_extr <= nbr_liste_cel);

  }
  else {
    nbr_cel_extr = CS_MIN(maillage->nbr_cel, nbr_liste_cel);
    ind_cel_extr = NULL;
  }

  /* Extraction de la connectivité "cellules -> faces" */

  cs_maillage_ret_cel_fac(maillage,
                          nbr_cel_extr,
                          ind_cel_extr,
                          &pos_cel_fac,
                          &val_cel_fac);

  if (ind_cel_extr != NULL)
    BFT_FREE(ind_cel_extr);

  /* Construction de la connectivité nodale */

  dec_num_faces[0] = 0;
  dec_num_faces[1] = maillage->nbr_fbr + dec_num_faces[0];
  dec_num_faces[2] = maillage->nbr_fac + dec_num_faces[1];

  pos_faces_som[0] = maillage->pos_fbr_som;
  pos_faces_som[1] = maillage->pos_fac_som;
  val_faces_som[0] = maillage->val_fbr_som;
  val_faces_som[1] = maillage->val_fac_som;

  maillage_ext = fvm_nodal_create(nom);

  fvm_nodal_from_desc_add_cells(maillage_ext,
                                nbr_cel_extr,
                                NULL,
                                2,
                                dec_num_faces,
                                (const fvm_lnum_t **) pos_faces_som,
                                (const fvm_lnum_t **) val_faces_som,
                                pos_cel_fac,
                                val_cel_fac,
                                liste_cel,
                                &faces_polyedres);

  fvm_nodal_set_shared_vertices(maillage_ext,
                                maillage->coo_som);

  /* Libération mémoire */

  BFT_FREE(faces_polyedres);

  BFT_FREE(pos_cel_fac);
  BFT_FREE(val_cel_fac);

  /* Tri des cellules par numéro ou indice global croissant */

  fvm_nodal_order_cells(maillage_ext, maillage->num_cel);
  fvm_nodal_init_io_num(maillage_ext, maillage->num_cel, 3);

  /* Tri des sommets par numéro ou indice global croissant */

  fvm_nodal_order_vertices(maillage_ext, maillage->num_som);
  fvm_nodal_init_io_num(maillage_ext, maillage->num_som, 0);

  /* On a terminé */

  return maillage_ext;

}


/*----------------------------------------------------------------------------
 * Extraction et conversion en connectivité nodale externe d'un sous-ensemble
 * des faces d'un maillage.
 *
 * Les listes des faces à traiter sont optionnelles (si aucune des deux
 * n'est fournie, on extrait les faces de bord par défaut); elle peuvent
 * ne pas être ordonnées en entrée, elle le seront toujours en sortie
 * (les faces étant extraites au cours d'un parcours en ordre croissant,
 * la liste est réordonnée pour assurer la cohérence des liens des faces
 * extraites vers leurs faces parentes, construites à partir de cette liste).
 *----------------------------------------------------------------------------*/

fvm_nodal_t  * cs_maillage_extrait_fac_nodal
(
 const cs_maillage_t  *const maillage,      /* --> maillage                   */
 const char           *const nom,           /* --> nom à affecter             */
 const cs_int_t              nbr_liste_fac, /* --> taille de liste_fac[]      */
 const cs_int_t              nbr_liste_fbr, /* --> taille de liste_fbr[]      */
       cs_int_t              liste_fac[],   /* <-> liste optionnelle des faces
                                             *     internes à traiter (1 à n) */
       cs_int_t              liste_fbr[]    /* <-> liste optionnelle des faces
                                             *     de bord à traiter (1 à n)  */
)
{

  cs_int_t    ifac, i ;

  cs_int_t    nbr_fac_max = 0;
  cs_int_t    nbr_fbr_liste = 0 ;
  cs_int_t    nbr_fac_liste = 0 ;
  cs_int_t    nbr_fac_extr = 0 ;
  cs_int_t  * ind_fac_extr = NULL ;
  cs_int_t  * liste_fac_extr = NULL ;

  fvm_lnum_t  dec_num_faces[3];
  fvm_lnum_t  *pos_faces_som[2];
  fvm_lnum_t  *val_faces_som[2];

  fvm_gnum_t  *num_glob_fac = NULL;

  fvm_nodal_t  *maillage_ext;

  /* Vérification que le maillage contient bien les connectivités
     faces->sommets */

  if (maillage->pos_fbr_som == NULL || maillage->pos_fac_som == NULL)
    bft_error(__FILE__, __LINE__, 0,
              _("Le maillage principal ne contient pas de connectivité\n"
                "faces->sommets, indispensable à la reconstruction\n"
                "de la connectivité nodale (cs_maillage_extrait_cel_nodal)."));

  /* Comptage du nombre de faces à convertir */

  nbr_fac_max = maillage->nbr_fac + maillage->nbr_fbr;
  BFT_MALLOC(ind_fac_extr, nbr_fac_max, cs_int_t);

  /* Initialisation sous forme de marqueur */

  for (ifac = 0 ; ifac < nbr_fac_max ; ifac++)
    ind_fac_extr[ifac] = -1;

  if (nbr_liste_fbr == maillage->nbr_fbr) {
    for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++)
      ind_fac_extr[ifac] = 1;
  }
  else if (liste_fbr != NULL) {
    for (ifac = 0 ; ifac < nbr_liste_fbr ; ifac++)
      ind_fac_extr[liste_fbr[ifac] - 1] = 1;
  }

  if (nbr_liste_fac == maillage->nbr_fac) {
    for (ifac = 0 ; ifac < maillage->nbr_fac ; ifac++)
      ind_fac_extr[ifac + maillage->nbr_fbr] = 1;
  }
  else if (liste_fac != NULL) {
    for (ifac = 0 ; ifac < nbr_liste_fac ; ifac++)
      ind_fac_extr[liste_fac[ifac] - 1 + maillage->nbr_fbr] = 1;
  }

  /* conversion indices marqués comme utilisés en pointeurs (1 à n)
     et reconstruction des valeurs de liste_fbr[] et liste_fac[]
     de manière à s'assurer qu'elles soient triées */

  nbr_fbr_liste = 0;
  nbr_fac_liste = 0;

  if (liste_fbr != NULL) {
    for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++) {
      if (ind_fac_extr[ifac] == 1) {
        liste_fbr[nbr_fbr_liste] = ifac + 1;
        nbr_fbr_liste++;
      }
    }
  }
  else
    nbr_fbr_liste = CS_MIN(nbr_liste_fbr, maillage->nbr_fbr);

  if (liste_fac != NULL) {
    for (ifac = 0, i = maillage->nbr_fbr ;
         ifac < maillage->nbr_fac ;
         ifac++, i++) {
      if (ind_fac_extr[i] == 1) {
        liste_fac[nbr_fac_liste] = ifac + 1;
        nbr_fac_liste++;
      }
    }
  }
  else
    nbr_fac_liste = CS_MIN(nbr_liste_fac, maillage->nbr_fac);

  BFT_FREE(ind_fac_extr);

  /* Construction d'une liste continue (faces de bord, faces) */

  nbr_fac_extr = nbr_fbr_liste + nbr_fac_liste;

  BFT_MALLOC(liste_fac_extr, nbr_fac_extr, cs_int_t);

  if (liste_fbr != NULL) {
    for (ifac = 0 ; ifac < nbr_fbr_liste ; ifac++)
      liste_fac_extr[ifac] = liste_fbr[ifac];
  }
  else if (liste_fbr == NULL) { /* faces de bord par défaut si aucune liste */
    for (ifac = 0 ; ifac < nbr_fbr_liste ; ifac++)
      liste_fac_extr[ifac] = ifac + 1;
  }

  if (liste_fac != NULL) {
    for (ifac = 0, i = nbr_fbr_liste ; ifac < nbr_fac_liste ; ifac++, i++)
      liste_fac_extr[i] = liste_fac[ifac] + maillage->nbr_fbr;
  }
  else if (liste_fac == NULL) {
    for (ifac = 0, i = nbr_fbr_liste ; ifac < nbr_fac_liste ; ifac++, i++)
      liste_fac_extr[i] = ifac + maillage->nbr_fbr + 1;
  }


  /* Construction de la connectivité nodale */

  dec_num_faces[0] = 0;
  dec_num_faces[1] = maillage->nbr_fbr + dec_num_faces[0];
  dec_num_faces[2] = maillage->nbr_fac + dec_num_faces[1];

  pos_faces_som[0] = maillage->pos_fbr_som;
  pos_faces_som[1] = maillage->pos_fac_som;
  val_faces_som[0] = maillage->val_fbr_som;
  val_faces_som[1] = maillage->val_fac_som;

  maillage_ext = fvm_nodal_create(nom);

  fvm_nodal_from_desc_add_faces(maillage_ext,
                                nbr_fac_extr,
                                liste_fac_extr,
                                2,
                                dec_num_faces,
                                (const fvm_lnum_t **) pos_faces_som,
                                (const fvm_lnum_t **) val_faces_som,
                                NULL);

  fvm_nodal_set_shared_vertices(maillage_ext,
                                maillage->coo_som);

  BFT_FREE(liste_fac_extr);

  /* En cas de parallélisme, tri des faces par numéro ou indice
     global croissant */

  if (cs_glob_base_nbr > 1) {

    BFT_MALLOC(num_glob_fac, nbr_fac_max, fvm_gnum_t);

    if (maillage->num_fbr_ini == NULL) {
      for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++)
        num_glob_fac[ifac] = maillage->num_fbr[ifac];
    }
    else {
      for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++)
        num_glob_fac[ifac] = maillage->num_fbr[maillage->num_fbr_ini[ifac] - 1];
    }

    assert(maillage->nbr_fbr_glob + maillage->nbr_fac_glob > 0);

    if (maillage->num_fac_ini == NULL) {
      for (ifac = 0, i = maillage->nbr_fbr ;
           ifac < maillage->nbr_fac ;
           ifac++, i++)
        num_glob_fac[i] = maillage->num_fac[ifac] + maillage->nbr_fbr_glob;
    }
    else {
      for (ifac = 0, i = maillage->nbr_fbr ;
           ifac < maillage->nbr_fac ;
           ifac++, i++)
        num_glob_fac[i] =   maillage->num_fac[maillage->num_fac_ini[ifac] - 1]
                          + maillage->nbr_fbr_glob;
    }

  }

  /* Sans parallélisme, on doit tout de même tenir compte d'une éventuelle
     renumérotation des faces */

  else if (maillage->num_fac_ini != NULL || maillage->num_fbr_ini != NULL) {

    BFT_MALLOC(num_glob_fac, nbr_fac_max, fvm_gnum_t);

    if (maillage->num_fbr_ini == NULL) {
      for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++)
        num_glob_fac[ifac] = ifac + 1;
    }
    else {
      for (ifac = 0 ; ifac < maillage->nbr_fbr ; ifac++)
        num_glob_fac[ifac] = maillage->num_fbr_ini[ifac] - 1;
    }

    if (maillage->num_fac_ini == NULL) {
      for (ifac = 0, i = maillage->nbr_fbr ;
           ifac < maillage->nbr_fac ;
           ifac++, i++)
        num_glob_fac[i] = maillage->nbr_fbr + ifac + 1;
    }
    else {
      for (ifac = 0, i = maillage->nbr_fbr ;
           ifac < maillage->nbr_fac ;
           ifac++, i++)
        num_glob_fac[i] = maillage->nbr_fbr + maillage->num_fac_ini[ifac];
    }

  }

  fvm_nodal_order_faces(maillage_ext, num_glob_fac);
  fvm_nodal_init_io_num(maillage_ext, num_glob_fac, 2);

  if (num_glob_fac != NULL)
    BFT_FREE(num_glob_fac);

  /* Tri des sommets par numéro ou indice global croissant */

  fvm_nodal_order_vertices(maillage_ext, maillage->num_som);
  fvm_nodal_init_io_num(maillage_ext, maillage->num_som, 0);

  /* On a terminé */

  return maillage_ext;

}


/*============================================================================
 * Fonctions privées
 *============================================================================*/

#ifdef __cplusplus
}
#endif /* __cplusplus */


syntax highlighted by Code2HTML, v. 0.9.1