Whamcloud - gitweb
Fix edge case when i_size doesn't get fixed until second e2fsck run
[tools/e2fsprogs.git] / e2fsck / dirinfo.c
index 5218234..516c461 100644 (file)
@@ -5,50 +5,46 @@
  * under the terms of the GNU Public License.
  */
 
-#include <et/com_err.h>
 #include "e2fsck.h"
 
-static int             dir_info_count = 0;
-static int             dir_info_size = 0;
-static struct dir_info *dir_info = 0;
-
-int get_num_dirs(ext2_filsys fs)
-{
-       int     i, num_dirs;
-
-       num_dirs = 0;
-       for (i = 0; i < fs->group_desc_count; i++)
-               num_dirs += fs->group_desc[i].bg_used_dirs_count;
-
-       return num_dirs;
-}
-
 /*
  * This subroutine is called during pass1 to create a directory info
  * entry.  During pass1, the passed-in parent is 0; it will get filled
  * in during pass2.  
  */
-void add_dir_info(ext2_filsys fs, ino_t ino, ino_t parent)
+void e2fsck_add_dir_info(e2fsck_t ctx, ext2_ino_t ino, ext2_ino_t parent)
 {
        struct dir_info *dir;
-       int     i, j;
+       int             i, j;
+       ext2_ino_t      num_dirs;
+       errcode_t       retval;
+       unsigned long   old_size;
 
 #if 0
        printf("add_dir_info for inode %lu...\n", ino);
 #endif
-       if (!dir_info) {
-               dir_info_count = 0;
-               dir_info_size = get_num_dirs(fs) + 10;
-
-               dir_info  = allocate_memory(dir_info_size *
-                                          sizeof (struct dir_info),
-                                          "directory map");
+       if (!ctx->dir_info) {
+               ctx->dir_info_count = 0;
+               retval = ext2fs_get_num_dirs(ctx->fs, &num_dirs);
+               if (retval)
+                       num_dirs = 1024;        /* Guess */
+               ctx->dir_info_size = num_dirs + 10;
+               ctx->dir_info  = (struct dir_info *)
+                       e2fsck_allocate_memory(ctx, ctx->dir_info_size
+                                              * sizeof (struct dir_info),
+                                              "directory map");
        }
        
-       if (dir_info_count >= dir_info_size) {
-               dir_info_size += 10;
-               dir_info = realloc(dir_info,
-                                 dir_info_size * sizeof(struct dir_info));
+       if (ctx->dir_info_count >= ctx->dir_info_size) {
+               old_size = ctx->dir_info_size * sizeof(struct dir_info);
+               ctx->dir_info_size += 10;
+               retval = ext2fs_resize_mem(old_size, ctx->dir_info_size *
+                                          sizeof(struct dir_info),
+                                          &ctx->dir_info);
+               if (retval) {
+                       ctx->dir_info_size -= 10;
+                       return;
+               }
        }
 
        /*
@@ -60,16 +56,17 @@ void add_dir_info(ext2_filsys fs, ino_t ino, ino_t parent)
         * the dir_info array needs to be sorted by inode number for
         * get_dir_info()'s sake.
         */
-       if (dir_info_count && dir_info[dir_info_count-1].ino >= ino) {
-               for (i = dir_info_count-1; i > 0; i--)
-                       if (dir_info[i-1].ino < ino)
+       if (ctx->dir_info_count &&
+           ctx->dir_info[ctx->dir_info_count-1].ino >= ino) {
+               for (i = ctx->dir_info_count-1; i > 0; i--)
+                       if (ctx->dir_info[i-1].ino < ino)
                                break;
-               dir = &dir_info[i];
+               dir = &ctx->dir_info[i];
                if (dir->ino != ino) 
-                       for (j = dir_info_count++; j > i; j--)
-                               dir_info[j] = dir_info[j-1];
+                       for (j = ctx->dir_info_count++; j > i; j--)
+                               ctx->dir_info[j] = ctx->dir_info[j-1];
        } else
-               dir = &dir_info[dir_info_count++];
+               dir = &ctx->dir_info[ctx->dir_info_count++];
        
        dir->ino = ino;
        dir->dotdot = parent;
@@ -80,26 +77,26 @@ void add_dir_info(ext2_filsys fs, ino_t ino, ino_t parent)
  * get_dir_info() --- given an inode number, try to find the directory
  * information entry for it.
  */
-struct dir_info *get_dir_info(ino_t ino)
+struct dir_info *e2fsck_get_dir_info(e2fsck_t ctx, ext2_ino_t ino)
 {
        int     low, high, mid;
 
        low = 0;
-       high = dir_info_count-1;
-       if (!dir_info)
+       high = ctx->dir_info_count-1;
+       if (!ctx->dir_info)
                return 0;
-       if (ino == dir_info[low].ino)
-               return &dir_info[low];
-       if  (ino == dir_info[high].ino)
-               return &dir_info[high];
+       if (ino == ctx->dir_info[low].ino)
+               return &ctx->dir_info[low];
+       if  (ino == ctx->dir_info[high].ino)
+               return &ctx->dir_info[high];
 
        while (low < high) {
                mid = (low+high)/2;
                if (mid == low || mid == high)
                        break;
-               if (ino == dir_info[mid].ino)
-                       return &dir_info[mid];
-               if (ino < dir_info[mid].ino)
+               if (ino == ctx->dir_info[mid].ino)
+                       return &ctx->dir_info[mid];
+               if (ino < ctx->dir_info[mid].ino)
                        high = mid;
                else
                        low = mid;
@@ -110,23 +107,31 @@ struct dir_info *get_dir_info(ino_t ino)
 /*
  * Free the dir_info structure when it isn't needed any more.
  */
-void free_dir_info(ext2_filsys fs)
+void e2fsck_free_dir_info(e2fsck_t ctx)
 {
-       if (dir_info) {
-               free(dir_info);
-               dir_info = 0;
+       if (ctx->dir_info) {
+               ext2fs_free_mem(&ctx->dir_info);
+               ctx->dir_info = 0;
        }
-       dir_info_size = 0;
-       dir_info_count = 0;
+       ctx->dir_info_size = 0;
+       ctx->dir_info_count = 0;
+}
+
+/*
+ * Return the count of number of directories in the dir_info structure
+ */
+int e2fsck_get_num_dirinfo(e2fsck_t ctx)
+{
+       return ctx->dir_info_count;
 }
 
 /*
  * A simple interator function
  */
-struct dir_info *dir_info_iter(int *control)
+struct dir_info *e2fsck_dir_info_iter(e2fsck_t ctx, int *control)
 {
-       if (*control >= dir_info_count)
+       if (*control >= ctx->dir_info_count)
                return 0;
 
-       return(dir_info + (*control)++);
+       return(ctx->dir_info + (*control)++);
 }