pkgsublist.cc 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. /*
  2. * dselect - Debian package maintenance user interface
  3. * pkgsublist.cc - status modification and recursive package list handling
  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/i18n.h>
  26. #include <dpkg/dpkg.h>
  27. #include <dpkg/dpkg-db.h>
  28. #include "dselect.h"
  29. #include "bindings.h"
  30. void packagelist::add(pkginfo *pkg) {
  31. if (debug) fprintf(debug,"packagelist[%p]::add(pkginfo %s)\n",this,pkg->name);
  32. if (!recursive || // never add things to top level
  33. !pkg->clientdata || // don't add pure virtual packages
  34. pkg->clientdata->uprec) // don't add ones already in the recursive list
  35. return;
  36. if (debug) fprintf(debug,"packagelist[%p]::add(pkginfo %s) adding\n",this,pkg->name);
  37. perpackagestate *state= &datatable[nitems];
  38. state->pkg= pkg;
  39. state->direct= state->original= pkg->clientdata->selected;
  40. state->suggested= state->selected= pkg->clientdata->selected;
  41. state->spriority= sp_inherit; state->dpriority= dp_none;
  42. state->uprec= pkg->clientdata;
  43. state->relations.init();
  44. pkg->clientdata= state;
  45. table[nitems]= state;
  46. nitems++;
  47. }
  48. void packagelist::add(pkginfo *pkg, pkginfo::pkgwant nw) {
  49. if (debug) fprintf(debug,"packagelist[%p]::add(pkginfo %s, %s)\n",
  50. this, pkg->name, wantstrings[nw]);
  51. add(pkg); if (!pkg->clientdata) return;
  52. pkg->clientdata->direct= nw;
  53. selpriority np;
  54. np= would_like_to_install(nw,pkg) ? sp_selecting : sp_deselecting;
  55. if (pkg->clientdata->spriority > np) return;
  56. if (debug) fprintf(debug,"packagelist[%p]::add(pkginfo %s, %s) setting\n",
  57. this, pkg->name, wantstrings[nw]);
  58. pkg->clientdata->suggested= pkg->clientdata->selected= nw;
  59. pkg->clientdata->spriority= np;
  60. }
  61. void packagelist::add(pkginfo *pkg, const char *extrainfo, showpriority showimp) {
  62. if (debug)
  63. fprintf(debug,"packagelist[%p]::add(pkginfo %s, \"...\", showpriority %d)\n",
  64. this,pkg->name,showimp);
  65. add(pkg); if (!pkg->clientdata) return;
  66. if (pkg->clientdata->dpriority < showimp) pkg->clientdata->dpriority= showimp;
  67. pkg->clientdata->relations(extrainfo);
  68. pkg->clientdata->relations.terminate();
  69. }
  70. int packagelist::alreadydone(doneent **done, void *check) {
  71. doneent *search = *done;
  72. while (search && search->dep != check)
  73. search = search->next;
  74. if (search) return 1;
  75. if (debug) fprintf(debug,"packagelist[%p]::alreadydone(%p,%p) new\n",
  76. this,done,check);
  77. search= new doneent;
  78. search->next= *done;
  79. search->dep= check;
  80. *done= search;
  81. return 0;
  82. }
  83. void packagelist::addunavailable(deppossi *possi) {
  84. if (debug) fprintf(debug,"packagelist[%p]::addunavail(%p)\n",this,possi);
  85. if (!recursive) return;
  86. if (alreadydone(&unavdone,possi)) return;
  87. assert(possi->up->up->clientdata);
  88. assert(possi->up->up->clientdata->uprec);
  89. varbuf& vb= possi->up->up->clientdata->relations;
  90. vb(possi->ed->name);
  91. vb(_(" does not appear to be available\n"));
  92. }
  93. int packagelist::add(dependency *depends, showpriority displayimportance) {
  94. if (debug) fprintf(debug,"packagelist[%p]::add(dependency[%p])\n",this,depends);
  95. if (alreadydone(&depsdone,depends)) return 0;
  96. const char *comma= "";
  97. varbuf info;
  98. info(depends->up->name);
  99. info(' ');
  100. info(gettext(relatestrings[depends->type]));
  101. info(' ');
  102. deppossi *possi;
  103. for (possi=depends->list;
  104. possi;
  105. possi=possi->next, comma=(possi && possi->next ? ", " : _(" or "))) {
  106. info(comma);
  107. info(possi->ed->name);
  108. if (possi->verrel != dvr_none) {
  109. switch (possi->verrel) {
  110. case dvr_earlierequal: info(" (<= "); break;
  111. case dvr_laterequal: info(" (>= "); break;
  112. case dvr_earlierstrict: info(" (<< "); break;
  113. case dvr_laterstrict: info(" (>> "); break;
  114. case dvr_exact: info(" (= "); break;
  115. default: internerr("unknown verrel");
  116. }
  117. info(versiondescribe(&possi->version, vdew_nonambig));
  118. info(")");
  119. }
  120. }
  121. info('\n');
  122. add(depends->up,info.string(),displayimportance);
  123. for (possi=depends->list; possi; possi=possi->next) {
  124. add(possi->ed,info.string(),displayimportance);
  125. if (possi->verrel == dvr_none && depends->type != dep_provides) {
  126. // providers aren't relevant if a version was specified, or
  127. // if we're looking at a provider relationship already
  128. deppossi *provider;
  129. for (provider= possi->ed->available.valid ? possi->ed->available.depended : 0;
  130. provider;
  131. provider=provider->nextrev) {
  132. if (provider->up->type != dep_provides) continue;
  133. add(provider->up->up,info.string(),displayimportance);
  134. add(provider->up,displayimportance);
  135. }
  136. }
  137. }
  138. return 1;
  139. }
  140. void repeatedlydisplay(packagelist *sub,
  141. showpriority initial,
  142. packagelist *unredisplay) {
  143. pkginfo **newl;
  144. keybindings *kb;
  145. if (debug) fprintf(debug,"repeatedlydisplay(packagelist[%p])\n",sub);
  146. if (sub->resolvesuggest() != 0 && sub->deletelessimp_anyleft(initial)) {
  147. if (debug) fprintf(debug,"repeatedlydisplay(packagelist[%p]) once\n",sub);
  148. if (unredisplay) unredisplay->enddisplay();
  149. for (;;) {
  150. manual_install = 0; /* Remove flag now that resolvesuggest has seen it. */
  151. newl= sub->display();
  152. if (!newl) break;
  153. if (debug) fprintf(debug,"repeatedlydisplay(packagelist[%p]) newl\n",sub);
  154. kb= sub->bindings; delete sub;
  155. sub= new packagelist(kb,newl);
  156. if (sub->resolvesuggest() <= 1) break;
  157. if (!sub->deletelessimp_anyleft(dp_must)) break;
  158. if (debug) fprintf(debug,"repeatedlydisplay(packagelist[%p]) again\n",sub);
  159. }
  160. if (unredisplay) unredisplay->startdisplay();
  161. }
  162. delete sub;
  163. if (debug) fprintf(debug,"repeatedlydisplay(packagelist[%p]) done\n",sub);
  164. }
  165. int packagelist::deletelessimp_anyleft(showpriority than) {
  166. if (debug)
  167. fprintf(debug,"packagelist[%p]::dli_al(%d): nitems=%d\n",this,than,nitems);
  168. int insat, runthr;
  169. for (runthr=0, insat=0;
  170. runthr < nitems;
  171. runthr++) {
  172. if (table[runthr]->dpriority < than) {
  173. table[runthr]->free(recursive);
  174. } else {
  175. if (insat != runthr) table[insat]= table[runthr];
  176. insat++;
  177. }
  178. }
  179. nitems= insat;
  180. if (debug) fprintf(debug,"packagelist[%p]::dli_al(%d) done; nitems=%d\n",
  181. this,than,nitems);
  182. return nitems;
  183. }