]> git.saurik.com Git - apt.git/commitdiff
merge from the debian-sid branch
authorDavid Kalnischkies <kalnischkies@gmail.com>
Tue, 4 May 2010 11:26:49 +0000 (13:26 +0200)
committerDavid Kalnischkies <kalnischkies@gmail.com>
Tue, 4 May 2010 11:26:49 +0000 (13:26 +0200)
1  2 
apt-pkg/acquire-item.cc
apt-pkg/depcache.cc
apt-pkg/policy.cc
debian/changelog

diff --combined apt-pkg/acquire-item.cc
index 916fca71eeaaf0e1722129c71d90010e36c2cc23,6d4336425482bfa01d1d0deef027078a68f75455..1f253bb81b96966d3ca9982aaaa713618eb9d086
@@@ -1327,8 -1327,7 +1327,8 @@@ pkgAcqArchive::pkgAcqArchive(pkgAcquir
     the archive is already available in the cache and stashs the MD5 for
     checking later. */
  bool pkgAcqArchive::QueueNext()
 -{   
 +{
 +   string const ForceHash = _config->Find("Acquire::ForceHash");
     for (; Vf.end() == false; Vf++)
     {
        // Ignore not source sources
         return false;
        
        string PkgFile = Parse.FileName();
 -      if(Parse.SHA256Hash() != "")
 -       ExpectedHash = HashString("SHA256", Parse.SHA256Hash());
 -      else if (Parse.SHA1Hash() != "")
 -       ExpectedHash = HashString("SHA1", Parse.SHA1Hash());
 -      else 
 -       ExpectedHash = HashString("MD5Sum", Parse.MD5Hash());
 +      if (ForceHash.empty() == false)
 +      {
 +       if(stringcasecmp(ForceHash, "sha256") == 0)
 +          ExpectedHash = HashString("SHA256", Parse.SHA256Hash());
 +       else if (stringcasecmp(ForceHash, "sha1") == 0)
 +          ExpectedHash = HashString("SHA1", Parse.SHA1Hash());
 +       else
 +          ExpectedHash = HashString("MD5Sum", Parse.MD5Hash());
 +      }
 +      else
 +      {
 +       string Hash;
 +       if ((Hash = Parse.SHA256Hash()).empty() == false)
 +          ExpectedHash = HashString("SHA256", Hash);
 +       else if ((Hash = Parse.SHA1Hash()).empty() == false)
 +          ExpectedHash = HashString("SHA1", Hash);
 +       else
 +          ExpectedHash = HashString("MD5Sum", Parse.MD5Hash());
 +      }
        if (PkgFile.empty() == true)
         return _error->Error(_("The package index files are corrupted. No Filename: "
                              "field for package %s."),
@@@ -1561,8 -1547,9 +1561,9 @@@ void pkgAcqArchive::Finished(
  /* The file is added to the queue */
  pkgAcqFile::pkgAcqFile(pkgAcquire *Owner,string URI,string Hash,
                       unsigned long Size,string Dsc,string ShortDesc,
-                      const string &DestDir, const string &DestFilename) :
-                        Item(Owner), ExpectedHash(Hash)
+                      const string &DestDir, const string &DestFilename,
+                        bool IsIndexFile) :
+                        Item(Owner), ExpectedHash(Hash), IsIndexFile(IsIndexFile)
  {
     Retries = _config->FindI("Acquire::Retries",0);
     
@@@ -1677,3 -1664,12 +1678,12 @@@ void pkgAcqFile::Failed(string Message,
     Item::Failed(Message,Cnf);
  }
                                                                        /*}}}*/
+ // AcqIndex::Custom600Headers - Insert custom request headers         /*{{{*/
+ // ---------------------------------------------------------------------
+ /* The only header we use is the last-modified header. */
+ string pkgAcqFile::Custom600Headers()
+ {
+    if (IsIndexFile)
+       return "\nIndex-File: true";
+ }
+                                                                       /*}}}*/
diff --combined apt-pkg/depcache.cc
index 659c4227e3f9245e8df6abafa573c285caed7ebf,0f30f13c244f824fb7aed35e6b836f7b1ca859d9..4d1a08eb628da10d00a6f4c379dda58c07afc243
@@@ -17,7 -17,6 +17,7 @@@
  #include <apt-pkg/fileutl.h>
  #include <apt-pkg/strutl.h>
  #include <apt-pkg/configuration.h>
 +#include <apt-pkg/aptconfiguration.h>
  #include <apt-pkg/pkgsystem.h>
  #include <apt-pkg/tagfile.h>
  
@@@ -165,46 -164,34 +165,46 @@@ bool pkgDepCache::Init(OpProgress *Prog
  bool pkgDepCache::readStateFile(OpProgress *Prog)                     /*{{{*/
  {
     FileFd state_file;
 -   string state = _config->FindDir("Dir::State") + "extended_states";
 +   string const state = _config->FindDir("Dir::State") + "extended_states";
     if(FileExists(state)) {
        state_file.Open(state, FileFd::ReadOnly);
 -      int file_size = state_file.Size();
 +      int const file_size = state_file.Size();
        if(Prog != NULL)
         Prog->OverallProgress(0, file_size, 1, 
                               _("Reading state information"));
  
        pkgTagFile tagfile(&state_file);
        pkgTagSection section;
 -      int amt=0;
 -      bool debug_autoremove=_config->FindB("Debug::pkgAutoRemove",false);
 +      int amt = 0;
 +      bool const debug_autoremove = _config->FindB("Debug::pkgAutoRemove",false);
        while(tagfile.Step(section)) {
 -       string pkgname = section.FindS("Package");
 -       pkgCache::PkgIterator pkg=Cache->FindPkg(pkgname);
 -       // Silently ignore unknown packages and packages with no actual
 -       // version.
 -       if(!pkg.end() && !pkg.VersionList().end()) {
 -          short reason = section.FindI("Auto-Installed", 0);
 -          if(reason > 0)
 -             PkgState[pkg->ID].Flags  |= Flag::Auto;
 -          if(debug_autoremove)
 -             std::clog << "Auto-Installed : " << pkgname << std::endl;
 -          amt+=section.size();
 -          if(Prog != NULL)
 -             Prog->OverallProgress(amt, file_size, 1, 
 -                                   _("Reading state information"));
 +       string const pkgname = section.FindS("Package");
 +       string pkgarch = section.FindS("Architecture");
 +       if (pkgarch.empty() == true)
 +          pkgarch = "any";
 +       pkgCache::PkgIterator pkg = Cache->FindPkg(pkgname, pkgarch);
 +       // Silently ignore unknown packages and packages with no actual version.
 +       if(pkg.end() == true || pkg->VersionList == 0)
 +          continue;
 +
 +       short const reason = section.FindI("Auto-Installed", 0);
 +       if(reason > 0)
 +       {
 +          PkgState[pkg->ID].Flags |= Flag::Auto;
 +          if (unlikely(debug_autoremove))
-              std::cout << "Auto-Installed : " << pkg.FullName() << std::endl;
++             std::clog << "Auto-Installed : " << pkg.FullName() << std::endl;
 +          if (pkgarch == "any")
 +          {
 +             pkgCache::GrpIterator G = pkg.Group();
 +             for (pkg = G.NextPkg(pkg); pkg.end() != true; pkg = G.NextPkg(pkg))
 +                if (pkg->VersionList != 0)
 +                   PkgState[pkg->ID].Flags |= Flag::Auto;
 +          }
         }
 +       amt += section.size();
 +       if(Prog != NULL)
 +          Prog->OverallProgress(amt, file_size, 1, 
 +                                _("Reading state information"));
        }
        if(Prog != NULL)
         Prog->OverallProgress(file_size, file_size, 1,
                                                                        /*}}}*/
  bool pkgDepCache::writeStateFile(OpProgress *prog, bool InstalledOnly)        /*{{{*/
  {
 -   bool debug_autoremove = _config->FindB("Debug::pkgAutoRemove",false);
 +   bool const debug_autoremove = _config->FindB("Debug::pkgAutoRemove",false);
     
     if(debug_autoremove)
        std::clog << "pkgDepCache::writeStateFile()" << std::endl;
  
     FileFd StateFile;
 -   string state = _config->FindDir("Dir::State") + "extended_states";
 +   string const state = _config->FindDir("Dir::State") + "extended_states";
  
     // if it does not exist, create a empty one
     if(!FileExists(state)) 
                           state.c_str());
  
     FILE *OutFile;
 -   string outfile = state + ".tmp";
 +   string const outfile = state + ".tmp";
     if((OutFile = fopen(outfile.c_str(),"w")) == NULL)
        return _error->Error(_("Failed to write temporary StateFile %s"),
                           outfile.c_str());
     std::set<string> pkgs_seen;
     const char *nullreorderlist[] = {0};
     while(tagfile.Step(section)) {
 -       string pkgname = section.FindS("Package");
 +       string const pkgname = section.FindS("Package");
 +       string pkgarch = section.FindS("Architecture");
 +       if (pkgarch.empty() == true)
 +          pkgarch = "native";
         // Silently ignore unknown packages and packages with no actual
         // version.
 -       pkgCache::PkgIterator pkg=Cache->FindPkg(pkgname);
 +       pkgCache::PkgIterator pkg = Cache->FindPkg(pkgname, pkgarch);
         if(pkg.end() || pkg.VersionList().end()) 
            continue;
 -       bool newAuto = (PkgState[pkg->ID].Flags & Flag::Auto);
 +       StateCache const &P = PkgState[pkg->ID];
 +       bool newAuto = (P.Flags & Flag::Auto);
 +       // skip not installed or now-removed ones if requested
 +       if (InstalledOnly && (
 +           (pkg->CurrentVer == 0 && P.Mode != ModeInstall) ||
 +           (pkg->CurrentVer != 0 && P.Mode == ModeDelete)))
 +       {
 +          // The section is obsolete if it contains no other tag
 +          unsigned int const count = section.Count();
 +          if (count < 2 ||
 +              (count == 2 && section.Exists("Auto-Installed")) ||
 +              (count == 3 && section.Exists("Auto-Installed") && section.Exists("Architecture")))
 +             continue;
 +          else
 +             newAuto = false;
 +       }
         if(_config->FindB("Debug::pkgAutoRemove",false))
            std::clog << "Update existing AutoInstall info: " 
 -                    << pkg.Name() << std::endl;
 -       TFRewriteData rewrite[2];
 -       rewrite[0].Tag = "Auto-Installed";
 -       rewrite[0].Rewrite = newAuto ? "1" : "0";
 +                    << pkg.FullName() << std::endl;
 +       TFRewriteData rewrite[3];
 +       rewrite[0].Tag = "Architecture";
 +       rewrite[0].Rewrite = pkg.Arch();
         rewrite[0].NewTag = 0;
 -       rewrite[1].Tag = 0;
 +       rewrite[1].Tag = "Auto-Installed";
 +       rewrite[1].Rewrite = newAuto ? "1" : "0";
 +       rewrite[1].NewTag = 0;
 +       rewrite[2].Tag = 0;
         TFRewrite(OutFile, section, nullreorderlist, rewrite);
         fprintf(OutFile,"\n");
 -       pkgs_seen.insert(pkgname);
 +       pkgs_seen.insert(pkg.FullName());
     }
     
     // then write the ones we have not seen yet
     std::ostringstream ostr;
     for(pkgCache::PkgIterator pkg=Cache->PkgBegin(); !pkg.end(); pkg++) {
 -      if(PkgState[pkg->ID].Flags & Flag::Auto) {
 -       if (pkgs_seen.find(pkg.Name()) != pkgs_seen.end()) {
 +      StateCache const &P = PkgState[pkg->ID];
 +      if(P.Flags & Flag::Auto) {
 +       if (pkgs_seen.find(pkg.FullName()) != pkgs_seen.end()) {
            if(debug_autoremove)
 -             std::clog << "Skipping already written " << pkg.Name() << std::endl;
 +             std::clog << "Skipping already written " << pkg.FullName() << std::endl;
            continue;
         }
 -         // skip not installed ones if requested
 -         if(InstalledOnly && pkg->CurrentVer == 0)
 -            continue;
 +       // skip not installed ones if requested
 +       if (InstalledOnly && (
 +           (pkg->CurrentVer == 0 && P.Mode != ModeInstall) ||
 +           (pkg->CurrentVer != 0 && P.Mode == ModeDelete)))
 +          continue;
 +       const char* const pkgarch = pkg.Arch();
 +       if (strcmp(pkgarch, "all") == 0)
 +          continue;
         if(debug_autoremove)
 -          std::clog << "Writing new AutoInstall: " 
 -                    << pkg.Name() << std::endl;
 +          std::clog << "Writing new AutoInstall: " << pkg.FullName() << std::endl;
         ostr.str(string(""));
 -       ostr << "Package: " << pkg.Name() 
 +       ostr << "Package: " << pkg.Name()
 +            << "\nArchitecture: " << pkgarch
              << "\nAuto-Installed: 1\n\n";
         fprintf(OutFile,"%s",ostr.str().c_str());
 -       fprintf(OutFile,"\n");
        }
     }
     fclose(OutFile);
@@@ -635,107 -596,6 +635,107 @@@ void pkgDepCache::UpdateVerState(PkgIte
     }
  }
                                                                        /*}}}*/
 +// DepCache::RemovePseudoInstalledPkg - MultiArch helper for Update() /*{{{*/
 +// ---------------------------------------------------------------------
 +/* We "install" arch all packages for all archs if it is installed. Many
 +   of these will be broken. This method will look at these broken Pkg and
 +   "remove" it. */
 +bool pkgDepCache::RemovePseudoInstalledPkg(PkgIterator &Pkg, std::set<unsigned long> &recheck) {
 +   if (unlikely(Pkg->CurrentVer == 0))
 +      return false;
 +
 +   VerIterator V = Pkg.CurrentVer();
 +   if (V->MultiArch != Version::All)
 +      return false;
 +
 +   // Never ever kill an "all" package - they have no dependency so they can't be broken
 +   if (strcmp(Pkg.Arch(),"all") == 0)
 +      return false;
 +
 +   unsigned char const CurDepState = VersionState(V.DependsList(),DepInstall,DepInstMin,DepInstPolicy);
 +   if ((CurDepState & DepInstMin) == DepInstMin) {
 +      // okay, the package isn't broken, but is the package also required?
 +      // If it has no real dependencies, no installed rdepends and doesn't
 +      // provide something of value, we will kill it as not required.
 +      // These pseudopackages have otherwise interesting effects if they get
 +      // a new dependency in a newer version…
 +      for (pkgCache::DepIterator D = V.DependsList();
 +         D.end() != true; ++D)
 +       if (D.IsCritical() == true && D.ParentPkg()->Group != Pkg->Group)
 +          return false;
 +      for (DepIterator D = Pkg.RevDependsList(); D.end() != true; ++D)
 +      {
 +       if (D.IsCritical() == false)
 +          continue;
 +       PkgIterator const P = D.ParentPkg();
 +       if (P->Group == Pkg->Group)
 +          continue;
 +       if (P->CurrentVer != 0)
 +          return false;
 +      }
 +      for (PrvIterator Prv = V.ProvidesList(); Prv.end() != true; Prv++)
 +       for (DepIterator d = Prv.ParentPkg().RevDependsList();
 +            d.end() != true; ++d)
 +       {
 +          PkgIterator const P = d.ParentPkg();
 +          if (P->CurrentVer != 0 &&
 +              P->Group != Pkg->Group)
 +             return false;
 +       }
 +   }
 +
 +   // Dependencies for this arch all package are not statisfied
 +   // so we installed it only for our convenience: get right of it now.
 +   RemoveSizes(Pkg);
 +   RemoveStates(Pkg);
 +
 +   Pkg->CurrentVer = 0;
 +   PkgState[Pkg->ID].InstallVer = 0;
 +
 +   AddStates(Pkg);
 +   Update(Pkg);
 +   AddSizes(Pkg);
 +
 +   // After the remove previously satisfied pseudo pkg could be now
 +   // no longer satisfied, so we need to recheck the reverse dependencies
 +   for (DepIterator d = Pkg.RevDependsList(); d.end() != true; ++d)
 +   {
 +      PkgIterator const P = d.ParentPkg();
 +      if (P->CurrentVer != 0)
 +       recheck.insert(P.Index());
 +   }
 +
 +   for (DepIterator d = V.DependsList(); d.end() != true; ++d)
 +   {
 +      PkgIterator const P = d.TargetPkg();
 +      for (PrvIterator Prv = P.ProvidesList(); Prv.end() != true; ++Prv)
 +      {
 +       PkgIterator const O = Prv.OwnerPkg();
 +       if (O->CurrentVer != 0)
 +          recheck.insert(O.Index());
 +      }
 +
 +      if (P->CurrentVer != 0)
 +       recheck.insert(P.Index());
 +   }
 +
 +   for (PrvIterator Prv = V.ProvidesList(); Prv.end() != true; Prv++)
 +   {
 +      for (DepIterator d = Prv.ParentPkg().RevDependsList();
 +         d.end() != true; ++d)
 +      {
 +       PkgIterator const P = d.ParentPkg();
 +       if (P->CurrentVer == 0)
 +          continue;
 +
 +          recheck.insert(P.Index());
 +      }
 +   }
 +
 +
 +   return true;
 +}
 +                                                                      /*}}}*/
  // DepCache::Update - Figure out all the state information            /*{{{*/
  // ---------------------------------------------------------------------
  /* This will figure out the state of all the packages and all the 
@@@ -749,13 -609,9 +749,13 @@@ void pkgDepCache::Update(OpProgress *Pr
     iKeepCount = 0;
     iBrokenCount = 0;
     iBadCount = 0;
 -   
 +
 +   std::set<unsigned long> recheck;
 +
     // Perform the depends pass
     int Done = 0;
 +   bool const checkMultiArch = APT::Configuration::getArchitectures().size() > 1;
 +   unsigned long killed = 0;
     for (PkgIterator I = PkgBegin(); I.end() != true; I++,Done++)
     {
        if (Prog != 0 && Done%20 == 0)
        for (VerIterator V = I.VersionList(); V.end() != true; V++)
        {
         unsigned char Group = 0;
 -       
 +
         for (DepIterator D = V.DependsList(); D.end() != true; D++)
         {
            // Build the dependency state.
                D->Type == Dep::DpkgBreaks ||
                D->Type == Dep::Obsoletes)
               State = ~State;
 -       }       
 +       }
        }
  
 -      // Compute the pacakge dependency state and size additions
 +      // Compute the package dependency state and size additions
        AddSizes(I);
        UpdateVerState(I);
        AddStates(I);
 +
 +      if (checkMultiArch != true || I->CurrentVer == 0)
 +       continue;
 +
 +      VerIterator const V = I.CurrentVer();
 +      if (V->MultiArch != Version::All)
 +       continue;
 +
 +      recheck.insert(I.Index());
 +      --Done; // no progress if we need to recheck the package
 +   }
 +
 +   if (checkMultiArch == true) {
 +      /* FIXME: recheck breaks proper progress reporting as we don't know
 +              how many packages we need to recheck. To lower the effect
 +              a bit we increase with a kill, but we should do something more clever… */
 +      for(std::set<unsigned long>::const_iterator p = recheck.begin();
 +        p != recheck.end(); ++p) {
 +       if (Prog != 0 && Done%20 == 0)
 +          Prog->Progress(Done);
 +       PkgIterator P = PkgIterator(*Cache, Cache->PkgP + *p);
 +       if (RemovePseudoInstalledPkg(P, recheck) == true) {
 +          ++killed;
 +          ++Done;
 +       }
 +       recheck.erase(p);
 +      }
 +
 +      /* Okay, we have killed a great amount of pseudopackages -
 +       we have killed so many that we have now arch "all" packages
 +       without an installed pseudo package, but we NEED an installed
 +       pseudo package, so we will search now for a pseudo package
 +       we can install without breaking everything. */
 +      for (GrpIterator G = Cache->GrpBegin(); G.end() != true; ++G)
 +      {
 +       PkgIterator P = G.FindPkg("all");
 +       if (P.end() == true)
 +          continue;
 +       if (P->CurrentVer == 0)
 +          continue;
 +       bool installed = false;
 +       for (P = G.FindPkg("any"); P.end() != true; P = G.NextPkg(P))
 +       {
 +          if (strcmp(P.Arch(), "all") == 0)
 +             continue;
 +          if (P->CurrentVer == 0)
 +             continue;
 +          installed = true;
 +          break;
 +       }
 +       if (installed == false)
 +          recheck.insert(G.Index());
 +      }
 +
 +      while (recheck.empty() != true)
 +      {
 +       std::set<unsigned long>::const_iterator g = recheck.begin();
 +       unsigned long const G = *g;
 +       recheck.erase(g);
 +       if (unlikely(ReInstallPseudoForGroup(G, recheck) == false))
 +          _error->Warning(_("Internal error, group '%s' has no installable pseudo package"), GrpIterator(*Cache, Cache->GrpP + *g).Name());
 +      }
     }
  
 -   if (Prog != 0)      
 +   if (Prog != 0)
        Prog->Progress(Done);
  
     readStateFile(Prog);
  }
                                                                        /*}}}*/
 +// DepCache::ReInstallPseudoForGroup - MultiArch helper for Update()  /*{{{*/
 +// ---------------------------------------------------------------------
 +/* RemovePseudoInstalledPkg() is very successful. It even kills packages
 +   to an amount that no pseudo package is left, but we need a pseudo package
 +   for upgrading senarios so we need to reinstall one pseudopackage which
 +   doesn't break everything. Thankfully we can't have architecture depending
 +   negative dependencies so this problem is already eliminated */
 +bool pkgDepCache::ReInstallPseudoForGroup(pkgCache::PkgIterator const &P, std::set<unsigned long> &recheck)
 +{
 +   if (P->CurrentVer != 0)
 +      return true;
 +   // recursive call for packages which provide this package
 +   for (pkgCache::PrvIterator Prv = P.ProvidesList(); Prv.end() != true; ++Prv)
 +      ReInstallPseudoForGroup(Prv.OwnerPkg(), recheck);
 +   // check if we actually need to look at this group
 +   unsigned long const G = P->Group;
 +   std::set<unsigned long>::const_iterator Pi = recheck.find(G);
 +   if (Pi == recheck.end())
 +      return true;
 +   recheck.erase(Pi); // remove here, so we can't fall into an endless loop
 +   if (unlikely(ReInstallPseudoForGroup(G, recheck) == false))
 +   {
 +      recheck.insert(G);
 +      return false;
 +   }
 +   return true;
 +}
 +bool pkgDepCache::ReInstallPseudoForGroup(unsigned long const &G, std::set<unsigned long> &recheck)
 +{
 +   std::vector<std::string> static const Archs = APT::Configuration::getArchitectures();
 +   pkgCache::GrpIterator Grp(*Cache, Cache->GrpP + G);
 +   if (unlikely(Grp.end() == true))
 +      return false;
 +   for (std::vector<std::string>::const_iterator a = Archs.begin();
 +        a != Archs.end(); ++a)
 +   {
 +      pkgCache::PkgIterator P = Grp.FindPkg(*a);
 +      if (P.end() == true)
 +       continue;
 +      pkgCache::VerIterator allV = Grp.FindPkg("all").CurrentVer();
 +      for (VerIterator V = P.VersionList(); V.end() != true; ++V)
 +      {
 +       // search for the same version as the all package
 +       if (allV->Hash != V->Hash || strcmp(allV.VerStr(),V.VerStr()) != 0)
 +          continue;
 +       unsigned char const CurDepState = VersionState(V.DependsList(),DepInstall,DepInstMin,DepInstPolicy);
 +       // If it is broken, try to install dependencies first before retry
 +       if ((CurDepState & DepInstMin) != DepInstMin)
 +       {
 +          for (pkgCache::DepIterator D = V.DependsList(); D.end() != true; ++D)
 +          {
 +             if (D->Type != pkgCache::Dep::PreDepends && D->Type != pkgCache::Dep::Depends)
 +                continue;
 +             ReInstallPseudoForGroup(D.TargetPkg(), recheck);
 +          }
 +          unsigned char const CurDepState = VersionState(V.DependsList(),DepInstall,DepInstMin,DepInstPolicy);
 +          // if package ist still broken… try another arch
 +          if ((CurDepState & DepInstMin) != DepInstMin)
 +             break;
 +       }
 +       // dependencies satisfied: reinstall the package
 +       RemoveSizes(P);
 +       RemoveStates(P);
 +       P->CurrentVer = V.Index();
 +       PkgState[P->ID].InstallVer = V;
 +       AddStates(P);
 +       Update(P);
 +       AddSizes(P);
 +       return true;
 +      }
 +   }
 +   return false;
 +}
 +                                                                      /*}}}*/
  // DepCache::Update - Update the deps list of a package                       /*{{{*/
  // ---------------------------------------------------------------------
  /* This is a helper for update that only does the dep portion of the scan. 
@@@ -1093,10 -813,6 +1093,10 @@@ void pkgDepCache::MarkDelete(PkgIterato
     AddStates(Pkg);   
     Update(Pkg);
     AddSizes(Pkg);
 +
 +   // if we remove the pseudo package, we also need to remove the "real"
 +   if (Pkg->CurrentVer != 0 && Pkg.CurrentVer().Pseudo() == true)
 +      MarkDelete(Pkg.Group().FindPkg("all"), rPurge, Depth+1, FromUser);
  }
                                                                        /*}}}*/
  // DepCache::IsDeleteOk - check if it is ok to remove this package    /*{{{*/
@@@ -1623,7 -1339,7 +1623,7 @@@ bool pkgDepCache::MarkRequired(InRootSe
  
        // debug output
        if(debug_autoremove && PkgState[p->ID].Flags & Flag::Auto)
 -       std::clog << "AutoDep: " << p.Name() << std::endl;
 +       std::clog << "AutoDep: " << p.FullName() << std::endl;
     }
  
     // init vars
  // MarkPackage - mark a single package in Mark-and-Sweep              /*{{{*/
  void pkgDepCache::MarkPackage(const pkgCache::PkgIterator &pkg,
                              const pkgCache::VerIterator &ver,
 -                            bool follow_recommends,
 -                            bool follow_suggests)
 +                            bool const &follow_recommends,
 +                            bool const &follow_suggests)
  {
     pkgDepCache::StateCache &state = PkgState[pkg->ID];
 -   VerIterator currver            = pkg.CurrentVer();
 -   VerIterator candver            = state.CandidateVerIter(*this);
 -   VerIterator instver            = state.InstVerIter(*this);
 +
 +   // if we are marked already we are done
 +   if(state.Marked)
 +      return;
 +
 +   VerIterator const currver = pkg.CurrentVer();
 +   VerIterator const candver = state.CandidateVerIter(*this);
 +   VerIterator const instver = state.InstVerIter(*this);
  
  #if 0
     // If a package was garbage-collected but is now being marked, we
        !(ver == currver && instver.end() && !ver.end()))
        return;
  
 -   // if we are marked already we are done
 -   if(state.Marked)
 -      return;
 +   bool const debug_autoremove = _config->FindB("Debug::pkgAutoRemove", false);
  
 -   bool debug_autoremove = _config->FindB("Debug::pkgAutoRemove", false);
 -   
     if(debug_autoremove)
       {
 -       std::clog << "Marking: " << pkg.Name();
 +       std::clog << "Marking: " << pkg.FullName();
         if(!ver.end())
         std::clog << " " << ver.VerStr();
         if(!currver.end())
  
     state.Marked=true;
  
 -   if(!ver.end())
 +   if(ver.end() == true)
 +      return;
 +
 +   // If the version belongs to a Multi-Arch all package
 +   // we will mark all others in this Group with this version also
 +   // Beware: We compare versions here the lazy way: string comparision
 +   // this is bad if multiple repositories provide different versions
 +   // of the package with an identical version number - but even in this
 +   // case the dependencies are likely the same.
 +   if (ver->MultiArch == pkgCache::Version::All &&
 +      strcmp(ver.Arch(true), "all") == 0)
     {
 +      GrpIterator G = pkg.Group();
 +      const char* const VerStr = ver.VerStr();
 +      for (PkgIterator P = G.FindPkg("any");
 +         P.end() != true; P = G.NextPkg(P))
 +      {
 +       for (VerIterator V = P.VersionList();
 +            V.end() != true; ++V)
 +       {
 +          if (strcmp(VerStr, V.VerStr()) != 0)
 +             continue;
 +          MarkPackage(P, V, follow_recommends, follow_suggests);
 +          break;
 +       }
 +      }
 +   }
 +
       for(DepIterator d = ver.DependsList(); !d.end(); ++d)
       {
        if(d->Type == Dep::Depends ||
              {
                if(debug_autoremove)
                  {
 -                  std::clog << "Following dep: " << d.ParentPkg().Name()
 +                  std::clog << "Following dep: " << d.ParentPkg().FullName()
                              << " " << d.ParentVer().VerStr() << " "
 -                            << d.DepType() << " "
 -                            << d.TargetPkg().Name();
 +                            << d.DepType() << " " << d.TargetPkg().FullName();
                    if((d->CompareOp & ~pkgCache::Dep::Or) != pkgCache::Dep::NoOp)
                      {
                        std::clog << " (" << d.CompType() << " "
                      }
                    std::clog << std::endl;
                  }
 -               MarkPackage(V.ParentPkg(), V, 
 +               MarkPackage(V.ParentPkg(), V,
                             follow_recommends, follow_suggests);
              }
           }
              {
                if(debug_autoremove)
                  {
 -                  std::clog << "Following dep: " << d.ParentPkg().Name()
 -                            << " " << d.ParentVer().VerStr() << " "
 -                            << d.DepType() << " "
 -                            << d.TargetPkg().Name();
 +                  std::clog << "Following dep: " << d.ParentPkg().FullName() << " "
 +                            << d.ParentVer().VerStr() << " "
 +                            << d.DepType() << " " << d.TargetPkg().FullName() << " ";
                    if((d->CompareOp & ~pkgCache::Dep::Or) != pkgCache::Dep::NoOp)
                      {
                        std::clog << " (" << d.CompType() << " "
                                  << d.TargetVer() << ")";
                      }
                    std::clog << ", provided by "
 -                            << prv.OwnerPkg().Name() << " "
 +                            << prv.OwnerPkg().FullName() << " "
                              << prv.OwnerVer().VerStr()
                              << std::endl;
                  }
           }
        }
       }
 -   }
  }
                                                                        /*}}}*/
  bool pkgDepCache::Sweep()                                             /*{{{*/
       {
        state.Garbage=true;
        if(debug_autoremove)
-          std::cout << "Garbage: " << p.FullName() << std::endl;
 -         std::clog << "Garbage: " << p.Name() << std::endl;
++         std::clog << "Garbage: " << p.FullName() << std::endl;
       }
    }   
  
diff --combined apt-pkg/policy.cc
index 9b24c2ef183dad1b8fe7e26f50e94e836055c4e6,4996007a39c697e8676f2d97e713f79ae9b99d8e..479cf393549ef6eb67c760e1f63552a33b78c6ce
@@@ -106,7 -106,7 +106,7 @@@ bool pkgPolicy::InitDefaults(
  
     if (_config->FindB("Debug::pkgPolicy",false) == true)
        for (pkgCache::PkgFileIterator F = Cache->FileBegin(); F != Cache->FileEnd(); F++)
-        cout << "Prio of " << F.FileName() << ' ' << PFPriority[F->ID] << endl; 
+        std::clog << "Prio of " << F.FileName() << ' ' << PFPriority[F->ID] << std::endl; 
     
     return true;   
  }
@@@ -201,38 -201,35 +201,38 @@@ pkgCache::VerIterator pkgPolicy::GetCan
  void pkgPolicy::CreatePin(pkgVersionMatch::MatchType Type,string Name,
                          string Data,signed short Priority)
  {
 -   Pin *P = 0;
 -   
     if (Name.empty() == true)
 -      P = &*Defaults.insert(Defaults.end(),PkgPin());
 -   else
     {
 -      // Get a spot to put the pin
 -      pkgCache::PkgIterator Pkg = Cache->FindPkg(Name);
 -      if (Pkg.end() == true)
 +      Pin *P = &*Defaults.insert(Defaults.end(),PkgPin());
 +      P->Type = Type;
 +      P->Priority = Priority;
 +      P->Data = Data;
 +      return;
 +   }
 +
 +   // Get a spot to put the pin
 +   pkgCache::GrpIterator Grp = Cache->FindGrp(Name);
 +   for (pkgCache::PkgIterator Pkg = Grp.FindPkg("any");
 +      Pkg.end() != true; Pkg = Grp.NextPkg(Pkg))
 +   {
 +      Pin *P = 0;
 +      if (Pkg.end() == false)
 +       P = Pins + Pkg->ID;
 +      else
        {
         // Check the unmatched table
 -       for (vector<PkgPin>::iterator I = Unmatched.begin(); 
 +       for (vector<PkgPin>::iterator I = Unmatched.begin();
              I != Unmatched.end() && P == 0; I++)
            if (I->Pkg == Name)
               P = &*I;
 -       
 +
         if (P == 0)
 -          P = &*Unmatched.insert(Unmatched.end(),PkgPin());      
 +          P = &*Unmatched.insert(Unmatched.end(),PkgPin());
        }
 -      else
 -      {
 -       P = Pins + Pkg->ID;
 -      }      
 +      P->Type = Type;
 +      P->Priority = Priority;
 +      P->Data = Data;
     }
 -   
 -   // Set..
 -   P->Type = Type;
 -   P->Priority = Priority;
 -   P->Data = Data;
  }
                                                                        /*}}}*/
  // Policy::GetMatch - Get the matching version for a package pin      /*{{{*/
diff --combined debian/changelog
index 3fd3f0a33592facc112bdd9259818177ba7d979d,eb618118d4826d22ba4ecc02c8cf602131fe04f7..0f01d1867dc5a1ca11f602345eba6caed9f6ca5b
@@@ -1,64 -1,17 +1,74 @@@
- apt (0.7.26~exp4) experimental; urgency=low
 -apt (0.7.26~exp4) unstable; urgency=low
++apt (0.7.26~exp4) UNRELEASED; urgency=low
++  [ Michael Vogt ]
+   * [ Abi break ] apt-pkg/acquire-item.{cc,h}:
+     - add "IsIndexFile" to constructor of pkgAcqFile so that it sends
+       the right cache control headers
+   * apt-pkg/depcache.cc:
+     - fix incorrect std::cout usage for debug output
+   * test/libapt/getlanguages_test.cc:
+     - Add test for Esperanto that has nocounty associated with them
+       (LP: #560956)
  
 - -- Michael Vogt <michael.vogt@ubuntu.com>  Tue, 04 May 2010 09:55:08 +0200
 +  [ David Kalnischkies ]
 +  * apt-pkg/depcache.cc:
 +    - rewrite the pseudo package reinstaller to be more intelligent
 +      in his package choices
 +  * apt-pkg/packagemanager.cc:
 +    - don't try to "unpack" pseudo packages twice
 +  * apt-pkg/contrib/fileutl.cc:
 +    - add a parent-guarded "mkdir -p" as CreateDirectory()
 +  * apt-pkg/acquire.{cc,h}:
 +    - add a delayed constructor with Setup() for success reporting
 +    - check for and create directories in Setup if needed instead of
 +      error out unfriendly in the Constructor (Closes: #523920, #525783)
 +    - optional handle a lock file in Setup()
 +  * apt-pkg/acquire-item.cc:
 +    - Acquire::ForceHash to force method for expected hash
 +  * cmdline/apt-get.cc:
 +    - remove the lock file handling and let Acquire take care of it instead
 +    - display MD5Sum in --print-uris if not forced to use another method
 +      instead of displaying the strongest available (Closes: #576420)
 +    - regex for package names executed on Grp- not PkgIterator
 +    - show non-candidates as fallback for virtual packages (Closes: #578385)
 +    - set also "all" to this version for pseudo packages in TryToChangeVer
 +  * apt-pkg/deb/dpkgpm.cc:
 +    - remove Chroot-Directory from files passed to install commands.
 +      Thanks to Kel Modderman for report & patch! (Closes: #577226)
 +  * ftparchive/writer.cc:
 +    - remove 999 chars Files and Checksums rewrite limit (Closes: #577759)
 +  * cmdline/apt-cache.cc:
 +    - align Installed and Candidate Version in policy so they can be compared
 +      easier, thanks Ralf Gesellensetter for the pointer! (Closes: #578657)
 +    - use GroupCount for package names in stats and add a package struct line
 +  * doc/apt.ent:
 +    - Add a note about APT_CONFIG in the -c description (Closes: #578267)
 +  * doc/po/de.po:
 +    - correct typos in german apt_preferences manpage, thanks Chris Leick!
 +  * apt-pkg/sourcelist.cc:
 +    - be less strict and accept [option=value] as well
 +  * apt-pkg/contrib/configuration.cc:
 +    - error out if #clear directive has no argument
 +  * doc/files.sgml:
 +    - sync documentation with status quo, regarding files/directories in
 +      use, extended_states and uri schemes.
 +  * doc/cache.sgml:
 +    - drop the file in favor of inplace documentation with doxygen
 +  * apt-pkg/pkgcache.h:
 +    - enhance the Groups ABI by providing a ID as the other structs does
 +    - check also the size of the Group struct then checking for the others
 +
 +  [ Jari Aalto ]
 +  * cmdline/apt-get.cc:
 +    - replace backticks with single quotes around fix-broken command
 +      in the broken packages message. (Closes: #577168)
 +  * dselect/install:
 +    - modernize if-statements not to use 'x' (Closes: #577117)
 +    - replace backticks with POSIX $() (Closes: #577116)
  
 -apt (0.7.26~exp3) UNRELEASED; urgency=low
 + -- David Kalnischkies <kalnischkies@gmail.com>  Sat, 03 Apr 2010 14:58:39 +0200
 +
 +apt (0.7.26~exp3) experimental; urgency=low
  
    [ Christian Perrier ]
    * German translation update. Closes: #571037
    * Add "manpages-pl (<< 20060617-3~)" to avoid file conflicts with
      that package that was providing some manpages for APT utilities.
  
 +  [ David Kalnischkies ]
 +  * [BREAK] merge MultiArch-ABI. We don't support MultiArch,
 +    but we support the usage of the new ABI so libapt users
 +    can start to prepare for MultiArch (Closes: #536029)
 +  * Ignore :qualifiers after package name in build dependencies
 +    in the library by default, but try to honour them in apt-get
 +    as we have some sort of MultiArch support ready (Closes: #558103)
 +  * add translation of the manpages to PT (portuguese)
 +    Thanks to Américo Monteiro!
 +  * Switch to dpkg-source 3.0 (native) format
 +  * apt-pkg/depcache.cc:
 +    - remove Auto-Installed information from extended_states
 +      together with the package itself (Closes: #572364)
 +  * cmdline/apt-mark:
 +    - don't crash if no arguments are given (Closes: #570962)
 +  * debian/control:
 +    - remove some years old and obsolete Replaces
 +    - add automake/conf build-depends/conflicts as recommend by
 +      the autotools-dev README (Closes: #572615)
 +  * apt-pkg/contrib/mmap.{h,cc}:
 +    - add char[] fallback for filesystems without shared writable
 +      mmap() like JFFS2. Thanks to Marius Vollmer for writing
 +      and to Loïc Minier for pointing to the patch! (Closes: #314334)
 +  * doc/apt_preferences.5.xml:
 +    - fix two typos and be more verbose in the novice warning.
 +      Thanks to Osamu Aoki for pointing it out! (Closes: #567669)
 +    - fix a=sid vs. n=sid typo, thanks Ansgar Burchardt!
 +    - origin can be used to match a hostname (Closes: #352667)
 +    - remove wrong pin-priority is optional remark (Closes: #574944)
 +  * apt-pkg/deb/dpkgpm.cc:
 +    - fix error message construction in OpenLog()
 +    - if available store the Commandline in the history
 +  * cmdline/apt-get.cc:
 +    - add a --only-upgrade flag to install command (Closes: #572259)
 +    - fix memory leaks in error conditions in DoSource()
 +    - try version match in FindSrc first exact than fuzzy (LP: #551178)
 +  * apt-pkg/contrib/cmndline.cc:
 +    - save Commandline in Commandline::AsString for logging
 +  * apt-pkg/deb/debversion.cc:
 +    - consider absent of debian revision equivalent to 0 (Closes: #573592)
 +  * doc/makefile, doc/*:
 +    - generate subdirectories for building the manpages in on the fly
 +      depending on the po files we have.
 +  * apt-pkg/pkgcachegen.cc:
 +    - merge versions correctly even if multiple different versions
 +      with the same version number are available.
 +      Thanks to Magnus Holmgren for the patch! (Closes: #351056)
 +  * ftparchive/writer.cc:
 +    - write LongDescriptions if they shouldn't be included in Packages
 +      file into i18n/Translation-en by default.
 +  * doc/po/de.po:
 +    - correct a few typos in the german manpage translation.
 +      Thanks to Chris Leick and Georg Koppen! (Closes: #574962)
 +  * apt-pkg/contrib/strutl.cc:
 +    - convert all toupper calls to tolower_ascii for a little speedup
 +
 +  [ Jean-Baptiste Lallement ]
 +  * apt-pkg/contrib/strutl.cc:
 +    - always escape '%' (LP: #130289) (Closes: #500560)
 +    - unescape '%' sequence only if followed by 2 hex digit
 +    - username/password are urlencoded in proxy string (RFC 3986)
 +
    [ Julian Andres Klode ]
    * cmdline/apt-cache.cc:
      - Change behavior of showsrc to match the one of show (Closes: #512046).
    * doc/examples/configure-index:
      - add missing Debug::pkgPackageManager option
  
 - -- Christian Perrier <bubulle@debian.org>  Wed, 24 Feb 2010 22:13:50 +0100
 + -- Michael Vogt <mvo@debian.org>  Thu, 01 Apr 2010 17:30:43 +0200
  
  apt (0.7.26~exp2) experimental; urgency=low