pkgdepcon.cc 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. /*
  2. * dselect - Debian package maintenance user interface
  3. * pkgdepcon.cc - dependency and conflict resolution
  4. *
  5. * Copyright © 1995 Ian Jackson <ian@chiark.greenend.org.uk>
  6. *
  7. * This is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as
  9. * published by the Free Software Foundation; either version 2,
  10. * or (at your option) any later version.
  11. *
  12. * This is distributed in the hope that it will be useful, but
  13. * WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. */
  20. #include <config.h>
  21. #include <compat.h>
  22. #include <assert.h>
  23. #include <string.h>
  24. #include <stdio.h>
  25. #include <dpkg/dpkg.h>
  26. #include <dpkg/dpkg-db.h>
  27. #include "dselect.h"
  28. #include "pkglist.h"
  29. static const int depdebug= 1;
  30. int packagelist::useavailable(pkginfo *pkg) {
  31. if (pkg->clientdata &&
  32. pkg->clientdata->selected == pkginfo::want_install &&
  33. informative(pkg,&pkg->available) &&
  34. (!(pkg->status == pkginfo::stat_installed ||
  35. pkg->status == pkginfo::stat_triggersawaited ||
  36. pkg->status == pkginfo::stat_triggerspending) ||
  37. versioncompare(&pkg->available.version,&pkg->installed.version) > 0))
  38. return 1;
  39. else
  40. return 0;
  41. }
  42. pkginfoperfile *packagelist::findinfo(pkginfo *pkg) {
  43. pkginfoperfile *r;
  44. r= useavailable(pkg) ? &pkg->available : &pkg->installed;
  45. if (debug)
  46. fprintf(debug,"packagelist[%p]::findinfo(%s) useavailable=%d\n",this,pkg->name,useavailable(pkg));
  47. if (!r->valid) blankpackageperfile(r);
  48. return r;
  49. }
  50. int packagelist::checkdependers(pkginfo *pkg, int changemade) {
  51. struct deppossi *possi;
  52. if (pkg->available.valid) {
  53. for (possi= pkg->available.depended; possi; possi= possi->nextrev) {
  54. if (!useavailable(possi->up->up)) continue;
  55. changemade= greaterint(changemade,resolvedepcon(possi->up));
  56. }
  57. }
  58. if (pkg->installed.valid) {
  59. for (possi= pkg->installed.depended; possi; possi= possi->nextrev) {
  60. if (useavailable(possi->up->up)) continue;
  61. changemade= greaterint(changemade,resolvedepcon(possi->up));
  62. }
  63. }
  64. return changemade;
  65. }
  66. int packagelist::resolvesuggest() {
  67. // We continually go around looking for things to change, but we may
  68. // only change the `suggested' value if we also increase the `priority'
  69. // Return 2 if we made a change due to a Recommended, Depends or Conficts,
  70. // or 1 if we offered or made a change because of an Optional line.
  71. if (debug)
  72. fprintf(debug,"packagelist[%p]::resolvesuggest()\n",this);
  73. int changemade, maxchangemade;
  74. maxchangemade= 0;
  75. for (;;) {
  76. changemade= 0;
  77. int index;
  78. for (index=0; index<nitems; index++) {
  79. if (!table[index]->pkg->name) continue;
  80. if (depdebug && debug)
  81. fprintf(debug,"packagelist[%p]::resolvesuggest() loop[%i] %s / %d\n",
  82. this, index, table[index]->pkg->name, changemade);
  83. dependency *depends;
  84. for (depends= findinfo(table[index]->pkg)->depends;
  85. depends;
  86. depends= depends->next) {
  87. changemade= greaterint(changemade,resolvedepcon(depends));
  88. }
  89. changemade= checkdependers(table[index]->pkg,changemade);
  90. for (depends= findinfo(table[index]->pkg)->depends;
  91. depends;
  92. depends= depends->next) {
  93. if (depends->type != dep_provides) continue;
  94. changemade= checkdependers(depends->list->ed,changemade);
  95. }
  96. if (depdebug && debug)
  97. fprintf(debug,"packagelist[%p]::resolvesuggest() loop[%i] %s / -> %d\n",
  98. this, index, table[index]->pkg->name, changemade);
  99. }
  100. if (!changemade) break;
  101. maxchangemade= greaterint(maxchangemade, changemade);
  102. }
  103. if (debug)
  104. fprintf(debug,"packagelist[%p]::resolvesuggest() done; maxchangemade=%d\n",
  105. this,maxchangemade);
  106. return maxchangemade;
  107. }
  108. static int dep_update_best_to_change_stop(perpackagestate *& best, pkginfo *trythis) {
  109. // There's no point trying to select a pure virtual package.
  110. if (!trythis->clientdata) return 0;
  111. if (depdebug && debug)
  112. fprintf(debug,"update_best_to_change(best=%s{%d}, test=%s{%d});\n",
  113. best ? best->pkg->name : "", best ? (int)best->spriority : -1,
  114. trythis->name, trythis->clientdata->spriority);
  115. // If the problem is caused by us deselecting one of these packages
  116. // we should not try to select another one instead.
  117. if (trythis->clientdata->spriority == sp_deselecting) return 1;
  118. // If we haven't found anything yet then this is our best so far.
  119. if (!best) goto yes;
  120. // If only one of the packages is available, use that one
  121. if (!informative(trythis,&trythis->available) &&
  122. informative(best->pkg,&best->pkg->available)) return 0;
  123. if (informative(trythis,&trythis->available) &&
  124. !informative(best->pkg,&best->pkg->available)) goto yes;
  125. // Select the package with the lowest priority (ie, the one of whom
  126. // we were least sure we wanted it deselected).
  127. if (trythis->clientdata->spriority > best->spriority) return 0;
  128. if (trythis->clientdata->spriority < best->spriority) goto yes;
  129. // Pick the package with the must fundamental recommendation level.
  130. if (trythis->priority > best->pkg->priority) return 0;
  131. if (trythis->priority < best->pkg->priority) goto yes;
  132. // If we're still unsure we'll change the first one in the list.
  133. return 0;
  134. yes:
  135. if (depdebug && debug) fprintf(debug,"update_best_to_change(); yes\n");
  136. best=trythis->clientdata; return 0;
  137. }
  138. int packagelist::deselect_one_of(pkginfo *per, pkginfo *ped, dependency *display) {
  139. perpackagestate *er= per->clientdata;
  140. perpackagestate *ed= ped->clientdata;
  141. if (!er || !would_like_to_install(er->selected,per) ||
  142. !ed || !would_like_to_install(ed->selected,ped)) return 0;
  143. add(display,dp_must);
  144. er= per->clientdata; // these can be changed by add
  145. ed= ped->clientdata;
  146. if (depdebug && debug)
  147. fprintf(debug,"packagelist[%p]::deselect_one_of(): er %s{%d} ed %s{%d} [%p]\n",
  148. this, er->pkg->name, er->spriority, ed->pkg->name, ed->spriority, display);
  149. perpackagestate *best;
  150. // Try not keep packages needing reinstallation.
  151. if (per->eflag & pkginfo::eflag_reinstreq)
  152. best = ed;
  153. else if (ped->eflag & pkginfo::eflag_reinstreq)
  154. best = er;
  155. else if (er->spriority < ed->spriority) best= er; // We'd rather change the
  156. else if (er->spriority > ed->spriority) best= ed; // one with the lowest priority.
  157. // ... failing that the one with the highest priority
  158. else if (er->pkg->priority > ed->pkg->priority)
  159. best = er;
  160. else if (er->pkg->priority < ed->pkg->priority)
  161. best = ed;
  162. else best= ed; // ... failing that, the second
  163. if (depdebug && debug)
  164. fprintf(debug,"packagelist[%p]::deselect_one_of(): best %s{%d}\n",
  165. this, best->pkg->name, best->spriority);
  166. if (best->spriority >= sp_deselecting) return 0;
  167. best->suggested=
  168. best->pkg->status == pkginfo::stat_notinstalled
  169. ? pkginfo::want_purge : pkginfo::want_deinstall; // FIXME: configurable.
  170. best->selected= best->suggested;
  171. best->spriority= sp_deselecting;
  172. return 2;
  173. }
  174. int packagelist::resolvedepcon(dependency *depends) {
  175. perpackagestate *best, *fixbyupgrade;
  176. deppossi *possi, *provider;
  177. int r, foundany;
  178. if (depdebug && debug) {
  179. fprintf(debug,"packagelist[%p]::resolvedepcon([%p] %s --%s-->",
  180. this, depends, depends->up->name, relatestrings[depends->type]);
  181. for (possi=depends->list; possi; possi=possi->next)
  182. fprintf(debug," %s",possi->ed->name);
  183. fprintf(debug,"); (ing)->want=%s\n",
  184. depends->up->clientdata
  185. ? wantstrings[depends->up->clientdata->suggested]
  186. : "(no clientdata)");
  187. }
  188. if (!depends->up->clientdata) return 0;
  189. switch (depends->type) {
  190. case dep_provides:
  191. case dep_replaces:
  192. return 0;
  193. case dep_enhances:
  194. case dep_suggests:
  195. case dep_recommends:
  196. case dep_depends:
  197. case dep_predepends:
  198. if (would_like_to_install(depends->up->clientdata->selected,depends->up) <= 0)
  199. return 0;
  200. fixbyupgrade= 0;
  201. possi = depends->list;
  202. while (possi && !deppossatisfied(possi, &fixbyupgrade))
  203. possi = possi->next;
  204. if (depdebug && debug)
  205. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): depends found %s\n",
  206. this,depends,
  207. possi ? possi->ed->name : "[none]");
  208. if (possi) return 0;
  209. // Ensures all in the recursive list; adds info strings; ups priorities
  210. switch (depends->type) {
  211. case dep_enhances:
  212. case dep_suggests:
  213. r= add(depends, dp_may);
  214. return r;
  215. case dep_recommends:
  216. r= add(depends, dp_should);
  217. break;
  218. default:
  219. r= add(depends, dp_must);
  220. }
  221. if (fixbyupgrade) {
  222. if (depdebug && debug) fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): "
  223. "fixbyupgrade %s\n", this,depends,fixbyupgrade->pkg->name);
  224. best= fixbyupgrade;
  225. } else {
  226. best= 0;
  227. for (possi= depends->list;
  228. possi;
  229. possi= possi->next) {
  230. foundany= 0;
  231. if (possi->ed->clientdata) foundany= 1;
  232. if (dep_update_best_to_change_stop(best, possi->ed)) goto mustdeselect;
  233. for (provider= possi->ed->available.valid ? possi->ed->available.depended : 0;
  234. provider;
  235. provider= provider->nextrev) {
  236. if (provider->up->type != dep_provides) continue;
  237. if (provider->up->up->clientdata) foundany= 1;
  238. if (dep_update_best_to_change_stop(best, provider->up->up)) goto mustdeselect;
  239. }
  240. if (!foundany) addunavailable(possi);
  241. }
  242. if (!best) {
  243. if (depdebug && debug) fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): "
  244. "mustdeselect nobest\n", this,depends);
  245. return r;
  246. }
  247. }
  248. if (depdebug && debug)
  249. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): select best=%s{%d}\n",
  250. this,depends, best->pkg->name, best->spriority);
  251. if (best->spriority >= sp_selecting) return r;
  252. /* Always select depends. Only select recommends if we got here because
  253. * of a manually-initiated install request. */
  254. if (depends->type != dep_recommends || manual_install) {
  255. best->selected= best->suggested= pkginfo::want_install;
  256. best->spriority= sp_selecting;
  257. }
  258. return r ? 2 : 0;
  259. mustdeselect:
  260. best= depends->up->clientdata;
  261. if (depdebug && debug)
  262. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): mustdeselect best=%s{%d}\n",
  263. this,depends, best->pkg->name, best->spriority);
  264. if (best->spriority >= sp_deselecting) return r;
  265. /* Always remove depends, but never remove recommends. */
  266. if (depends->type != dep_recommends) {
  267. best->selected= best->suggested=
  268. best->pkg->status == pkginfo::stat_notinstalled
  269. ? pkginfo::want_purge : pkginfo::want_deinstall; // FIXME: configurable
  270. best->spriority= sp_deselecting;
  271. }
  272. return r ? 2 : 0;
  273. case dep_conflicts:
  274. case dep_breaks:
  275. if (depdebug && debug)
  276. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): conflict\n",
  277. this,depends);
  278. if (would_like_to_install(depends->up->clientdata->selected,depends->up) == 0)
  279. return 0;
  280. if (depdebug && debug)
  281. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): conflict installing 1\n",
  282. this,depends);
  283. if (!deppossatisfied(depends->list,0)) return 0;
  284. if (depdebug && debug)
  285. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): conflict satisfied - ouch\n",
  286. this,depends);
  287. if (depends->up != depends->list->ed) {
  288. r= deselect_one_of(depends->up, depends->list->ed, depends); if (r) return r;
  289. }
  290. for (provider= depends->list->ed->available.valid ?
  291. depends->list->ed->available.depended : 0;
  292. provider;
  293. provider= provider->nextrev) {
  294. if (provider->up->type != dep_provides) continue;
  295. if (provider->up->up == depends->up) continue; // conflicts & provides same thing
  296. r= deselect_one_of(depends->up, provider->up->up, depends); if (r) return r;
  297. }
  298. if (depdebug && debug)
  299. fprintf(debug,"packagelist[%p]::resolvedepcon([%p]): no desel\n", this,depends);
  300. return 0;
  301. default:
  302. internerr("unknown deptype");
  303. }
  304. /* never reached, make gcc happy */
  305. return 1;
  306. }
  307. int packagelist::deppossatisfied(deppossi *possi, perpackagestate **fixbyupgrade) {
  308. // `satisfied' here for Conflicts and Breaks means that the
  309. // restriction is violated ie that the target package is wanted
  310. int would;
  311. pkginfo::pkgwant want= pkginfo::want_purge;
  312. if (possi->ed->clientdata) {
  313. want= possi->ed->clientdata->selected;
  314. would= would_like_to_install(want,possi->ed);
  315. } else {
  316. would= 0;
  317. }
  318. if ((possi->up->type == dep_conflicts || possi->up->type == dep_breaks)
  319. ? possi->up->up != possi->ed && would != 0
  320. : would > 0) {
  321. // If it's to be installed or left installed, then either it's of
  322. // the right version, and therefore OK, or a version must have
  323. // been specified, in which case we don't need to look at the rest
  324. // anyway.
  325. if (useavailable(possi->ed)) {
  326. assert(want == pkginfo::want_install);
  327. return versionsatisfied(&possi->ed->available,possi);
  328. } else {
  329. if (versionsatisfied(&possi->ed->installed,possi)) return 1;
  330. if (want == pkginfo::want_hold && fixbyupgrade && !*fixbyupgrade &&
  331. versionsatisfied(&possi->ed->available,possi) &&
  332. versioncompare(&possi->ed->available.version,
  333. &possi->ed->installed.version) > 1)
  334. *fixbyupgrade= possi->ed->clientdata;
  335. return 0;
  336. }
  337. }
  338. if (possi->verrel != dvr_none) return 0;
  339. deppossi *provider;
  340. if (possi->ed->installed.valid) {
  341. for (provider= possi->ed->installed.depended;
  342. provider;
  343. provider= provider->nextrev) {
  344. if (provider->up->type == dep_provides &&
  345. provider->up->up->clientdata &&
  346. !useavailable(provider->up->up) &&
  347. would_like_to_install(provider->up->up->clientdata->selected,
  348. provider->up->up))
  349. return 1;
  350. }
  351. }
  352. if (possi->ed->available.valid) {
  353. for (provider= possi->ed->available.depended;
  354. provider;
  355. provider= provider->nextrev) {
  356. if (provider->up->type != dep_provides ||
  357. !provider->up->up->clientdata ||
  358. !would_like_to_install(provider->up->up->clientdata->selected,
  359. provider->up->up))
  360. continue;
  361. if (useavailable(provider->up->up))
  362. return 1;
  363. if (fixbyupgrade && !*fixbyupgrade &&
  364. (!(provider->up->up->status == pkginfo::stat_installed ||
  365. provider->up->up->status == pkginfo::stat_triggerspending ||
  366. provider->up->up->status == pkginfo::stat_triggersawaited) ||
  367. versioncompare(&provider->up->up->available.version,
  368. &provider->up->up->installed.version) > 1))
  369. *fixbyupgrade= provider->up->up->clientdata;
  370. }
  371. }
  372. return 0;
  373. }