summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/apk_defines.h3
-rw-r--r--src/apk_package.h6
-rw-r--r--src/apk_print.h1
-rw-r--r--src/commit.c357
-rw-r--r--src/package.c34
-rw-r--r--src/print.c12
-rw-r--r--src/solver.c3
7 files changed, 314 insertions, 102 deletions
diff --git a/src/apk_defines.h b/src/apk_defines.h
index ea4da76..629ca7b 100644
--- a/src/apk_defines.h
+++ b/src/apk_defines.h
@@ -143,6 +143,9 @@ void *apk_array_resize(void *array, size_t new_size, size_t elem_size);
APK_ARRAY(apk_string_array, char *);
+#define foreach_array_item(iter, array) \
+ for (iter = &(array)->item[0]; iter < &(array)->item[(array)->num]; iter++)
+
#define LIST_END (void *) 0xe01
#define LIST_POISON1 (void *) 0xdeadbeef
#define LIST_POISON2 (void *) 0xabbaabba
diff --git a/src/apk_package.h b/src/apk_package.h
index a5cda2f..44e463c 100644
--- a/src/apk_package.h
+++ b/src/apk_package.h
@@ -37,6 +37,10 @@ struct apk_provider;
#define APK_SIGN_GENERATE 4
#define APK_SIGN_VERIFY_AND_GENERATE 5
+#define APK_DEP_IRRELEVANT 0
+#define APK_DEP_SATISFIED 1
+#define APK_DEP_CONFLICTED 2
+
struct apk_sign_ctx {
int keys_fd;
int action;
@@ -129,6 +133,8 @@ void apk_dep_from_pkg(struct apk_dependency *dep, struct apk_database *db,
int apk_dep_is_materialized(struct apk_dependency *dep, struct apk_package *pkg);
int apk_dep_is_materialized_or_provided(struct apk_dependency *dep, struct apk_package *pkg);
int apk_dep_is_provided(struct apk_dependency *dep, struct apk_provider *p);
+int apk_dep_analyze(struct apk_dependency *dep, struct apk_package *pkg);
+char *apk_dep_snprintf(char *buf, size_t n, struct apk_dependency *dep);
void apk_blob_push_dep(apk_blob_t *to, struct apk_database *, struct apk_dependency *dep);
void apk_blob_push_deps(apk_blob_t *to, struct apk_database *, struct apk_dependency_array *deps);
diff --git a/src/apk_print.h b/src/apk_print.h
index 1dba85e..4c004be 100644
--- a/src/apk_print.h
+++ b/src/apk_print.h
@@ -31,5 +31,6 @@ struct apk_indent {
int apk_print_indented(struct apk_indent *i, apk_blob_t blob);
void apk_print_indented_words(struct apk_indent *i, const char *text);
+void apk_print_indented_fmt(struct apk_indent *i, const char *fmt, ...);
#endif
diff --git a/src/commit.c b/src/commit.c
index f426ba0..a4ba492 100644
--- a/src/commit.c
+++ b/src/commit.c
@@ -18,6 +18,46 @@
#include "apk_print.h"
+static void foreach_package_reverse_dependency2(
+ struct apk_package *pkg,
+ struct apk_name_array *rdepends,
+ int match,
+ void cb(struct apk_package *pkg0, struct apk_dependency *d0, void *ctx),
+ void *ctx)
+{
+ int i, j, k;
+
+ for (i = 0; i < rdepends->num; i++) {
+ struct apk_name *name0 = rdepends->item[i];
+
+ for (j = 0; j < name0->providers->num; j++) {
+ struct apk_package *pkg0 = name0->providers->item[j].pkg;
+
+ for (k = 0; k < pkg0->depends->num; k++) {
+ struct apk_dependency *d0 = &pkg0->depends->item[k];
+ if (apk_dep_analyze(d0, pkg) == match)
+ cb(pkg0, d0, ctx);
+ }
+ }
+ }
+}
+
+static void foreach_package_reverse_dependency(
+ struct apk_package *pkg,
+ int match,
+ void cb(struct apk_package *pkg0, struct apk_dependency *d0, void *ctx),
+ void *ctx)
+{
+ int i;
+
+ if (pkg == NULL)
+ return;
+
+ foreach_package_reverse_dependency2(pkg, pkg->name->rdepends, match, cb, ctx);
+ for (i = 0; i < pkg->provides->num; i++)
+ foreach_package_reverse_dependency2(pkg, pkg->provides->item[i].name->rdepends, match, cb, ctx);
+}
+
static inline int pkg_available(struct apk_database *db, struct apk_package *pkg)
{
if (pkg->repos & db->available_repos)
@@ -370,27 +410,46 @@ all_done:
return r;
}
-static void add_name_if_virtual(struct apk_name_array **names, struct apk_name *name)
-{
- int i;
-
- if (name->providers->num == 0)
- return;
-
- for (i = 0; i < name->providers->num; i++)
- if (name->providers->item[i].pkg->name == name)
- return;
+enum {
+ STATE_UNSET = 0,
+ STATE_PRESENT,
+ STATE_MISSING
+};
- for (i = 0; i < (*names)->num; i++)
- if ((*names)->item[i] == name)
- return;
+struct print_state {
+ struct apk_database *db;
+ struct apk_dependency_array *world;
+ struct apk_indent i;
+ struct apk_name_array *missing;
+ const char *label;
+ int num_labels;
+ int match;
+};
- *apk_name_array_add(names) = name;
+static void label_start(struct print_state *ps, const char *text)
+{
+ if (ps->label) {
+ printf(" %s:\n", ps->label);
+ ps->label = NULL;
+ ps->i.x = ps->i.indent = 0;
+ ps->num_labels++;
+ }
+ if (ps->i.x == 0) {
+ ps->i.x = printf(" %s", text);
+ ps->i.indent = ps->i.x + 1;
+ }
+}
+static void label_end(struct print_state *ps)
+{
+ if (ps->i.x != 0) {
+ printf("\n");
+ ps->i.x = ps->i.indent = 0;
+ }
}
-static void print_pinning_errors(struct apk_database *db, char *label,
- struct apk_package *pkg, unsigned int tag)
+static void print_pinning_errors(struct print_state *ps, struct apk_package *pkg, unsigned int tag)
{
+ struct apk_database *db = ps->db;
int i;
if (pkg->ipkg != NULL)
@@ -398,68 +457,146 @@ static void print_pinning_errors(struct apk_database *db, char *label,
if (pkg->repos & apk_db_get_pinning_mask_repos(db, APK_DEFAULT_PINNING_MASK | BIT(tag)))
return;
- printf(" %s: not pinned:", label);
for (i = 0; i < db->num_repo_tags; i++) {
- if (pkg->repos & db->repo_tags[i].allowed_repos)
- printf(" @" BLOB_FMT, BLOB_PRINTF(*db->repo_tags[i].name));
+ if (pkg->repos & db->repo_tags[i].allowed_repos) {
+ label_start(ps, "masked in:");
+ apk_print_indented(&ps->i, *db->repo_tags[i].name);
+ }
}
- printf("\n");
+ label_end(ps);
}
-static void print_dep_errors(struct apk_database *db, char *label,
- struct apk_dependency_array *deps,
- struct apk_name_array **names)
+static void print_conflicts(struct print_state *ps, struct apk_package *pkg)
{
- int i, print_label = 1;
- char buf[256];
- apk_blob_t p;
- struct apk_indent indent;
-
- for (i = 0; i < deps->num; i++) {
- struct apk_dependency *dep = &deps->item[i];
- struct apk_package *pkg;
+ struct apk_provider *p;
+ struct apk_dependency *d;
- pkg = (struct apk_package*) (dep->name->state_int & ~1);
- if (apk_dep_is_materialized_or_provided(dep, pkg))
+ foreach_array_item(p, pkg->name->providers) {
+ if (p->pkg == pkg || p->pkg->state_ptr == STATE_UNSET)
continue;
+ label_start(ps, "conflicts:");
+ apk_print_indented_fmt(&ps->i, PKG_VER_FMT, PKG_VER_PRINTF(p->pkg));
+ }
+ foreach_array_item(d, pkg->provides) {
+ foreach_array_item(p, d->name->providers) {
+ if (p->pkg == pkg || p->pkg->state_ptr == STATE_UNSET)
+ continue;
+ label_start(ps, "conflicts:");
+ apk_print_indented_fmt(
+ &ps->i,
+ PKG_VER_FMT "[%s]",
+ PKG_VER_PRINTF(p->pkg),
+ d->name->name);
+ }
+ }
+ label_end(ps);
+}
- if (pkg == NULL)
- add_name_if_virtual(names, dep->name);
+static void print_dep(struct apk_package *pkg0, struct apk_dependency *d0, void *ctx)
+{
+ struct print_state *ps = (struct print_state *) ctx;
+ const char *label = (ps->match == APK_DEP_SATISFIED) ? "satisfies:" : "breaks:";
+ char tmp[256];
- if (print_label) {
- print_label = 0;
- indent.x = printf(" %s:", label);
- indent.indent = indent.x + 1;
- }
- p = APK_BLOB_BUF(buf);
- apk_blob_push_dep(&p, db, dep);
- p = apk_blob_pushed(APK_BLOB_BUF(buf), p);
- apk_print_indented(&indent, p);
+ if (pkg0 != NULL && pkg0->state_ptr == STATE_UNSET)
+ return;
+
+ label_start(ps, label);
+ if (pkg0 == NULL)
+ apk_print_indented_fmt(&ps->i, "world[%s]", apk_dep_snprintf(tmp, sizeof(tmp), d0));
+ else
+ apk_print_indented_fmt(&ps->i, PKG_VER_FMT "[%s]",
+ PKG_VER_PRINTF(pkg0),
+ apk_dep_snprintf(tmp, sizeof(tmp), d0));
+}
+
+static void print_deps(struct print_state *ps, struct apk_package *pkg, int match)
+{
+ struct apk_dependency *d0;
+
+ ps->match = match;
+ foreach_array_item(d0, ps->world) {
+ if (apk_dep_analyze(d0, pkg) == match)
+ print_dep(NULL, d0, ps);
}
- if (!print_label)
- printf("\n");
+ foreach_package_reverse_dependency(pkg, ps->match, print_dep, ps);
+ label_end(ps);
}
-static void print_provides_errors(struct apk_database *db, char *label,
- struct apk_package *pkg)
+static void analyze_package(struct print_state *ps, struct apk_package *pkg, unsigned int tag)
{
- int i;
+ char pkgtext[256];
- for (i = 0; i < pkg->provides->num; i++) {
- struct apk_name *pn = pkg->provides->item[i].name;
- struct apk_provider p = APK_PROVIDER_FROM_PROVIDES(pkg, &pkg->provides->item[i]);
- struct apk_package *pkgC = (struct apk_package *) (pn->state_int & ~1);
+ snprintf(pkgtext, sizeof(pkgtext), PKG_VER_FMT, PKG_VER_PRINTF(pkg));
+ ps->label = pkgtext;
- if (pkgC == NULL && p.version == &apk_null_blob)
- continue;
+ print_pinning_errors(ps, pkg, tag);
+ print_conflicts(ps, pkg);
+ print_deps(ps, pkg, APK_DEP_CONFLICTED);
+ if (ps->label == NULL)
+ print_deps(ps, pkg, APK_DEP_SATISFIED);
+}
+
+static void analyze_name(struct print_state *ps, struct apk_name *name)
+{
+ struct apk_name **pname0, *name0;
+ struct apk_provider *p0;
+ struct apk_dependency *d0;
+ char tmp[256];
+
+ if (name->providers->num) {
+ snprintf(tmp, sizeof(tmp), "%s (virtual)", name->name);
+ ps->label = tmp;
+
+ label_start(ps, "provided by:");
+ foreach_array_item(p0, name->providers) {
+ /* FIXME: print only name if all pkgs provide it */
+ struct apk_package *pkg = p0->pkg;
+ apk_print_indented_fmt(&ps->i, PKG_VER_FMT, PKG_VER_PRINTF(pkg));
+ }
+ label_end(ps);
+ } else {
+ snprintf(tmp, sizeof(tmp), "%s (missing)", name->name);
+ ps->label = tmp;
+ }
- if ((pkgC == pkg) && (pn->state_int & 1) == 0) {
- pn->state_int |= 1;
+ label_start(ps, "required by:");
+ foreach_array_item(d0, ps->world) {
+ if (d0->name != name || d0->conflict)
continue;
+ apk_print_indented_fmt(&ps->i, "world[%s]",
+ apk_dep_snprintf(tmp, sizeof(tmp), d0));
+ }
+ foreach_array_item(pname0, name->rdepends) {
+ name0 = *pname0;
+ foreach_array_item(p0, name0->providers) {
+ if (p0->pkg->state_ptr == STATE_UNSET)
+ continue;
+ foreach_array_item(d0, p0->pkg->depends) {
+ if (d0->name != name || d0->conflict)
+ continue;
+ apk_print_indented_fmt(&ps->i,
+ PKG_VER_FMT "[%s]",
+ PKG_VER_PRINTF(p0->pkg),
+ apk_dep_snprintf(tmp, sizeof(tmp), d0));
+ break;
+ }
+ if (d0 != NULL)
+ break;
}
+ }
+ label_end(ps);
+}
- printf(" %s provides " PROVIDER_FMT " conflicting with " PKG_VER_FMT "\n",
- label, PROVIDER_PRINTF(pn, &p), PKG_VER_PRINTF(pkgC));
+static void analyze_deps(struct print_state *ps, struct apk_dependency_array *deps)
+{
+ struct apk_dependency *d0;
+
+ foreach_array_item(d0, deps) {
+ if (d0->name->state_int != STATE_UNSET)
+ continue;
+ d0->name->state_int = STATE_MISSING;
+ analyze_name(ps, d0->name);
}
}
@@ -467,60 +604,76 @@ void apk_solver_print_errors(struct apk_database *db,
struct apk_changeset *changeset,
struct apk_dependency_array *world)
{
- struct apk_name_array *names;
- char pkgtext[256];
- int i, j;
-
- apk_name_array_init(&names);
- /* FIXME: Print more rational error messages. Group failed
- * dependencies by package name. As in,
- * pkg-1.2 selected, but:
- * world: pkg>2
+ struct print_state ps;
+ struct apk_change *change;
+ struct apk_dependency *p;
+
+ /* ERROR: unsatisfiable dependencies:
+ * name:
+ * required by: a b c d e
+ * not available in any repository
+ * name (virtual):
+ * required by: a b c d e
+ * provided by: foo bar zed
+ * pkg-1.2:
+ * masked by: @testing
+ * satisfies: a[pkg]
+ * conflicts: pkg-2.0 foo-1.2 bar-1.2
+ * breaks: b[pkg>2] c[foo>2] d[!pkg]
+ *
+ * When two packages provide same name 'foo':
+ * a-1:
+ * satisfies: world[a]
+ * conflicts: b-1[foo]
+ * b-1:
+ * satisfies: world[b]
+ * conflicts: a-1[foo]
+ *
+ * c-1:
+ * satisfies: world[a]
+ * conflicts: c-1[foo] (self-conflict by providing foo twice)
+ *
+ * When two packages get pulled in:
+ * a-1:
+ * satisfies: app1[so:a.so.1]
+ * conflicts: a-2
+ * a-2:
+ * satisfies: app2[so:a.so.2]
+ * conflicts: a-1
+ *
+ * satisfies lists all dependencies that is not satisfiable by
+ * any other selected version. or all of them with -v.
*/
- apk_error("unsatisfiable dependencies:");
+
+ apk_error("unsatisfiable constraints:");
- for (i = 0; i < changeset->changes->num; i++) {
- struct apk_package *pkg = changeset->changes->item[i].new_pkg;
+ /* Construct information about names */
+ foreach_array_item(change, changeset->changes) {
+ struct apk_package *pkg = change->new_pkg;
if (pkg == NULL)
continue;
- pkg->state_int = 1;
- pkg->name->state_ptr = pkg;
- for (j = 0; j < pkg->provides->num; j++) {
- if (pkg->provides->item[j].version == &apk_null_blob)
- continue;
- if (pkg->provides->item[j].name->state_ptr == NULL)
- pkg->provides->item[j].name->state_ptr = pkg;
- }
+ pkg->state_int = STATE_PRESENT;
+ pkg->name->state_int = STATE_PRESENT;
+ foreach_array_item(p, pkg->provides)
+ p->name->state_int = STATE_PRESENT;
}
- print_dep_errors(db, "world", world, &names);
- for (i = 0; i < changeset->changes->num; i++) {
- struct apk_change *change = &changeset->changes->item[i];
+ /* Analyze is package, and missing names referred to */
+ ps = (struct print_state) {
+ .db = db,
+ .world = world,
+ };
+ analyze_deps(&ps, world);
+ foreach_array_item(change, changeset->changes) {
struct apk_package *pkg = change->new_pkg;
if (pkg == NULL)
continue;
- snprintf(pkgtext, sizeof(pkgtext), PKG_VER_FMT, PKG_VER_PRINTF(pkg));
- print_pinning_errors(db, pkgtext, pkg, change->new_repository_tag);
- print_dep_errors(db, pkgtext, pkg->depends, &names);
- print_provides_errors(db, pkgtext, pkg);
+ analyze_package(&ps, pkg, change->new_repository_tag);
+ analyze_deps(&ps, pkg->depends);
}
- for (i = 0; i < names->num; i++) {
- struct apk_indent indent;
- struct apk_name *name = names->item[i];
-
- printf("\n %s is a virtual package provided by:\n ", name->name);
- indent.x = 4;
- indent.indent = 4;
-
- for (j = 0; j < name->providers->num; j++) {
- struct apk_package *pkg = name->providers->item[j].pkg;
- snprintf(pkgtext, sizeof(pkgtext), PKG_VER_FMT, PKG_VER_PRINTF(pkg));
- apk_print_indented(&indent, APK_BLOB_STR(pkgtext));
- }
- printf("\n");
- }
- apk_name_array_free(&names);
+ if (ps.num_labels == 0)
+ printf(" Huh? Error reporter did not find the broken constraints.\n");
}
int apk_solver_commit(struct apk_database *db,
diff --git a/src/package.c b/src/package.c
index ecf88c2..e9eb45a 100644
--- a/src/package.c
+++ b/src/package.c
@@ -399,6 +399,40 @@ int apk_dep_is_materialized_or_provided(struct apk_dependency *dep, struct apk_p
return dep->conflict;
}
+int apk_dep_analyze(struct apk_dependency *dep, struct apk_package *pkg)
+{
+ int i;
+
+ if (pkg == NULL)
+ return APK_DEP_IRRELEVANT;
+
+ if (dep->name == pkg->name)
+ return apk_dep_is_materialized(dep, pkg) ? APK_DEP_SATISFIED : APK_DEP_CONFLICTED;
+
+ for (i = 0; i < pkg->provides->num; i++) {
+ struct apk_provider p;
+
+ if (pkg->provides->item[i].name != dep->name)
+ continue;
+
+ p = APK_PROVIDER_FROM_PROVIDES(pkg, &pkg->provides->item[i]);
+ return apk_dep_is_provided(dep, &p) ? APK_DEP_SATISFIED : APK_DEP_CONFLICTED;
+ }
+
+ return APK_DEP_IRRELEVANT;
+}
+
+char *apk_dep_snprintf(char *buf, size_t n, struct apk_dependency *dep)
+{
+ apk_blob_t b = APK_BLOB_PTR_LEN(buf, n);
+ apk_blob_push_dep(&b, NULL, dep);
+ if (b.len)
+ apk_blob_push_blob(&b, APK_BLOB_PTR_LEN("", 1));
+ else
+ b.ptr[-1] = 0;
+ return buf;
+}
+
void apk_blob_push_dep(apk_blob_t *to, struct apk_database *db, struct apk_dependency *dep)
{
int result_mask = dep->result_mask;
diff --git a/src/print.c b/src/print.c
index b47d8d7..93ef931 100644
--- a/src/print.c
+++ b/src/print.c
@@ -57,6 +57,18 @@ void apk_print_indented_words(struct apk_indent *i, const char *text)
(apk_blob_cb) apk_print_indented, i);
}
+void apk_print_indented_fmt(struct apk_indent *i, const char *fmt, ...)
+{
+ char tmp[256];
+ size_t n;
+ va_list va;
+
+ va_start(va, fmt);
+ n = vsnprintf(tmp, sizeof(tmp), fmt, va);
+ apk_print_indented(i, APK_BLOB_PTR_LEN(tmp, n));
+ va_end(va);
+}
+
const char *apk_error_str(int error)
{
if (error < 0)
diff --git a/src/solver.c b/src/solver.c
index 71a4d41..58bb30b 100644
--- a/src/solver.c
+++ b/src/solver.c
@@ -437,6 +437,9 @@ static int compare_providers(struct apk_solver_state *ss,
r = (int)pkgA->ss.dependencies_used - (int)pkgB->ss.dependencies_used;
if (r)
return r;
+ r = pkgB->ss.conflicts - pkgA->ss.conflicts;
+ if (r)
+ return r;
/* Prefer installed on self-upgrade */
solver_flags = pkgA->ss.solver_flags | pkgB->ss.solver_flags;