]>
git.saurik.com Git - apt.git/blob - apt-pkg/edsp.cc
1 // -*- mode: cpp; mode: fold -*-
3 /* ######################################################################
4 Set of methods to help writing and reading everything needed for EDSP
5 ##################################################################### */
7 // Include Files /*{{{*/
10 #include <apt-pkg/error.h>
11 #include <apt-pkg/cacheset.h>
12 #include <apt-pkg/depcache.h>
13 #include <apt-pkg/pkgcache.h>
14 #include <apt-pkg/cacheiterators.h>
15 #include <apt-pkg/progress.h>
16 #include <apt-pkg/fileutl.h>
17 #include <apt-pkg/edsp.h>
18 #include <apt-pkg/tagfile.h>
19 #include <apt-pkg/strutl.h>
35 // we could use pkgCache::DepType and ::Priority, but these would be localized stringsā¦
36 const char * const PrioMap
[] = {0, "important", "required", "standard",
38 const char * const DepMap
[] = {"", "Depends", "Pre-Depends", "Suggests",
39 "Recommends" , "Conflicts", "Replaces",
40 "Obsoletes", "Breaks", "Enhances"};
43 // WriteScenarioVersion /*{{{*/
44 static void WriteScenarioVersion(pkgDepCache
&Cache
, FILE* output
, pkgCache::PkgIterator
const &Pkg
,
45 pkgCache::VerIterator
const &Ver
)
47 fprintf(output
, "Package: %s\n", Pkg
.Name());
48 fprintf(output
, "Source: %s\n", Ver
.SourcePkgName());
49 fprintf(output
, "Architecture: %s\n", Ver
.Arch());
50 fprintf(output
, "Version: %s\n", Ver
.VerStr());
51 fprintf(output
, "Source-Version: %s\n", Ver
.SourceVerStr());
52 if (Pkg
.CurrentVer() == Ver
)
53 fprintf(output
, "Installed: yes\n");
54 if (Pkg
->SelectedState
== pkgCache::State::Hold
||
55 (Cache
[Pkg
].Keep() == true && Cache
[Pkg
].Protect() == true))
56 fprintf(output
, "Hold: yes\n");
57 fprintf(output
, "APT-ID: %d\n", Ver
->ID
);
58 fprintf(output
, "Priority: %s\n", PrioMap
[Ver
->Priority
]);
59 if ((Pkg
->Flags
& pkgCache::Flag::Essential
) == pkgCache::Flag::Essential
)
60 fprintf(output
, "Essential: yes\n");
61 fprintf(output
, "Section: %s\n", Ver
.Section());
62 if ((Ver
->MultiArch
& pkgCache::Version::Allowed
) == pkgCache::Version::Allowed
)
63 fprintf(output
, "Multi-Arch: allowed\n");
64 else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
)
65 fprintf(output
, "Multi-Arch: foreign\n");
66 else if ((Ver
->MultiArch
& pkgCache::Version::Same
) == pkgCache::Version::Same
)
67 fprintf(output
, "Multi-Arch: same\n");
68 signed short Pin
= std::numeric_limits
<signed short>::min();
69 std::set
<string
> Releases
;
70 for (pkgCache::VerFileIterator I
= Ver
.FileList(); I
.end() == false; ++I
) {
71 pkgCache::PkgFileIterator File
= I
.File();
72 signed short const p
= Cache
.GetPolicy().GetPriority(File
);
75 if (File
.Flagged(pkgCache::Flag::NotSource
) == false) {
76 string Release
= File
.RelStr();
78 Releases
.insert(Release
);
81 if (!Releases
.empty()) {
82 fprintf(output
, "APT-Release:\n");
83 for (std::set
<string
>::iterator R
= Releases
.begin(); R
!= Releases
.end(); ++R
)
84 fprintf(output
, " %s\n", R
->c_str());
86 fprintf(output
, "APT-Pin: %d\n", Pin
);
87 if (Cache
.GetCandidateVer(Pkg
) == Ver
)
88 fprintf(output
, "APT-Candidate: yes\n");
89 if ((Cache
[Pkg
].Flags
& pkgCache::Flag::Auto
) == pkgCache::Flag::Auto
)
90 fprintf(output
, "APT-Automatic: yes\n");
93 // WriteScenarioDependency /*{{{*/
94 static void WriteScenarioDependency( FILE* output
, pkgCache::VerIterator
const &Ver
)
96 std::string dependencies
[pkgCache::Dep::Enhances
+ 1];
98 for (pkgCache::DepIterator Dep
= Ver
.DependsList(); Dep
.end() == false; ++Dep
)
100 if (Dep
.IsImplicit() == true)
102 if (orGroup
== false)
103 dependencies
[Dep
->Type
].append(", ");
104 dependencies
[Dep
->Type
].append(Dep
.TargetPkg().Name());
105 if (Dep
->Version
!= 0)
106 dependencies
[Dep
->Type
].append(" (").append(pkgCache::CompTypeDeb(Dep
->CompareOp
)).append(" ").append(Dep
.TargetVer()).append(")");
107 if ((Dep
->CompareOp
& pkgCache::Dep::Or
) == pkgCache::Dep::Or
)
109 dependencies
[Dep
->Type
].append(" | ");
115 for (int i
= 1; i
< pkgCache::Dep::Enhances
+ 1; ++i
)
116 if (dependencies
[i
].empty() == false)
117 fprintf(output
, "%s: %s\n", DepMap
[i
], dependencies
[i
].c_str()+2);
119 for (pkgCache::PrvIterator Prv
= Ver
.ProvidesList(); Prv
.end() == false; ++Prv
)
121 if (Prv
.IsMultiArchImplicit() == true)
123 provides
.append(", ").append(Prv
.Name());
125 if (provides
.empty() == false)
126 fprintf(output
, "Provides: %s\n", provides
.c_str()+2);
129 // WriteScenarioLimitedDependency /*{{{*/
130 static void WriteScenarioLimitedDependency(FILE* output
,
131 pkgCache::VerIterator
const &Ver
,
132 APT::PackageSet
const &pkgset
)
134 std::string dependencies
[pkgCache::Dep::Enhances
+ 1];
135 bool orGroup
= false;
136 for (pkgCache::DepIterator Dep
= Ver
.DependsList(); Dep
.end() == false; ++Dep
)
138 if (Dep
.IsImplicit() == true)
140 if (orGroup
== false)
142 if (pkgset
.find(Dep
.TargetPkg()) == pkgset
.end())
144 dependencies
[Dep
->Type
].append(", ");
146 else if (pkgset
.find(Dep
.TargetPkg()) == pkgset
.end())
148 if ((Dep
->CompareOp
& pkgCache::Dep::Or
) == pkgCache::Dep::Or
)
150 dependencies
[Dep
->Type
].erase(dependencies
[Dep
->Type
].end()-3, dependencies
[Dep
->Type
].end());
154 dependencies
[Dep
->Type
].append(Dep
.TargetPkg().Name());
155 if (Dep
->Version
!= 0)
156 dependencies
[Dep
->Type
].append(" (").append(pkgCache::CompTypeDeb(Dep
->CompareOp
)).append(" ").append(Dep
.TargetVer()).append(")");
157 if ((Dep
->CompareOp
& pkgCache::Dep::Or
) == pkgCache::Dep::Or
)
159 dependencies
[Dep
->Type
].append(" | ");
165 for (int i
= 1; i
< pkgCache::Dep::Enhances
+ 1; ++i
)
166 if (dependencies
[i
].empty() == false)
167 fprintf(output
, "%s: %s\n", DepMap
[i
], dependencies
[i
].c_str()+2);
169 for (pkgCache::PrvIterator Prv
= Ver
.ProvidesList(); Prv
.end() == false; ++Prv
)
171 if (Prv
.IsMultiArchImplicit() == true)
173 if (pkgset
.find(Prv
.ParentPkg()) == pkgset
.end())
175 provides
.append(", ").append(Prv
.Name());
177 if (provides
.empty() == false)
178 fprintf(output
, "Provides: %s\n", provides
.c_str()+2);
181 // EDSP::WriteScenario - to the given file descriptor /*{{{*/
182 bool EDSP::WriteScenario(pkgDepCache
&Cache
, FILE* output
, OpProgress
*Progress
)
184 if (Progress
!= NULL
)
185 Progress
->SubProgress(Cache
.Head().VersionCount
, _("Send scenario to solver"));
187 std::vector
<std::string
> archs
= APT::Configuration::getArchitectures();
188 for (pkgCache::PkgIterator Pkg
= Cache
.PkgBegin(); Pkg
.end() == false; ++Pkg
)
190 std::string
const arch
= Pkg
.Arch();
191 if (std::find(archs
.begin(), archs
.end(), arch
) == archs
.end())
193 for (pkgCache::VerIterator Ver
= Pkg
.VersionList(); Ver
.end() == false; ++Ver
, ++p
)
195 WriteScenarioVersion(Cache
, output
, Pkg
, Ver
);
196 WriteScenarioDependency(output
, Ver
);
197 fprintf(output
, "\n");
198 if (Progress
!= NULL
&& p
% 100 == 0)
199 Progress
->Progress(p
);
205 // EDSP::WriteLimitedScenario - to the given file descriptor /*{{{*/
206 bool EDSP::WriteLimitedScenario(pkgDepCache
&Cache
, FILE* output
,
207 APT::PackageSet
const &pkgset
,
208 OpProgress
*Progress
)
210 if (Progress
!= NULL
)
211 Progress
->SubProgress(Cache
.Head().VersionCount
, _("Send scenario to solver"));
213 for (APT::PackageSet::const_iterator Pkg
= pkgset
.begin(); Pkg
!= pkgset
.end(); ++Pkg
, ++p
)
214 for (pkgCache::VerIterator Ver
= Pkg
.VersionList(); Ver
.end() == false; ++Ver
)
216 WriteScenarioVersion(Cache
, output
, Pkg
, Ver
);
217 WriteScenarioLimitedDependency(output
, Ver
, pkgset
);
218 fprintf(output
, "\n");
219 if (Progress
!= NULL
&& p
% 100 == 0)
220 Progress
->Progress(p
);
222 if (Progress
!= NULL
)
227 // EDSP::WriteRequest - to the given file descriptor /*{{{*/
228 bool EDSP::WriteRequest(pkgDepCache
&Cache
, FILE* output
, bool const Upgrade
,
229 bool const DistUpgrade
, bool const AutoRemove
,
230 OpProgress
*Progress
)
232 if (Progress
!= NULL
)
233 Progress
->SubProgress(Cache
.Head().PackageCount
, _("Send request to solver"));
236 for (pkgCache::PkgIterator Pkg
= Cache
.PkgBegin(); Pkg
.end() == false; ++Pkg
, ++p
)
238 if (Progress
!= NULL
&& p
% 100 == 0)
239 Progress
->Progress(p
);
241 pkgDepCache::StateCache
&P
= Cache
[Pkg
];
242 if (P
.Delete() == true)
244 else if (P
.NewInstall() == true || P
.Upgrade() == true || P
.ReInstall() == true ||
245 (P
.Mode
== pkgDepCache::ModeKeep
&& (P
.iFlags
& pkgDepCache::Protected
) == pkgDepCache::Protected
))
249 req
->append(" ").append(Pkg
.FullName());
251 fprintf(output
, "Request: EDSP 0.5\n");
253 const char *arch
= _config
->Find("APT::Architecture").c_str();
254 std::vector
<string
> archs
= APT::Configuration::getArchitectures();
255 fprintf(output
, "Architecture: %s\n", arch
);
256 fprintf(output
, "Architectures:");
257 for (std::vector
<string
>::const_iterator a
= archs
.begin(); a
!= archs
.end(); ++a
)
258 fprintf(output
, " %s", a
->c_str());
259 fprintf(output
, "\n");
261 if (del
.empty() == false)
262 fprintf(output
, "Remove: %s\n", del
.c_str()+1);
263 if (inst
.empty() == false)
264 fprintf(output
, "Install: %s\n", inst
.c_str()+1);
266 fprintf(output
, "Upgrade: yes\n");
267 if (DistUpgrade
== true)
268 fprintf(output
, "Dist-Upgrade: yes\n");
269 if (AutoRemove
== true)
270 fprintf(output
, "Autoremove: yes\n");
271 if (_config
->FindB("APT::Solver::Strict-Pinning", true) == false)
272 fprintf(output
, "Strict-Pinning: no\n");
273 string
solverpref("APT::Solver::");
274 solverpref
.append(_config
->Find("APT::Solver", "internal")).append("::Preferences");
275 if (_config
->Exists(solverpref
) == true)
276 fprintf(output
, "Preferences: %s\n", _config
->Find(solverpref
,"").c_str());
277 fprintf(output
, "\n");
282 // EDSP::ReadResponse - from the given file descriptor /*{{{*/
283 bool EDSP::ReadResponse(int const input
, pkgDepCache
&Cache
, OpProgress
*Progress
) {
284 /* We build an map id to mmap offset here
285 In theory we could use the offset as ID, but then VersionCount
286 couldn't be used to create other versionmappings anymore and it
287 would be too easy for a (buggy) solver to segfault APTā¦ */
288 unsigned long long const VersionCount
= Cache
.Head().VersionCount
;
289 unsigned long VerIdx
[VersionCount
];
290 for (pkgCache::PkgIterator P
= Cache
.PkgBegin(); P
.end() == false; ++P
) {
291 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; ++V
)
292 VerIdx
[V
->ID
] = V
.Index();
293 Cache
[P
].Marked
= true;
294 Cache
[P
].Garbage
= false;
298 in
.OpenDescriptor(input
, FileFd::ReadOnly
);
299 pkgTagFile
response(&in
, 100);
300 pkgTagSection section
;
302 while (response
.Step(section
) == true) {
304 if (section
.Exists("Install") == true)
306 else if (section
.Exists("Remove") == true)
308 else if (section
.Exists("Progress") == true) {
309 if (Progress
!= NULL
) {
310 string msg
= section
.FindS("Message");
311 if (msg
.empty() == true)
312 msg
= _("Prepare for receiving solution");
313 Progress
->SubProgress(100, msg
, section
.FindI("Percentage", 0));
316 } else if (section
.Exists("Error") == true) {
317 std::string msg
= SubstVar(SubstVar(section
.FindS("Message"), "\n .\n", "\n\n"), "\n ", "\n");
318 if (msg
.empty() == true) {
319 msg
= _("External solver failed without a proper error message");
320 _error
->Error("%s", msg
.c_str());
322 _error
->Error("External solver failed with: %s", msg
.substr(0,msg
.find('\n')).c_str());
323 if (Progress
!= NULL
)
325 std::cerr
<< "The solver encountered an error of type: " << section
.FindS("Error") << std::endl
;
326 std::cerr
<< "The following information might help you to understand what is wrong:" << std::endl
;
327 std::cerr
<< msg
<< std::endl
<< std::endl
;
329 } else if (section
.Exists("Autoremove") == true)
334 size_t const id
= section
.FindULL(type
.c_str(), VersionCount
);
335 if (id
== VersionCount
) {
336 _error
->Warning("Unable to parse %s request with id value '%s'!", type
.c_str(), section
.FindS(type
.c_str()).c_str());
338 } else if (id
> Cache
.Head().VersionCount
) {
339 _error
->Warning("ID value '%s' in %s request stanza is to high to refer to a known version!", section
.FindS(type
.c_str()).c_str(), type
.c_str());
343 pkgCache::VerIterator
Ver(Cache
.GetCache(), Cache
.GetCache().VerP
+ VerIdx
[id
]);
344 Cache
.SetCandidateVersion(Ver
);
345 if (type
== "Install")
347 pkgCache::PkgIterator
const P
= Ver
.ParentPkg();
348 if (Cache
[P
].Mode
!= pkgDepCache::ModeInstall
)
349 Cache
.MarkInstall(P
, false, 0, false);
351 else if (type
== "Remove")
352 Cache
.MarkDelete(Ver
.ParentPkg(), false);
353 else if (type
== "Autoremove") {
354 Cache
[Ver
.ParentPkg()].Marked
= false;
355 Cache
[Ver
.ParentPkg()].Garbage
= true;
361 // ReadLine - first line from the given file descriptor /*{{{*/
362 // ---------------------------------------------------------------------
363 /* Little helper method to read a complete line into a string. Similar to
364 fgets but we need to use the low-level read() here as otherwise the
365 listparser will be confused later on as mixing of fgets and read isn't
366 a supported action according to the manpages and results are undefined */
367 static bool ReadLine(int const input
, std::string
&line
) {
372 while ((data
= read(input
, &one
, sizeof(one
))) != -1) {
379 if (line
.empty() == true && isblank(one
) != 0)
386 // StringToBool - convert yes/no to bool /*{{{*/
387 // ---------------------------------------------------------------------
388 /* we are not as lazy as we are in the global StringToBool as we really
389 only accept yes/no here - but we will ignore leading spaces */
390 static bool StringToBool(char const *answer
, bool const defValue
) {
391 for (; isspace(*answer
) != 0; ++answer
);
392 if (strncasecmp(answer
, "yes", 3) == 0)
394 else if (strncasecmp(answer
, "no", 2) == 0)
397 _error
->Warning("Value '%s' is not a boolean 'yes' or 'no'!", answer
);
401 // EDSP::ReadRequest - first stanza from the given file descriptor /*{{{*/
402 bool EDSP::ReadRequest(int const input
, std::list
<std::string
> &install
,
403 std::list
<std::string
> &remove
, bool &upgrade
,
404 bool &distUpgrade
, bool &autoRemove
)
412 while (ReadLine(input
, line
) == true)
414 // Skip empty lines before request
415 if (line
.empty() == true)
417 // The first Tag must be a request, so search for it
418 if (line
.compare(0, 8, "Request:") != 0)
421 while (ReadLine(input
, line
) == true)
423 // empty lines are the end of the request
424 if (line
.empty() == true)
427 std::list
<std::string
> *request
= NULL
;
428 if (line
.compare(0, 8, "Install:") == 0)
433 else if (line
.compare(0, 7, "Remove:") == 0)
438 else if (line
.compare(0, 8, "Upgrade:") == 0)
439 upgrade
= StringToBool(line
.c_str() + 9, false);
440 else if (line
.compare(0, 13, "Dist-Upgrade:") == 0)
441 distUpgrade
= StringToBool(line
.c_str() + 14, false);
442 else if (line
.compare(0, 11, "Autoremove:") == 0)
443 autoRemove
= StringToBool(line
.c_str() + 12, false);
444 else if (line
.compare(0, 13, "Architecture:") == 0)
445 _config
->Set("APT::Architecture", line
.c_str() + 14);
446 else if (line
.compare(0, 14, "Architectures:") == 0)
448 std::string
const archs
= line
.c_str() + 15;
449 _config
->Set("APT::Architectures", SubstVar(archs
, " ", ","));
452 _error
->Warning("Unknown line in EDSP Request stanza: %s", line
.c_str());
456 size_t end
= line
.length();
458 size_t begin
= line
.rfind(' ');
459 if (begin
== std::string::npos
)
461 request
->push_back(line
.substr(0, end
));
464 else if (begin
< end
)
465 request
->push_back(line
.substr(begin
+ 1, end
));
467 end
= line
.find_last_not_of(' ');
468 } while (end
!= std::string::npos
);
474 // EDSP::ApplyRequest - first stanza from the given file descriptor /*{{{*/
475 bool EDSP::ApplyRequest(std::list
<std::string
> const &install
,
476 std::list
<std::string
> const &remove
,
479 for (std::list
<std::string
>::const_iterator i
= install
.begin();
480 i
!= install
.end(); ++i
) {
481 pkgCache::PkgIterator P
= Cache
.FindPkg(*i
);
483 _error
->Warning("Package %s is not known, so can't be installed", i
->c_str());
485 Cache
.MarkInstall(P
, false);
488 for (std::list
<std::string
>::const_iterator i
= remove
.begin();
489 i
!= remove
.end(); ++i
) {
490 pkgCache::PkgIterator P
= Cache
.FindPkg(*i
);
492 _error
->Warning("Package %s is not known, so can't be installed", i
->c_str());
499 // EDSP::WriteSolution - to the given file descriptor /*{{{*/
500 bool EDSP::WriteSolution(pkgDepCache
&Cache
, FILE* output
)
502 bool const Debug
= _config
->FindB("Debug::EDSP::WriteSolution", false);
503 for (pkgCache::PkgIterator Pkg
= Cache
.PkgBegin(); Pkg
.end() == false; ++Pkg
)
505 if (Cache
[Pkg
].Delete() == true)
507 fprintf(output
, "Remove: %d\n", Pkg
.CurrentVer()->ID
);
509 fprintf(output
, "Package: %s\nVersion: %s\n", Pkg
.FullName().c_str(), Pkg
.CurrentVer().VerStr());
511 else if (Cache
[Pkg
].NewInstall() == true || Cache
[Pkg
].Upgrade() == true)
513 fprintf(output
, "Install: %d\n", Cache
.GetCandidateVer(Pkg
)->ID
);
515 fprintf(output
, "Package: %s\nVersion: %s\n", Pkg
.FullName().c_str(), Cache
.GetCandidateVer(Pkg
).VerStr());
517 else if (Cache
[Pkg
].Garbage
== true)
519 fprintf(output
, "Autoremove: %d\n", Pkg
.CurrentVer()->ID
);
521 fprintf(output
, "Package: %s\nVersion: %s\n", Pkg
.FullName().c_str(), Pkg
.CurrentVer().VerStr());
525 fprintf(output
, "\n");
531 // EDSP::WriteProgess - pulse to the given file descriptor /*{{{*/
532 bool EDSP::WriteProgress(unsigned short const percent
, const char* const message
, FILE* output
) {
533 fprintf(output
, "Progress: %s\n", TimeRFC1123(time(NULL
)).c_str());
534 fprintf(output
, "Percentage: %d\n", percent
);
535 fprintf(output
, "Message: %s\n\n", message
);
540 // EDSP::WriteError - format an error message to be send to file descriptor /*{{{*/
541 bool EDSP::WriteError(char const * const uuid
, std::string
const &message
, FILE* output
) {
542 fprintf(output
, "Error: %s\n", uuid
);
543 fprintf(output
, "Message: %s\n\n", SubstVar(SubstVar(message
, "\n\n", "\n.\n"), "\n", "\n ").c_str());
547 // EDSP::ExecuteSolver - fork requested solver and setup ipc pipes {{{*/
548 pid_t
EDSP::ExecuteSolver(const char* const solver
, int * const solver_in
, int * const solver_out
, bool) {
549 std::vector
<std::string
> const solverDirs
= _config
->FindVector("Dir::Bin::Solvers");
551 for (std::vector
<std::string
>::const_iterator dir
= solverDirs
.begin();
552 dir
!= solverDirs
.end(); ++dir
) {
553 file
= flCombine(*dir
, solver
);
554 if (RealFileExists(file
.c_str()) == true)
559 if (file
.empty() == true)
561 _error
->Error("Can't call external solver '%s' as it is not in a configured directory!", solver
);
564 int external
[4] = {-1, -1, -1, -1};
565 if (pipe(external
) != 0 || pipe(external
+ 2) != 0)
567 _error
->Errno("Resolve", "Can't create needed IPC pipes for EDSP");
570 for (int i
= 0; i
< 4; ++i
)
571 SetCloseExec(external
[i
], true);
573 pid_t Solver
= ExecFork();
575 dup2(external
[0], STDIN_FILENO
);
576 dup2(external
[3], STDOUT_FILENO
);
577 const char* calling
[2] = { file
.c_str(), 0 };
578 execv(calling
[0], (char**) calling
);
579 std::cerr
<< "Failed to execute solver '" << solver
<< "'!" << std::endl
;
585 if (WaitFd(external
[1], true, 5) == false)
587 _error
->Errno("Resolve", "Timed out while Waiting on availability of solver stdin");
591 *solver_in
= external
[1];
592 *solver_out
= external
[2];
595 bool EDSP::ExecuteSolver(const char* const solver
, int *solver_in
, int *solver_out
) {
596 if (ExecuteSolver(solver
, solver_in
, solver_out
, true) == 0)
601 // EDSP::ResolveExternal - resolve problems by asking external for help {{{*/
602 bool EDSP::ResolveExternal(const char* const solver
, pkgDepCache
&Cache
,
603 bool const upgrade
, bool const distUpgrade
,
604 bool const autoRemove
, OpProgress
*Progress
) {
605 int solver_in
, solver_out
;
606 pid_t
const solver_pid
= EDSP::ExecuteSolver(solver
, &solver_in
, &solver_out
, true);
610 FILE* output
= fdopen(solver_in
, "w");
612 return _error
->Errno("Resolve", "fdopen on solver stdin failed");
614 if (Progress
!= NULL
)
615 Progress
->OverallProgress(0, 100, 5, _("Execute external solver"));
616 EDSP::WriteRequest(Cache
, output
, upgrade
, distUpgrade
, autoRemove
, Progress
);
617 if (Progress
!= NULL
)
618 Progress
->OverallProgress(5, 100, 20, _("Execute external solver"));
619 EDSP::WriteScenario(Cache
, output
, Progress
);
622 if (Progress
!= NULL
)
623 Progress
->OverallProgress(25, 100, 75, _("Execute external solver"));
624 if (EDSP::ReadResponse(solver_out
, Cache
, Progress
) == false)
627 return ExecWait(solver_pid
, solver
);