/* Chrysalide - Outil d'analyse de fichiers binaires
 * shuffle.c - permtutation aléatoire des éléments d'un ensemble fini
 *
 * Copyright (C) 2016-2018 Cyrille Bagard
 *
 *  This file is part of Chrysalide.
 *
 *  Chrysalide 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 3 of the License, or
 *  (at your option) any later version.
 *
 *  Chrysalide 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 Chrysalide.  If not, see <http://www.gnu.org/licenses/>.
 */


#include "shuffle.h"


#include <malloc.h>
#include <stdlib.h>
#include <string.h>



/******************************************************************************
*                                                                             *
*  Paramètres  : panel = instance d'objet GLib à traiter.                     *
*                                                                             *
*  Description : Mélange le contenu d'une liste.                              *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

void shuffle(void *base, size_t nmemb, size_t size)
{
    char *list;                             /* Conversion pour le confort  */
    char *tmp;                              /* Lieu de transition          */
    size_t i;                               /* Boucle de parcours          */
    size_t j;                               /* Emplacement aléatoire       */

    /**
     * Application de l'algorithme Fisher-Yates.
     *
     * Cf. https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle
     *
     * En version hors-ligne, cela donne :
     *
     *    -- To shuffle an array a of n elements (indices 0..n-1):
     *    for i from 0 to n−2 do
     *        j ← random integer such that i ≤ j < n
     *        exchange a[i] and a[j]
     *
     */

    if (nmemb > 1)
    {
        list = (char *)base;

        tmp = malloc(size);

        for (i = 0; i < (nmemb - 1); i++)
        {
            j = i + rand() % (nmemb - i);

            if (j != i)
            {
                memcpy(tmp, list + i * size, size);
                memcpy(list + i * size, list + j * size, size);
                memcpy(list + j * size, tmp, size);
            }

        }

        free(tmp);

    }

}