From 0d73c71e57bf5bcb773d1fb07bd16f45d24c688f Mon Sep 17 00:00:00 2001 From: ringabout <43030857+ringabout@users.noreply.github.com> Date: Tue, 5 Mar 2024 13:46:36 +0000 Subject: [PATCH] clone sat --- .github/workflows/ci.yml | 1 + config.nims | 4 + dist/sat | 1 + nim-testutils | 1 + nim-unittest2 | 1 + nimble | 1 + packages | 1 + src/atlas.nim | 4 +- src/depgraphs.nim | 5 +- src/nimbleparser.nim | 4 +- src/sat.nim | 781 --------------------------------------- src/satvars.nim | 106 ------ test12/nim-testutils | 1 + test12/nim-unittest2 | 1 + test12/packages | 1 + test12/testutils | 1 + test3/nim-testutils | 1 + test3/nim-unittest2 | 1 + test3/packages | 1 + test3/testutils | 1 + testatlas/nim-testutils | 1 + testatlas/nim-unittest2 | 1 + testatlas/packages | 1 + testatlas/testutils | 1 + 24 files changed, 32 insertions(+), 890 deletions(-) create mode 160000 dist/sat create mode 160000 nim-testutils create mode 160000 nim-unittest2 create mode 160000 nimble create mode 160000 packages delete mode 100644 src/sat.nim delete mode 100644 src/satvars.nim create mode 160000 test12/nim-testutils create mode 160000 test12/nim-unittest2 create mode 160000 test12/packages create mode 160000 test12/testutils create mode 160000 test3/nim-testutils create mode 160000 test3/nim-unittest2 create mode 160000 test3/packages create mode 160000 test3/testutils create mode 160000 testatlas/nim-testutils create mode 160000 testatlas/nim-unittest2 create mode 160000 testatlas/packages create mode 160000 testatlas/testutils diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 14b0095..da4677a 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -131,6 +131,7 @@ jobs: - name: Run tests run: | cd atlas + nim cloneDeps nim c -r tests/tester.nim - name: Test install with Nimble diff --git a/config.nims b/config.nims index 5e97af7..056e5da 100644 --- a/config.nims +++ b/config.nims @@ -8,7 +8,11 @@ task unitTests, "Runs unit tests": task tester, "Runs integration tests": exec "nim c -d:debug -r tests/tester.nim" +task cloneDeps, "clone Atlas deps": + exec "nim r src/private/clone.nim" + task buildRelease, "Build release": + exec "nim r src/private/clone.nim" when defined(macosx): let x86Args = "\"-target x86_64-apple-macos11 -arch x86_64 -DARCH=x86_64\"" exec "nim c -d:release --passC:" & x86args & " --passL:" & x86args & " -o:./atlas_x86_64 src/atlas.nim" diff --git a/dist/sat b/dist/sat new file mode 160000 index 0000000..ff49b7d --- /dev/null +++ b/dist/sat @@ -0,0 +1 @@ +Subproject commit ff49b7d31a9afa4b50a27c91b9d4737e687f6f48 diff --git a/nim-testutils b/nim-testutils new file mode 160000 index 0000000..460c90a --- /dev/null +++ b/nim-testutils @@ -0,0 +1 @@ +Subproject commit 460c90a5223ed84c321ddd001424c3a2a546b81c diff --git a/nim-unittest2 b/nim-unittest2 new file mode 160000 index 0000000..183eaaa --- /dev/null +++ b/nim-unittest2 @@ -0,0 +1 @@ +Subproject commit 183eaaa57714694694f5dea5f896556db4b96325 diff --git a/nimble b/nimble new file mode 160000 index 0000000..3575fd5 --- /dev/null +++ b/nimble @@ -0,0 +1 @@ +Subproject commit 3575fd54a890d910ace56678aa74b4237d604175 diff --git a/packages b/packages new file mode 160000 index 0000000..b03a72e --- /dev/null +++ b/packages @@ -0,0 +1 @@ +Subproject commit b03a72e7be742c7d68b3314d65b33efecf6b67f6 diff --git a/src/atlas.nim b/src/atlas.nim index 0712156..20edb9c 100644 --- a/src/atlas.nim +++ b/src/atlas.nim @@ -10,12 +10,14 @@ ## a Nimble dependency and its dependencies recursively. import std / [parseopt, strutils, os, osproc, tables, sets, json, jsonutils] -import versions, context, osutils, packagesjson, sat, gitops, nimenv, lockfiles, +import versions, context, osutils, packagesjson, gitops, nimenv, lockfiles, depgraphs, confighandler, configutils, cloner, nimblechecksums, reporters, nimbleparser, pkgurls from std/terminal import isatty +import ../dist/sat/src/sat/sat + const AtlasVersion = block: diff --git a/src/depgraphs.nim b/src/depgraphs.nim index 681462a..daf6b6b 100644 --- a/src/depgraphs.nim +++ b/src/depgraphs.nim @@ -8,7 +8,10 @@ import std / [sets, tables, os, strutils, streams, json, jsonutils, algorithm] -import context, satvars, sat, gitops, runners, reporters, nimbleparser, pkgurls, cloner, versions +import context, gitops, runners, reporters, nimbleparser, pkgurls, cloner, versions + +import ../dist/sat/src/sat/[sat, satvars] + type DependencyVersion* = object # Represents a specific version of a project. diff --git a/src/nimbleparser.nim b/src/nimbleparser.nim index 73a5b79..ce058d4 100644 --- a/src/nimbleparser.nim +++ b/src/nimbleparser.nim @@ -7,7 +7,9 @@ # import std / [os, strutils, tables, unicode, hashes] -import versions, satvars, packagesjson, reporters, gitops, parse_requires, pkgurls, compiledpatterns +import versions, packagesjson, reporters, gitops, parse_requires, pkgurls, compiledpatterns + +import ../dist/sat/src/sat/satvars type DependencyStatus* = enum diff --git a/src/sat.nim b/src/sat.nim deleted file mode 100644 index cc3ee83..0000000 --- a/src/sat.nim +++ /dev/null @@ -1,781 +0,0 @@ -## SAT solver -## (c) 2021 Andreas Rumpf -## Based on explanations and Haskell code from -## https://andrew.gibiansky.com/blog/verification/writing-a-sat-solver/ - -## Formulars as packed ASTs, no pointers no cry. Solves formulars with many -## thousands of variables in no time. - -import satvars - -type - FormKind* = enum - FalseForm, TrueForm, VarForm, NotForm, AndForm, OrForm, ExactlyOneOfForm, ZeroOrOneOfForm # 8 so the last 3 bits - Atom = distinct BaseType - Formular* = seq[Atom] # linear storage - -const - KindBits = 3 - KindMask = 0b111 - -template kind(a: Atom): FormKind = FormKind(BaseType(a) and KindMask) -template intVal(a: Atom): BaseType = BaseType(a) shr KindBits - -proc newVar*(val: VarId): Atom {.inline.} = - Atom((BaseType(val) shl KindBits) or BaseType(VarForm)) - -proc newOperation(k: FormKind; val: BaseType): Atom {.inline.} = - Atom((val shl KindBits) or BaseType(k)) - -proc trueLit*(): Atom {.inline.} = Atom(TrueForm) -proc falseLit*(): Atom {.inline.} = Atom(FalseForm) - -proc lit(k: FormKind): Atom {.inline.} = Atom(k) - -when false: - proc isTrueLit(a: Atom): bool {.inline.} = a.kind == TrueForm - proc isFalseLit(a: Atom): bool {.inline.} = a.kind == FalseForm - -proc varId(a: Atom): VarId = - assert a.kind == VarForm - result = VarId(BaseType(a) shr KindBits) - -type - PatchPos = distinct int - FormPos = distinct int - -template firstSon(n: FormPos): FormPos = FormPos(n.int+1) - -proc prepare(dest: var Formular; source: Formular; sourcePos: FormPos): PatchPos = - result = PatchPos dest.len - dest.add source[sourcePos.int] - -proc prepare(dest: var Formular; k: FormKind): PatchPos = - result = PatchPos dest.len - dest.add newOperation(k, 1) - -proc patch(f: var Formular; pos: PatchPos) = - let pos = pos.int - let k = f[pos].kind - assert k > VarForm - let distance = int32(f.len - pos) - f[pos] = newOperation(k, distance) - -proc nextChild(f: Formular; pos: var int) {.inline.} = - let x = f[int pos] - pos += (if x.kind <= VarForm: 1 else: int(intVal(x))) - -proc sons2(f: Formular; n: FormPos): (FormPos, FormPos) {.inline.} = - var p = n.int + 1 - nextChild(f, p) - result = (n.firstSon, FormPos(p)) - -iterator sonsReadonly(f: Formular; n: FormPos): FormPos = - var pos = n.int - assert f[pos].kind > VarForm - let last = pos + f[pos].intVal - inc pos - while pos < last: - yield FormPos pos - nextChild f, pos - -iterator sons(dest: var Formular; source: Formular; n: FormPos): FormPos = - let patchPos = prepare(dest, source, n) - for x in sonsReadonly(source, n): yield x - patch dest, patchPos - -proc copyTree(dest: var Formular; source: Formular; n: FormPos) = - let x = source[int n] - let len = (if x.kind <= VarForm: 1 else: int(intVal(x))) - for i in 0..= 0 - assert n.int < f.len - case f[n.int].kind - of FalseForm: dest.add 'F' - of TrueForm: dest.add 'T' - of VarForm: - varRepr dest, varId(f[n.int]).int - else: - case f[n.int].kind - of AndForm: - dest.add "(&" - of OrForm: - dest.add "(|" - of ExactlyOneOfForm: - dest.add "(1==" - of NotForm: - dest.add "(~" - of ZeroOrOneOfForm: - dest.add "(1>=" - else: assert false, "cannot happen" - var i = 0 - for child in sonsReadonly(f, n): - if i > 0: dest.add ' ' - toString(dest, f, child, varRepr) - inc i - dest.add ')' - -proc `$`*(f: Formular): string = - assert f.len > 0 - toString(result, f, FormPos 0, proc (dest: var string; x: int) = - dest.add 'v' - dest.addInt x - ) - -proc `$`*(f: Formular; varRepr: proc (dest: var string; i: int)): string = - assert f.len > 0 - toString(result, f, FormPos 0, varRepr) - -type - Builder* = object - f: Formular - toPatch: seq[PatchPos] - -proc isEmpty*(b: Builder): bool {.inline.} = - b.f.len == 0 or b.f.len == 1 and b.f[0].kind in {NotForm, AndForm, OrForm, ExactlyOneOfForm, ZeroOrOneOfForm} - -proc openOpr*(b: var Builder; k: FormKind) = - b.toPatch.add PatchPos b.f.len - b.f.add newOperation(k, 0) - -proc closeOpr*(b: var Builder) = - patch(b.f, b.toPatch.pop()) - -proc add*(b: var Builder; a: Atom) = - b.f.add a - -proc add*(b: var Builder; a: VarId) = - b.f.add newVar(a) - -proc addNegated*(b: var Builder; a: VarId) = - b.openOpr NotForm - b.f.add newVar(a) - b.closeOpr - -proc getPatchPos*(b: Builder): PatchPos = - PatchPos b.f.len - -proc resetToPatchPos*(b: var Builder; p: PatchPos) = - b.f.setLen p.int - -proc deleteLastNode*(b: var Builder) = - b.f.setLen b.f.len - 1 - -type - BuilderPos* = distinct int - -proc rememberPos*(b: Builder): BuilderPos {.inline.} = BuilderPos b.f.len -proc rewind*(b: var Builder; pos: BuilderPos) {.inline.} = setLen b.f, int(pos) - -proc toForm*(b: var Builder): Formular = - assert b.toPatch.len == 0, "missing `closeOpr` calls" - result = move b.f - -proc isValid*(v: VarId): bool {.inline.} = v.int32 >= 0 - -proc freeVariable(f: Formular): VarId = - ## returns NoVar if there is no free variable. - for i in 0..= 2: - setLen dest, initialLen - dest.add lit FalseForm - result = FalseForm - elif trueCount == childCount: - setLen dest, initialLen - if trueCount <= 1: - dest.add lit TrueForm - result = TrueForm - else: - dest.add lit FalseForm - result = FalseForm - elif childCount == 1: - setLen dest, initialLen - dest.add lit TrueForm - result = TrueForm - - of ExactlyOneOfForm: - let initialLen = dest.len - var childCount = 0 - var trueCount = 0 - for child in sons(dest, source, n): - let oldLen = dest.len - - let inner = simplify(dest, source, child, sol) - # ignore 'exactlyOneOf F' subexpressions: - if inner == FalseForm: - setLen dest, oldLen - else: - if inner == TrueForm: - inc trueCount - inc childCount - - if trueCount >= 2: - setLen dest, initialLen - dest.add lit FalseForm - result = FalseForm - elif trueCount == childCount: - setLen dest, initialLen - if trueCount != 1: - dest.add lit FalseForm - result = FalseForm - else: - dest.add lit TrueForm - result = TrueForm - elif childCount == 1: - for i in initialLen..= 0 - assert n.int < f.len - case f[n.int].kind - of FalseForm: result = false - of TrueForm: result = true - of VarForm: - let v = varId(f[n.int]) - result = s.isTrue(v) - else: - case f[n.int].kind - of AndForm: - for child in sonsReadonly(f, n): - if not eval(f, child, s): return false - return true - of OrForm: - for child in sonsReadonly(f, n): - if eval(f, child, s): return true - return false - of ExactlyOneOfForm: - var conds = 0 - for child in sonsReadonly(f, n): - if eval(f, child, s): inc conds - result = conds == 1 - of NotForm: - for child in sonsReadonly(f, n): - if not eval(f, child, s): return true - return false - of ZeroOrOneOfForm: - var conds = 0 - for child in sonsReadonly(f, n): - if eval(f, child, s): inc conds - result = conds <= 1 - else: assert false, "cannot happen" - -proc eval*(f: Formular; s: Solution): bool = - eval(f, FormPos(0), s) - -proc trivialVars(f: Formular; n: FormPos; val: uint64; sol: var Solution) = - case f[n.int].kind - of FalseForm, TrueForm: discard - of VarForm: - let v = varId(f[n.int]) - sol.setVar(v, val or sol.getVar(v)) - of NotForm: - let newVal = if val == SetToFalse: SetToTrue else: SetToFalse - trivialVars(f, n.firstSon, newVal, sol) - of OrForm: - if val == SetToTrue: - # XXX We assume here that it's an implication. We should test that instead: - let (a, b) = sons2(f, n) - if eval(f, a.firstSon, sol): - trivialVars(f, b, SetToTrue, sol) - of AndForm: - if val == SetToTrue: - for child in sonsReadonly(f, n): - trivialVars(f, child, val, sol) - of ExactlyOneOfForm, ZeroOrOneOfForm: - if val == SetToTrue: - var trueAt = -1 - for ch in sonsReadonly(f, n): - if f[ch.int].kind == VarForm: - let v = varId(f[ch.int]) - if sol.getVar(v) == SetToTrue: - trueAt = ch.int - if trueAt >= 0: - # All others must be false: - for ch in sonsReadonly(f, n): - if f[ch.int].kind == VarForm: - let v = varId(f[ch.int]) - if ch.int != trueAt: - sol.setVar(v, SetToFalse or sol.getVar(v)) - -proc satisfiable*(f: Formular; sout: var Solution): bool = - let v = freeVariable(f) - if v == NoVar: - result = f[0].kind == TrueForm - else: - var s = sout - trivialVars(f, FormPos(0), SetToTrue, s) - if containsInvalid(s): - sout = s - return false - - result = false - # We have a variable to guess. - # Construct the two guesses. - # Return whether either one of them works. - let prevValue = s.getVar(v) - s.setVar(v, SetToFalse) - - var falseGuess: Formular - let res = simplify(falseGuess, f, FormPos 0, s) - - if res == TrueForm: - result = true - else: - result = satisfiable(falseGuess, s) - if not result: - s.setVar(v, SetToTrue) - - var trueGuess: Formular - let res = simplify(trueGuess, f, FormPos 0, s) - - if res == TrueForm: - result = true - else: - result = satisfiable(trueGuess, s) - #if not result: - # Revert the assignment after trying the second option - # s.setVar(v, prevValue) - if result: - sout = s - -type - Space = seq[Solution] - -proc mul(a, b: Space): Space = - result = @[] - for i in 0..= 0 - assert n.int < f.len - result = @[] - case f[n.int].kind - of FalseForm: - # We want `true` but got `false`: - if wanted == SetToTrue: - result.add createSolution(maxVar) - result[0].invalid = true - of TrueForm: - # We want `false` but got `true`: - if wanted == SetToFalse: - result.add createSolution(maxVar) - result[0].invalid = true - of VarForm: - if wanted == DontCare: return result - result.add createSolution(maxVar) - let v = varId(f[n.int]) - result[0].setVar(v, wanted) - - of AndForm: - case wanted - of SetToFalse: - assert false, "not yet implemented: ~(& ...)" - of SetToTrue: - result.add createSolution(maxVar) - for child in sonsReadonly(f, n): - let inner = solutionSpace(f, child, maxVar, SetToTrue) - result = mul(result, inner) - else: - discard "well we don't care about the value for the AND expression" - - of OrForm: - case wanted - of SetToFalse: - # ~(A | B) == ~A & ~B - # all children must be false: - result.add createSolution(maxVar) - for child in sonsReadonly(f, n): - let inner = solutionSpace(f, child, maxVar, SetToFalse) - result = mul(result, inner) - of SetToTrue: - # any of the children need to be true: - for child in sonsReadonly(f, n): - let inner = solutionSpace(f, child, maxVar, SetToTrue) - for a in inner: result.add a - else: - discard "well we don't care about the value for the OR expression" - - of ExactlyOneOfForm: - if wanted == DontCare: return - assert wanted == SetToTrue - var children: seq[FormPos] = @[] - for child in sonsReadonly(f, n): children.add child - for i in 0..=") - else: - quit "unknown operator: " & s[result] - else: - quit "( expected, but got: " & s[result] - -when isMainModule: - proc main = - var b: Builder - b.openOpr(AndForm) - - b.openOpr(OrForm) - b.add newVar(VarId 1) - b.add newVar(VarId 2) - b.add newVar(VarId 3) - b.add newVar(VarId 4) - b.closeOpr - - b.openOpr(ExactlyOneOfForm) - b.add newVar(VarId 5) - b.add newVar(VarId 6) - b.add newVar(VarId 7) - - #b.openOpr(NotForm) - b.add newVar(VarId 8) - #b.closeOpr - b.closeOpr - - b.add newVar(VarId 5) - b.add newVar(VarId 6) - b.closeOpr - - let f = toForm(b) - echo "original: " - echo f - - let m = maxVariable(f) - var s = createSolution(m) - echo "is solvable? ", satisfiable(f, s) - echo "solution" - for i in 0..v0 (1==v6)) (<->v1 (1==v7 v8)) (<->v2 (1==v9 v10)) (<->v3 (1==v11)) (<->v4 (1==v12 v13)) (<->v14 (1==v8 v7)) (<->v15 (1==v9)) (<->v16 (1==v10 v9)) (<->v17 (1==v11)) (<->v18 (1==v11)) (<->v19 (1==v13)) (|(~v6) v14) (|(~v7) v15) (|(~v8) v16) (|(~v9) v17) (|(~v10) v18) (|(~v11) v19) (|(~v12) v20))""" - myFormular = """(&(1==v0) (1==v1) (1>=v2 v3 v4 v5 v6 v7 v8 v9 v10 v11 v12 v13) -(1>=v14 v15 v16 v17 v18 v19 v20 v21 v22 v23 v24 v25 v26 v27 v28) -(1>=v29 v30 v31) (1>=v32) -(1>=v33 v34 v35 v36 v37 v38 v39 v40 v41 v42 v43 v44 v45 v46 v47 v48 v49 v50 v51 v52 v53 v54 v55 v56 v57 v58) -(1>=v59 v60 v61 v62) -(1>=v63 v64 v65 v66 v67 v68 v69 v70 v71 v72 v73 v74) (1>=v75) (1>=v76 v77) -(1>=v78 v79 v80 v81 v82 v83 v84 v85 v86 v87 v88 v89 v90 v91 v92 v93 v94 v95 v96 v97 v98 v99 v100 v101 v102 v103 v104 v105 v106 v107 v108 v109 v110 v111 v112 v113 v114 v115 v116 v117 v118 v119 v120 v121 v122 v123 v124 v125 v126 v127 v128 v129 v130 v131 v132 v133 v134 v135 v136 v137 v138 v139 v140 v141 v142 v143 v144 v145 v146 v147 v148 v149 v150 v151 v152 v153 v154 v155) (1>=v156 v157 v158 v159 v160 v161 v162 v163 v164 v165 v166 v167 v168 v169 v170 v171 v172) (1>=v173 v174 v175 v176 v177 v178 v179 v180 v181 v182 v183 v184 v185 v186 v187 v188) -(1>=v189) (1>=v190) -(1>=v191 v192 v193 v194 v195 v196 v197 v198 v199 v200 v201 v202) -(1>=v203 v204 v205 v206 v207 v208 v209) -(|(~v210) (1==v1)) (|(~v211) (&(1==v2) (1==v14) (1==v29) (1==v32) (1==v33) (1==v59) -(1==v63) (1==v75) (1==v76) (1==v78) (1==v156) (1==v173))) (|(~v212) (1==v189)) -(|(~v214) (1==v190)) (|(~v215) (&(1==v202 v201 v200 v199 v198 v197 v196 v195 v194 v193 v192 v191) -(1==v204 v203))) (|(~v216) (&(1==v202 v201 v200 v199 v198 v197 v196 v195 v194 v193 v192 v191) -(1==v209 v208 v207 v206 v205 v204 v203))) (|(~v217) (1==v76)) (|(~v0) v210) (|(~v1) v211) -(|(~v2) v212) (|(~v4) v212) (|(~v5) v212) (|(~v6) v212) (|(~v7) v214) (|(~v8) v214) -(|(~v9) v214) (|(~v14) v215) (|(~v15) v215) (|(~v16) v215) (|(~v17) v215) (|(~v18) v215) -(|(~v19) v215) (|(~v20) v215) (|(~v21) v216) (|(~v22) v216) (|(~v75) v217))""" - - mySol = @[ - SetToTrue, #v0 - SetToFalse, #v1 - SetToTrue, #v2 - SetToFalse, #v3 - SetToTrue, #v4 - SetToTrue, #v5 - SetToFalse, #v6 - SetToTrue, #v7 - SetToTrue, #v8 - SetToFalse, #v9 - SetToTrue, # v10 - SetToFalse, # v11 - SetToTrue, # v12 - SetToTrue, # v13 - SetToFalse, - SetToFalse, - SetToFalse, - SetToFalse, - SetToFalse, - SetToFalse, - SetToFalse - ] - - proc main3() = - var b: Builder - - discard parseFormular(myFormular, 0, b) - - let f = toForm(b) - echo "original: " - echo f - - var s = createSolution(f) - echo "is solvable? ", satisfiable(f, s) - - echo "SOLUTION" - let max = maxVariable(f) - for i in 0..