Skip to content

Commit 734c91f

Browse files
committed
fast-import: Avoid infinite loop after reset
Johannes Sixt noticed that a 'reset' command applied to a branch that is already active in the branch LRU cache can cause fast-import to relink the same branch into the LRU cache twice. This will cause the LRU cache to contain a cycle, making unload_one_branch run in an infinite loop as it tries to select the oldest branch for eviction. I have trivially fixed the problem by adding an active bit to each branch object; this bit indicates if the branch is already in the LRU and allows us to avoid trying to add it a second time. Converting the pack_id field into a bitfield makes this change take up no additional memory. Signed-off-by: Shawn O. Pearce <spearce@spearce.org>
1 parent 7193db3 commit 734c91f

File tree

1 file changed

+11
-5
lines changed

1 file changed

+11
-5
lines changed

fast-import.c

Lines changed: 11 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -220,7 +220,8 @@ struct branch
220220
const char *name;
221221
struct tree_entry branch_tree;
222222
uintmax_t last_commit;
223-
unsigned int pack_id;
223+
unsigned active : 1;
224+
unsigned pack_id : PACK_ID_BITS;
224225
unsigned char sha1[20];
225226
};
226227

@@ -528,6 +529,7 @@ static struct branch *new_branch(const char *name)
528529
b->table_next_branch = branch_table[hc];
529530
b->branch_tree.versions[0].mode = S_IFDIR;
530531
b->branch_tree.versions[1].mode = S_IFDIR;
532+
b->active = 0;
531533
b->pack_id = MAX_PACK_ID;
532534
branch_table[hc] = b;
533535
branch_count++;
@@ -1547,6 +1549,7 @@ static void unload_one_branch(void)
15471549
e = active_branches;
15481550
active_branches = e->active_next_branch;
15491551
}
1552+
e->active = 0;
15501553
e->active_next_branch = NULL;
15511554
if (e->branch_tree.tree) {
15521555
release_tree_content_recursive(e->branch_tree.tree);
@@ -1559,10 +1562,13 @@ static void unload_one_branch(void)
15591562
static void load_branch(struct branch *b)
15601563
{
15611564
load_tree(&b->branch_tree);
1562-
b->active_next_branch = active_branches;
1563-
active_branches = b;
1564-
cur_active_branches++;
1565-
branch_load_count++;
1565+
if (!b->active) {
1566+
b->active = 1;
1567+
b->active_next_branch = active_branches;
1568+
active_branches = b;
1569+
cur_active_branches++;
1570+
branch_load_count++;
1571+
}
15661572
}
15671573

15681574
static void file_change_m(struct branch *b)

0 commit comments

Comments
 (0)