/* * Copyright (c) 2014, Novell Inc. * * This program is licensed under the BSD license, read LICENSE.BSD * for further information */ /* * cplxdeps.c * * normalize complex dependencies into CNF/DNF form */ #include #include #include #include #include #include "pool.h" #include "cplxdeps.h" #ifdef ENABLE_COMPLEX_DEPS #undef CPLXDEBUG int pool_is_complex_dep_rd(Pool *pool, Reldep *rd) { for (;;) { if (rd->flags == REL_AND || rd->flags == REL_COND) /* those two are the complex ones */ return 1; if (rd->flags != REL_OR) return 0; if (ISRELDEP(rd->name) && pool_is_complex_dep_rd(pool, GETRELDEP(pool, rd->name))) return 1; if (!ISRELDEP(rd->evr)) return 0; rd = GETRELDEP(pool, rd->evr); } } /* expand simple dependencies into package lists */ static int expand_simpledeps(Pool *pool, Queue *bq, int start, int split) { int end = bq->count; int i, x; int newsplit = 0; for (i = start; i < end; i++) { if (i == split) newsplit = bq->count - (end - start); x = bq->elements[i]; if (x == pool->nsolvables) { Id *dp = pool->whatprovidesdata + bq->elements[++i]; for (; *dp; dp++) queue_push(bq, *dp); } else queue_push(bq, x); } if (i == split) newsplit = bq->count - (end - start); queue_deleten(bq, start, end - start); return newsplit; } /* invert all literals in the blocks. note that this also turns DNF into CNF and vice versa */ static void invert_depblocks(Pool *pool, Queue *bq, int start) { int i, j, end; expand_simpledeps(pool, bq, start, 0); end = bq->count; for (i = j = start; i < end; i++) { if (bq->elements[i]) { bq->elements[i] = -bq->elements[i]; continue; } /* end of block reached, reorder */ if (i - 1 > j) { int k; for (k = i - 1; j < k; j++, k--) { Id t = bq->elements[j]; bq->elements[j] = bq->elements[k]; bq->elements[k] = t; } } j = i + 1; } } /* * returns: * 0: no blocks * 1: matches all * -1: at least one block */ static int normalize_dep(Pool *pool, Id dep, Queue *bq, int todnf) { int bqcnt = bq->count; int bqcnt2; Id dp; #ifdef CPLXDEBUG printf("normalize_dep %s todnf:%d\n", pool_dep2str(pool, dep), todnf); #endif if (pool_is_complex_dep(pool, dep)) { Reldep *rd = GETRELDEP(pool, dep); if (rd->flags == REL_AND || rd->flags == REL_OR || rd->flags == REL_COND) { int flags = rd->flags; int r, mode; /* in inverted mode, COND means AND. otherwise it means OR NOT */ if (flags == REL_COND && todnf) flags = REL_AND; mode = flags == REL_AND ? 0 : 1; /* get blocks of first argument */ r = normalize_dep(pool, rd->name, bq, todnf); if (r == 0) { if (flags == REL_AND) return 0; if (flags == REL_COND) { r = normalize_dep(pool, rd->evr, bq, todnf ^ 1); if (r == 0 || r == 1) return r == 0 ? 1 : 0; invert_depblocks(pool, bq, bqcnt); /* invert block for COND */ return r; } return normalize_dep(pool, rd->evr, bq, todnf); } if (r == 1) { if (flags != REL_AND) return 1; return normalize_dep(pool, rd->evr, bq, todnf); } /* get blocks of second argument */ bqcnt2 = bq->count; /* COND is OR with NEG on evr block, so we invert the todnf flag in that case*/ r = normalize_dep(pool, rd->evr, bq, flags == REL_COND ? todnf ^ 1 : todnf); if (r == 0) { if (flags == REL_OR) return -1; queue_truncate(bq, bqcnt); return flags == REL_COND ? 1 : 0; } if (r == 1) { if (flags == REL_OR) { queue_truncate(bq, bqcnt); return 1; } return -1; } if (flags == REL_COND) invert_depblocks(pool, bq, bqcnt2); /* invert 2nd block */ if (mode == todnf) { /* simple case: just join em. nothing more to do here. */ #ifdef CPLXDEBUG printf("SIMPLE JOIN %d %d %d\n", bqcnt, bqcnt2, bq->count); #endif return -1; } else { /* complex case: mix em */ int i, j, bqcnt3; #ifdef CPLXDEBUG printf("COMPLEX JOIN %d %d %d\n", bqcnt, bqcnt2, bq->count); #endif bqcnt2 = expand_simpledeps(pool, bq, bqcnt, bqcnt2); bqcnt3 = bq->count; for (i = bqcnt; i < bqcnt2; i++) { for (j = bqcnt2; j < bqcnt3; j++) { int a, b; int bqcnt4 = bq->count; int k = i; /* mix i block with j block, both blocks are sorted */ while (bq->elements[k] && bq->elements[j]) { if (bq->elements[k] < bq->elements[j]) queue_push(bq, bq->elements[k++]); else { if (bq->elements[k] == bq->elements[j]) k++; queue_push(bq, bq->elements[j++]); } } while (bq->elements[j]) queue_push(bq, bq->elements[j++]); while (bq->elements[k]) queue_push(bq, bq->elements[k++]); /* block is finished, check for A + -A */ for (a = bqcnt4, b = bq->count - 1; a < b; ) { if (-bq->elements[a] == bq->elements[b]) break; if (-bq->elements[a] > bq->elements[b]) a++; else b--; } if (a < b) queue_truncate(bq, bqcnt4); /* ignore this block */ else queue_push(bq, 0); /* finish block */ } /* advance to next block */ while (bq->elements[i]) i++; } i = -1; if (bqcnt3 == bq->count) /* ignored all blocks? */ i = todnf ? 0 : 1; queue_deleten(bq, bqcnt, bqcnt3 - bqcnt); return i; } } } /* fallback case: just use package list */ dp = pool_whatprovides(pool, dep); if (dp <= 2 || !pool->whatprovidesdata[dp]) return dp == 2 ? 1 : 0; if (pool->whatprovidesdata[dp] == SYSTEMSOLVABLE) return 1; if (todnf) { for (; pool->whatprovidesdata[dp]; dp++) queue_push2(bq, pool->whatprovidesdata[dp], 0); } else { queue_push2(bq, pool->nsolvables, dp); /* not yet expanded marker */ queue_push(bq, 0); } return -1; } int pool_normalize_complex_dep(Pool *pool, Id dep, Queue *bq, int flags) { int i, bqcnt = bq->count; i = normalize_dep(pool, dep, bq, (flags & CPLXDEPS_TODNF) ? 1 : 0); if ((flags & CPLXDEPS_EXPAND) != 0) { if (i != 0 && i != 1) expand_simpledeps(pool, bq, bqcnt, 0); } if ((flags & CPLXDEPS_INVERT) != 0) { if (i == 0 || i == 1) i ^= 1; else invert_depblocks(pool, bq, bqcnt); } #ifdef CPLXDEBUG if (i == 0) printf("NONE\n"); else if (i == 1) printf("ALL\n"); else { for (i = bqcnt; i < bq->count; i++) { if (bq->elements[i] == pool->nsolvables) { Id *dp = pool->whatprovidesdata + bq->elements[++i]; printf(" ("); while (*dp) printf(" %s", pool_solvid2str(pool, *dp++)); printf(" )"); } else if (bq->elements[i] > 0) printf(" %s", pool_solvid2str(pool, bq->elements[i])); else if (bq->elements[i] < 0) printf(" -%s", pool_solvid2str(pool, -bq->elements[i])); else printf(" ||"); } printf("\n"); i = -1; } #endif return i; } #endif /* ENABLE_COMPLEX_DEPS */