+ /* TODO Manejo de errores */
+ free(block);
+ PERR("Error al leer bloque");
+ *err = 3; /* EMUFS_ERROR_FILE_READ */
+ return NULL;
+ }
+ fclose(file);
+ return block;
+}
+
+EMUFS_REG_ID emufs_tipo1_grabar_registro(EMUFS* efs, void* reg, EMUFS_REG_SIZE reg_size, int* err)
+{
+ EMUFS_TIPO1_REG_HEADER reg_header; /* cabecera del registro a guardar */
+ EMUFS_FREE fs; /* espacio libre en el bloque */
+ EMUFS_BLOCK_ID block_id; /* identificador del 1er bloque */
+ char* block; /* buffer del bloque a guardar en disco */
+ char name_f[255];
+
+ strcpy(name_f, efs->nombre);
+ strcat(name_f, ".dat");
+
+ /* pongo tamaño del registro en la cabecera. */
+ reg_header.size = reg_size;
+ /* busco lugar para el registro en un bloque existente */
+ block_id = emufs_fsc_buscar_lugar(efs, sizeof(EMUFS_TIPO1_REG_HEADER)
+ + reg_size, &fs);
+ /* si no hay bloques con suficiente espacio creo un bloque nuevo */
+ if (block_id == EMUFS_NOT_FOUND) {
+ /* tamaño máximo ultilizable para datos en un bloque */
+ EMUFS_BLOCK_SIZE block_space = efs->tam_bloque - sizeof(EMUFS_TIPO1_REG_HEADER);
+ /* identificador del bloque que se guarda */
+ EMUFS_BLOCK_ID curr_block_id = EMUFS_NOT_FOUND;
+ /* tamaño de la porción de registro que se guarda */
+ EMUFS_REG_SIZE chunk_size = 0;
+ /* puntero a la poción del registro */
+ char* chunk_ptr = reg;
+
+ /* crear un nuevo bloque en memoria */
+ block = (char*) malloc(efs->tam_bloque);
+ if (block == NULL) {
+ /* TODO Manejo de errores */
+ PERR("No hay memoria");
+ *err = 2; /* EMUFS_ERROR_OUT_OF_MEMORY */
+ return EMUFS_NOT_FOUND;
+ }
+ reg_header.id = emufs_idx_get_new_id(efs, err);
+ do {
+ memset(block, 0, efs->tam_bloque); /* inicializa bloque */
+ chunk_ptr += chunk_size; /* Avanzo para guardar prox chunk */
+ reg_header.size -= chunk_size; /* Resto lo que ya guardé */
+ chunk_size = MIN(reg_header.size, block_space);
+ /* graba porción del registro en bloque */
+ emufs_tipo1_escribir_reg_chunk_en_memoria(block, reg_header, chunk_ptr, chunk_size);
+ /* graba el bloque en el archivo */
+ curr_block_id = emufs_tipo1_grabar_bloque(efs, block, EMUFS_NOT_FOUND, err);
+ if (*err) {
+ PERR("error al grabar bloque");
+ free(block);
+ return EMUFS_NOT_FOUND;
+ }
+ /* grabo el nuevo registro en el archivo de espacios libres */
+ *err = emufs_fsc_agregar(efs, curr_block_id, block_space - chunk_size);
+ if (*err) {
+ PERR("No se pudo agregar fsc");
+ free(block);
+ return EMUFS_NOT_FOUND;
+ }
+ /* si es el primer id de bloque obtenido, lo guardo para
+ * agregarlo después al archivo de índices. */
+ if (block_id == EMUFS_NOT_FOUND) {
+ block_id = curr_block_id;
+ }
+ } while (reg_header.size > block_space);
+ free(block);
+
+ /* Encontró espacio en un bloque existente, graba registro ahí */
+ } else {
+ /* cargo el bloque en block_id */
+ if (!(block = (char*) emufs_tipo1_leer_bloque(efs, block_id, err))) {
+ /* TODO Manejo de errores */
+ PERR("no se pudo leer el bloque");
+ return EMUFS_NOT_FOUND;
+ }
+ /* inserta el registro en el bloque */
+ /* tengo que buscar un ID válido para el nuevo registro */
+ reg_header.id = emufs_idx_get_new_id(efs, err);
+ /* graba registro en bloque */
+ emufs_tipo1_escribir_reg_en_memoria(block + efs->tam_bloque - fs,
+ reg_header, reg);
+ /* graba el bloque en el archivo */
+ block_id = emufs_tipo1_grabar_bloque(efs, block, block_id, err);
+ if (*err) {
+ PERR("error al grabar bloque");
+ free(block);
+ return EMUFS_NOT_FOUND;
+ }
+ free(block);
+ /* actualizo el archivo de espacios libres */
+ *err = emufs_fsc_actualizar(efs, block_id, fs - reg_size
+ - sizeof(EMUFS_TIPO1_REG_HEADER));
+ if (*err) {
+ PERR("No se pudo actualizar fsc");
+ return EMUFS_NOT_FOUND;
+ }
+ }
+
+ /* actualizo el indice de bloques y registros */
+ *err = emufs_idx_agregar(efs, reg_header.id, block_id);
+ if (*err){
+ PERR("No se pudo agregar idx");
+ return EMUFS_NOT_FOUND;
+ }
+
+ return reg_header.id;
+}
+
+/*Graba un bloque en el archivo*/
+EMUFS_BLOCK_ID emufs_tipo1_grabar_bloque(EMUFS *efs, void *block,
+ EMUFS_BLOCK_ID block_id, int* err)
+{
+ FILE* file;
+ char name_f[255];
+
+ strcpy(name_f,efs->nombre);
+ strcat(name_f,".dat");
+
+ if ((file = fopen(name_f, "r+b")) == NULL) {
+ /* TODO Manejo de errores */
+ PERR("Error al abrir archivo");
+ *err = 4; /* EMUFS_ERROR_CANT_OPEN_FILE */
+ return EMUFS_NOT_FOUND;
+ }
+ /* Si es NOT_FOUND tengo que agregar un bloque al final del archivo */
+ if (block_id == EMUFS_NOT_FOUND) {
+ /* me paro al final del archivo */
+ if (fseek(file, 0l, SEEK_END)) {
+ /* TODO Manejo de errores */
+ PERR("No se pudo hacer fseek()");
+ fclose(file);
+ *err = 8; /* EMUFS_ERROR_SEEK_FILE */
+ return EMUFS_NOT_FOUND;
+ }
+ /* Obtengo ID del bloque nuevo */
+ block_id = (ftell(file) - emufs_tipo1_header_size()) / efs->tam_bloque;
+ /* Si es un ID válido, salto hasta ese bloque. */
+ } else {
+ /* Salta el header del archivo */
+ if ((*err = emufs_tipo1_header_jump(file))) {
+ PERR("no se pudo saltar la cabecera del archivo");
+ fclose(file);
+ return EMUFS_NOT_FOUND;
+ }
+ /* Salta bloques */
+ if ((*err = emufs_tipo1_block_jump(efs, file, block_id))) {
+ PERR("no se pudo saltar la cabecera del bloque");
+ fclose(file);
+ return EMUFS_NOT_FOUND;
+ }
+ }
+ /* Grabo el bloque */
+ if (fwrite(block, efs->tam_bloque, 1, file) != 1) {
+ PERR("No se pudo escribir el archivo");