]> git.llucax.com Git - z.facultad/75.06/emufs.git/blobdiff - emufs/indice_b.c
Detalles de la lista.
[z.facultad/75.06/emufs.git] / emufs / indice_b.c
index afb6b8ecdf68ba818eb271806ec547a492f06aaa..2e1615b25fe2360d77b8933c126d32db80841107 100644 (file)
@@ -2,7 +2,7 @@
 #include "indice_b.h"
 #include "common.h"
 #include "emufs.h"
-#include <curses.h>
+#include "form.h"
 
 /* Cantidad de claves por nodo */
 #define CANT_HIJOS(x) ((x->tam_bloque-sizeof(B_NodoHeader))/sizeof(B_NodoEntry))
 static void b_grabar_nodo(INDICE *idx, int id, char *data);
 /** Da el ID del proximo nodo a poder ser utilizado */
 static int b_ultimo_id(INDICE *idx);
-/** Lee un nodo desde el archivo */
-static char *b_leer_nodo(INDICE *idx, int id);
 /** Crea un nodo en el archivo y lo retorna. En i se pone el ID asignado */
 static char *b_crear_nodo(INDICE *idx, int *i);
-/** Lee el header de un nodo y lo guarda en header */
-static void b_leer_header(char *src, B_NodoHeader *header);
 /** Actualiza el header de un nodo desde header */
 static void b_actualizar_header(char *src, B_NodoHeader *header);
-/** Retorna el array de claves del nodo (esta data modifica directamente el bloque
- *  por eso no es necesario usar un actualizar_claves 
- */
-static B_NodoEntry *b_leer_claves(char *src, B_NodoHeader *header);
 /** Inserta una clave en el nodo de manera iterativa.
  * \param idx Índice en donde insertar la clave.
  * \param clave Clave a insertar.
@@ -49,7 +41,9 @@ static void b_pasar_clave_a_derecha(INDICE*, char*, int, char*, int, int, B_Nodo
 /** Le pasa al hermano izquierdo una clave cuando se insertan claves */
 static void b_pasar_clave_a_izquierda(INDICE*, char*, int, char*, int, int, B_NodoEntry, int, int);
 /** Junta 2 nodos y hace uno solo */
-static void b_fundir_nodo(char *, int, char *, int, char *, int, int);
+static void b_fundir_nodo(INDICE *,char *, int, char *, int, char *, int, int);
+/** Crea 3 nodos a partir de 2 llenos */
+static void b_partir_dos_nodos_en_tres(INDICE*, int nodo_izq, int nodo_der, int padre, B_NodoEntry nuevo_entry);
                        
 static EMUFS_REG_ID b_insertar_dup_en_pos(INDICE *idx, INDICE_DATO pos, INDICE_DATO nuevo);
 
@@ -222,6 +216,7 @@ int emufs_indice_b_borrar(INDICE *idx, CLAVE k)
 
        if (encontrado) {
                PERR("Clave encontrada, borrando ...");
+               fprintf(stderr, "La clave a borrar esta en el nodo %d\n", nodo_id);
                b_borrar_clave(idx, nodo, nodo_id, k);
        } else {
                PERR("Clave no encontrada");
@@ -262,7 +257,7 @@ static char *b_crear_nodo(INDICE *idx, int *id)
        return bloque;
 }
 
-static char *b_leer_nodo(INDICE *idx, int id)
+char *b_leer_nodo(INDICE *idx, int id)
 {
        FILE *fp;
        char *out;
@@ -317,7 +312,7 @@ static void b_grabar_nodo(INDICE *idx, int id, char *data)
        fclose(fp);
 }
 
-static void b_leer_header(char *src, B_NodoHeader *header)
+void b_leer_header(char *src, B_NodoHeader *header)
 {
        if (!src) return;
 
@@ -330,7 +325,7 @@ static void b_actualizar_header(char *src, B_NodoHeader *header)
        memcpy(src, header, sizeof(B_NodoHeader));
 }
 
-static B_NodoEntry *b_leer_claves(char *src, B_NodoHeader *header)
+B_NodoEntry *b_leer_claves(char *src, B_NodoHeader *header)
 {
        return (B_NodoEntry *)(src+sizeof(B_NodoHeader));
 }
@@ -357,11 +352,52 @@ static void b_insertar_en_nodo(INDICE *idx, CLAVE clave, INDICE_DATO dato, int n
 
                if (nodo_header.cant == CANT_HIJOS(idx)) {
                        int total;
-                       /* TODO: Si es B*, hay que chequear si alguno de los 2
-                        *       nodos hermanos pueden prestarme espacio (y
-                        *       desplazar si es así). Si no pueden, hay que
-                        *       hacer un split de 2 nodos en 3.
-                        *       Si no es B*, hay que hacer lo que sigue:
+                       /*
+                        * TODO FIXME XXX TODO FIXME XXX TODO FIXME XXX
+                        *
+                        *******************************************************
+                        * Pseudocódigo que explica que hay que hacer si es B*
+                        *
+                        * OJO! Si el nodo en el cual estoy insertando es el
+                        * raíz, se maneja exactamente igual que en el B común,
+                        * así que el if sería algo como:
+                        * if (idx->tipo == IND_B_ASC && !es_raiz(nodo_id))
+                        *******************************************************
+                        *
+                        * nuevo_entry = new entry(clave, dato, hijo_der)
+                        * padre = get_padre(nodo)
+                        *
+                        * // Veo si puedo pasar a derecha
+                        * hijo_derecho = get_hijo_derecho(padre)
+                        * if (hijo_derecho != NULL && hijo_derecho.cantidad_entries < MAX_ENTRIES)
+                        *      buffer = new entries[MAX_ENTRIES+1]
+                        *      copiar_entries(buffer, nodo)
+                        *      insertar_ordenado(buffer, nuevo_entry)
+                        *      entry_a_pasar = get_entry_extremo_derecho(buffer)
+                        *      b_pasar_clave_a_derecha(idx, hijo_derecho, hijo_derecho.id, padre, padre.id, padre.posicion, entry_a_pasar)
+                        *      SALIR
+                        *
+                        * // Veo si puedo pasar a izquierda
+                        * hijo_izquierdo = get_hijo_izquierdo(padre)
+                        * if (hijo_izquierdo != NULL && hijo_izquierdo.cantidad_entries < MAX_ENTRIES)
+                        *      buffer = new entries[MAX_ENTRIES+1]
+                        *      copiar_entries(buffer, nodo)
+                        *      insertar_ordenado(buffer, nuevo_entry)
+                        *      entry_a_pasar = get_entry_extremo_izquierdo(buffer)
+                        *      b_pasar_clave_a_izquierda(idx, hijo_izquierdo, hijo_izquierdo.id, padre, padre.id, padre.posicion, entry_a_pasar)
+                        *      SALIR
+                        *
+                        * // Parto 2 nodos en 3.
+                        * if (hijo_izquierdo != NULL)
+                        *      b_partir_dos_nodos_en_tres(idx, hijo_izquierdo, nodo, padre, nuevo_entry)
+                        * else // Siempre alguno tiene que existir.
+                        *      b_partir_dos_nodos_en_tres(idx, nodo, hijo_derecho, padre, nuevo_entry)
+                        *
+                        * SALIR
+                        *
+                        **********************************************************************************
+                        * Fin de pseudocódigo, si no es B* se sigue haciendo lo que dice a continuación. *
+                        **********************************************************************************
                         */
                        nuevo = b_crear_nodo(idx, &nuevo_id);
                        i=0;
@@ -601,7 +637,7 @@ INDICE_DATO *emufs_indice_b_buscar_muchos(INDICE *idx, CLAVE clave, int *cant)
 
 static void b_borrar_clave(INDICE *idx, char *nodo, int nodo_id, CLAVE k)
 {
-       int pos, actual_id, padre_id, i, pos_padre, izquierda_id, derecha_id;
+       int pos, actual_id, padre_id, i, pos_padre, izquierda_id, derecha_id, p;
        B_NodoHeader header, header_actual, header_padre, header_izq, header_der;
        B_NodoEntry *claves, *claves_actual, *claves_padre;/*, *claves_izq, *claves_der;*/
        char *actual, *padre, *izq, *der;
@@ -616,13 +652,15 @@ static void b_borrar_clave(INDICE *idx, char *nodo, int nodo_id, CLAVE k)
        while (emufs_indice_es_menor(idx, claves[pos].clave, k)) pos++;
 
        /* Es el nodo una hoja? */
+       fprintf(stderr, "La clave esta en la pos = %d\n", pos);
        if (header.hijo_izquierdo != -1) {
                PERR("Nodo no es hoja, intercambio");
-               /* No!, es un nodo intermedio!! */
-               if (pos == 0)
-                       actual = b_leer_nodo(idx, header.hijo_izquierdo);
-               else
-                       actual = b_leer_nodo(idx, claves[pos+1].hijo_derecho);
+/*             if (pos == 0) {
+                       actual = b_leer_nodo(idx, nodo_header.hijo_izquierdo);
+               else*/
+               actual = b_leer_nodo(idx, claves[pos].hijo_derecho);
+               actual_id = claves[pos].hijo_derecho;
+               p = claves[pos].hijo_derecho;
 
                b_leer_header(actual, &header_actual);
                while (header_actual.hijo_izquierdo != -1) {
@@ -631,20 +669,24 @@ static void b_borrar_clave(INDICE *idx, char *nodo, int nodo_id, CLAVE k)
                        actual = b_leer_nodo(idx, actual_id);
                        b_leer_header(actual, &header_actual);
                }
-               claves_actual = b_leer_claves(actual, &header);
+               claves_actual = b_leer_claves(actual, &header_actual);
 
                claves[pos] = claves_actual[0];
+               claves[pos].hijo_derecho = p;
                pos = 0;
                b_grabar_nodo(idx, nodo_id, nodo);
                PERR("Listo");
        } else {
                PERR("Nodo es hoja");
                actual = nodo;
+               header_actual = header;
+               claves_actual = claves;
+               actual_id = nodo_id;
        }
 
        /* Borro la clave */
        PERR("Borrando clave");
-       for(i=pos; i < header_actual.cant; i++) {
+       for(i=pos; i < header_actual.cant-1; i++) {
                claves_actual[i] = claves_actual[i+1];
        }
        PERR("Borrado completo");
@@ -655,6 +697,7 @@ static void b_borrar_clave(INDICE *idx, char *nodo, int nodo_id, CLAVE k)
 
        /* Se cumple la condicion de hijos? */
        PERR("Dejo todo consistente");
+       fprintf(stderr, "Condicion : %d >= %d\n", header_actual.cant, MIN_HIJOS(idx));
        if (header_actual.cant >= MIN_HIJOS(idx)) {
                PERR("Borrar completo sin fundir");
                return;
@@ -678,7 +721,7 @@ static void b_borrar_clave(INDICE *idx, char *nodo, int nodo_id, CLAVE k)
                        b_leer_header(der, &header_der);
                } else {
                        PERR("Buscando que hijo soy");
-                       for(pos_padre=0; (pos_padre<header_padre.cant) && (claves_padre[pos_padre].hijo_derecho != actual_id); pos_padre++)     {       }
+                       for(pos_padre=0; (claves_padre[pos_padre].hijo_derecho != actual_id); pos_padre++)      {       }
 
                        if (pos_padre == header_padre.cant) {
                                PERR("ERROR GRAVE. Padre no me contiene :-(");
@@ -708,34 +751,44 @@ static void b_borrar_clave(INDICE *idx, char *nodo, int nodo_id, CLAVE k)
                PERR("Ta calcule lo que tengo que hacer");
                if ((derecha_id != -1) && (header_der.cant > MIN_HIJOS(idx))) {
                        PERR("Le pido clave a derecha");
+                       fprintf(stderr, "ANTES DE PEDIR DERECHA TENGO %d claves\n", header_actual.cant);
                        b_pedir_clave_derecha(der, derecha_id, padre, padre_id, actual, actual_id, pos_padre);
                        PERR("listo");
+                       b_leer_header(der, &header_der);
+                       b_leer_header(padre, &header_padre);
+                       b_leer_header(actual, &header_actual);
+                       fprintf(stderr, "DESPUES DE PEDIR DERECHA TENGO %d claves\n", header_actual.cant);
                } else if ((izquierda_id != -1) && (header_izq.cant > MIN_HIJOS(idx))) {
                        PERR("Le pido clave a izquierda");
-                       b_pedir_clave_izquierda(izq, izquierda_id, padre, padre_id, actual, actual_id, pos_padre-1);
+                       b_pedir_clave_izquierda(izq, izquierda_id, padre, padre_id, actual, actual_id, pos_padre);
+                       /* como se modificaron cosas, leo de nuevo los headers */
+                       b_leer_header(izq, &header_izq);
+                       b_leer_header(padre, &header_padre);
+                       b_leer_header(actual, &header_actual);
                        PERR("Listo");
                } else {
                        /* No pude pasar clave, tengo que fundir :-( */
                        PERR("Fundo nodos!");
                        if (derecha_id != -1) {
-                               b_fundir_nodo(actual, actual_id, padre, padre_id, der, derecha_id, pos_padre);
+                               b_fundir_nodo(idx, actual, actual_id, padre, padre_id, der, derecha_id, pos_padre);
                        } else {
-                               b_fundir_nodo(izq, izquierda_id, padre, padre_id, actual, actual_id, pos_padre-1);
+                               b_fundir_nodo(idx, izq, izquierda_id, padre, padre_id, actual, actual_id, pos_padre);
                        }
                }
 
                /* TODO que guardo ?, todo ? */
                b_grabar_nodo(idx, actual_id, actual);
-               b_grabar_nodo(idx, izquierda_id, izq);
-               b_grabar_nodo(idx, derecha_id, der);
-               b_grabar_nodo(idx, padre_id, padre);
+               if (izquierda_id != -1) b_grabar_nodo(idx, izquierda_id, izq);
+               if (derecha_id != -1) b_grabar_nodo(idx, derecha_id, der);
+               if (padre_id != -1) b_grabar_nodo(idx, padre_id, padre);
                if (actual_id != -1) free(actual);
-               /*if (padre_id != -1) free(padre);*/
                if (derecha_id != -1) free(der);
                if (izquierda_id != -1) free(izq);
                actual = padre;
                actual_id = padre_id;
-       } while ((actual_id != -1) && (header_actual.cant < MIN_HIJOS(idx)));
+               b_leer_header(actual, &header_actual);
+               claves_actual = b_leer_claves(actual, &header_actual);
+       } while ((actual_id != -1) && (actual_id != 0) && (header_actual.cant < MIN_HIJOS(idx)));
 }
 
 static void b_pedir_clave_derecha(char *der, int der_id, char *padre, int padre_id, char *nodo, int nodo_id, int pos_clave)
@@ -751,14 +804,14 @@ static void b_pedir_clave_derecha(char *der, int der_id, char *padre, int padre_
        b_leer_header(padre, &h_padre);
        c_padre = b_leer_claves(padre, &h_padre);
 
-       c_nodo[h_nodo.cant] = c_padre[pos_clave];
+       c_nodo[h_nodo.cant] = c_padre[pos_clave+1];
        c_nodo[h_nodo.cant].hijo_derecho = -1; /* XXX */
 
-       c_padre[pos_clave] = c_der[0];
-       c_padre[pos_clave].hijo_derecho = der_id;
+       c_padre[pos_clave+1] = c_der[0];
+       c_padre[pos_clave+1].hijo_derecho = der_id;
        
        /* Muevo las claves de derecho */
-       for(i=0; i<h_der.cant; i++) {
+       for(i=0; i<h_der.cant-1; i++) {
                c_der[i] = c_der[i+1];
        }
        h_der.cant--;
@@ -797,19 +850,24 @@ void b_pedir_clave_izquierda(char *izq, int izq_id, char *padre, int padre_id, c
        b_leer_header(padre, &h_padre);
        c_padre = b_leer_claves(padre, &h_padre);
 
-       for(i=h_nodo.cant; i>0;i++)
+       PERR("Muevo las claves");
+       for(i=h_nodo.cant; i>0;i--)
                c_nodo[i] = c_nodo[i-1];
 
        h_nodo.cant++;
+       PERR("Paso clave de padre a nodo");
        c_nodo[0] = c_padre[pos_clave];
        c_nodo[0].hijo_derecho = -1; /* XXX */
+       PERR("Paso clave de izquierda a padre");
        c_padre[pos_clave] = c_izq[h_izq.cant-1];
-       c_padre[pos_clave].hijo_derecho = izq_id;
+       c_padre[pos_clave].hijo_derecho = nodo_id;
        h_izq.cant--;
 
+       PERR("ACTUALIZO")
        b_actualizar_header(izq, &h_izq);
        b_actualizar_header(padre, &h_padre);
        b_actualizar_header(nodo, &h_nodo);
+       PERR("Salgo");
 }
 
 void b_pasar_clave_a_izquierda(INDICE* idx, char *izq, int izq_id, char *padre, int padre_id, int padre_pos, B_NodoEntry entry, int id_entry_hijo_izq, int id_entry_nodo)
@@ -827,8 +885,35 @@ void b_pasar_clave_a_izquierda(INDICE* idx, char *izq, int izq_id, char *padre,
        padre_entries[padre_pos] = entry;
 }
 
-static void b_fundir_nodo(char *izq, int izq_id, char *padre, int padre_id, char *der, int der_id, int pos_clave)
+static void b_fundir_nodo(INDICE *idx, char *izq, int izq_id, char *padre, int padre_id, char *der, int der_id, int pos_padre)
 {
+       int i;
+       B_NodoHeader h_izq, h_padre, h_der;
+       B_NodoEntry *c_izq, *c_padre, *c_der;
+
+       b_leer_header(der, &h_der);
+       c_der = b_leer_claves(der, &h_der);
+       b_leer_header(izq, &h_izq);
+       c_izq = b_leer_claves(izq, &h_izq);
+       b_leer_header(padre, &h_padre);
+       c_padre = b_leer_claves(padre, &h_padre);
+
+       c_izq[h_izq.cant] = c_padre[pos_padre];
+       h_padre.cant--;
+       for(i=pos_padre; i<h_padre.cant; i++)
+               c_padre[i] = c_padre[i+1];
+       h_izq.cant++;
+       for(i=0; i<h_der.cant; i++)
+               c_izq[h_izq.cant+i] = c_der[i];
+
+       h_izq.cant += h_der.cant;
+       
+       b_actualizar_header(izq, &h_izq);
+       b_actualizar_header(padre, &h_padre);
+
+       /* TODO Aca queda libre el nodo der, ver de recuperar! */
+       memset(der, 'X', idx->tam_bloque);
+       b_grabar_nodo(idx, der_id, der);
 }
 
 static EMUFS_REG_ID b_insertar_dup_en_pos(INDICE *idx, INDICE_DATO pos, INDICE_DATO nuevo)
@@ -969,58 +1054,32 @@ static void desabreviar_claves(INDICE *idx, B_NodoEntry *array, B_NodoHeader *he
        free(primera);
 }
 
-int emufs_indice_b_ver(INDICE *idx, WINDOW *win, int w, int h, int id)
+static void b_partir_dos_nodos_en_tres(INDICE* idx, int nodo_izq, int nodo_der, int padre, B_NodoEntry nuevo_entry)
 {
-       int y=0;
-       B_NodoHeader header;
-       B_NodoEntry *claves;
-       char *nodo;
-       char tmp[100];
-       int i;
-       int proximo;
-
-       mvwaddstr(win, y++, 0, "Nombre : ");
-       waddstr(win, idx->nombre);
-
-       /* Muestro la raiz */
-       nodo = b_leer_nodo(idx, id);
-       b_leer_header(nodo, &header);
-       claves = b_leer_claves(nodo, &header);
-
-       mvwaddstr(win, y++, 0, "Nodo Nro ");
-       sprintf(tmp, "%d", id);
-       waddstr(win, tmp);
-       mvwaddstr(win, y++, 0, "Nivel =  ");
-       sprintf(tmp, "%d", header.nivel);
-       waddstr(win, tmp);
-       mvwaddstr(win, y++, 0, "Cantidad de hijo = ");
-       sprintf(tmp, "%d", header.cant);
-       waddstr(win, tmp);
-       mvwaddstr(win, y++, 0, "Padre = ");
-       sprintf(tmp, "%d", header.padre);
-       waddstr(win, tmp);
-
-       /* Muestro las claves */
-       mvwaddstr(win, y++, 0, "Claves");
-       wmove(win, y, 0);
-       sprintf(tmp, "%d", header.hijo_izquierdo);
-       waddstr(win, tmp);
-       for(i=0; i<header.cant; i++) {
-               sprintf(tmp, "(%d)%d",
-                               claves[i].clave.i_clave,
-/*                             claves[i].dato.id,
-                               claves[i].dato.bloque,*/
-                               claves[i].hijo_derecho
-               );
-               waddstr(win, tmp);
-       }
-       free(nodo);
-       wrefresh(win);
-       proximo = getch()-'0';
-       werase(win);
-       wrefresh(win);
-       emufs_indice_b_ver(idx, win, w, h, proximo);
-       werase(win);
-       wrefresh(win);
+       /*
+        * PSEUDOCODIGO    TODO FIXME XXX TODO FIXME XXX TODO FIXME XXX
+        *
+        * // Creo un buffer con todos los entries (las claves) de ambos nodos, mas el padre y la nueva, ordenadas
+        * buffer_size = 2*MAX_ENTRIES+2
+        * buffer = new entries[buffer_size]
+        * copiar_entries(buffer, nodo_izq)
+        * concatenar_entries(buffer, padre)
+        * concatenar_entries(buffer, nodo_der)
+        * insertar_ordenado(buffer, nuevo_entry)
+        * // Borro los 2 nodos viejos para reutilizarlos y creo el tercero
+        * borrar_entries(nodo_izq)
+        * borrar_entries(nodo_der)
+        * nodo_nuevo = new nodo()
+        * // Copio de a tercios del buffer en los nuevos nodos, excluyendo las 2 claves 'limítrofes' para insertarlas luego en el padre
+        * copiar_algunos_entries(nodo_izq, buffer, 0, (buffer_size/3)-1)
+        * entry_promovido1 = buffer[buffer_size/3]
+        * copiar_algunos_entries(nodo_izq, buffer, (buffer_size/3)+1, 2*(buffer_size/3))
+        * entry_promovido2 = buffer[(2*(buffer_size/3))+1]
+        * copiar_algunos_entries(nodo_nuevo, buffer, (2*(buffer_size/3))+2, buffer_size-1))
+        * // Finalmente inserto (recursivamente, porque esta funcion es llamada desde b_insertar_en_nodo()) las claves promovidas en el padre
+        * b_insertar_en_nodo(idx, entry_promovido.clave, entry_promovido.dato, entry_promovido.id, entry_promovido, nodo_izq.id, nodo_der.id)
+        * b_insertar_en_nodo(idx, entry_promovido.clave, entry_promovido.dato, entry_promovido.id, entry_promovido, nodo_der.id, nodo_nuevo.id)
+        *
+        */
 }