solver: transitive dependency requiring

If n+1 packages depend A, and A depend on B. Add n+1 dependencies
to B. Otherwise if someone conflicts B, B might be left out.

Leaving package unassigned is no longer a non-preferred action,
this fixes the final test case that was failing.

And with --force we might even install that scenario.
Add also some debug checks.
cute-signatures
Timo Teräs 2012-02-22 09:14:46 +02:00
parent 955153eac2
commit a7500a9df5
1 changed files with 41 additions and 12 deletions

View File

@ -73,6 +73,7 @@ struct apk_decision {
}; };
#ifdef DEBUG_CHECKS #ifdef DEBUG_CHECKS
struct apk_score saved_score; struct apk_score saved_score;
unsigned short saved_requirers;
#endif #endif
unsigned no_package : 1; unsigned no_package : 1;
@ -119,6 +120,7 @@ struct apk_name_state {
unsigned originally_installed : 1; unsigned originally_installed : 1;
unsigned has_available_pkgs : 1; unsigned has_available_pkgs : 1;
unsigned in_changeset : 1; unsigned in_changeset : 1;
unsigned in_world_dependency : 1;
/* dynamic state flags */ /* dynamic state flags */
unsigned none_excluded : 1; unsigned none_excluded : 1;
@ -636,7 +638,6 @@ static void get_unassigned_score(struct apk_name *name, struct apk_score *score)
*score = (struct apk_score){ *score = (struct apk_score){
.conflicts = ns->requirers, .conflicts = ns->requirers,
.non_preferred_actions = 1,
.preference = name->pkgs->num, .preference = name->pkgs->num,
}; };
} }
@ -973,6 +974,7 @@ static solver_result_t push_decision(struct apk_solver_state *ss,
#ifdef DEBUG_CHECKS #ifdef DEBUG_CHECKS
d->saved_score = ss->score; d->saved_score = ss->score;
d->saved_requirers = name_to_ns(name)->requirers;
#endif #endif
d->type = primary_decision; d->type = primary_decision;
d->branching_point = branching_point; d->branching_point = branching_point;
@ -995,6 +997,8 @@ static int next_branch(struct apk_solver_state *ss)
while (ss->num_decisions > 0) { while (ss->num_decisions > 0) {
struct apk_decision *d = &ss->decisions[ss->num_decisions]; struct apk_decision *d = &ss->decisions[ss->num_decisions];
struct apk_name *name = decision_to_name(d);
struct apk_name_state *ns = name_to_ns(name);
undo_decision(ss, d); undo_decision(ss, d);
@ -1003,6 +1007,8 @@ static int next_branch(struct apk_solver_state *ss)
"ERROR! saved_score "SCORE_FMT" != score "SCORE_FMT"\n", "ERROR! saved_score "SCORE_FMT" != score "SCORE_FMT"\n",
SCORE_PRINTF(&d->saved_score), SCORE_PRINTF(&d->saved_score),
SCORE_PRINTF(&ss->score)); SCORE_PRINTF(&ss->score));
ASSERT(d->saved_requirers == ns->requirers,
"ERROR! requirers not restored between decisions\n");
#endif #endif
if (backup_until >= ss->num_decisions && if (backup_until >= ss->num_decisions &&
@ -1013,8 +1019,6 @@ static int next_branch(struct apk_solver_state *ss)
} }
if (d->no_package && !d->found_solution) { if (d->no_package && !d->found_solution) {
struct apk_name *name = decision_to_name(d);
struct apk_name_state *ns = name_to_ns(name);
if (ns->last_touched_decision < backup_until) if (ns->last_touched_decision < backup_until)
backup_until = ns->last_touched_decision; backup_until = ns->last_touched_decision;
} }
@ -1030,7 +1034,15 @@ static void apply_constraint(struct apk_solver_state *ss, struct apk_dependency
{ {
struct apk_name *name = dep->name; struct apk_name *name = dep->name;
struct apk_name_state *ns = name_to_ns(name); struct apk_name_state *ns = name_to_ns(name);
int i; int i, strength;
if (ss->num_decisions > 0) {
struct apk_name *name0 = decision_to_name(&ss->decisions[ss->num_decisions]);
struct apk_name_state *ns0 = name_to_ns(name0);
strength = ns0->requirers ?: 1;
} else {
strength = 1;
}
if (ns->locked) { if (ns->locked) {
if (ns->chosen) if (ns->chosen)
@ -1040,12 +1052,12 @@ static void apply_constraint(struct apk_solver_state *ss, struct apk_dependency
dbg_printf("%s: locked to empty\n", dbg_printf("%s: locked to empty\n",
name->name); name->name);
if (!apk_dep_is_satisfied(dep, ns->chosen)) if (!apk_dep_is_satisfied(dep, ns->chosen))
ss->score.conflicts++; ss->score.conflicts += strength;
return; return;
} }
if (name->pkgs->num == 0) { if (name->pkgs->num == 0) {
if (!dep->optional) if (!dep->optional)
ss->score.conflicts++; ss->score.conflicts += strength;
return; return;
} }
@ -1083,7 +1095,7 @@ static void apply_constraint(struct apk_solver_state *ss, struct apk_dependency
} }
if (!dep->optional) if (!dep->optional)
ns->requirers++; ns->requirers += strength;
update_name_state(ss, name); update_name_state(ss, name);
} }
@ -1092,7 +1104,15 @@ static void undo_constraint(struct apk_solver_state *ss, struct apk_dependency *
{ {
struct apk_name *name = dep->name; struct apk_name *name = dep->name;
struct apk_name_state *ns = name_to_ns(name); struct apk_name_state *ns = name_to_ns(name);
int i; int i, strength;
if (ss->num_decisions > 0) {
struct apk_name *name0 = decision_to_name(&ss->decisions[ss->num_decisions]);
struct apk_name_state *ns0 = name_to_ns(name0);
strength = ns0->requirers ?: 1;
} else {
strength = 1;
}
if (ns->locked) { if (ns->locked) {
if (ns->chosen != NULL) { if (ns->chosen != NULL) {
@ -1103,12 +1123,12 @@ static void undo_constraint(struct apk_solver_state *ss, struct apk_dependency *
name->name); name->name);
} }
if (!apk_dep_is_satisfied(dep, ns->chosen)) if (!apk_dep_is_satisfied(dep, ns->chosen))
ss->score.conflicts--; ss->score.conflicts -= strength;
return; return;
} }
if (name->pkgs->num == 0) { if (name->pkgs->num == 0) {
if (!dep->optional) if (!dep->optional)
ss->score.conflicts--; ss->score.conflicts -= strength;
return; return;
} }
@ -1142,7 +1162,7 @@ static void undo_constraint(struct apk_solver_state *ss, struct apk_dependency *
} }
if (!dep->optional) if (!dep->optional)
ns->requirers--; ns->requirers -= strength;
update_name_state(ss, name); update_name_state(ss, name);
} }
@ -1431,6 +1451,9 @@ static int free_state(apk_hash_item item, void *ctx)
struct apk_name_state *ns = (struct apk_name_state *) name->state_ptr; struct apk_name_state *ns = (struct apk_name_state *) name->state_ptr;
if (ns != NULL) { if (ns != NULL) {
#ifdef DEBUG_CHECKS
ASSERT(ns->requirers == 0, "Requirers is not zero after cleanup\n");
#endif
free(ns); free(ns);
name->state_ptr = NULL; name->state_ptr = NULL;
} }
@ -1481,8 +1504,10 @@ int apk_solver_solve(struct apk_database *db,
ss->best_score = (struct apk_score){ .conflicts = -1 }; ss->best_score = (struct apk_score){ .conflicts = -1 };
list_init(&ss->unsolved_list_head); list_init(&ss->unsolved_list_head);
for (i = 0; i < world->num; i++) for (i = 0; i < world->num; i++) {
sort_name(ss, world->item[i].name); sort_name(ss, world->item[i].name);
name_to_ns(world->item[i].name)->in_world_dependency = 1;
}
list_for_each_entry(ipkg, &db->installed.packages, installed_pkgs_list) { list_for_each_entry(ipkg, &db->installed.packages, installed_pkgs_list) {
sort_name(ss, ipkg->pkg->name); sort_name(ss, ipkg->pkg->name);
name_to_ns(ipkg->pkg->name)->originally_installed = 1; name_to_ns(ipkg->pkg->name)->originally_installed = 1;
@ -1525,6 +1550,10 @@ int apk_solver_solve(struct apk_database *db,
/* STOP or EXPAND */ /* STOP or EXPAND */
} while (r != SOLVERR_STOP); } while (r != SOLVERR_STOP);
#ifdef DEBUG_CHECKS
foreach_dependency(ss, world, undo_constraint);
#endif
/* collect packages */ /* collect packages */
dbg_printf("finished. best score "SCORE_FMT". solution has %d packages.\n", dbg_printf("finished. best score "SCORE_FMT". solution has %d packages.\n",
SCORE_PRINTF(&ss->best_score), SCORE_PRINTF(&ss->best_score),