]>
git.saurik.com Git - apt.git/blob - apt-pkg/depcache.cc
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 /*{{{*/
12 #pragma implementation "apt-pkg/depcache.h"
14 #include <apt-pkg/depcache.h>
15 #include <apt-pkg/version.h>
16 #include <apt-pkg/error.h>
17 #include <apt-pkg/sptr.h>
18 #include <apt-pkg/algorithms.h>
20 #include <apt-pkg/fileutl.h>
21 #include <apt-pkg/configuration.h>
22 #include <apt-pkg/pkgsystem.h>
23 #include <apt-pkg/tagfile.h>
31 pkgDepCache::ActionGroup::ActionGroup(pkgDepCache
&cache
) :
32 cache(cache
), released(false)
37 void pkgDepCache::ActionGroup::release()
41 if(cache
.group_level
== 0)
42 std::cerr
<< "W: Unbalanced action groups, expect badness" << std::endl
;
47 if(cache
.group_level
== 0)
55 pkgDepCache::ActionGroup::~ActionGroup()
60 // DepCache::pkgDepCache - Constructors /*{{{*/
61 // ---------------------------------------------------------------------
63 pkgDepCache::pkgDepCache(pkgCache
*pCache
,Policy
*Plcy
) :
64 group_level(0), Cache(pCache
), PkgState(0), DepState(0)
69 delLocalPolicy
= LocalPolicy
= new Policy
;
72 // DepCache::~pkgDepCache - Destructor /*{{{*/
73 // ---------------------------------------------------------------------
75 pkgDepCache::~pkgDepCache()
79 delete delLocalPolicy
;
82 // DepCache::Init - Generate the initial extra structures. /*{{{*/
83 // ---------------------------------------------------------------------
84 /* This allocats the extension buffers and initializes them. */
85 bool pkgDepCache::Init(OpProgress
*Prog
)
87 // Suppress mark updates during this operation (just in case) and
88 // run a mark operation when Init terminates.
89 ActionGroup
actions(*this);
93 PkgState
= new StateCache
[Head().PackageCount
];
94 DepState
= new unsigned char[Head().DependsCount
];
95 memset(PkgState
,0,sizeof(*PkgState
)*Head().PackageCount
);
96 memset(DepState
,0,sizeof(*DepState
)*Head().DependsCount
);
100 Prog
->OverallProgress(0,2*Head().PackageCount
,Head().PackageCount
,
101 _("Building dependency tree"));
102 Prog
->SubProgress(Head().PackageCount
,_("Candidate versions"));
105 /* Set the current state of everything. In this state all of the
106 packages are kept exactly as is. See AllUpgrade */
108 for (PkgIterator I
= PkgBegin(); I
.end() != true; I
++,Done
++)
111 Prog
->Progress(Done
);
113 // Find the proper cache slot
114 StateCache
&State
= PkgState
[I
->ID
];
117 // Figure out the install version
118 State
.CandidateVer
= GetCandidateVer(I
);
119 State
.InstallVer
= I
.CurrentVer();
120 State
.Mode
= ModeKeep
;
122 State
.Update(I
,*this);
128 Prog
->OverallProgress(Head().PackageCount
,2*Head().PackageCount
,
130 _("Building dependency tree"));
131 Prog
->SubProgress(Head().PackageCount
,_("Dependency generation"));
143 bool pkgDepCache::readStateFile(OpProgress
*Prog
)
146 string state
= _config
->FindDir("Dir::State") + "extended_states";
147 if(FileExists(state
)) {
148 state_file
.Open(state
, FileFd::ReadOnly
);
149 int file_size
= state_file
.Size();
151 Prog
->OverallProgress(0, file_size
, 1,
152 _("Reading state information"));
154 pkgTagFile
tagfile(&state_file
);
155 pkgTagSection section
;
157 while(tagfile
.Step(section
)) {
158 string pkgname
= section
.FindS("Package");
159 pkgCache::PkgIterator pkg
=Cache
->FindPkg(pkgname
);
160 // Silently ignore unknown packages and packages with no actual
162 if(!pkg
.end() && !pkg
.VersionList().end()) {
163 short reason
= section
.FindI("Auto-Installed", 0);
165 PkgState
[pkg
->ID
].Flags
|= Flag::Auto
;
166 if(_config
->FindB("Debug::pkgAutoRemove",false))
167 std::cout
<< "Auto-Installed : " << pkgname
<< std::endl
;
170 Prog
->OverallProgress(amt
, file_size
, 1,
171 _("Reading state information"));
174 Prog
->OverallProgress(file_size
, file_size
, 1,
175 _("Reading state information"));
182 bool pkgDepCache::writeStateFile(OpProgress
*prog
)
184 if(_config
->FindB("Debug::pkgAutoRemove",false))
185 std::clog
<< "pkgDepCache::writeStateFile()" << std::endl
;
188 string state
= _config
->FindDir("Dir::State") + "extended_states";
190 // if it does not exist, create a empty one
191 if(!FileExists(state
))
193 StateFile
.Open(state
, FileFd::WriteEmpty
);
198 if(!StateFile
.Open(state
, FileFd::ReadOnly
))
199 return _error
->Error(_("Failed to open StateFile %s"),
203 string outfile
= state
+ ".tmp";
204 if((OutFile
= fopen(outfile
.c_str(),"w")) == NULL
)
205 return _error
->Error(_("Failed to write temporary StateFile %s"),
208 // first merge with the existing sections
209 pkgTagFile
tagfile(&StateFile
);
210 pkgTagSection section
;
211 std::set
<string
> pkgs_seen
;
212 const char *nullreorderlist
[] = {0};
213 while(tagfile
.Step(section
)) {
214 string pkgname
= section
.FindS("Package");
215 // Silently ignore unknown packages and packages with no actual
217 pkgCache::PkgIterator pkg
=Cache
->FindPkg(pkgname
);
218 if(pkg
.end() || pkg
.VersionList().end())
220 bool oldAuto
= section
.FindI("Auto-Installed");
221 bool newAuto
= (PkgState
[pkg
->ID
].Flags
& Flag::Auto
);
222 if(_config
->FindB("Debug::pkgAutoRemove",false))
223 std::clog
<< "Update exisiting AutoInstall info: "
224 << pkg
.Name() << std::endl
;
225 TFRewriteData rewrite
[2];
226 rewrite
[0].Tag
= "Auto-Installed";
227 rewrite
[0].Rewrite
= newAuto
? "1" : "0";
228 rewrite
[0].NewTag
= 0;
230 TFRewrite(OutFile
, section
, nullreorderlist
, rewrite
);
231 fprintf(OutFile
,"\n");
232 pkgs_seen
.insert(pkgname
);
235 // then write the ones we have not seen yet
236 std::ostringstream ostr
;
237 for(pkgCache::PkgIterator pkg
=Cache
->PkgBegin(); !pkg
.end(); pkg
++) {
238 if(PkgState
[pkg
->ID
].Flags
& Flag::Auto
) {
239 if (pkgs_seen
.find(pkg
.Name()) != pkgs_seen
.end()) {
240 if(_config
->FindB("Debug::pkgAutoRemove",false))
241 std::clog
<< "Skipping already written " << pkg
.Name() << std::endl
;
244 if(_config
->FindB("Debug::pkgAutoRemove",false))
245 std::clog
<< "Writing new AutoInstall: "
246 << pkg
.Name() << std::endl
;
247 ostr
.str(string(""));
248 ostr
<< "Package: " << pkg
.Name()
249 << "\nAuto-Installed: 1\n\n";
250 fprintf(OutFile
,ostr
.str().c_str());
251 fprintf(OutFile
,"\n");
256 // move the outfile over the real file
257 rename(outfile
.c_str(), state
.c_str());
262 // DepCache::CheckDep - Checks a single dependency /*{{{*/
263 // ---------------------------------------------------------------------
264 /* This first checks the dependency against the main target package and
265 then walks along the package provides list and checks if each provides
266 will be installed then checks the provides against the dep. Res will be
267 set to the package which was used to satisfy the dep. */
268 bool pkgDepCache::CheckDep(DepIterator Dep
,int Type
,PkgIterator
&Res
)
270 Res
= Dep
.TargetPkg();
272 /* Check simple depends. A depends -should- never self match but
273 we allow it anyhow because dpkg does. Technically it is a packaging
274 bug. Conflicts may never self match */
275 if (Dep
.TargetPkg() != Dep
.ParentPkg() ||
276 (Dep
->Type
!= Dep::Conflicts
&& Dep
->Type
!= Dep::Obsoletes
))
278 PkgIterator Pkg
= Dep
.TargetPkg();
279 // Check the base package
280 if (Type
== NowVersion
&& Pkg
->CurrentVer
!= 0)
281 if (VS().CheckDep(Pkg
.CurrentVer().VerStr(),Dep
->CompareOp
,
282 Dep
.TargetVer()) == true)
285 if (Type
== InstallVersion
&& PkgState
[Pkg
->ID
].InstallVer
!= 0)
286 if (VS().CheckDep(PkgState
[Pkg
->ID
].InstVerIter(*this).VerStr(),
287 Dep
->CompareOp
,Dep
.TargetVer()) == true)
290 if (Type
== CandidateVersion
&& PkgState
[Pkg
->ID
].CandidateVer
!= 0)
291 if (VS().CheckDep(PkgState
[Pkg
->ID
].CandidateVerIter(*this).VerStr(),
292 Dep
->CompareOp
,Dep
.TargetVer()) == true)
296 if (Dep
->Type
== Dep::Obsoletes
)
299 // Check the providing packages
300 PrvIterator P
= Dep
.TargetPkg().ProvidesList();
301 PkgIterator Pkg
= Dep
.ParentPkg();
302 for (; P
.end() != true; P
++)
304 /* Provides may never be applied against the same package if it is
305 a conflicts. See the comment above. */
306 if (P
.OwnerPkg() == Pkg
&& Dep
->Type
== Dep::Conflicts
)
309 // Check if the provides is a hit
310 if (Type
== NowVersion
)
312 if (P
.OwnerPkg().CurrentVer() != P
.OwnerVer())
316 if (Type
== InstallVersion
)
318 StateCache
&State
= PkgState
[P
.OwnerPkg()->ID
];
319 if (State
.InstallVer
!= (Version
*)P
.OwnerVer())
323 if (Type
== CandidateVersion
)
325 StateCache
&State
= PkgState
[P
.OwnerPkg()->ID
];
326 if (State
.CandidateVer
!= (Version
*)P
.OwnerVer())
330 // Compare the versions.
331 if (VS().CheckDep(P
.ProvideVersion(),Dep
->CompareOp
,Dep
.TargetVer()) == true)
341 // DepCache::AddSizes - Add the packages sizes to the counters /*{{{*/
342 // ---------------------------------------------------------------------
343 /* Call with Mult = -1 to preform the inverse opration */
344 void pkgDepCache::AddSizes(const PkgIterator
&Pkg
,signed long Mult
)
346 StateCache
&P
= PkgState
[Pkg
->ID
];
348 if (Pkg
->VersionList
== 0)
351 if (Pkg
.State() == pkgCache::PkgIterator::NeedsConfigure
&&
355 // Compute the size data
356 if (P
.NewInstall() == true)
358 iUsrSize
+= (signed)(Mult
*P
.InstVerIter(*this)->InstalledSize
);
359 iDownloadSize
+= (signed)(Mult
*P
.InstVerIter(*this)->Size
);
364 if (Pkg
->CurrentVer
!= 0 &&
365 (P
.InstallVer
!= (Version
*)Pkg
.CurrentVer() ||
366 (P
.iFlags
& ReInstall
) == ReInstall
) && P
.InstallVer
!= 0)
368 iUsrSize
+= (signed)(Mult
*((signed)P
.InstVerIter(*this)->InstalledSize
-
369 (signed)Pkg
.CurrentVer()->InstalledSize
));
370 iDownloadSize
+= (signed)(Mult
*P
.InstVerIter(*this)->Size
);
375 if (Pkg
.State() == pkgCache::PkgIterator::NeedsUnpack
&&
378 iDownloadSize
+= (signed)(Mult
*P
.InstVerIter(*this)->Size
);
383 if (Pkg
->CurrentVer
!= 0 && P
.InstallVer
== 0)
385 iUsrSize
-= (signed)(Mult
*Pkg
.CurrentVer()->InstalledSize
);
390 // DepCache::AddStates - Add the package to the state counter /*{{{*/
391 // ---------------------------------------------------------------------
392 /* This routine is tricky to use, you must make sure that it is never
393 called twice for the same package. This means the Remove/Add section
394 should be as short as possible and not encompass any code that will
395 calld Remove/Add itself. Remember, dependencies can be circular so
396 while processing a dep for Pkg it is possible that Add/Remove
397 will be called on Pkg */
398 void pkgDepCache::AddStates(const PkgIterator
&Pkg
,int Add
)
400 StateCache
&State
= PkgState
[Pkg
->ID
];
402 // The Package is broken (either minimal dep or policy dep)
403 if ((State
.DepState
& DepInstMin
) != DepInstMin
)
405 if ((State
.DepState
& DepInstPolicy
) != DepInstPolicy
)
406 iPolicyBrokenCount
+= Add
;
409 if (Pkg
.State() != PkgIterator::NeedsNothing
)
413 if (Pkg
->CurrentVer
== 0)
415 if (State
.Mode
== ModeDelete
&&
416 (State
.iFlags
| Purge
) == Purge
&& Pkg
.Purge() == false)
419 if (State
.Mode
== ModeInstall
)
424 // Installed, no upgrade
425 if (State
.Status
== 0)
427 if (State
.Mode
== ModeDelete
)
430 if ((State
.iFlags
& ReInstall
) == ReInstall
)
436 // Alll 3 are possible
437 if (State
.Mode
== ModeDelete
)
439 if (State
.Mode
== ModeKeep
)
441 if (State
.Mode
== ModeInstall
)
445 // DepCache::BuildGroupOrs - Generate the Or group dep data /*{{{*/
446 // ---------------------------------------------------------------------
447 /* The or group results are stored in the last item of the or group. This
448 allows easy detection of the state of a whole or'd group. */
449 void pkgDepCache::BuildGroupOrs(VerIterator
const &V
)
451 unsigned char Group
= 0;
453 for (DepIterator D
= V
.DependsList(); D
.end() != true; D
++)
455 // Build the dependency state.
456 unsigned char &State
= DepState
[D
->ID
];
458 /* Invert for Conflicts. We have to do this twice to get the
459 right sense for a conflicts group */
460 if (D
->Type
== Dep::Conflicts
|| D
->Type
== Dep::Obsoletes
)
463 // Add to the group if we are within an or..
467 if ((D
->CompareOp
& Dep::Or
) != Dep::Or
)
470 // Invert for Conflicts
471 if (D
->Type
== Dep::Conflicts
|| D
->Type
== Dep::Obsoletes
)
476 // DepCache::VersionState - Perform a pass over a dependency list /*{{{*/
477 // ---------------------------------------------------------------------
478 /* This is used to run over a dependency list and determine the dep
479 state of the list, filtering it through both a Min check and a Policy
480 check. The return result will have SetMin/SetPolicy low if a check
481 fails. It uses the DepState cache for it's computations. */
482 unsigned char pkgDepCache::VersionState(DepIterator D
,unsigned char Check
,
483 unsigned char SetMin
,
484 unsigned char SetPolicy
)
486 unsigned char Dep
= 0xFF;
488 while (D
.end() != true)
490 // Compute a single dependency element (glob or)
491 DepIterator Start
= D
;
492 unsigned char State
= 0;
493 for (bool LastOR
= true; D
.end() == false && LastOR
== true; D
++)
495 State
|= DepState
[D
->ID
];
496 LastOR
= (D
->CompareOp
& Dep::Or
) == Dep::Or
;
499 // Minimum deps that must be satisfied to have a working package
500 if (Start
.IsCritical() == true)
501 if ((State
& Check
) != Check
)
504 // Policy deps that must be satisfied to install the package
505 if (IsImportantDep(Start
) == true &&
506 (State
& Check
) != Check
)
513 // DepCache::DependencyState - Compute the 3 results for a dep /*{{{*/
514 // ---------------------------------------------------------------------
515 /* This is the main dependency computation bit. It computes the 3 main
516 results for a dependencys, Now, Install and Candidate. Callers must
517 invert the result if dealing with conflicts. */
518 unsigned char pkgDepCache::DependencyState(DepIterator
&D
)
520 unsigned char State
= 0;
522 if (CheckDep(D
,NowVersion
) == true)
524 if (CheckDep(D
,InstallVersion
) == true)
526 if (CheckDep(D
,CandidateVersion
) == true)
532 // DepCache::UpdateVerState - Compute the Dep member of the state /*{{{*/
533 // ---------------------------------------------------------------------
534 /* This determines the combined dependency representation of a package
535 for its two states now and install. This is done by using the pre-generated
536 dependency information. */
537 void pkgDepCache::UpdateVerState(PkgIterator Pkg
)
539 // Empty deps are always true
540 StateCache
&State
= PkgState
[Pkg
->ID
];
541 State
.DepState
= 0xFF;
543 // Check the Current state
544 if (Pkg
->CurrentVer
!= 0)
546 DepIterator D
= Pkg
.CurrentVer().DependsList();
547 State
.DepState
&= VersionState(D
,DepNow
,DepNowMin
,DepNowPolicy
);
550 /* Check the candidate state. We do not compare against the whole as
551 a candidate state but check the candidate version against the
553 if (State
.CandidateVer
!= 0)
555 DepIterator D
= State
.CandidateVerIter(*this).DependsList();
556 State
.DepState
&= VersionState(D
,DepInstall
,DepCandMin
,DepCandPolicy
);
559 // Check target state which can only be current or installed
560 if (State
.InstallVer
!= 0)
562 DepIterator D
= State
.InstVerIter(*this).DependsList();
563 State
.DepState
&= VersionState(D
,DepInstall
,DepInstMin
,DepInstPolicy
);
567 // DepCache::Update - Figure out all the state information /*{{{*/
568 // ---------------------------------------------------------------------
569 /* This will figure out the state of all the packages and all the
570 dependencies based on the current policy. */
571 void pkgDepCache::Update(OpProgress
*Prog
)
581 // Perform the depends pass
583 for (PkgIterator I
= PkgBegin(); I
.end() != true; I
++,Done
++)
585 if (Prog
!= 0 && Done%20
== 0)
586 Prog
->Progress(Done
);
587 for (VerIterator V
= I
.VersionList(); V
.end() != true; V
++)
589 unsigned char Group
= 0;
591 for (DepIterator D
= V
.DependsList(); D
.end() != true; D
++)
593 // Build the dependency state.
594 unsigned char &State
= DepState
[D
->ID
];
595 State
= DependencyState(D
);
597 // Add to the group if we are within an or..
600 if ((D
->CompareOp
& Dep::Or
) != Dep::Or
)
603 // Invert for Conflicts
604 if (D
->Type
== Dep::Conflicts
|| D
->Type
== Dep::Obsoletes
)
609 // Compute the pacakge dependency state and size additions
616 Prog
->Progress(Done
);
621 // DepCache::Update - Update the deps list of a package /*{{{*/
622 // ---------------------------------------------------------------------
623 /* This is a helper for update that only does the dep portion of the scan.
624 It is mainly meant to scan reverse dependencies. */
625 void pkgDepCache::Update(DepIterator D
)
627 // Update the reverse deps
628 for (;D
.end() != true; D
++)
630 unsigned char &State
= DepState
[D
->ID
];
631 State
= DependencyState(D
);
633 // Invert for Conflicts
634 if (D
->Type
== Dep::Conflicts
|| D
->Type
== Dep::Obsoletes
)
637 RemoveStates(D
.ParentPkg());
638 BuildGroupOrs(D
.ParentVer());
639 UpdateVerState(D
.ParentPkg());
640 AddStates(D
.ParentPkg());
644 // DepCache::Update - Update the related deps of a package /*{{{*/
645 // ---------------------------------------------------------------------
646 /* This is called whenever the state of a package changes. It updates
647 all cached dependencies related to this package. */
648 void pkgDepCache::Update(PkgIterator
const &Pkg
)
650 // Recompute the dep of the package
655 // Update the reverse deps
656 Update(Pkg
.RevDependsList());
658 // Update the provides map for the current ver
659 if (Pkg
->CurrentVer
!= 0)
660 for (PrvIterator P
= Pkg
.CurrentVer().ProvidesList();
661 P
.end() != true; P
++)
662 Update(P
.ParentPkg().RevDependsList());
664 // Update the provides map for the candidate ver
665 if (PkgState
[Pkg
->ID
].CandidateVer
!= 0)
666 for (PrvIterator P
= PkgState
[Pkg
->ID
].CandidateVerIter(*this).ProvidesList();
667 P
.end() != true; P
++)
668 Update(P
.ParentPkg().RevDependsList());
673 // DepCache::MarkKeep - Put the package in the keep state /*{{{*/
674 // ---------------------------------------------------------------------
676 void pkgDepCache::MarkKeep(PkgIterator
const &Pkg
, bool Soft
, bool FromUser
)
678 // Simplifies other routines.
679 if (Pkg
.end() == true)
682 /* Reject an attempt to keep a non-source broken installed package, those
684 if (Pkg
.State() == PkgIterator::NeedsUnpack
&&
685 Pkg
.CurrentVer().Downloadable() == false)
688 /** \todo Can this be moved later in the method? */
689 ActionGroup
group(*this);
691 /* We changed the soft state all the time so the UI is a bit nicer
693 StateCache
&P
= PkgState
[Pkg
->ID
];
695 P
.iFlags
|= AutoKept
;
697 P
.iFlags
&= ~AutoKept
;
699 // Check that it is not already kept
700 if (P
.Mode
== ModeKeep
)
703 // We dont even try to keep virtual packages..
704 if (Pkg
->VersionList
== 0)
707 if(FromUser
&& !P
.Marked
)
708 P
.Flags
&= ~Flag::Auto
;
713 if (Pkg
->CurrentVer
== 0)
716 P
.InstallVer
= Pkg
.CurrentVer();
725 // DepCache::MarkDelete - Put the package in the delete state /*{{{*/
726 // ---------------------------------------------------------------------
728 void pkgDepCache::MarkDelete(PkgIterator
const &Pkg
, bool rPurge
)
730 // Simplifies other routines.
731 if (Pkg
.end() == true)
734 ActionGroup
group(*this);
736 // Check that it is not already marked for delete
737 StateCache
&P
= PkgState
[Pkg
->ID
];
738 P
.iFlags
&= ~(AutoKept
| Purge
);
742 if ((P
.Mode
== ModeDelete
|| P
.InstallVer
== 0) &&
743 (Pkg
.Purge() == true || rPurge
== false))
746 // We dont even try to delete virtual packages..
747 if (Pkg
->VersionList
== 0)
753 if (Pkg
->CurrentVer
== 0 && (Pkg
.Purge() == true || rPurge
== false))
764 // DepCache::MarkInstall - Put the package in the install state /*{{{*/
765 // ---------------------------------------------------------------------
767 void pkgDepCache::MarkInstall(PkgIterator
const &Pkg
,bool AutoInst
,
768 unsigned long Depth
, bool FromUser
,
769 bool ForceImportantDeps
)
774 // Simplifies other routines.
775 if (Pkg
.end() == true)
778 ActionGroup
group(*this);
780 /* Check that it is not already marked for install and that it can be
782 StateCache
&P
= PkgState
[Pkg
->ID
];
783 P
.iFlags
&= ~AutoKept
;
784 if ((P
.InstPolicyBroken() == false && P
.InstBroken() == false) &&
785 (P
.Mode
== ModeInstall
||
786 P
.CandidateVer
== (Version
*)Pkg
.CurrentVer()))
788 if (P
.CandidateVer
== (Version
*)Pkg
.CurrentVer() && P
.InstallVer
== 0)
789 MarkKeep(Pkg
, false, FromUser
);
793 // See if there is even any possible instalation candidate
794 if (P
.CandidateVer
== 0)
796 // We dont even try to install virtual packages..
797 if (Pkg
->VersionList
== 0)
799 /* Target the candidate version and remove the autoflag. We reset the
800 autoflag below if this was called recursively. Otherwise the user
801 should have the ability to de-auto a package by changing its state */
805 P
.Mode
= ModeInstall
;
806 P
.InstallVer
= P
.CandidateVer
;
810 // Set it to manual if it's a new install or cancelling the
811 // removal of a garbage package.
812 if(P
.Status
== 2 || (!Pkg
.CurrentVer().end() && !P
.Marked
))
813 P
.Flags
&= ~Flag::Auto
;
817 // Set it to auto if this is a new install.
819 P
.Flags
|= Flag::Auto
;
821 if (P
.CandidateVer
== (Version
*)Pkg
.CurrentVer())
828 if (AutoInst
== false)
831 DepIterator Dep
= P
.InstVerIter(*this).DependsList();
832 for (; Dep
.end() != true;)
835 DepIterator Start
= Dep
;
838 for (bool LastOR
= true; Dep
.end() == false && LastOR
== true; Dep
++,Ors
++)
840 LastOR
= (Dep
->CompareOp
& Dep::Or
) == Dep::Or
;
842 if ((DepState
[Dep
->ID
] & DepInstall
) == DepInstall
)
846 // Dep is satisfied okay.
850 /* Check if this dep should be consider for install. If it is a user
851 defined important dep and we are installed a new package then
852 it will be installed. Otherwise we only check for important
853 deps that have changed from the installed version
855 if (IsImportantDep(Start
) == false)
858 /* check if any ImportantDep() (but not Critial) where added
859 * since we installed the package
861 bool isNewImportantDep
= false;
862 if(!ForceImportantDeps
&& !Start
.IsCritical())
865 VerIterator instVer
= Pkg
.CurrentVer();
867 for (DepIterator D
= instVer
.DependsList(); D
.end() != true; D
++)
869 //FIXME: deal better with or-groups(?)
870 DepIterator LocalStart
= D
;
872 if(IsImportantDep(D
) && Start
.TargetPkg() == D
.TargetPkg())
875 // this is a new dep if it was not found to be already
876 // a important dep of the installed pacakge
877 isNewImportantDep
= !found
;
879 if(isNewImportantDep
)
880 if(_config
->FindB("Debug::pkgDepCache::AutoInstall",false) == true)
881 std::clog
<< "new important dependency: "
882 << Start
.TargetPkg().Name() << std::endl
;
884 // skip important deps if the package is already installed
885 if (Pkg
->CurrentVer
!= 0 && Start
.IsCritical() == false
886 && !isNewImportantDep
&& !ForceImportantDeps
)
889 /* If we are in an or group locate the first or that can
890 succeed. We have already cached this.. */
891 for (; Ors
> 1 && (DepState
[Start
->ID
] & DepCVer
) != DepCVer
; Ors
--)
894 /* This bit is for processing the possibilty of an install/upgrade
895 fixing the problem */
896 SPtrArray
<Version
*> List
= Start
.AllTargets();
897 if ((DepState
[Start
->ID
] & DepCVer
) == DepCVer
)
899 // Right, find the best version to install..
900 Version
**Cur
= List
;
901 PkgIterator P
= Start
.TargetPkg();
902 PkgIterator
InstPkg(*Cache
,0);
904 // See if there are direct matches (at the start of the list)
905 for (; *Cur
!= 0 && (*Cur
)->ParentPkg
== P
.Index(); Cur
++)
907 PkgIterator
Pkg(*Cache
,Cache
->PkgP
+ (*Cur
)->ParentPkg
);
908 if (PkgState
[Pkg
->ID
].CandidateVer
!= *Cur
)
914 // Select the highest priority providing package
915 if (InstPkg
.end() == true)
917 pkgPrioSortList(*Cache
,Cur
);
918 for (; *Cur
!= 0; Cur
++)
920 PkgIterator
Pkg(*Cache
,Cache
->PkgP
+ (*Cur
)->ParentPkg
);
921 if (PkgState
[Pkg
->ID
].CandidateVer
!= *Cur
)
928 if (InstPkg
.end() == false)
930 if(_config
->FindB("Debug::pkgDepCache::AutoInstall",false) == true)
931 std::clog
<< "Installing " << InstPkg
.Name()
932 << " as dep of " << Pkg
.Name()
934 MarkInstall(InstPkg
,true,Depth
+ 1, false, ForceImportantDeps
);
936 // Set the autoflag, after MarkInstall because MarkInstall unsets it
937 if (P
->CurrentVer
== 0)
938 PkgState
[InstPkg
->ID
].Flags
|= Flag::Auto
;
943 /* For conflicts we just de-install the package and mark as auto,
944 Conflicts may not have or groups */
945 if (Start
->Type
== Dep::Conflicts
|| Start
->Type
== Dep::Obsoletes
)
947 for (Version
**I
= List
; *I
!= 0; I
++)
949 VerIterator
Ver(*this,*I
);
950 PkgIterator Pkg
= Ver
.ParentPkg();
959 // DepCache::SetReInstall - Set the reinstallation flag /*{{{*/
960 // ---------------------------------------------------------------------
962 void pkgDepCache::SetReInstall(PkgIterator
const &Pkg
,bool To
)
964 ActionGroup
group(*this);
969 StateCache
&P
= PkgState
[Pkg
->ID
];
971 P
.iFlags
|= ReInstall
;
973 P
.iFlags
&= ~ReInstall
;
979 // DepCache::SetCandidateVersion - Change the candidate version /*{{{*/
980 // ---------------------------------------------------------------------
982 void pkgDepCache::SetCandidateVersion(VerIterator TargetVer
)
984 ActionGroup
group(*this);
986 pkgCache::PkgIterator Pkg
= TargetVer
.ParentPkg();
987 StateCache
&P
= PkgState
[Pkg
->ID
];
992 if (P
.CandidateVer
== P
.InstallVer
)
993 P
.InstallVer
= (Version
*)TargetVer
;
994 P
.CandidateVer
= (Version
*)TargetVer
;
1002 void pkgDepCache::MarkAuto(const PkgIterator
&Pkg
, bool Auto
)
1004 StateCache
&state
= PkgState
[Pkg
->ID
];
1006 ActionGroup
group(*this);
1009 state
.Flags
|= Flag::Auto
;
1011 state
.Flags
&= ~Flag::Auto
;
1014 // StateCache::Update - Compute the various static display things /*{{{*/
1015 // ---------------------------------------------------------------------
1016 /* This is called whenever the Candidate version changes. */
1017 void pkgDepCache::StateCache::Update(PkgIterator Pkg
,pkgCache
&Cache
)
1020 VerIterator Ver
= CandidateVerIter(Cache
);
1022 // Use a null string or the version string
1023 if (Ver
.end() == true)
1026 CandVersion
= Ver
.VerStr();
1028 // Find the current version
1030 if (Pkg
->CurrentVer
!= 0)
1031 CurVersion
= Pkg
.CurrentVer().VerStr();
1033 // Strip off the epochs for display
1034 CurVersion
= StripEpoch(CurVersion
);
1035 CandVersion
= StripEpoch(CandVersion
);
1037 // Figure out if its up or down or equal
1038 Status
= Ver
.CompareVer(Pkg
.CurrentVer());
1039 if (Pkg
->CurrentVer
== 0 || Pkg
->VersionList
== 0 || CandidateVer
== 0)
1043 // StateCache::StripEpoch - Remove the epoch specifier from the version /*{{{*/
1044 // ---------------------------------------------------------------------
1046 const char *pkgDepCache::StateCache::StripEpoch(const char *Ver
)
1052 for (const char *I
= Ver
; *I
!= 0; I
++)
1059 // Policy::GetCandidateVer - Returns the Candidate install version /*{{{*/
1060 // ---------------------------------------------------------------------
1061 /* The default just returns the highest available version that is not
1062 a source and automatic. */
1063 pkgCache::VerIterator
pkgDepCache::Policy::GetCandidateVer(PkgIterator Pkg
)
1065 /* Not source/not automatic versions cannot be a candidate version
1066 unless they are already installed */
1067 VerIterator
Last(*(pkgCache
*)this,0);
1069 for (VerIterator I
= Pkg
.VersionList(); I
.end() == false; I
++)
1071 if (Pkg
.CurrentVer() == I
)
1074 for (VerFileIterator J
= I
.FileList(); J
.end() == false; J
++)
1076 if ((J
.File()->Flags
& Flag::NotSource
) != 0)
1079 /* Stash the highest version of a not-automatic source, we use it
1080 if there is nothing better */
1081 if ((J
.File()->Flags
& Flag::NotAutomatic
) != 0)
1083 if (Last
.end() == true)
1097 pkgDepCache::DefaultRootSetFunc::DefaultRootSetFunc()
1098 : constructedSuccessfully(false)
1100 Configuration::Item
const *Opts
;
1101 Opts
= _config
->Tree("APT::NeverAutoRemove");
1102 if (Opts
!= 0 && Opts
->Child
!= 0)
1105 for (; Opts
!= 0; Opts
= Opts
->Next
)
1107 if (Opts
->Value
.empty() == true)
1110 regex_t
*p
= new regex_t
;
1111 if(regcomp(p
,Opts
->Value
.c_str(),
1112 REG_EXTENDED
| REG_ICASE
| REG_NOSUB
) != 0)
1116 _error
->Error("Regex compilation error for APT::NeverAutoRemove");
1120 rootSetRegexp
.push_back(p
);
1124 constructedSuccessfully
= true;
1127 pkgDepCache::DefaultRootSetFunc::~DefaultRootSetFunc()
1129 for(unsigned int i
= 0; i
< rootSetRegexp
.size(); i
++)
1131 regfree(rootSetRegexp
[i
]);
1132 delete rootSetRegexp
[i
];
1137 bool pkgDepCache::DefaultRootSetFunc::InRootSet(const pkgCache::PkgIterator
&pkg
)
1139 for(unsigned int i
= 0; i
< rootSetRegexp
.size(); i
++)
1140 if (regexec(rootSetRegexp
[i
], pkg
.Name(), 0, 0, 0) == 0)
1146 pkgDepCache::InRootSetFunc
*pkgDepCache::GetRootSetFunc()
1148 DefaultRootSetFunc
*f
= new DefaultRootSetFunc
;
1149 if(f
->wasConstructedSuccessfully())
1158 bool pkgDepCache::MarkFollowsRecommends()
1160 return _config
->FindB("APT::AutoRemove::RecommendsImportant", true);
1163 bool pkgDepCache::MarkFollowsSuggests()
1165 return _config
->FindB("APT::AutoRemove::SuggestsImportant", false);
1168 // the main mark algorithm
1169 bool pkgDepCache::MarkRequired(InRootSetFunc
&userFunc
)
1171 bool follow_recommends
;
1172 bool follow_suggests
;
1175 for(PkgIterator p
= PkgBegin(); !p
.end(); ++p
)
1177 PkgState
[p
->ID
].Marked
= false;
1178 PkgState
[p
->ID
].Garbage
= false;
1181 if(_config
->FindB("Debug::pkgAutoRemove",false)
1182 && PkgState
[p
->ID
].Flags
& Flag::Auto
)
1183 std::clog
<< "AutoDep: " << p
.Name() << std::endl
;
1187 follow_recommends
= MarkFollowsRecommends();
1188 follow_suggests
= MarkFollowsSuggests();
1192 // do the mark part, this is the core bit of the algorithm
1193 for(PkgIterator p
= PkgBegin(); !p
.end(); ++p
)
1195 if(!(PkgState
[p
->ID
].Flags
& Flag::Auto
) ||
1196 (p
->Flags
& Flag::Essential
) ||
1197 userFunc
.InRootSet(p
))
1200 // the package is installed (and set to keep)
1201 if(PkgState
[p
->ID
].Keep() && !p
.CurrentVer().end())
1202 MarkPackage(p
, p
.CurrentVer(),
1203 follow_recommends
, follow_suggests
);
1204 // the package is to be installed
1205 else if(PkgState
[p
->ID
].Install())
1206 MarkPackage(p
, PkgState
[p
->ID
].InstVerIter(*this),
1207 follow_recommends
, follow_suggests
);
1214 // mark a single package in Mark-and-Sweep
1215 void pkgDepCache::MarkPackage(const pkgCache::PkgIterator
&pkg
,
1216 const pkgCache::VerIterator
&ver
,
1217 bool follow_recommends
,
1218 bool follow_suggests
)
1220 pkgDepCache::StateCache
&state
= PkgState
[pkg
->ID
];
1221 VerIterator candver
= state
.CandidateVerIter(*this);
1222 VerIterator instver
= state
.InstVerIter(*this);
1225 // If a package was garbage-collected but is now being marked, we
1226 // should re-select it
1227 // For cases when a pkg is set to upgrade and this trigger the
1228 // removal of a no-longer used dependency. if the pkg is set to
1229 // keep again later it will result in broken deps
1230 if(state
.Delete() && state
.RemoveReason
= Unused
)
1233 mark_install(pkg
, false, false, NULL
);
1234 else if(ver
==pkg
.CurrentVer())
1235 MarkKeep(pkg
, false, false);
1237 instver
=state
.InstVerIter(*this);
1241 // Ignore versions other than the InstVer, and ignore packages
1242 // that are already going to be removed or just left uninstalled.
1243 if(!(ver
== instver
&& !instver
.end()))
1246 // if we are marked already we are done
1250 //std::cout << "Setting Marked for: " << pkg.Name() << std::endl;
1255 for(DepIterator d
= ver
.DependsList(); !d
.end(); ++d
)
1257 if(d
->Type
== Dep::Depends
||
1258 d
->Type
== Dep::PreDepends
||
1259 (follow_recommends
&&
1260 d
->Type
== Dep::Recommends
) ||
1262 d
->Type
== Dep::Suggests
))
1264 // Try all versions of this package.
1265 for(VerIterator V
= d
.TargetPkg().VersionList();
1268 if(_system
->VS
->CheckDep(V
.VerStr(), d
->CompareOp
, d
.TargetVer()))
1270 MarkPackage(V
.ParentPkg(), V
,
1271 follow_recommends
, follow_suggests
);
1274 // Now try virtual packages
1275 for(PrvIterator prv
=d
.TargetPkg().ProvidesList();
1278 if(_system
->VS
->CheckDep(prv
.ProvideVersion(), d
->CompareOp
,
1281 MarkPackage(prv
.OwnerPkg(), prv
.OwnerVer(),
1282 follow_recommends
, follow_suggests
);
1290 bool pkgDepCache::Sweep()
1293 for(PkgIterator p
=PkgBegin(); !p
.end(); ++p
)
1295 StateCache
&state
=PkgState
[p
->ID
];
1297 // if it is not marked and it is installed, it's garbage
1298 if(!state
.Marked
&& (!p
.CurrentVer().end() || state
.Install()) &&
1302 if(_config
->FindB("Debug::pkgAutoRemove",false))
1303 std::cout
<< "Garbage: " << p
.Name() << std::endl
;
1310 // Policy::IsImportantDep - True if the dependency is important /*{{{*/
1311 // ---------------------------------------------------------------------
1313 bool pkgDepCache::Policy::IsImportantDep(DepIterator Dep
)
1315 if(Dep
.IsCritical())
1317 else if(Dep
->Type
== pkgCache::Dep::Recommends
)
1318 return _config
->FindB("APT::Install-Recommends", false);
1319 else if(Dep
->Type
== pkgCache::Dep::Suggests
)
1320 return _config
->FindB("APT::Install-Suggests", false);