1 // -*- mode: cpp; mode: fold -*-
3 // $Id: deblistparser.cc,v 1.29.2.5 2004/01/06 01:43:44 mdz Exp $
4 /* ######################################################################
6 Package Cache Generator - Generator for the cache structure.
8 This builds the cache structure from the abstract package list parser.
10 ##################################################################### */
12 // Include Files /*{{{*/
15 #include <apt-pkg/deblistparser.h>
16 #include <apt-pkg/error.h>
17 #include <apt-pkg/configuration.h>
18 #include <apt-pkg/cachefilter.h>
19 #include <apt-pkg/aptconfiguration.h>
20 #include <apt-pkg/strutl.h>
21 #include <apt-pkg/crc-16.h>
22 #include <apt-pkg/md5.h>
23 #include <apt-pkg/pkgcache.h>
24 #include <apt-pkg/cacheiterators.h>
25 #include <apt-pkg/tagfile.h>
26 #include <apt-pkg/macros.h>
38 static const debListParser::WordList PrioList
[] = {
39 {"required",pkgCache::State::Required
},
40 {"important",pkgCache::State::Important
},
41 {"standard",pkgCache::State::Standard
},
42 {"optional",pkgCache::State::Optional
},
43 {"extra",pkgCache::State::Extra
},
46 // ListParser::debListParser - Constructor /*{{{*/
47 // ---------------------------------------------------------------------
48 /* Provide an architecture and only this one and "all" will be accepted
49 in Step(), if no Architecture is given we will accept every arch
50 we would accept in general with checkArchitecture() */
51 debListParser::debListParser(FileFd
*File
) :
52 pkgCacheListParser(), d(NULL
), Tags(File
)
56 // ListParser::Package - Return the package name /*{{{*/
57 // ---------------------------------------------------------------------
58 /* This is to return the name of the package this section describes */
59 string
debListParser::Package() {
60 string Result
= Section
.FindS("Package");
62 // Normalize mixed case package names to lower case, like dpkg does
63 // See Bug#807012 for details
64 std::transform(Result
.begin(), Result
.end(), Result
.begin(), tolower_ascii
);
66 if(unlikely(Result
.empty() == true))
67 _error
->Error("Encountered a section with no Package: header");
71 // ListParser::Architecture - Return the package arch /*{{{*/
72 // ---------------------------------------------------------------------
73 /* This will return the Architecture of the package this section describes */
74 string
debListParser::Architecture() {
75 std::string
const Arch
= Section
.FindS("Architecture");
76 return Arch
.empty() ? "none" : Arch
;
79 // ListParser::ArchitectureAll /*{{{*/
80 // ---------------------------------------------------------------------
82 bool debListParser::ArchitectureAll() {
83 return Section
.FindS("Architecture") == "all";
86 // ListParser::Version - Return the version string /*{{{*/
87 // ---------------------------------------------------------------------
88 /* This is to return the string describing the version in debian form,
89 epoch:upstream-release. If this returns the blank string then the
90 entry is assumed to only describe package properties */
91 string
debListParser::Version()
93 return Section
.FindS("Version");
96 unsigned char debListParser::ParseMultiArch(bool const showErrors
) /*{{{*/
99 string
const MultiArch
= Section
.FindS("Multi-Arch");
100 if (MultiArch
.empty() == true || MultiArch
== "no")
101 MA
= pkgCache::Version::No
;
102 else if (MultiArch
== "same") {
103 if (ArchitectureAll() == true)
105 if (showErrors
== true)
106 _error
->Warning("Architecture: all package '%s' can't be Multi-Arch: same",
107 Section
.FindS("Package").c_str());
108 MA
= pkgCache::Version::No
;
111 MA
= pkgCache::Version::Same
;
113 else if (MultiArch
== "foreign")
114 MA
= pkgCache::Version::Foreign
;
115 else if (MultiArch
== "allowed")
116 MA
= pkgCache::Version::Allowed
;
119 if (showErrors
== true)
120 _error
->Warning("Unknown Multi-Arch type '%s' for package '%s'",
121 MultiArch
.c_str(), Section
.FindS("Package").c_str());
122 MA
= pkgCache::Version::No
;
125 if (ArchitectureAll() == true)
126 MA
|= pkgCache::Version::All
;
131 // ListParser::NewVersion - Fill in the version structure /*{{{*/
132 // ---------------------------------------------------------------------
134 bool debListParser::NewVersion(pkgCache::VerIterator
&Ver
)
140 if (Section
.Find("Section",Start
,Stop
) == true)
142 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::SECTION
, Start
, Stop
- Start
);
145 // Parse the source package name
146 pkgCache::GrpIterator
const G
= Ver
.ParentPkg().Group();
147 Ver
->SourcePkgName
= G
->Name
;
148 Ver
->SourceVerStr
= Ver
->VerStr
;
149 if (Section
.Find("Source",Start
,Stop
) == true)
151 const char * const Space
= (const char * const) memchr(Start
, ' ', Stop
- Start
);
152 pkgCache::VerIterator V
;
157 const char * const Open
= (const char * const) memchr(Space
, '(', Stop
- Space
);
158 if (likely(Open
!= NULL
))
160 const char * const Close
= (const char * const) memchr(Open
, ')', Stop
- Open
);
161 if (likely(Close
!= NULL
))
163 std::string
const version(Open
+ 1, (Close
- Open
) - 1);
164 if (version
!= Ver
.VerStr())
166 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::VERSIONNUMBER
, version
);
167 Ver
->SourceVerStr
= idx
;
173 std::string
const pkgname(Start
, Stop
- Start
);
174 if (pkgname
!= G
.Name())
176 for (pkgCache::PkgIterator P
= G
.PackageList(); P
.end() == false; P
= G
.NextPkg(P
))
178 for (V
= P
.VersionList(); V
.end() == false; ++V
)
180 if (pkgname
== V
.SourcePkgName())
182 Ver
->SourcePkgName
= V
->SourcePkgName
;
186 if (V
.end() == false)
191 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::PKGNAME
, pkgname
);
192 Ver
->SourcePkgName
= idx
;
197 Ver
->MultiArch
= ParseMultiArch(true);
199 Ver
->Size
= Section
.FindULL("Size");
200 // Unpacked Size (in K)
201 Ver
->InstalledSize
= Section
.FindULL("Installed-Size");
202 Ver
->InstalledSize
*= 1024;
205 if (Section
.Find("Priority",Start
,Stop
) == true)
207 if (GrabWord(string(Start
,Stop
-Start
),PrioList
,Ver
->Priority
) == false)
208 Ver
->Priority
= pkgCache::State::Extra
;
211 if (ParseDepends(Ver
,"Pre-Depends",pkgCache::Dep::PreDepends
) == false)
213 if (ParseDepends(Ver
,"Depends",pkgCache::Dep::Depends
) == false)
215 if (ParseDepends(Ver
,"Conflicts",pkgCache::Dep::Conflicts
) == false)
217 if (ParseDepends(Ver
,"Breaks",pkgCache::Dep::DpkgBreaks
) == false)
219 if (ParseDepends(Ver
,"Recommends",pkgCache::Dep::Recommends
) == false)
221 if (ParseDepends(Ver
,"Suggests",pkgCache::Dep::Suggests
) == false)
223 if (ParseDepends(Ver
,"Replaces",pkgCache::Dep::Replaces
) == false)
225 if (ParseDepends(Ver
,"Enhances",pkgCache::Dep::Enhances
) == false)
228 if (ParseDepends(Ver
,"Optional",pkgCache::Dep::Suggests
) == false)
231 if (ParseProvides(Ver
) == false)
237 // ListParser::Description - Return the description string /*{{{*/
238 // ---------------------------------------------------------------------
239 /* This is to return the string describing the package in debian
240 form. If this returns the blank string then the entry is assumed to
241 only describe package properties */
242 string
debListParser::Description(std::string
const &lang
)
245 return Section
.FindS("Description");
247 return Section
.FindS(string("Description-").append(lang
).c_str());
250 // ListParser::AvailableDescriptionLanguages /*{{{*/
251 std::vector
<std::string
> debListParser::AvailableDescriptionLanguages()
253 std::vector
<std::string
> const understood
= APT::Configuration::getLanguages();
254 std::vector
<std::string
> avail
;
255 if (Section
.Exists("Description") == true)
257 for (std::vector
<std::string
>::const_iterator lang
= understood
.begin(); lang
!= understood
.end(); ++lang
)
259 std::string
const tagname
= "Description-" + *lang
;
260 if (Section
.Exists(tagname
.c_str()) == true)
261 avail
.push_back(*lang
);
266 // ListParser::Description_md5 - Return the description_md5 MD5SumValue /*{{{*/
267 // ---------------------------------------------------------------------
268 /* This is to return the md5 string to allow the check if it is the right
269 description. If no Description-md5 is found in the section it will be
272 MD5SumValue
debListParser::Description_md5()
274 string
const value
= Section
.FindS("Description-md5");
275 if (value
.empty() == true)
277 std::string
const desc
= Description("") + "\n";
279 return MD5SumValue();
282 md5
.Add(desc
.c_str());
285 else if (likely(value
.size() == 32))
287 if (likely(value
.find_first_not_of("0123456789abcdefABCDEF") == string::npos
))
288 return MD5SumValue(value
);
289 _error
->Error("Malformed Description-md5 line; includes invalid character '%s'", value
.c_str());
290 return MD5SumValue();
292 _error
->Error("Malformed Description-md5 line; doesn't have the required length (32 != %d) '%s'", (int)value
.size(), value
.c_str());
293 return MD5SumValue();
296 // ListParser::UsePackage - Update a package structure /*{{{*/
297 // ---------------------------------------------------------------------
298 /* This is called to update the package with any new information
299 that might be found in the section */
300 bool debListParser::UsePackage(pkgCache::PkgIterator
&Pkg
,
301 pkgCache::VerIterator
&Ver
)
303 string
const static myArch
= _config
->Find("APT::Architecture");
304 // Possible values are: "all", "native", "installed" and "none"
305 // The "installed" mode is handled by ParseStatus(), See #544481 and friends.
306 string
const static essential
= _config
->Find("pkgCacheGen::Essential", "all");
307 if (essential
== "all" ||
308 (essential
== "native" && Pkg
->Arch
!= 0 && myArch
== Pkg
.Arch()))
309 if (Section
.FindFlag("Essential",Pkg
->Flags
,pkgCache::Flag::Essential
) == false)
311 if (Section
.FindFlag("Important",Pkg
->Flags
,pkgCache::Flag::Important
) == false)
314 if (strcmp(Pkg
.Name(),"apt") == 0)
316 if ((essential
== "native" && Pkg
->Arch
!= 0 && myArch
== Pkg
.Arch()) ||
318 Pkg
->Flags
|= pkgCache::Flag::Essential
| pkgCache::Flag::Important
;
320 Pkg
->Flags
|= pkgCache::Flag::Important
;
323 if (ParseStatus(Pkg
,Ver
) == false)
328 // ListParser::VersionHash - Compute a unique hash for this version /*{{{*/
329 // ---------------------------------------------------------------------
331 unsigned short debListParser::VersionHash()
333 const char *Sections
[] ={"Installed-Size",
341 unsigned long Result
= INIT_FCS
;
343 for (const char * const *I
= Sections
; *I
!= 0; ++I
)
347 if (Section
.Find(*I
,Start
,End
) == false || End
- Start
>= (signed)sizeof(S
))
350 /* Strip out any spaces from the text, this undoes dpkgs reformatting
351 of certain fields. dpkg also has the rather interesting notion of
352 reformatting depends operators < -> <= */
354 for (; Start
!= End
; ++Start
)
356 if (isspace(*Start
) != 0)
358 *J
++ = tolower_ascii(*Start
);
360 if ((*Start
== '<' || *Start
== '>') && Start
[1] != *Start
&& Start
[1] != '=')
364 Result
= AddCRC16(Result
,S
,J
- S
);
370 // StatusListParser::ParseStatus - Parse the status field /*{{{*/
371 // ---------------------------------------------------------------------
372 /* Status lines are of the form,
373 Status: want flag status
374 want = unknown, install, hold, deinstall, purge
376 status = not-installed, config-files, half-installed, unpacked,
377 half-configured, triggers-awaited, triggers-pending, installed
379 bool debListParser::ParseStatus(pkgCache::PkgIterator
&,
380 pkgCache::VerIterator
&)
384 bool debStatusListParser::ParseStatus(pkgCache::PkgIterator
&Pkg
,
385 pkgCache::VerIterator
&Ver
)
389 if (Section
.Find("Status",Start
,Stop
) == false)
392 // UsePackage() is responsible for setting the flag in the default case
393 bool const static essential
= _config
->Find("pkgCacheGen::Essential", "") == "installed";
394 if (essential
== true &&
395 Section
.FindFlag("Essential",Pkg
->Flags
,pkgCache::Flag::Essential
) == false)
398 // Isolate the first word
399 const char *I
= Start
;
400 for(; I
< Stop
&& *I
!= ' '; I
++);
401 if (I
>= Stop
|| *I
!= ' ')
402 return _error
->Error("Malformed Status line");
404 // Process the want field
405 WordList WantList
[] = {{"unknown",pkgCache::State::Unknown
},
406 {"install",pkgCache::State::Install
},
407 {"hold",pkgCache::State::Hold
},
408 {"deinstall",pkgCache::State::DeInstall
},
409 {"purge",pkgCache::State::Purge
},
411 if (GrabWord(string(Start
,I
-Start
),WantList
,Pkg
->SelectedState
) == false)
412 return _error
->Error("Malformed 1st word in the Status line");
414 // Isloate the next word
417 for(; I
< Stop
&& *I
!= ' '; I
++);
418 if (I
>= Stop
|| *I
!= ' ')
419 return _error
->Error("Malformed status line, no 2nd word");
421 // Process the flag field
422 WordList FlagList
[] = {{"ok",pkgCache::State::Ok
},
423 {"reinstreq",pkgCache::State::ReInstReq
},
424 {"hold",pkgCache::State::HoldInst
},
425 {"hold-reinstreq",pkgCache::State::HoldReInstReq
},
427 if (GrabWord(string(Start
,I
-Start
),FlagList
,Pkg
->InstState
) == false)
428 return _error
->Error("Malformed 2nd word in the Status line");
430 // Isloate the last word
433 for(; I
< Stop
&& *I
!= ' '; I
++);
435 return _error
->Error("Malformed Status line, no 3rd word");
437 // Process the flag field
438 WordList StatusList
[] = {{"not-installed",pkgCache::State::NotInstalled
},
439 {"config-files",pkgCache::State::ConfigFiles
},
440 {"half-installed",pkgCache::State::HalfInstalled
},
441 {"unpacked",pkgCache::State::UnPacked
},
442 {"half-configured",pkgCache::State::HalfConfigured
},
443 {"triggers-awaited",pkgCache::State::TriggersAwaited
},
444 {"triggers-pending",pkgCache::State::TriggersPending
},
445 {"installed",pkgCache::State::Installed
},
447 if (GrabWord(string(Start
,I
-Start
),StatusList
,Pkg
->CurrentState
) == false)
448 return _error
->Error("Malformed 3rd word in the Status line");
450 /* A Status line marks the package as indicating the current
451 version as well. Only if it is actually installed.. Otherwise
452 the interesting dpkg handling of the status file creates bogus
454 if (!(Pkg
->CurrentState
== pkgCache::State::NotInstalled
||
455 Pkg
->CurrentState
== pkgCache::State::ConfigFiles
))
457 if (Ver
.end() == true)
458 _error
->Warning("Encountered status field in a non-version description");
460 Pkg
->CurrentVer
= Ver
.Index();
466 const char *debListParser::ConvertRelation(const char *I
,unsigned int &Op
)
468 // Determine the operator
476 Op
= pkgCache::Dep::LessEq
;
483 Op
= pkgCache::Dep::Less
;
487 // < is the same as <= and << is really Cs < for some reason
488 Op
= pkgCache::Dep::LessEq
;
496 Op
= pkgCache::Dep::GreaterEq
;
503 Op
= pkgCache::Dep::Greater
;
507 // > is the same as >= and >> is really Cs > for some reason
508 Op
= pkgCache::Dep::GreaterEq
;
512 Op
= pkgCache::Dep::Equals
;
516 // HACK around bad package definitions
518 Op
= pkgCache::Dep::Equals
;
524 // ListParser::ParseDepends - Parse a dependency element /*{{{*/
525 // ---------------------------------------------------------------------
526 /* This parses the dependency elements out of a standard string in place,
528 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
529 std::string
&Package
,std::string
&Ver
,unsigned int &Op
)
530 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, false, true, false); }
531 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
532 std::string
&Package
,std::string
&Ver
,unsigned int &Op
,
533 bool const &ParseArchFlags
)
534 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, ParseArchFlags
, true, false); }
535 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
536 std::string
&Package
,std::string
&Ver
,unsigned int &Op
,
537 bool const &ParseArchFlags
, bool const &StripMultiArch
)
538 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, ParseArchFlags
, StripMultiArch
, false); }
539 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
540 string
&Package
,string
&Ver
,
541 unsigned int &Op
, bool const &ParseArchFlags
,
542 bool const &StripMultiArch
,
543 bool const &ParseRestrictionsList
)
545 // Strip off leading space
546 for (;Start
!= Stop
&& isspace(*Start
) != 0; ++Start
);
548 // Parse off the package name
549 const char *I
= Start
;
550 for (;I
!= Stop
&& isspace(*I
) == 0 && *I
!= '(' && *I
!= ')' &&
551 *I
!= ',' && *I
!= '|' && *I
!= '[' && *I
!= ']' &&
552 *I
!= '<' && *I
!= '>'; ++I
);
555 if (I
!= Stop
&& *I
== ')')
561 // Stash the package name
562 Package
.assign(Start
,I
- Start
);
564 // We don't want to confuse library users which can't handle MultiArch
565 string
const arch
= _config
->Find("APT::Architecture");
566 if (StripMultiArch
== true) {
567 size_t const found
= Package
.rfind(':');
568 if (found
!= string::npos
&&
569 (strcmp(Package
.c_str() + found
, ":any") == 0 ||
570 strcmp(Package
.c_str() + found
, ":native") == 0 ||
571 strcmp(Package
.c_str() + found
+ 1, arch
.c_str()) == 0))
572 Package
= Package
.substr(0,found
);
575 // Skip white space to the '('
576 for (;I
!= Stop
&& isspace(*I
) != 0 ; I
++);
579 if (I
!= Stop
&& *I
== '(')
582 for (I
++; I
!= Stop
&& isspace(*I
) != 0 ; I
++);
585 I
= ConvertRelation(I
,Op
);
588 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
590 I
= (const char*) memchr(I
, ')', Stop
- I
);
591 if (I
== NULL
|| Start
== I
)
594 // Skip trailing whitespace
596 for (; End
> Start
&& isspace(End
[-1]); End
--);
598 Ver
.assign(Start
,End
-Start
);
604 Op
= pkgCache::Dep::NoOp
;
608 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
610 if (ParseArchFlags
== true)
612 APT::CacheFilter::PackageArchitectureMatchesSpecification
matchesArch(arch
, false);
614 // Parse an architecture
615 if (I
!= Stop
&& *I
== '[')
619 if (unlikely(I
== Stop
))
624 bool NegArch
= false;
627 // look for whitespace or ending ']'
628 for (;End
!= Stop
&& !isspace(*End
) && *End
!= ']'; ++End
);
630 if (unlikely(End
== Stop
))
639 std::string
arch(I
, End
);
640 if (arch
.empty() == false && matchesArch(arch
.c_str()) == true)
645 // we found a match, so fast-forward to the end of the wildcards
646 for (; End
!= Stop
&& *End
!= ']'; ++End
);
655 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
662 Package
= ""; /* not for this arch */
666 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
669 if (ParseRestrictionsList
== true)
671 // Parse a restrictions formula which is in disjunctive normal form:
672 // (foo AND bar) OR (blub AND bla)
674 std::vector
<string
> const profiles
= APT::Configuration::getBuildProfiles();
676 // if the next character is a restriction list, then by default the
677 // dependency does not apply and the conditions have to be checked
678 // if the next character is not a restriction list, then by default the
679 // dependency applies
680 bool applies1
= (*I
!= '<');
688 if (unlikely(I
== Stop
))
693 // if of the prior restriction list is already fulfilled, then
694 // we can just skip to the end of the current list
696 for (;End
!= Stop
&& *End
!= '>'; ++End
);
699 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
701 bool applies2
= true;
702 // all the conditions inside a restriction list have to be
703 // met so once we find one that is not met, we can skip to
704 // the end of this list
707 // look for whitespace or ending '>'
708 // End now points to the character after the current term
709 for (;End
!= Stop
&& !isspace(*End
) && *End
!= '>'; ++End
);
711 if (unlikely(End
== Stop
))
714 bool NegRestriction
= false;
717 NegRestriction
= true;
721 std::string
restriction(I
, End
);
723 if (restriction
.empty() == false && profiles
.empty() == false &&
724 std::find(profiles
.begin(), profiles
.end(), restriction
) != profiles
.end())
726 if (NegRestriction
) {
728 // since one of the terms does not apply we don't have to check the others
729 for (; End
!= Stop
&& *End
!= '>'; ++End
);
732 if (!NegRestriction
) {
734 // since one of the terms does not apply we don't have to check the others
735 for (; End
!= Stop
&& *End
!= '>'; ++End
);
742 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
748 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
756 if (applies1
== false) {
757 Package
= ""; //not for this restriction
761 if (I
!= Stop
&& *I
== '|')
762 Op
|= pkgCache::Dep::Or
;
764 if (I
== Stop
|| *I
== ',' || *I
== '|')
767 for (I
++; I
!= Stop
&& isspace(*I
) != 0; I
++);
774 // ListParser::ParseDepends - Parse a dependency list /*{{{*/
775 // ---------------------------------------------------------------------
776 /* This is the higher level depends parser. It takes a tag and generates
777 a complete depends tree for the given version. */
778 bool debListParser::ParseDepends(pkgCache::VerIterator
&Ver
,
779 const char *Tag
,unsigned int Type
)
783 if (Section
.Find(Tag
,Start
,Stop
) == false)
786 string
const pkgArch
= Ver
.Arch();
794 Start
= ParseDepends(Start
, Stop
, Package
, Version
, Op
, false, false, false);
796 return _error
->Error("Problem parsing dependency %s",Tag
);
797 size_t const found
= Package
.rfind(':');
799 if (found
== string::npos
)
801 if (NewDepends(Ver
,Package
,pkgArch
,Version
,Op
,Type
) == false)
804 else if (strcmp(Package
.c_str() + found
, ":any") == 0)
806 if (NewDepends(Ver
,Package
,"any",Version
,Op
,Type
) == false)
811 // Such dependencies are not supposed to be accepted …
812 // … but this is probably the best thing to do anyway
813 if (Package
.substr(found
+ 1) == "native")
814 Package
= Package
.substr(0, found
) + ':' + Ver
.Cache()->NativeArch();
816 if (NewDepends(Ver
, Package
, "any", Version
, Op
| pkgCache::Dep::ArchSpecific
, Type
) == false)
826 // ListParser::ParseProvides - Parse the provides list /*{{{*/
827 // ---------------------------------------------------------------------
829 bool debListParser::ParseProvides(pkgCache::VerIterator
&Ver
)
831 /* it is unlikely, but while parsing dependencies, we might have already
832 picked up multi-arch implicit provides which we do not want to duplicate here */
833 bool hasProvidesAlready
= false;
834 std::string
const spzName
= Ver
.ParentPkg().FullName(false);
836 for (pkgCache::PrvIterator Prv
= Ver
.ProvidesList(); Prv
.end() == false; ++Prv
)
838 if (Prv
.IsMultiArchImplicit() == false || (Prv
->Flags
& pkgCache::Flag::ArchSpecific
) == 0)
840 if (spzName
!= Prv
.OwnerPkg().FullName(false))
842 hasProvidesAlready
= true;
847 string
const Arch
= Ver
.Arch();
850 if (Section
.Find("Provides",Start
,Stop
) == true)
858 Start
= ParseDepends(Start
,Stop
,Package
,Version
,Op
, false, false, false);
859 const size_t archfound
= Package
.rfind(':');
861 return _error
->Error("Problem parsing Provides line");
862 if (Op
!= pkgCache::Dep::NoOp
&& Op
!= pkgCache::Dep::Equals
) {
863 _error
->Warning("Ignoring Provides line with non-equal DepCompareOp for package %s", Package
.c_str());
864 } else if (archfound
!= string::npos
) {
865 std::string spzArch
= Package
.substr(archfound
+ 1);
866 if (spzArch
!= "any")
868 if (NewProvides(Ver
, Package
.substr(0, archfound
), spzArch
, Version
, pkgCache::Flag::MultiArchImplicit
| pkgCache::Flag::ArchSpecific
) == false)
871 if (NewProvides(Ver
, Package
, "any", Version
, pkgCache::Flag::ArchSpecific
) == false)
873 } else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
) {
874 if (APT::Configuration::checkArchitecture(Arch
))
875 if (NewProvidesAllArch(Ver
, Package
, Version
, 0) == false)
878 if ((Ver
->MultiArch
& pkgCache::Version::Allowed
) == pkgCache::Version::Allowed
)
880 if (NewProvides(Ver
, Package
+ ":any", "any", Version
, pkgCache::Flag::MultiArchImplicit
) == false)
883 if (NewProvides(Ver
, Package
, Arch
, Version
, 0) == false)
886 if (archfound
== std::string::npos
)
888 std::string
const spzName
= Package
+ ':' + Ver
.ParentPkg().Arch();
889 pkgCache::PkgIterator
const spzPkg
= Ver
.Cache()->FindPkg(spzName
, "any");
890 if (spzPkg
.end() == false)
892 if (NewProvides(Ver
, spzName
, "any", Version
, pkgCache::Flag::MultiArchImplicit
| pkgCache::Flag::ArchSpecific
) == false)
896 } while (Start
!= Stop
);
899 if (APT::Configuration::checkArchitecture(Arch
))
901 if ((Ver
->MultiArch
& pkgCache::Version::Allowed
) == pkgCache::Version::Allowed
)
903 string
const Package
= string(Ver
.ParentPkg().Name()).append(":").append("any");
904 if (NewProvides(Ver
, Package
, "any", Ver
.VerStr(), pkgCache::Flag::MultiArchImplicit
) == false)
907 else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
)
909 if (NewProvidesAllArch(Ver
, Ver
.ParentPkg().Name(), Ver
.VerStr(), pkgCache::Flag::MultiArchImplicit
) == false)
914 if (hasProvidesAlready
== false)
916 pkgCache::PkgIterator
const spzPkg
= Ver
.Cache()->FindPkg(spzName
, "any");
917 if (spzPkg
.end() == false)
919 if (NewProvides(Ver
, spzName
, "any", Ver
.VerStr(), pkgCache::Flag::MultiArchImplicit
| pkgCache::Flag::ArchSpecific
) == false)
926 // ListParser::GrabWord - Matches a word and returns /*{{{*/
927 // ---------------------------------------------------------------------
928 /* Looks for a word in a list of words - for ParseStatus */
929 bool debListParser::GrabWord(string Word
,const WordList
*List
,unsigned char &Out
)
931 for (unsigned int C
= 0; List
[C
].Str
!= 0; C
++)
933 if (strcasecmp(Word
.c_str(),List
[C
].Str
) == 0)
942 // ListParser::Step - Move to the next section in the file /*{{{*/
943 // ---------------------------------------------------------------------
944 /* This has to be careful to only process the correct architecture */
945 bool debListParser::Step()
947 iOffset
= Tags
.Offset();
948 return Tags
.Step(Section
);
951 // ListParser::GetPrio - Convert the priority from a string /*{{{*/
952 // ---------------------------------------------------------------------
954 unsigned char debListParser::GetPrio(string Str
)
957 if (GrabWord(Str
,PrioList
,Out
) == false)
958 Out
= pkgCache::State::Extra
;
963 bool debListParser::SameVersion(unsigned short const Hash
, /*{{{*/
964 pkgCache::VerIterator
const &Ver
)
966 if (pkgCacheListParser::SameVersion(Hash
, Ver
) == false)
968 // status file has no (Download)Size, but all others are fair game
969 // status file is parsed last, so the first version we encounter is
970 // probably also the version we have downloaded
971 unsigned long long const Size
= Section
.FindULL("Size");
972 if (Size
!= 0 && Size
!= Ver
->Size
)
974 // available everywhere, but easier to check here than to include in VersionHash
975 unsigned char MultiArch
= ParseMultiArch(false);
976 if (MultiArch
!= Ver
->MultiArch
)
978 // for all practical proposes (we can check): same version
983 debDebFileParser::debDebFileParser(FileFd
*File
, std::string
const &DebFile
)
984 : debListParser(File
), DebFile(DebFile
)
988 bool debDebFileParser::UsePackage(pkgCache::PkgIterator
&Pkg
,
989 pkgCache::VerIterator
&Ver
)
991 bool res
= debListParser::UsePackage(Pkg
, Ver
);
992 // we use the full file path as a provides so that the file is found
994 if(NewProvides(Ver
, DebFile
, Pkg
.Cache()->NativeArch(), Ver
.VerStr(), 0) == false)
999 debListParser::~debListParser() {}