- 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;
- }