1 // -*- mode: cpp; mode: fold -*- 
   3 // $Id: depcache.cc,v 1.25 2001/05/27 05:36:04 jgg Exp $ 
   4 /* ###################################################################### 
   6    Dependency Cache - Caches Dependency information. 
   8    ##################################################################### */ 
  10 // Include Files                                                        /*{{{*/ 
  13 #include <apt-pkg/depcache.h> 
  14 #include <apt-pkg/version.h> 
  15 #include <apt-pkg/versionmatch.h> 
  16 #include <apt-pkg/error.h> 
  17 #include <apt-pkg/sptr.h> 
  18 #include <apt-pkg/algorithms.h> 
  19 #include <apt-pkg/fileutl.h> 
  20 #include <apt-pkg/strutl.h> 
  21 #include <apt-pkg/configuration.h> 
  22 #include <apt-pkg/aptconfiguration.h> 
  23 #include <apt-pkg/pkgsystem.h> 
  24 #include <apt-pkg/tagfile.h> 
  25 #include <apt-pkg/progress.h> 
  26 #include <apt-pkg/cacheset.h> 
  40 // helper for Install-Recommends-Sections and Never-MarkAuto-Sections   /*{{{*/ 
  42 ConfigValueInSubTree(const char* SubTree
, const char *needle
) 
  44    Configuration::Item 
const *Opts
; 
  45    Opts 
= _config
->Tree(SubTree
); 
  46    if (Opts 
!= 0 && Opts
->Child 
!= 0) 
  49       for (; Opts 
!= 0; Opts 
= Opts
->Next
) 
  51          if (Opts
->Value
.empty() == true) 
  53          if (strcmp(needle
, Opts
->Value
.c_str()) == 0) 
  60 pkgDepCache::ActionGroup::ActionGroup(pkgDepCache 
&cache
) :             /*{{{*/ 
  61   cache(cache
), released(false) 
  66 void pkgDepCache::ActionGroup::release() 
  70       if(cache
.group_level 
== 0) 
  71         std::cerr 
<< "W: Unbalanced action groups, expect badness" << std::endl
; 
  76           if(cache
.group_level 
== 0) 
  84 pkgDepCache::ActionGroup::~ActionGroup() 
  89 // DepCache::pkgDepCache - Constructors                                 /*{{{*/ 
  90 // --------------------------------------------------------------------- 
  92 pkgDepCache::pkgDepCache(pkgCache 
*pCache
,Policy 
*Plcy
) : 
  93   group_level(0), Cache(pCache
), PkgState(0), DepState(0) 
  95    DebugMarker 
= _config
->FindB("Debug::pkgDepCache::Marker", false); 
  96    DebugAutoInstall 
= _config
->FindB("Debug::pkgDepCache::AutoInstall", false); 
 100       delLocalPolicy 
= LocalPolicy 
= new Policy
; 
 103 // DepCache::~pkgDepCache - Destructor                                  /*{{{*/ 
 104 // --------------------------------------------------------------------- 
 106 pkgDepCache::~pkgDepCache() 
 110    delete delLocalPolicy
; 
 113 // DepCache::Init - Generate the initial extra structures.              /*{{{*/ 
 114 // --------------------------------------------------------------------- 
 115 /* This allocats the extension buffers and initializes them. */ 
 116 bool pkgDepCache::Init(OpProgress 
*Prog
) 
 118    // Suppress mark updates during this operation (just in case) and 
 119    // run a mark operation when Init terminates. 
 120    ActionGroup 
actions(*this); 
 124    PkgState 
= new StateCache
[Head().PackageCount
]; 
 125    DepState 
= new unsigned char[Head().DependsCount
]; 
 126    memset(PkgState
,0,sizeof(*PkgState
)*Head().PackageCount
); 
 127    memset(DepState
,0,sizeof(*DepState
)*Head().DependsCount
);  
 131       Prog
->OverallProgress(0,2*Head().PackageCount
,Head().PackageCount
, 
 132                             _("Building dependency tree")); 
 133       Prog
->SubProgress(Head().PackageCount
,_("Candidate versions")); 
 136    /* Set the current state of everything. In this state all of the 
 137       packages are kept exactly as is. See AllUpgrade */ 
 139    for (PkgIterator I 
= PkgBegin(); I
.end() != true; ++I
, ++Done
) 
 141       if (Prog 
!= 0 && Done%20 
== 0) 
 142          Prog
->Progress(Done
); 
 144       // Find the proper cache slot 
 145       StateCache 
&State 
= PkgState
[I
->ID
]; 
 148       // Figure out the install version 
 149       State
.CandidateVer 
= GetCandidateVer(I
); 
 150       State
.InstallVer 
= I
.CurrentVer(); 
 151       State
.Mode 
= ModeKeep
; 
 153       State
.Update(I
,*this); 
 159       Prog
->OverallProgress(Head().PackageCount
,2*Head().PackageCount
, 
 161                             _("Building dependency tree")); 
 162       Prog
->SubProgress(Head().PackageCount
,_("Dependency generation")); 
 173 bool pkgDepCache::readStateFile(OpProgress 
*Prog
)                       /*{{{*/ 
 176    string 
const state 
= _config
->FindFile("Dir::State::extended_states"); 
 177    if(RealFileExists(state
)) { 
 178       state_file
.Open(state
, FileFd::ReadOnly
); 
 179       off_t 
const file_size 
= state_file
.Size(); 
 181          Prog
->OverallProgress(0, file_size
, 1,  
 182                                _("Reading state information")); 
 184       pkgTagFile 
tagfile(&state_file
); 
 185       pkgTagSection section
; 
 187       bool const debug_autoremove 
= _config
->FindB("Debug::pkgAutoRemove",false); 
 188       while(tagfile
.Step(section
)) { 
 189          string 
const pkgname 
= section
.FindS("Package"); 
 190          string pkgarch 
= section
.FindS("Architecture"); 
 191          if (pkgarch
.empty() == true) 
 193          pkgCache::PkgIterator pkg 
= Cache
->FindPkg(pkgname
, pkgarch
); 
 194          // Silently ignore unknown packages and packages with no actual version. 
 195          if(pkg
.end() == true || pkg
->VersionList 
== 0) 
 198          short const reason 
= section
.FindI("Auto-Installed", 0); 
 201             PkgState
[pkg
->ID
].Flags 
|= Flag::Auto
; 
 202             if (unlikely(debug_autoremove
)) 
 203                std::clog 
<< "Auto-Installed : " << pkg
.FullName() << std::endl
; 
 204             if (pkgarch 
== "any") 
 206                pkgCache::GrpIterator G 
= pkg
.Group(); 
 207                for (pkg 
= G
.NextPkg(pkg
); pkg
.end() != true; pkg 
= G
.NextPkg(pkg
)) 
 208                   if (pkg
->VersionList 
!= 0) 
 209                      PkgState
[pkg
->ID
].Flags 
|= Flag::Auto
; 
 212          amt 
+= section
.size(); 
 214             Prog
->OverallProgress(amt
, file_size
, 1,  
 215                                   _("Reading state information")); 
 218          Prog
->OverallProgress(file_size
, file_size
, 1, 
 219                                _("Reading state information")); 
 225 bool pkgDepCache::writeStateFile(OpProgress 
*prog
, bool InstalledOnly
)  /*{{{*/ 
 227    bool const debug_autoremove 
= _config
->FindB("Debug::pkgAutoRemove",false); 
 230       std::clog 
<< "pkgDepCache::writeStateFile()" << std::endl
; 
 233    string 
const state 
= _config
->FindFile("Dir::State::extended_states"); 
 235    // if it does not exist, create a empty one 
 236    if(!RealFileExists(state
))  
 238       StateFile
.Open(state
, FileFd::WriteAtomic
); 
 243    if(!StateFile
.Open(state
, FileFd::ReadOnly
)) 
 244       return _error
->Error(_("Failed to open StateFile %s"), 
 248    string 
const outfile 
= state 
+ ".tmp"; 
 249    if((OutFile 
= fopen(outfile
.c_str(),"w")) == NULL
) 
 250       return _error
->Error(_("Failed to write temporary StateFile %s"), 
 253    // first merge with the existing sections 
 254    pkgTagFile 
tagfile(&StateFile
); 
 255    pkgTagSection section
; 
 256    std::set
<string
> pkgs_seen
; 
 257    const char *nullreorderlist
[] = {0}; 
 258    while(tagfile
.Step(section
)) { 
 259          string 
const pkgname 
= section
.FindS("Package"); 
 260          string pkgarch 
= section
.FindS("Architecture"); 
 261          if (pkgarch
.empty() == true) 
 263          // Silently ignore unknown packages and packages with no actual 
 265          pkgCache::PkgIterator pkg 
= Cache
->FindPkg(pkgname
, pkgarch
); 
 266          if(pkg
.end() || pkg
.VersionList().end())  
 268          StateCache 
const &P 
= PkgState
[pkg
->ID
]; 
 269          bool newAuto 
= (P
.Flags 
& Flag::Auto
); 
 270          // skip not installed or now-removed ones if requested 
 271          if (InstalledOnly 
&& ( 
 272              (pkg
->CurrentVer 
== 0 && P
.Mode 
!= ModeInstall
) || 
 273              (pkg
->CurrentVer 
!= 0 && P
.Mode 
== ModeDelete
))) 
 275             // The section is obsolete if it contains no other tag 
 276             unsigned int const count 
= section
.Count(); 
 278                 (count 
== 2 && section
.Exists("Auto-Installed")) || 
 279                 (count 
== 3 && section
.Exists("Auto-Installed") && section
.Exists("Architecture"))) 
 284          if(_config
->FindB("Debug::pkgAutoRemove",false)) 
 285             std::clog 
<< "Update existing AutoInstall info: "  
 286                       << pkg
.FullName() << std::endl
; 
 287          TFRewriteData rewrite
[3]; 
 288          rewrite
[0].Tag 
= "Architecture"; 
 289          rewrite
[0].Rewrite 
= pkg
.Arch(); 
 290          rewrite
[0].NewTag 
= 0; 
 291          rewrite
[1].Tag 
= "Auto-Installed"; 
 292          rewrite
[1].Rewrite 
= newAuto 
? "1" : "0"; 
 293          rewrite
[1].NewTag 
= 0; 
 295          TFRewrite(OutFile
, section
, nullreorderlist
, rewrite
); 
 296          fprintf(OutFile
,"\n"); 
 297          pkgs_seen
.insert(pkg
.FullName()); 
 300    // then write the ones we have not seen yet 
 301    std::ostringstream ostr
; 
 302    for(pkgCache::PkgIterator pkg
=Cache
->PkgBegin(); !pkg
.end(); ++pkg
) { 
 303       StateCache 
const &P 
= PkgState
[pkg
->ID
]; 
 304       if(P
.Flags 
& Flag::Auto
) { 
 305          if (pkgs_seen
.find(pkg
.FullName()) != pkgs_seen
.end()) { 
 307                std::clog 
<< "Skipping already written " << pkg
.FullName() << std::endl
; 
 310          // skip not installed ones if requested 
 311          if (InstalledOnly 
&& ( 
 312              (pkg
->CurrentVer 
== 0 && P
.Mode 
!= ModeInstall
) || 
 313              (pkg
->CurrentVer 
!= 0 && P
.Mode 
== ModeDelete
))) 
 315          const char* const pkgarch 
= pkg
.Arch(); 
 316          if (strcmp(pkgarch
, "all") == 0) 
 319             std::clog 
<< "Writing new AutoInstall: " << pkg
.FullName() << std::endl
; 
 320          ostr
.str(string("")); 
 321          ostr 
<< "Package: " << pkg
.Name() 
 322               << "\nArchitecture: " << pkgarch
 
 323               << "\nAuto-Installed: 1\n\n"; 
 324          fprintf(OutFile
,"%s",ostr
.str().c_str()); 
 329    // move the outfile over the real file and set permissions 
 330    rename(outfile
.c_str(), state
.c_str()); 
 331    chmod(state
.c_str(), 0644); 
 336 // DepCache::CheckDep - Checks a single dependency                      /*{{{*/ 
 337 // --------------------------------------------------------------------- 
 338 /* This first checks the dependency against the main target package and 
 339    then walks along the package provides list and checks if each provides  
 340    will be installed then checks the provides against the dep. Res will be  
 341    set to the package which was used to satisfy the dep. */ 
 342 bool pkgDepCache::CheckDep(DepIterator Dep
,int Type
,PkgIterator 
&Res
) 
 344    Res 
= Dep
.TargetPkg(); 
 346    /* Check simple depends. A depends -should- never self match but  
 347       we allow it anyhow because dpkg does. Technically it is a packaging 
 348       bug. Conflicts may never self match */ 
 349    if (Dep
.TargetPkg() != Dep
.ParentPkg() || Dep
.IsNegative() == false) 
 351       PkgIterator Pkg 
= Dep
.TargetPkg(); 
 352       // Check the base package 
 353       if (Type 
== NowVersion 
&& Pkg
->CurrentVer 
!= 0) 
 354          if (VS().CheckDep(Pkg
.CurrentVer().VerStr(),Dep
->CompareOp
, 
 355                                  Dep
.TargetVer()) == true) 
 358       if (Type 
== InstallVersion 
&& PkgState
[Pkg
->ID
].InstallVer 
!= 0) 
 359          if (VS().CheckDep(PkgState
[Pkg
->ID
].InstVerIter(*this).VerStr(), 
 360                                  Dep
->CompareOp
,Dep
.TargetVer()) == true) 
 363       if (Type 
== CandidateVersion 
&& PkgState
[Pkg
->ID
].CandidateVer 
!= 0) 
 364          if (VS().CheckDep(PkgState
[Pkg
->ID
].CandidateVerIter(*this).VerStr(), 
 365                                  Dep
->CompareOp
,Dep
.TargetVer()) == true) 
 369    if (Dep
->Type 
== Dep::Obsoletes
) 
 372    // Check the providing packages 
 373    PrvIterator P 
= Dep
.TargetPkg().ProvidesList(); 
 374    for (; P
.end() != true; ++P
) 
 376       if (Dep
.IsIgnorable(P
) == true) 
 379       // Check if the provides is a hit 
 380       if (Type 
== NowVersion
) 
 382          if (P
.OwnerPkg().CurrentVer() != P
.OwnerVer()) 
 386       if (Type 
== InstallVersion
) 
 388          StateCache 
&State 
= PkgState
[P
.OwnerPkg()->ID
]; 
 389          if (State
.InstallVer 
!= (Version 
*)P
.OwnerVer()) 
 393       if (Type 
== CandidateVersion
) 
 395          StateCache 
&State 
= PkgState
[P
.OwnerPkg()->ID
]; 
 396          if (State
.CandidateVer 
!= (Version 
*)P
.OwnerVer()) 
 400       // Compare the versions. 
 401       if (VS().CheckDep(P
.ProvideVersion(),Dep
->CompareOp
,Dep
.TargetVer()) == true) 
 411 // DepCache::AddSizes - Add the packages sizes to the counters          /*{{{*/ 
 412 // --------------------------------------------------------------------- 
 413 /* Call with Inverse = true to preform the inverse opration */ 
 414 void pkgDepCache::AddSizes(const PkgIterator 
&Pkg
, bool const Inverse
) 
 416    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
 418    if (Pkg
->VersionList 
== 0) 
 421    if (Pkg
.State() == pkgCache::PkgIterator::NeedsConfigure 
&&  
 425    // Compute the size data 
 426    if (P
.NewInstall() == true) 
 428       if (Inverse 
== false) { 
 429          iUsrSize 
+= P
.InstVerIter(*this)->InstalledSize
; 
 430          iDownloadSize 
+= P
.InstVerIter(*this)->Size
; 
 432          iUsrSize 
-= P
.InstVerIter(*this)->InstalledSize
; 
 433          iDownloadSize 
-= P
.InstVerIter(*this)->Size
; 
 439    if (Pkg
->CurrentVer 
!= 0 &&  
 440        (P
.InstallVer 
!= (Version 
*)Pkg
.CurrentVer() ||  
 441         (P
.iFlags 
& ReInstall
) == ReInstall
) && P
.InstallVer 
!= 0) 
 443       if (Inverse 
== false) { 
 444          iUsrSize 
-= Pkg
.CurrentVer()->InstalledSize
; 
 445          iUsrSize 
+= P
.InstVerIter(*this)->InstalledSize
; 
 446          iDownloadSize 
+= P
.InstVerIter(*this)->Size
; 
 448          iUsrSize 
-= P
.InstVerIter(*this)->InstalledSize
; 
 449          iUsrSize 
+= Pkg
.CurrentVer()->InstalledSize
; 
 450          iDownloadSize 
-= P
.InstVerIter(*this)->Size
; 
 456    if (Pkg
.State() == pkgCache::PkgIterator::NeedsUnpack 
&& 
 459       if (Inverse 
== false) 
 460          iDownloadSize 
+= P
.InstVerIter(*this)->Size
; 
 462          iDownloadSize 
-= P
.InstVerIter(*this)->Size
; 
 467    if (Pkg
->CurrentVer 
!= 0 && P
.InstallVer 
== 0) 
 469       if (Inverse 
== false) 
 470          iUsrSize 
-= Pkg
.CurrentVer()->InstalledSize
; 
 472          iUsrSize 
+= Pkg
.CurrentVer()->InstalledSize
; 
 477 // DepCache::AddStates - Add the package to the state counter           /*{{{*/ 
 478 // --------------------------------------------------------------------- 
 479 /* This routine is tricky to use, you must make sure that it is never  
 480    called twice for the same package. This means the Remove/Add section 
 481    should be as short as possible and not encompass any code that will  
 482    calld Remove/Add itself. Remember, dependencies can be circular so 
 483    while processing a dep for Pkg it is possible that Add/Remove 
 484    will be called on Pkg */ 
 485 void pkgDepCache::AddStates(const PkgIterator 
&Pkg
, bool const Invert
) 
 487    signed char const Add 
= (Invert 
== false) ? 1 : -1; 
 488    StateCache 
&State 
= PkgState
[Pkg
->ID
]; 
 490    // The Package is broken (either minimal dep or policy dep) 
 491    if ((State
.DepState 
& DepInstMin
) != DepInstMin
) 
 493    if ((State
.DepState 
& DepInstPolicy
) != DepInstPolicy
) 
 494       iPolicyBrokenCount 
+= Add
; 
 497    if (Pkg
.State() != PkgIterator::NeedsNothing
) 
 501    if (Pkg
->CurrentVer 
== 0) 
 503       if (State
.Mode 
== ModeDelete 
&& 
 504           (State
.iFlags 
& Purge
) == Purge 
&& Pkg
.Purge() == false) 
 507       if (State
.Mode 
== ModeInstall
) 
 512    // Installed, no upgrade 
 513    if (State
.Status 
== 0) 
 515       if (State
.Mode 
== ModeDelete
) 
 518          if ((State
.iFlags 
& ReInstall
) == ReInstall
) 
 524    // Alll 3 are possible 
 525    if (State
.Mode 
== ModeDelete
) 
 527    if (State
.Mode 
== ModeKeep
) 
 529    if (State
.Mode 
== ModeInstall
) 
 533 // DepCache::BuildGroupOrs - Generate the Or group dep data             /*{{{*/ 
 534 // --------------------------------------------------------------------- 
 535 /* The or group results are stored in the last item of the or group. This 
 536    allows easy detection of the state of a whole or'd group. */ 
 537 void pkgDepCache::BuildGroupOrs(VerIterator 
const &V
) 
 539    unsigned char Group 
= 0; 
 541    for (DepIterator D 
= V
.DependsList(); D
.end() != true; ++D
) 
 543       // Build the dependency state. 
 544       unsigned char &State 
= DepState
[D
->ID
]; 
 546       /* Invert for Conflicts. We have to do this twice to get the 
 547          right sense for a conflicts group */ 
 548       if (D
.IsNegative() == true) 
 551       // Add to the group if we are within an or.. 
 555       if ((D
->CompareOp 
& Dep::Or
) != Dep::Or
) 
 558       // Invert for Conflicts 
 559       if (D
.IsNegative() == true) 
 564 // DepCache::VersionState - Perform a pass over a dependency list       /*{{{*/ 
 565 // --------------------------------------------------------------------- 
 566 /* This is used to run over a dependency list and determine the dep 
 567    state of the list, filtering it through both a Min check and a Policy 
 568    check. The return result will have SetMin/SetPolicy low if a check 
 569    fails. It uses the DepState cache for it's computations. */ 
 570 unsigned char pkgDepCache::VersionState(DepIterator D
,unsigned char Check
, 
 571                                        unsigned char SetMin
, 
 572                                        unsigned char SetPolicy
) 
 574    unsigned char Dep 
= 0xFF; 
 576    while (D
.end() != true) 
 578       // Compute a single dependency element (glob or) 
 579       DepIterator Start 
= D
; 
 580       unsigned char State 
= 0; 
 581       for (bool LastOR 
= true; D
.end() == false && LastOR 
== true; ++D
) 
 583          State 
|= DepState
[D
->ID
]; 
 584          LastOR 
= (D
->CompareOp 
& Dep::Or
) == Dep::Or
; 
 587       // Minimum deps that must be satisfied to have a working package 
 588       if (Start
.IsCritical() == true) 
 589          if ((State 
& Check
) != Check
) 
 592       // Policy deps that must be satisfied to install the package 
 593       if (IsImportantDep(Start
) == true &&  
 594           (State 
& Check
) != Check
) 
 601 // DepCache::DependencyState - Compute the 3 results for a dep          /*{{{*/ 
 602 // --------------------------------------------------------------------- 
 603 /* This is the main dependency computation bit. It computes the 3 main 
 604    results for a dependencys, Now, Install and Candidate. Callers must 
 605    invert the result if dealing with conflicts. */ 
 606 unsigned char pkgDepCache::DependencyState(DepIterator 
&D
) 
 608    unsigned char State 
= 0; 
 610    if (CheckDep(D
,NowVersion
) == true) 
 612    if (CheckDep(D
,InstallVersion
) == true) 
 614    if (CheckDep(D
,CandidateVersion
) == true) 
 620 // DepCache::UpdateVerState - Compute the Dep member of the state       /*{{{*/ 
 621 // --------------------------------------------------------------------- 
 622 /* This determines the combined dependency representation of a package 
 623    for its two states now and install. This is done by using the pre-generated 
 624    dependency information. */ 
 625 void pkgDepCache::UpdateVerState(PkgIterator Pkg
) 
 627    // Empty deps are always true 
 628    StateCache 
&State 
= PkgState
[Pkg
->ID
]; 
 629    State
.DepState 
= 0xFF; 
 631    // Check the Current state 
 632    if (Pkg
->CurrentVer 
!= 0) 
 634       DepIterator D 
= Pkg
.CurrentVer().DependsList(); 
 635       State
.DepState 
&= VersionState(D
,DepNow
,DepNowMin
,DepNowPolicy
); 
 638    /* Check the candidate state. We do not compare against the whole as 
 639       a candidate state but check the candidate version against the  
 641    if (State
.CandidateVer 
!= 0) 
 643       DepIterator D 
= State
.CandidateVerIter(*this).DependsList(); 
 644       State
.DepState 
&= VersionState(D
,DepInstall
,DepCandMin
,DepCandPolicy
); 
 647    // Check target state which can only be current or installed 
 648    if (State
.InstallVer 
!= 0) 
 650       DepIterator D 
= State
.InstVerIter(*this).DependsList(); 
 651       State
.DepState 
&= VersionState(D
,DepInstall
,DepInstMin
,DepInstPolicy
); 
 655 // DepCache::Update - Figure out all the state information              /*{{{*/ 
 656 // --------------------------------------------------------------------- 
 657 /* This will figure out the state of all the packages and all the  
 658    dependencies based on the current policy. */ 
 659 void pkgDepCache::Update(OpProgress 
*Prog
) 
 669    // Perform the depends pass 
 671    for (PkgIterator I 
= PkgBegin(); I
.end() != true; ++I
, ++Done
) 
 673       if (Prog 
!= 0 && Done%20 
== 0) 
 674          Prog
->Progress(Done
); 
 675       for (VerIterator V 
= I
.VersionList(); V
.end() != true; ++V
) 
 677          unsigned char Group 
= 0; 
 679          for (DepIterator D 
= V
.DependsList(); D
.end() != true; ++D
) 
 681             // Build the dependency state. 
 682             unsigned char &State 
= DepState
[D
->ID
]; 
 683             State 
= DependencyState(D
); 
 685             // Add to the group if we are within an or.. 
 688             if ((D
->CompareOp 
& Dep::Or
) != Dep::Or
) 
 691             // Invert for Conflicts 
 692             if (D
.IsNegative() == true) 
 697       // Compute the package dependency state and size additions 
 704       Prog
->Progress(Done
); 
 709 // DepCache::Update - Update the deps list of a package                 /*{{{*/ 
 710 // --------------------------------------------------------------------- 
 711 /* This is a helper for update that only does the dep portion of the scan.  
 712    It is mainly meant to scan reverse dependencies. */ 
 713 void pkgDepCache::Update(DepIterator D
) 
 715    // Update the reverse deps 
 716    for (;D
.end() != true; ++D
) 
 718       unsigned char &State 
= DepState
[D
->ID
]; 
 719       State 
= DependencyState(D
); 
 721       // Invert for Conflicts 
 722       if (D
.IsNegative() == true) 
 725       RemoveStates(D
.ParentPkg()); 
 726       BuildGroupOrs(D
.ParentVer()); 
 727       UpdateVerState(D
.ParentPkg()); 
 728       AddStates(D
.ParentPkg()); 
 732 // DepCache::Update - Update the related deps of a package              /*{{{*/ 
 733 // --------------------------------------------------------------------- 
 734 /* This is called whenever the state of a package changes. It updates 
 735    all cached dependencies related to this package. */ 
 736 void pkgDepCache::Update(PkgIterator 
const &Pkg
) 
 738    // Recompute the dep of the package 
 743    // Update the reverse deps 
 744    Update(Pkg
.RevDependsList()); 
 746    // Update the provides map for the current ver 
 747    if (Pkg
->CurrentVer 
!= 0) 
 748       for (PrvIterator P 
= Pkg
.CurrentVer().ProvidesList();  
 749            P
.end() != true; ++P
) 
 750          Update(P
.ParentPkg().RevDependsList()); 
 752    // Update the provides map for the candidate ver 
 753    if (PkgState
[Pkg
->ID
].CandidateVer 
!= 0) 
 754       for (PrvIterator P 
= PkgState
[Pkg
->ID
].CandidateVerIter(*this).ProvidesList(); 
 755            P
.end() != true; ++P
) 
 756          Update(P
.ParentPkg().RevDependsList()); 
 759 // DepCache::MarkKeep - Put the package in the keep state               /*{{{*/ 
 760 // --------------------------------------------------------------------- 
 762 bool pkgDepCache::MarkKeep(PkgIterator 
const &Pkg
, bool Soft
, bool FromUser
, 
 765    if (IsModeChangeOk(ModeKeep
, Pkg
, Depth
, FromUser
) == false) 
 768    /* Reject an attempt to keep a non-source broken installed package, those 
 770    if (Pkg
.State() == PkgIterator::NeedsUnpack 
&&  
 771        Pkg
.CurrentVer().Downloadable() == false) 
 774    /* We changed the soft state all the time so the UI is a bit nicer 
 776    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
 778    // Check that it is not already kept 
 779    if (P
.Mode 
== ModeKeep
) 
 783       P
.iFlags 
|= AutoKept
; 
 785       P
.iFlags 
&= ~AutoKept
; 
 787    ActionGroup 
group(*this); 
 789 #if 0 // reseting the autoflag here means we lose the  
 790       // auto-mark information if a user selects a package for removal 
 791       // but changes  his mind then and sets it for keep again 
 792       // - this makes sense as default when all Garbage dependencies 
 793       //   are automatically marked for removal (as aptitude does). 
 794       //   setting a package for keep then makes it no longer autoinstalled 
 795       //   for all other use-case this action is rather suprising 
 796    if(FromUser 
&& !P
.Marked
) 
 797      P
.Flags 
&= ~Flag::Auto
; 
 800    if (DebugMarker 
== true) 
 801       std::clog 
<< OutputInDepth(Depth
) << "MarkKeep " << Pkg 
<< " FU=" << FromUser 
<< std::endl
; 
 807    if (Pkg
->CurrentVer 
== 0) 
 810       P
.InstallVer 
= Pkg
.CurrentVer(); 
 819 // DepCache::MarkDelete - Put the package in the delete state           /*{{{*/ 
 820 // --------------------------------------------------------------------- 
 822 bool pkgDepCache::MarkDelete(PkgIterator 
const &Pkg
, bool rPurge
, 
 823                              unsigned long Depth
, bool FromUser
) 
 825    if (IsModeChangeOk(ModeDelete
, Pkg
, Depth
, FromUser
) == false) 
 828    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
 830    // Check that it is not already marked for delete 
 831    if ((P
.Mode 
== ModeDelete 
|| P
.InstallVer 
== 0) &&  
 832        (Pkg
.Purge() == true || rPurge 
== false)) 
 835    // check if we are allowed to remove the package 
 836    if (IsDeleteOk(Pkg
,rPurge
,Depth
,FromUser
) == false) 
 839    P
.iFlags 
&= ~(AutoKept 
| Purge
); 
 843    ActionGroup 
group(*this); 
 845    if (DebugMarker 
== true) 
 846       std::clog 
<< OutputInDepth(Depth
) << (rPurge 
? "MarkPurge " : "MarkDelete ") << Pkg 
<< " FU=" << FromUser 
<< std::endl
; 
 851    if (Pkg
->CurrentVer 
== 0 && (Pkg
.Purge() == true || rPurge 
== false)) 
 864 // DepCache::IsDeleteOk - check if it is ok to remove this package      /*{{{*/ 
 865 // --------------------------------------------------------------------- 
 866 /* The default implementation tries to prevent deletion of install requests. 
 867    dpkg holds are enforced by the private IsModeChangeOk */ 
 868 bool pkgDepCache::IsDeleteOk(PkgIterator 
const &Pkg
,bool rPurge
, 
 869                               unsigned long Depth
, bool FromUser
) 
 871    if (FromUser 
== false && Pkg
->CurrentVer 
== 0) 
 873       StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
 874       if (P
.InstallVer 
!= 0 && P
.Status 
== 2 && (P
.Flags 
& Flag::Auto
) != Flag::Auto
) 
 876          if (DebugMarker 
== true) 
 877             std::clog 
<< OutputInDepth(Depth
) << "Manual install request prevents MarkDelete of " << Pkg 
<< std::endl
; 
 884 // DepCache::IsModeChangeOk - check if it is ok to change the mode      /*{{{*/ 
 885 // --------------------------------------------------------------------- 
 886 /* this is used by all Mark methods on the very first line to check sanity 
 887    and prevents mode changes for packages on hold for example. 
 888    If you want to check Mode specific stuff you can use the virtual public 
 889    Is<Mode>Ok methods instead */ 
 890 char const* PrintMode(char const mode
) 
 894          case pkgDepCache::ModeInstall
: return "Install"; 
 895          case pkgDepCache::ModeKeep
: return "Keep"; 
 896          case pkgDepCache::ModeDelete
: return "Delete"; 
 897          default: return "UNKNOWN"; 
 900 bool pkgDepCache::IsModeChangeOk(ModeList 
const mode
, PkgIterator 
const &Pkg
, 
 901                                  unsigned long const Depth
, bool const FromUser
) 
 903    // we are not trying to hard… 
 904    if (unlikely(Depth 
> 100)) 
 908    if (unlikely(Pkg
.end() == true || Pkg
->VersionList 
== 0)) 
 911    // the user is always right 
 912    if (FromUser 
== true) 
 915    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
 917    // if previous state was set by user only user can reset it 
 918    if ((P
.iFlags 
& Protected
) == Protected
) 
 920       if (unlikely(DebugMarker 
== true) && P
.Mode 
!= mode
) 
 921          std::clog 
<< OutputInDepth(Depth
) << "Ignore Mark" << PrintMode(mode
) 
 922                    << " of " << Pkg 
<< " as its mode (" << PrintMode(P
.Mode
) 
 923                    << ") is protected" << std::endl
; 
 926    // enforce dpkg holds 
 927    else if (mode 
!= ModeKeep 
&& Pkg
->SelectedState 
== pkgCache::State::Hold 
&& 
 928             _config
->FindB("APT::Ignore-Hold",false) == false) 
 930       if (unlikely(DebugMarker 
== true) && P
.Mode 
!= mode
) 
 931          std::clog 
<< OutputInDepth(Depth
) << "Hold prevents Mark" << PrintMode(mode
) 
 932                    << " of " << Pkg 
<< std::endl
; 
 939 // DepCache::MarkInstall - Put the package in the install state         /*{{{*/ 
 940 // --------------------------------------------------------------------- 
 942 struct CompareProviders 
{ 
 943    pkgCache::PkgIterator 
const Pkg
; 
 944    CompareProviders(pkgCache::DepIterator 
const &Dep
) : Pkg(Dep
.TargetPkg()) {}; 
 945    //bool operator() (APT::VersionList::iterator const &AV, APT::VersionList::iterator const &BV) 
 946    bool operator() (pkgCache::VerIterator 
const &AV
, pkgCache::VerIterator 
const &BV
) 
 948       pkgCache::PkgIterator 
const A 
= AV
.ParentPkg(); 
 949       pkgCache::PkgIterator 
const B 
= BV
.ParentPkg(); 
 950       // Prefer packages in the same group as the target; e.g. foo:i386, foo:amd64 
 951       if (A
->Group 
!= B
->Group
) 
 953          if (A
->Group 
== Pkg
->Group 
&& B
->Group 
!= Pkg
->Group
) 
 955          else if (B
->Group 
== Pkg
->Group 
&& A
->Group 
!= Pkg
->Group
) 
 958       // we like essentials 
 959       if ((A
->Flags 
& pkgCache::Flag::Essential
) != (B
->Flags 
& pkgCache::Flag::Essential
)) 
 961          if ((A
->Flags 
& pkgCache::Flag::Essential
) == pkgCache::Flag::Essential
) 
 963          else if ((B
->Flags 
& pkgCache::Flag::Essential
) == pkgCache::Flag::Essential
) 
 966       // higher priority seems like a good idea 
 967       if (AV
->Priority 
!= BV
->Priority
) 
 968          return AV
->Priority 
< BV
->Priority
; 
 969       // prefer native architecture 
 970       if (strcmp(A
.Arch(), B
.Arch()) != 0) 
 972          if (strcmp(A
.Arch(), A
.Cache()->NativeArch()) == 0) 
 974          else if (strcmp(B
.Arch(), B
.Cache()->NativeArch()) == 0) 
 976          std::vector
<std::string
> archs 
= APT::Configuration::getArchitectures(); 
 977          for (std::vector
<std::string
>::const_iterator a 
= archs
.begin(); a 
!= archs
.end(); ++a
) 
 980             else if (*a 
== B
.Arch()) 
 983       // unable to decide… 
 984       return A
->ID 
< B
->ID
; 
 987 bool pkgDepCache::MarkInstall(PkgIterator 
const &Pkg
,bool AutoInst
, 
 988                               unsigned long Depth
, bool FromUser
, 
 989                               bool ForceImportantDeps
) 
 991    if (IsModeChangeOk(ModeInstall
, Pkg
, Depth
, FromUser
) == false) 
 994    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
 996    // See if there is even any possible instalation candidate 
 997    if (P
.CandidateVer 
== 0) 
1000    /* Check that it is not already marked for install and that it can be  
1002    if ((P
.InstPolicyBroken() == false && P
.InstBroken() == false) &&  
1003        (P
.Mode 
== ModeInstall 
|| 
1004         P
.CandidateVer 
== (Version 
*)Pkg
.CurrentVer())) 
1006       if (P
.CandidateVer 
== (Version 
*)Pkg
.CurrentVer() && P
.InstallVer 
== 0) 
1007          return MarkKeep(Pkg
, false, FromUser
, Depth
+1); 
1011    // check if we are allowed to install the package 
1012    if (IsInstallOk(Pkg
,AutoInst
,Depth
,FromUser
) == false) 
1015    ActionGroup 
group(*this); 
1016    P
.iFlags 
&= ~AutoKept
; 
1018    /* Target the candidate version and remove the autoflag. We reset the 
1019       autoflag below if this was called recursively. Otherwise the user 
1020       should have the ability to de-auto a package by changing its state */ 
1024    P
.Mode 
= ModeInstall
; 
1025    P
.InstallVer 
= P
.CandidateVer
; 
1029        // Set it to manual if it's a new install or already installed, 
1030        // but only if its not marked by the autoremover (aptitude depend on this behavior) 
1031        // or if we do automatic installation (aptitude never does it) 
1032        if(P
.Status 
== 2 || (Pkg
->CurrentVer 
!= 0 && (AutoInst 
== true || P
.Marked 
== false))) 
1033          P
.Flags 
&= ~Flag::Auto
; 
1037        // Set it to auto if this is a new install. 
1039          P
.Flags 
|= Flag::Auto
; 
1041    if (P
.CandidateVer 
== (Version 
*)Pkg
.CurrentVer()) 
1048    if (AutoInst 
== false || _config
->Find("APT::Solver", "internal") != "internal") 
1051    if (DebugMarker 
== true) 
1052       std::clog 
<< OutputInDepth(Depth
) << "MarkInstall " << Pkg 
<< " FU=" << FromUser 
<< std::endl
; 
1054    DepIterator Dep 
= P
.InstVerIter(*this).DependsList(); 
1055    for (; Dep
.end() != true;) 
1058       DepIterator Start 
= Dep
; 
1061       for (bool LastOR 
= true; Dep
.end() == false && LastOR 
== true; ++Dep
, ++Ors
) 
1063          LastOR 
= (Dep
->CompareOp 
& Dep::Or
) == Dep::Or
; 
1065          if ((DepState
[Dep
->ID
] & DepInstall
) == DepInstall
) 
1069       // Dep is satisfied okay. 
1070       if (Result 
== false) 
1073       /* Check if this dep should be consider for install. If it is a user 
1074          defined important dep and we are installed a new package then  
1075          it will be installed. Otherwise we only check for important 
1076          deps that have changed from the installed version 
1078       if (IsImportantDep(Start
) == false) 
1081       /* If we are in an or group locate the first or that can  
1082          succeed. We have already cached this.. */ 
1083       for (; Ors 
> 1 && (DepState
[Start
->ID
] & DepCVer
) != DepCVer
; --Ors
) 
1085       if (Ors 
== 1 && (DepState
[Start
->ID
] &DepCVer
) != DepCVer 
&& Start
.IsNegative() == false) 
1087          if(DebugAutoInstall 
== true) 
1088             std::clog 
<< OutputInDepth(Depth
) << Start 
<< " can't be satisfied!" << std::endl
; 
1089          if (Start
.IsCritical() == false) 
1091          // if the dependency was critical, we can't install it, so remove it again 
1092          MarkDelete(Pkg
,false,Depth 
+ 1, false); 
1096       /* Check if any ImportantDep() (but not Critical) were added 
1097        * since we installed the package.  Also check for deps that 
1098        * were satisfied in the past: for instance, if a version 
1099        * restriction in a Recommends was tightened, upgrading the 
1100        * package should follow that Recommends rather than causing the 
1101        * dependency to be removed. (bug #470115) 
1103       if (Pkg
->CurrentVer 
!= 0 && ForceImportantDeps 
== false && Start
.IsCritical() == false) 
1105          bool isNewImportantDep 
= true; 
1106          bool isPreviouslySatisfiedImportantDep 
= false; 
1107          for (DepIterator D 
= Pkg
.CurrentVer().DependsList(); D
.end() != true; ++D
) 
1109             //FIXME: Should we handle or-group better here? 
1110             // We do not check if the package we look for is part of the same or-group 
1111             // we might find while searching, but could that really be a problem? 
1112             if (D
.IsCritical() == true || IsImportantDep(D
) == false || 
1113                 Start
.TargetPkg() != D
.TargetPkg()) 
1116             isNewImportantDep 
= false; 
1118             while ((D
->CompareOp 
& Dep::Or
) != 0) 
1121             isPreviouslySatisfiedImportantDep 
= (((*this)[D
] & DepGNow
) != 0); 
1122             if (isPreviouslySatisfiedImportantDep 
== true) 
1126          if(isNewImportantDep 
== true) 
1128             if (DebugAutoInstall 
== true) 
1129                std::clog 
<< OutputInDepth(Depth
) << "new important dependency: " 
1130                          << Start
.TargetPkg().FullName() << std::endl
; 
1132          else if(isPreviouslySatisfiedImportantDep 
== true) 
1134             if (DebugAutoInstall 
== true) 
1135                std::clog 
<< OutputInDepth(Depth
) << "previously satisfied important dependency on " 
1136                          << Start
.TargetPkg().FullName() << std::endl
; 
1140             if (DebugAutoInstall 
== true) 
1141                std::clog 
<< OutputInDepth(Depth
) << "ignore old unsatisfied important dependency on " 
1142                          << Start
.TargetPkg().FullName() << std::endl
; 
1147       /* This bit is for processing the possibilty of an install/upgrade 
1148          fixing the problem */ 
1149       if (Start
->Type 
!= Dep::DpkgBreaks 
&& 
1150           (DepState
[Start
->ID
] & DepCVer
) == DepCVer
) 
1152          APT::VersionList verlist
; 
1153          pkgCache::VerIterator Cand 
= PkgState
[Start
.TargetPkg()->ID
].CandidateVerIter(*this); 
1154          if (Cand
.end() == false && VS().CheckDep(Cand
.VerStr(), Start
->CompareOp
, Start
.TargetVer()) == true) 
1155             verlist
.insert(Cand
); 
1156          for (PrvIterator Prv 
= Start
.TargetPkg().ProvidesList(); Prv
.end() != true; ++Prv
) 
1158             pkgCache::VerIterator V 
= Prv
.OwnerVer(); 
1159             pkgCache::VerIterator Cand 
= PkgState
[Prv
.OwnerPkg()->ID
].CandidateVerIter(*this); 
1160             if (Cand
.end() == true || V 
!= Cand 
|| 
1161                 VS().CheckDep(Cand
.VerStr(), Start
->CompareOp
, Start
.TargetVer()) == false) 
1163             verlist
.insert(Cand
); 
1165          CompareProviders 
comp(Start
); 
1166          APT::VersionList::iterator InstVer 
= std::max_element(verlist
.begin(), verlist
.end(), comp
); 
1168          if (InstVer 
!= verlist
.end()) 
1170             pkgCache::PkgIterator InstPkg 
= InstVer
.ParentPkg(); 
1171             if(DebugAutoInstall 
== true) 
1172                std::clog 
<< OutputInDepth(Depth
) << "Installing " << InstPkg
.Name() 
1173                          << " as " << Start
.DepType() << " of " << Pkg
.Name() 
1175             // now check if we should consider it a automatic dependency or not 
1176             if(Pkg
.Section() && ConfigValueInSubTree("APT::Never-MarkAuto-Sections", Pkg
.Section())) 
1178                if(DebugAutoInstall 
== true) 
1179                   std::clog 
<< OutputInDepth(Depth
) << "Setting NOT as auto-installed (direct " 
1180                             << Start
.DepType() << " of pkg in APT::Never-MarkAuto-Sections)" << std::endl
; 
1181                MarkInstall(InstPkg
,true,Depth 
+ 1, true); 
1185                // mark automatic dependency 
1186                MarkInstall(InstPkg
,true,Depth 
+ 1, false, ForceImportantDeps
); 
1187                // Set the autoflag, after MarkInstall because MarkInstall unsets it 
1188                if (InstPkg
->CurrentVer 
== 0) 
1189                   PkgState
[InstPkg
->ID
].Flags 
|= Flag::Auto
; 
1195       /* For conflicts we just de-install the package and mark as auto, 
1196          Conflicts may not have or groups.  For dpkg's Breaks we try to 
1197          upgrade the package. */ 
1198       if (Start
.IsNegative() == true) 
1200          SPtrArray
<Version 
*> List 
= Start
.AllTargets(); 
1201          for (Version 
**I 
= List
; *I 
!= 0; I
++) 
1203             VerIterator 
Ver(*this,*I
); 
1204             PkgIterator Pkg 
= Ver
.ParentPkg(); 
1206             /* The List includes all packages providing this dependency, 
1207                even providers which are not installed, so skip them. */ 
1208             if (PkgState
[Pkg
->ID
].InstallVer 
== 0) 
1211             if (PkgState
[Pkg
->ID
].CandidateVer 
!= *I 
&& 
1212                 Start
->Type 
== Dep::DpkgBreaks 
&& 
1213                 MarkInstall(Pkg
,true,Depth 
+ 1, false, ForceImportantDeps
) == true) 
1215             else if (MarkDelete(Pkg
,false,Depth 
+ 1, false) == false) 
1222    return Dep
.end() == true; 
1225 // DepCache::IsInstallOk - check if it is ok to install this package    /*{{{*/ 
1226 // --------------------------------------------------------------------- 
1227 /* The default implementation does nothing. 
1228    dpkg holds are enforced by the private IsModeChangeOk */ 
1229 bool pkgDepCache::IsInstallOk(PkgIterator 
const &Pkg
,bool AutoInst
, 
1230                               unsigned long Depth
, bool FromUser
) 
1235 // DepCache::SetReInstall - Set the reinstallation flag                 /*{{{*/ 
1236 // --------------------------------------------------------------------- 
1238 void pkgDepCache::SetReInstall(PkgIterator 
const &Pkg
,bool To
) 
1240    if (unlikely(Pkg
.end() == true)) 
1243    ActionGroup 
group(*this); 
1248    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
1250       P
.iFlags 
|= ReInstall
; 
1252       P
.iFlags 
&= ~ReInstall
; 
1258 // DepCache::SetCandidateVersion - Change the candidate version         /*{{{*/ 
1259 // --------------------------------------------------------------------- 
1261 void pkgDepCache::SetCandidateVersion(VerIterator TargetVer
) 
1263    pkgCache::PkgIterator Pkg 
= TargetVer
.ParentPkg(); 
1264    StateCache 
&P 
= PkgState
[Pkg
->ID
]; 
1266    if (P
.CandidateVer 
== TargetVer
) 
1269    ActionGroup 
group(*this); 
1274    if (P
.CandidateVer 
== P
.InstallVer 
&& P
.Install() == true) 
1275       P
.InstallVer 
= (Version 
*)TargetVer
; 
1276    P
.CandidateVer 
= (Version 
*)TargetVer
; 
1277    P
.Update(Pkg
,*this); 
1285 // DepCache::SetCandidateRelease - Change the candidate version         /*{{{*/ 
1286 // --------------------------------------------------------------------- 
1287 /* changes the candidate of a package and walks over all its dependencies 
1288    to check if it needs to change the candidate of the dependency, too, 
1289    to reach a installable versionstate */ 
1290 bool pkgDepCache::SetCandidateRelease(pkgCache::VerIterator TargetVer
, 
1291                                         std::string 
const &TargetRel
) 
1293    std::list
<std::pair
<pkgCache::VerIterator
, pkgCache::VerIterator
> > Changed
; 
1294    return SetCandidateRelease(TargetVer
, TargetRel
, Changed
); 
1296 bool pkgDepCache::SetCandidateRelease(pkgCache::VerIterator TargetVer
, 
1297                                         std::string 
const &TargetRel
, 
1298                                         std::list
<std::pair
<pkgCache::VerIterator
, pkgCache::VerIterator
> > &Changed
) 
1300    ActionGroup 
group(*this); 
1301    SetCandidateVersion(TargetVer
); 
1303    if (TargetRel 
== "installed" || TargetRel 
== "candidate") // both doesn't make sense in this context 
1306    pkgVersionMatch 
Match(TargetRel
, pkgVersionMatch::Release
); 
1307    // save the position of the last element we will not undo - if we have to 
1308    std::list
<std::pair
<pkgCache::VerIterator
, pkgCache::VerIterator
> >::iterator newChanged 
= --(Changed
.end()); 
1310    for (pkgCache::DepIterator D 
= TargetVer
.DependsList(); D
.end() == false; ++D
) 
1312       if (D
->Type 
!= pkgCache::Dep::PreDepends 
&& D
->Type 
!= pkgCache::Dep::Depends 
&& 
1313           ((D
->Type 
!= pkgCache::Dep::Recommends 
&& D
->Type 
!= pkgCache::Dep::Suggests
) || 
1314            IsImportantDep(D
) == false)) 
1317       // walk over an or-group and check if we need to do anything 
1318       // for simpilicity no or-group is handled as a or-group including one dependency 
1319       pkgCache::DepIterator Start 
= D
; 
1320       bool itsFine 
= false; 
1321       for (bool stillOr 
= true; stillOr 
== true; ++Start
) 
1323          stillOr 
= (Start
->CompareOp 
& Dep::Or
) == Dep::Or
; 
1324          pkgCache::PkgIterator 
const P 
= Start
.TargetPkg(); 
1325          // virtual packages can't be a solution 
1326          if (P
.end() == true || (P
->ProvidesList 
== 0 && P
->VersionList 
== 0)) 
1328          pkgCache::VerIterator 
const Cand 
= PkgState
[P
->ID
].CandidateVerIter(*this); 
1329          // no versioned dependency - but is it installable? 
1330          if (Start
.TargetVer() == 0 || Start
.TargetVer()[0] == '\0') 
1332             // Check if one of the providers is installable 
1333             if (P
->ProvidesList 
!= 0) 
1335                pkgCache::PrvIterator Prv 
= P
.ProvidesList(); 
1336                for (; Prv
.end() == false; ++Prv
) 
1338                   pkgCache::VerIterator 
const C 
= PkgState
[Prv
.OwnerPkg()->ID
].CandidateVerIter(*this); 
1339                   if (C
.end() == true || C 
!= Prv
.OwnerVer() || 
1340                       (VersionState(C
.DependsList(), DepInstall
, DepCandMin
, DepCandPolicy
) & DepCandMin
) != DepCandMin
) 
1344                if (Prv
.end() == true) 
1347             // no providers, so check if we have an installable candidate version 
1348             else if (Cand
.end() == true || 
1349                 (VersionState(Cand
.DependsList(), DepInstall
, DepCandMin
, DepCandPolicy
) & DepCandMin
) != DepCandMin
) 
1354          if (Cand
.end() == true) 
1356          // check if the current candidate is enough for the versioned dependency - and installable? 
1357          if (VS().CheckDep(P
.CandVersion(), Start
->CompareOp
, Start
.TargetVer()) == true && 
1358              (VersionState(Cand
.DependsList(), DepInstall
, DepCandMin
, DepCandPolicy
) & DepCandMin
) == DepCandMin
) 
1365       if (itsFine 
== true) { 
1366          // something in the or-group was fine, skip all other members 
1367          for (; (D
->CompareOp 
& Dep::Or
) == Dep::Or
; ++D
); 
1371       // walk again over the or-group and check each if a candidate switch would help 
1373       for (bool stillOr 
= true; stillOr 
== true; ++D
) 
1375          stillOr 
= (D
->CompareOp 
& Dep::Or
) == Dep::Or
; 
1376          // changing candidate will not help if the dependency is not versioned 
1377          if (D
.TargetVer() == 0 || D
.TargetVer()[0] == '\0') 
1379             if (stillOr 
== true) 
1384          pkgCache::VerIterator V
; 
1385          if (TargetRel 
== "newest") 
1386             V 
= D
.TargetPkg().VersionList(); 
1388             V 
= Match
.Find(D
.TargetPkg()); 
1390          // check if the version from this release could satisfy the dependency 
1391          if (V
.end() == true || VS().CheckDep(V
.VerStr(), D
->CompareOp
, D
.TargetVer()) == false) 
1393             if (stillOr 
== true) 
1398          pkgCache::VerIterator oldCand 
= PkgState
[D
.TargetPkg()->ID
].CandidateVerIter(*this); 
1401             // Do we already touched this Version? If so, their versioned dependencies are okay, no need to check again 
1402             for (std::list
<std::pair
<pkgCache::VerIterator
, pkgCache::VerIterator
> >::const_iterator c 
= Changed
.begin(); 
1403                  c 
!= Changed
.end(); ++c
) 
1405                if (c
->first
->ParentPkg 
!= V
->ParentPkg
) 
1412          if (itsFine 
== false) 
1414             // change the candidate 
1415             Changed
.push_back(make_pair(oldCand
, TargetVer
)); 
1416             if (SetCandidateRelease(V
, TargetRel
, Changed
) == false) 
1418                if (stillOr 
== false) 
1420                // undo the candidate changing 
1421                SetCandidateVersion(oldCand
); 
1428          // something in the or-group was fine, skip all other members 
1429          for (; (D
->CompareOp 
& Dep::Or
) == Dep::Or
; ++D
); 
1433       if (itsFine 
== false && (D
->Type 
== pkgCache::Dep::PreDepends 
|| D
->Type 
== pkgCache::Dep::Depends
)) 
1435          // undo all changes which aren't lead to a solution 
1436          for (std::list
<std::pair
<pkgCache::VerIterator
, pkgCache::VerIterator
> >::const_iterator c 
= ++newChanged
; 
1437               c 
!= Changed
.end(); ++c
) 
1438             SetCandidateVersion(c
->first
); 
1439          Changed
.erase(newChanged
, Changed
.end()); 
1446 // DepCache::MarkAuto - set the Auto flag for a package                 /*{{{*/ 
1447 // --------------------------------------------------------------------- 
1449 void pkgDepCache::MarkAuto(const PkgIterator 
&Pkg
, bool Auto
) 
1451   StateCache 
&state 
= PkgState
[Pkg
->ID
]; 
1453   ActionGroup 
group(*this); 
1456     state
.Flags 
|= Flag::Auto
; 
1458     state
.Flags 
&= ~Flag::Auto
; 
1461 // StateCache::Update - Compute the various static display things       /*{{{*/ 
1462 // --------------------------------------------------------------------- 
1463 /* This is called whenever the Candidate version changes. */ 
1464 void pkgDepCache::StateCache::Update(PkgIterator Pkg
,pkgCache 
&Cache
) 
1467    VerIterator Ver 
= CandidateVerIter(Cache
); 
1469    // Use a null string or the version string 
1470    if (Ver
.end() == true) 
1473       CandVersion 
= Ver
.VerStr(); 
1475    // Find the current version 
1477    if (Pkg
->CurrentVer 
!= 0) 
1478       CurVersion 
= Pkg
.CurrentVer().VerStr(); 
1480    // Strip off the epochs for display 
1481    CurVersion 
= StripEpoch(CurVersion
); 
1482    CandVersion 
= StripEpoch(CandVersion
); 
1484    // Figure out if its up or down or equal 
1485    Status 
= Ver
.CompareVer(Pkg
.CurrentVer()); 
1486    if (Pkg
->CurrentVer 
== 0 || Pkg
->VersionList 
== 0 || CandidateVer 
== 0) 
1490 // StateCache::StripEpoch - Remove the epoch specifier from the version /*{{{*/ 
1491 // --------------------------------------------------------------------- 
1493 const char *pkgDepCache::StateCache::StripEpoch(const char *Ver
) 
1499    for (const char *I 
= Ver
; *I 
!= 0; I
++) 
1505 // Policy::GetCandidateVer - Returns the Candidate install version      /*{{{*/ 
1506 // --------------------------------------------------------------------- 
1507 /* The default just returns the highest available version that is not 
1508    a source and automatic. */ 
1509 pkgCache::VerIterator 
pkgDepCache::Policy::GetCandidateVer(PkgIterator 
const &Pkg
) 
1511    /* Not source/not automatic versions cannot be a candidate version  
1512       unless they are already installed */ 
1513    VerIterator 
Last(*(pkgCache 
*)this,0); 
1515    for (VerIterator I 
= Pkg
.VersionList(); I
.end() == false; ++I
) 
1517       if (Pkg
.CurrentVer() == I
) 
1520       for (VerFileIterator J 
= I
.FileList(); J
.end() == false; ++J
) 
1522          if ((J
.File()->Flags 
& Flag::NotSource
) != 0) 
1525          /* Stash the highest version of a not-automatic source, we use it 
1526             if there is nothing better */ 
1527          if ((J
.File()->Flags 
& Flag::NotAutomatic
) != 0 || 
1528              (J
.File()->Flags 
& Flag::ButAutomaticUpgrades
) != 0) 
1530             if (Last
.end() == true) 
1542 // Policy::IsImportantDep - True if the dependency is important         /*{{{*/ 
1543 // --------------------------------------------------------------------- 
1545 bool pkgDepCache::Policy::IsImportantDep(DepIterator 
const &Dep
) 
1547    if(Dep
.IsCritical()) 
1549    else if(Dep
->Type 
== pkgCache::Dep::Recommends
)  
1551       if (InstallRecommends
) 
1553       // we suport a special mode to only install-recommends for certain 
1555       // FIXME: this is a meant as a temporarly solution until the  
1556       //        recommends are cleaned up 
1557       const char *sec 
= Dep
.ParentVer().Section(); 
1558       if (sec 
&& ConfigValueInSubTree("APT::Install-Recommends-Sections", sec
)) 
1561    else if(Dep
->Type 
== pkgCache::Dep::Suggests
) 
1562       return InstallSuggests
; 
1567 // Policy::GetPriority - Get the priority of the package pin            /*{{{*/ 
1568 signed short pkgDepCache::Policy::GetPriority(pkgCache::PkgIterator 
const &Pkg
) 
1570 signed short pkgDepCache::Policy::GetPriority(pkgCache::PkgFileIterator 
const &File
) 
1573 pkgDepCache::InRootSetFunc 
*pkgDepCache::GetRootSetFunc()               /*{{{*/ 
1575   DefaultRootSetFunc 
*f 
= new DefaultRootSetFunc
; 
1576   if(f
->wasConstructedSuccessfully()) 
1585 bool pkgDepCache::MarkFollowsRecommends() 
1587   return _config
->FindB("APT::AutoRemove::RecommendsImportant", true); 
1590 bool pkgDepCache::MarkFollowsSuggests() 
1592   return _config
->FindB("APT::AutoRemove::SuggestsImportant", true); 
1595 // pkgDepCache::MarkRequired - the main mark algorithm                  /*{{{*/ 
1596 bool pkgDepCache::MarkRequired(InRootSetFunc 
&userFunc
) 
1598    if (_config
->Find("APT::Solver", "internal") != "internal") 
1601    bool follow_recommends
; 
1602    bool follow_suggests
; 
1603    bool debug_autoremove 
= _config
->FindB("Debug::pkgAutoRemove",false); 
1606    for(PkgIterator p 
= PkgBegin(); !p
.end(); ++p
) 
1608       PkgState
[p
->ID
].Marked  
= false; 
1609       PkgState
[p
->ID
].Garbage 
= false; 
1612       if(debug_autoremove 
&& PkgState
[p
->ID
].Flags 
& Flag::Auto
) 
1613          std::clog 
<< "AutoDep: " << p
.FullName() << std::endl
; 
1617    follow_recommends 
= MarkFollowsRecommends(); 
1618    follow_suggests   
= MarkFollowsSuggests(); 
1622    // do the mark part, this is the core bit of the algorithm 
1623    for(PkgIterator p 
= PkgBegin(); !p
.end(); ++p
) 
1625       if(!(PkgState
[p
->ID
].Flags 
& Flag::Auto
) || 
1626           (p
->Flags 
& Flag::Essential
) || 
1627           userFunc
.InRootSet(p
) || 
1628           // be nice even then a required package violates the policy (#583517) 
1629           // and do the full mark process also for required packages 
1630           (p
.CurrentVer().end() != true && 
1631            p
.CurrentVer()->Priority 
== pkgCache::State::Required
)) 
1633          // the package is installed (and set to keep) 
1634          if(PkgState
[p
->ID
].Keep() && !p
.CurrentVer().end()) 
1635             MarkPackage(p
, p
.CurrentVer(), 
1636                         follow_recommends
, follow_suggests
); 
1637          // the package is to be installed  
1638          else if(PkgState
[p
->ID
].Install()) 
1639             MarkPackage(p
, PkgState
[p
->ID
].InstVerIter(*this), 
1640                         follow_recommends
, follow_suggests
); 
1647 // MarkPackage - mark a single package in Mark-and-Sweep                /*{{{*/ 
1648 void pkgDepCache::MarkPackage(const pkgCache::PkgIterator 
&pkg
, 
1649                               const pkgCache::VerIterator 
&ver
, 
1650                               bool const &follow_recommends
, 
1651                               bool const &follow_suggests
) 
1653    pkgDepCache::StateCache 
&state 
= PkgState
[pkg
->ID
]; 
1655    // if we are marked already we are done 
1659    VerIterator 
const currver 
= pkg
.CurrentVer(); 
1660    VerIterator 
const instver 
= state
.InstVerIter(*this); 
1663    VerIterator 
const candver 
= state
.CandidateVerIter(*this); 
1665    // If a package was garbage-collected but is now being marked, we 
1666    // should re-select it  
1667    // For cases when a pkg is set to upgrade and this trigger the 
1668    // removal of a no-longer used dependency.  if the pkg is set to 
1669    // keep again later it will result in broken deps 
1670    if(state
.Delete() && state
.RemoveReason 
= Unused
)  
1673          mark_install(pkg
, false, false, NULL
); 
1674       else if(ver
==pkg
.CurrentVer()) 
1675          MarkKeep(pkg
, false, false); 
1677       instver
=state
.InstVerIter(*this); 
1681    // For packages that are not going to be removed, ignore versions 
1682    // other than the InstVer.  For packages that are going to be 
1683    // removed, ignore versions other than the current version. 
1684    if(!(ver 
== instver 
&& !instver
.end()) && 
1685       !(ver 
== currver 
&& instver
.end() && !ver
.end())) 
1688    bool const debug_autoremove 
= _config
->FindB("Debug::pkgAutoRemove", false); 
1690    if(debug_autoremove
) 
1692        std::clog 
<< "Marking: " << pkg
.FullName(); 
1694          std::clog 
<< " " << ver
.VerStr(); 
1696          std::clog 
<< ", Curr=" << currver
.VerStr(); 
1698          std::clog 
<< ", Inst=" << instver
.VerStr(); 
1699        std::clog 
<< std::endl
; 
1704    if(ver
.end() == true) 
1707      for(DepIterator d 
= ver
.DependsList(); !d
.end(); ++d
) 
1709         if(d
->Type 
== Dep::Depends 
|| 
1710            d
->Type 
== Dep::PreDepends 
|| 
1711            (follow_recommends 
&& 
1712             d
->Type 
== Dep::Recommends
) || 
1714             d
->Type 
== Dep::Suggests
)) 
1716            // Try all versions of this package. 
1717            for(VerIterator V 
= d
.TargetPkg().VersionList();  
1720               if(_system
->VS
->CheckDep(V
.VerStr(), d
->CompareOp
, d
.TargetVer())) 
1722                 if(debug_autoremove
) 
1724                     std::clog 
<< "Following dep: " << d
.ParentPkg().FullName() 
1725                               << " " << d
.ParentVer().VerStr() << " " 
1726                               << d
.DepType() << " " << d
.TargetPkg().FullName(); 
1727                     if((d
->CompareOp 
& ~pkgCache::Dep::Or
) != pkgCache::Dep::NoOp
) 
1729                         std::clog 
<< " (" << d
.CompType() << " " 
1730                                   << d
.TargetVer() << ")"; 
1732                     std::clog 
<< std::endl
; 
1734                  MarkPackage(V
.ParentPkg(), V
, 
1735                              follow_recommends
, follow_suggests
); 
1738            // Now try virtual packages 
1739            for(PrvIterator prv
=d
.TargetPkg().ProvidesList();  
1742               if(_system
->VS
->CheckDep(prv
.ProvideVersion(), d
->CompareOp
,  
1745                 if(debug_autoremove
) 
1747                     std::clog 
<< "Following dep: " << d
.ParentPkg().FullName() << " " 
1748                               << d
.ParentVer().VerStr() << " " 
1749                               << d
.DepType() << " " << d
.TargetPkg().FullName() << " "; 
1750                     if((d
->CompareOp 
& ~pkgCache::Dep::Or
) != pkgCache::Dep::NoOp
) 
1752                         std::clog 
<< " (" << d
.CompType() << " " 
1753                                   << d
.TargetVer() << ")"; 
1755                     std::clog 
<< ", provided by " 
1756                               << prv
.OwnerPkg().FullName() << " " 
1757                               << prv
.OwnerVer().VerStr() 
1761                  MarkPackage(prv
.OwnerPkg(), prv
.OwnerVer(), 
1762                              follow_recommends
, follow_suggests
); 
1769 bool pkgDepCache::Sweep()                                               /*{{{*/ 
1771    bool debug_autoremove 
= _config
->FindB("Debug::pkgAutoRemove",false); 
1774    for(PkgIterator p
=PkgBegin(); !p
.end(); ++p
) 
1776      StateCache 
&state
=PkgState
[p
->ID
]; 
1778      // skip required packages 
1779      if (!p
.CurrentVer().end() &&  
1780          (p
.CurrentVer()->Priority 
== pkgCache::State::Required
)) 
1783      // if it is not marked and it is installed, it's garbage  
1784      if(!state
.Marked 
&& (!p
.CurrentVer().end() || state
.Install())) 
1787         if(debug_autoremove
) 
1788            std::clog 
<< "Garbage: " << p
.FullName() << std::endl
;