state: fix world dependencies to be honored always
previously they might have been skipped on certain situations. this also fixes some other reverse dependency enforcements and implements new "pending" state for locked name.cute-signatures
parent
8c19869c29
commit
ec2ade1542
|
@ -130,6 +130,9 @@ static int add_main(void *ctx, int argc, char **argv)
|
||||||
}
|
}
|
||||||
|
|
||||||
state = apk_state_new(&db);
|
state = apk_state_new(&db);
|
||||||
|
if (state == NULL)
|
||||||
|
goto err;
|
||||||
|
|
||||||
for (i = 0; (pkgs != NULL) && i < pkgs->num; i++) {
|
for (i = 0; (pkgs != NULL) && i < pkgs->num; i++) {
|
||||||
r = apk_state_lock_dependency(state, &pkgs->item[i]);
|
r = apk_state_lock_dependency(state, &pkgs->item[i]);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
|
|
|
@ -29,12 +29,15 @@ static int cache_download(struct apk_database *db)
|
||||||
struct apk_change *change;
|
struct apk_change *change;
|
||||||
struct apk_package *pkg;
|
struct apk_package *pkg;
|
||||||
char item[PATH_MAX], cacheitem[PATH_MAX];
|
char item[PATH_MAX], cacheitem[PATH_MAX];
|
||||||
int i, r;
|
int i, r = 0;
|
||||||
|
|
||||||
if (db->world == NULL)
|
if (db->world == NULL)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
state = apk_state_new(db);
|
state = apk_state_new(db);
|
||||||
|
if (state == NULL)
|
||||||
|
goto err;
|
||||||
|
|
||||||
for (i = 0; i < db->world->num; i++) {
|
for (i = 0; i < db->world->num; i++) {
|
||||||
r = apk_state_lock_dependency(state, &db->world->item[i]);
|
r = apk_state_lock_dependency(state, &db->world->item[i]);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
|
@ -65,6 +68,7 @@ static int cache_download(struct apk_database *db)
|
||||||
}
|
}
|
||||||
|
|
||||||
err:
|
err:
|
||||||
|
if (state != NULL)
|
||||||
apk_state_unref(state);
|
apk_state_unref(state);
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,6 +46,9 @@ static int del_main(void *ctx, int argc, char **argv)
|
||||||
}
|
}
|
||||||
|
|
||||||
state = apk_state_new(&db);
|
state = apk_state_new(&db);
|
||||||
|
if (state == NULL)
|
||||||
|
goto err;
|
||||||
|
|
||||||
for (i = 0; i < argc; i++) {
|
for (i = 0; i < argc; i++) {
|
||||||
struct apk_dependency dep;
|
struct apk_dependency dep;
|
||||||
|
|
||||||
|
@ -63,6 +66,7 @@ static int del_main(void *ctx, int argc, char **argv)
|
||||||
}
|
}
|
||||||
r = apk_state_commit(state, &db);
|
r = apk_state_commit(state, &db);
|
||||||
err:
|
err:
|
||||||
|
if (state != NULL)
|
||||||
apk_state_unref(state);
|
apk_state_unref(state);
|
||||||
out:
|
out:
|
||||||
apk_db_close(&db);
|
apk_db_close(&db);
|
||||||
|
|
|
@ -193,6 +193,9 @@ static int fetch_main(void *ctx, int argc, char **argv)
|
||||||
struct apk_change *change;
|
struct apk_change *change;
|
||||||
|
|
||||||
state = apk_state_new(&db);
|
state = apk_state_new(&db);
|
||||||
|
if (state == NULL)
|
||||||
|
goto err;
|
||||||
|
|
||||||
r = apk_state_lock_dependency(state, &dep);
|
r = apk_state_lock_dependency(state, &dep);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
apk_state_unref(state);
|
apk_state_unref(state);
|
||||||
|
|
111
src/state.c
111
src/state.c
|
@ -29,9 +29,22 @@ struct apk_deferred_state {
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
int apk_state_prune_dependency(struct apk_state *state,
|
||||||
|
struct apk_dependency *dep);
|
||||||
|
|
||||||
|
#define APK_NS_LOCKED 0x00000001
|
||||||
|
#define APK_NS_PENDING 0x00000002
|
||||||
|
|
||||||
static int inline ns_locked(apk_name_state_t name)
|
static int inline ns_locked(apk_name_state_t name)
|
||||||
{
|
{
|
||||||
if (((intptr_t) name) & 0x1)
|
if (((intptr_t) name) & APK_NS_LOCKED)
|
||||||
|
return TRUE;
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int inline ns_pending(apk_name_state_t name)
|
||||||
|
{
|
||||||
|
if (((intptr_t) name) & APK_NS_PENDING)
|
||||||
return TRUE;
|
return TRUE;
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
@ -43,12 +56,18 @@ static int ns_empty(apk_name_state_t name)
|
||||||
|
|
||||||
static apk_name_state_t ns_from_pkg(struct apk_package *pkg)
|
static apk_name_state_t ns_from_pkg(struct apk_package *pkg)
|
||||||
{
|
{
|
||||||
return (apk_name_state_t) (((intptr_t) pkg) | 0x1);
|
return (apk_name_state_t) (((intptr_t) pkg) | APK_NS_LOCKED | APK_NS_PENDING);
|
||||||
|
}
|
||||||
|
|
||||||
|
static apk_name_state_t ns_from_pkg_non_pending(struct apk_package *pkg)
|
||||||
|
{
|
||||||
|
return (apk_name_state_t) (((intptr_t) pkg) | APK_NS_LOCKED);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct apk_package *ns_to_pkg(apk_name_state_t name)
|
static struct apk_package *ns_to_pkg(apk_name_state_t name)
|
||||||
{
|
{
|
||||||
return (struct apk_package *) (((intptr_t) name) & ~0x1);
|
return (struct apk_package *)
|
||||||
|
(((intptr_t) name) & ~(APK_NS_LOCKED | APK_NS_PENDING));
|
||||||
}
|
}
|
||||||
|
|
||||||
static apk_name_state_t ns_from_choices(struct apk_name_choices *nc)
|
static apk_name_state_t ns_from_choices(struct apk_name_choices *nc)
|
||||||
|
@ -117,7 +136,7 @@ static void ns_free(apk_name_state_t name)
|
||||||
struct apk_state *apk_state_new(struct apk_database *db)
|
struct apk_state *apk_state_new(struct apk_database *db)
|
||||||
{
|
{
|
||||||
struct apk_state *state;
|
struct apk_state *state;
|
||||||
int num_bytes;
|
int num_bytes, i, r;
|
||||||
|
|
||||||
num_bytes = sizeof(struct apk_state) + db->name_id * sizeof(char *);
|
num_bytes = sizeof(struct apk_state) + db->name_id * sizeof(char *);
|
||||||
state = (struct apk_state*) calloc(1, num_bytes);
|
state = (struct apk_state*) calloc(1, num_bytes);
|
||||||
|
@ -125,7 +144,22 @@ struct apk_state *apk_state_new(struct apk_database *db)
|
||||||
state->num_names = db->name_id;
|
state->num_names = db->name_id;
|
||||||
list_init(&state->change_list_head);
|
list_init(&state->change_list_head);
|
||||||
|
|
||||||
|
/* Instantiate each 'name' target in world, and lockout incompatible
|
||||||
|
* choices */
|
||||||
|
for (i = 0; i < db->world->num; i++) {
|
||||||
|
r = apk_state_prune_dependency(state, &db->world->item[i]);
|
||||||
|
if (r < 0) {
|
||||||
|
apk_error("Top level dependencies for %s are "
|
||||||
|
"conflicting or unsatisfiable.",
|
||||||
|
db->world->item[i].name->name);
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return state;
|
return state;
|
||||||
|
err:
|
||||||
|
free(state);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct apk_state *apk_state_dup(struct apk_state *state)
|
struct apk_state *apk_state_dup(struct apk_state *state)
|
||||||
|
@ -159,20 +193,26 @@ static int apk_state_add_change(struct apk_state *state,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int apk_state_lock_dependency(struct apk_state *state,
|
/* returns:
|
||||||
|
* -1 error
|
||||||
|
* 0 locked entry matches and is ok
|
||||||
|
* +n this many candidates on apk_name_choices for the name
|
||||||
|
*/
|
||||||
|
int apk_state_prune_dependency(struct apk_state *state,
|
||||||
struct apk_dependency *dep)
|
struct apk_dependency *dep)
|
||||||
{
|
{
|
||||||
struct apk_name *name = dep->name;
|
struct apk_name *name = dep->name;
|
||||||
struct apk_name_choices *c;
|
struct apk_name_choices *c;
|
||||||
struct apk_package *installed = NULL, *latest = NULL, *use;
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (name->id >= state->num_names)
|
if (name->id >= state->num_names)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if (ns_empty(state->name[name->id])) {
|
if (ns_empty(state->name[name->id])) {
|
||||||
if (dep->result_mask == APK_DEPMASK_CONFLICT)
|
if (dep->result_mask == APK_DEPMASK_CONFLICT) {
|
||||||
return apk_state_lock_name(state, name, NULL);
|
state->name[name->id] = ns_from_pkg(NULL);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
/* This name has not been visited yet.
|
/* This name has not been visited yet.
|
||||||
* Construct list of candidates. */
|
* Construct list of candidates. */
|
||||||
|
@ -186,16 +226,17 @@ int apk_state_lock_dependency(struct apk_state *state,
|
||||||
|
|
||||||
/* Locked to not-installed / remove? */
|
/* Locked to not-installed / remove? */
|
||||||
if (pkg == NULL) {
|
if (pkg == NULL) {
|
||||||
if (dep->result_mask == APK_DEPMASK_CONFLICT)
|
if (dep->result_mask != APK_DEPMASK_CONFLICT)
|
||||||
return 0;
|
return -1;
|
||||||
|
} else {
|
||||||
|
if (!(apk_version_compare(pkg->version, dep->version)
|
||||||
|
& dep->result_mask))
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (apk_version_compare(pkg->version, dep->version)
|
if (ns_pending(state->name[name->id]))
|
||||||
& dep->result_mask)
|
return 1;
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Multiple candidates: prune incompatible versions. */
|
/* Multiple candidates: prune incompatible versions. */
|
||||||
|
@ -219,11 +260,30 @@ int apk_state_lock_dependency(struct apk_state *state,
|
||||||
if (c->num == 1) {
|
if (c->num == 1) {
|
||||||
struct apk_package *pkg = c->pkgs[0];
|
struct apk_package *pkg = c->pkgs[0];
|
||||||
name_choices_unref(c);
|
name_choices_unref(c);
|
||||||
state->name[name->id] = NULL;
|
state->name[name->id] = ns_from_pkg(pkg);
|
||||||
return apk_state_lock_name(state, name, pkg);
|
return 1;
|
||||||
}
|
}
|
||||||
state->name[name->id] = ns_from_choices(c);
|
|
||||||
|
|
||||||
|
state->name[name->id] = ns_from_choices(c);
|
||||||
|
return c->num;
|
||||||
|
}
|
||||||
|
|
||||||
|
int apk_state_lock_dependency(struct apk_state *state,
|
||||||
|
struct apk_dependency *dep)
|
||||||
|
{
|
||||||
|
struct apk_name *name = dep->name;
|
||||||
|
struct apk_name_choices *c;
|
||||||
|
struct apk_package *installed = NULL, *latest = NULL, *use;
|
||||||
|
int i, r;
|
||||||
|
|
||||||
|
r = apk_state_prune_dependency(state, dep);
|
||||||
|
if (r <= 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
if (ns_pending(state->name[name->id]))
|
||||||
|
return apk_state_lock_name(state, name, ns_to_pkg(state->name[name->id]));
|
||||||
|
|
||||||
|
c = ns_to_choices(state->name[name->id]);
|
||||||
#if 1
|
#if 1
|
||||||
/* Get latest and installed packages */
|
/* Get latest and installed packages */
|
||||||
for (i = 0; i < c->num; i++) {
|
for (i = 0; i < c->num; i++) {
|
||||||
|
@ -342,10 +402,10 @@ static int for_each_broken_reverse_depency(struct apk_state *state,
|
||||||
pkg0 = ns_to_pkg(state->name[name0->id]);
|
pkg0 = ns_to_pkg(state->name[name0->id]);
|
||||||
if (pkg0 == NULL)
|
if (pkg0 == NULL)
|
||||||
continue;
|
continue;
|
||||||
return call_if_dependency_broke(state, pkg0, name, cb);
|
r = call_if_dependency_broke(state, pkg0, name, cb);
|
||||||
}
|
if (r != 0)
|
||||||
|
return r;
|
||||||
if (!ns_empty(state->name[name0->id])) {
|
} else if (!ns_empty(state->name[name0->id])) {
|
||||||
struct apk_name_choices *ns =
|
struct apk_name_choices *ns =
|
||||||
ns_to_choices(state->name[name0->id]);
|
ns_to_choices(state->name[name0->id]);
|
||||||
|
|
||||||
|
@ -358,10 +418,9 @@ static int for_each_broken_reverse_depency(struct apk_state *state,
|
||||||
name, cb);
|
name, cb);
|
||||||
if (r != 0)
|
if (r != 0)
|
||||||
return r;
|
return r;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return 0;
|
} else {
|
||||||
}
|
|
||||||
|
|
||||||
for (j = 0; j < name0->pkgs->num; j++) {
|
for (j = 0; j < name0->pkgs->num; j++) {
|
||||||
pkg0 = name0->pkgs->item[j];
|
pkg0 = name0->pkgs->item[j];
|
||||||
|
|
||||||
|
@ -373,6 +432,8 @@ static int for_each_broken_reverse_depency(struct apk_state *state,
|
||||||
name, cb);
|
name, cb);
|
||||||
if (r != 0)
|
if (r != 0)
|
||||||
return r;
|
return r;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -407,7 +468,7 @@ int apk_state_lock_name(struct apk_state *state,
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
ns_free(state->name[name->id]);
|
ns_free(state->name[name->id]);
|
||||||
state->name[name->id] = ns_from_pkg(newpkg);
|
state->name[name->id] = ns_from_pkg_non_pending(newpkg);
|
||||||
|
|
||||||
if (name->pkgs != NULL) {
|
if (name->pkgs != NULL) {
|
||||||
for (i = 0; i < name->pkgs->num; i++) {
|
for (i = 0; i < name->pkgs->num; i++) {
|
||||||
|
|
|
@ -41,6 +41,9 @@ static int upgrade_main(void *ctx, int argc, char **argv)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
state = apk_state_new(&db);
|
state = apk_state_new(&db);
|
||||||
|
if (state == NULL)
|
||||||
|
goto err;
|
||||||
|
|
||||||
for (i = 0; i < db.world->num; i++) {
|
for (i = 0; i < db.world->num; i++) {
|
||||||
r = apk_state_lock_dependency(state, &db.world->item[i]);
|
r = apk_state_lock_dependency(state, &db.world->item[i]);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
|
|
Loading…
Reference in New Issue