5 #include <apt-pkg/error.h>
6 #include <apt-pkg/cdromutl.h>
7 #include <apt-pkg/strutl.h>
8 #include <apt-pkg/cdrom.h>
9 #include <apt-pkg/aptconfiguration.h>
10 #include <apt-pkg/configuration.h>
11 #include <apt-pkg/fileutl.h>
12 #include <apt-pkg/indexcopy.h>
32 // FindPackages - Find the package files on the CDROM /*{{{*/
33 // ---------------------------------------------------------------------
34 /* We look over the cdrom for package files. This is a recursive
35 search that short circuits when it his a package file in the dir.
36 This speeds it up greatly as the majority of the size is in the
38 bool pkgCdrom::FindPackages(string CD
,
40 vector
<string
> &SList
,
41 vector
<string
> &SigList
,
42 vector
<string
> &TransList
,
43 string
&InfoDir
, pkgCdromStatus
*log
,
46 static ino_t Inodes
[9];
49 // if we have a look we "pulse" now
56 if (CD
[CD
.length()-1] != '/')
59 if (chdir(CD
.c_str()) != 0)
60 return _error
->Errno("chdir","Unable to change to %s",CD
.c_str());
62 // Look for a .disk subdirectory
63 if (DirectoryExists(".disk") == true)
65 if (InfoDir
.empty() == true)
66 InfoDir
= CD
+ ".disk/";
69 // Don't look into directories that have been marked to ingore.
70 if (RealFileExists(".aptignr") == true)
73 /* Check _first_ for a signature file as apt-cdrom assumes that all files
74 under a Packages/Source file are in control of that file and stops
77 if (RealFileExists("Release.gpg") == true || RealFileExists("InRelease") == true)
79 SigList
.push_back(CD
);
82 /* Aha! We found some package files. We assume that everything under
83 this dir is controlled by those package files so we don't look down
85 std::vector
<APT::Configuration::Compressor
> const compressor
= APT::Configuration::getCompressors();
86 for (std::vector
<APT::Configuration::Compressor
>::const_iterator c
= compressor
.begin();
87 c
!= compressor
.end(); ++c
)
89 if (RealFileExists(std::string("Packages").append(c
->Extension
).c_str()) == false)
92 if (_config
->FindB("Debug::aptcdrom",false) == true)
93 std::clog
<< "Found Packages in " << CD
<< std::endl
;
96 // Continue down if thorough is given
97 if (_config
->FindB("APT::CDROM::Thorough",false) == false)
101 for (std::vector
<APT::Configuration::Compressor
>::const_iterator c
= compressor
.begin();
102 c
!= compressor
.end(); ++c
)
104 if (RealFileExists(std::string("Sources").append(c
->Extension
).c_str()) == false)
107 if (_config
->FindB("Debug::aptcdrom",false) == true)
108 std::clog
<< "Found Sources in " << CD
<< std::endl
;
111 // Continue down if thorough is given
112 if (_config
->FindB("APT::CDROM::Thorough",false) == false)
117 // see if we find translation indices
118 if (DirectoryExists("i18n") == true)
121 for (struct dirent
*Dir
= readdir(D
); Dir
!= 0; Dir
= readdir(D
))
123 if(strncmp(Dir
->d_name
, "Translation-", strlen("Translation-")) != 0)
125 string file
= Dir
->d_name
;
126 for (std::vector
<APT::Configuration::Compressor
>::const_iterator c
= compressor
.begin();
127 c
!= compressor
.end(); ++c
)
129 string fileext
= flExtension(file
);
132 else if (fileext
.empty() == false)
133 fileext
= "." + fileext
;
135 if (c
->Extension
== fileext
)
137 if (_config
->FindB("Debug::aptcdrom",false) == true)
138 std::clog
<< "Found translation " << Dir
->d_name
<< " in " << CD
<< "i18n/" << std::endl
;
139 file
.erase(file
.size() - fileext
.size());
140 TransList
.push_back(CD
+ "i18n/" + file
);
150 return _error
->Errno("opendir","Unable to read %s",CD
.c_str());
152 // Run over the directory
153 for (struct dirent
*Dir
= readdir(D
); Dir
!= 0; Dir
= readdir(D
))
156 if (strcmp(Dir
->d_name
,".") == 0 ||
157 strcmp(Dir
->d_name
,"..") == 0 ||
158 //strcmp(Dir->d_name,"source") == 0 ||
159 strcmp(Dir
->d_name
,".disk") == 0 ||
160 strcmp(Dir
->d_name
,"experimental") == 0 ||
161 strcmp(Dir
->d_name
,"binary-all") == 0 ||
162 strcmp(Dir
->d_name
,"debian-installer") == 0)
165 // See if the name is a sub directory
167 if (stat(Dir
->d_name
,&Buf
) != 0)
170 if (S_ISDIR(Buf
.st_mode
) == 0)
174 for (I
= 0; I
!= Depth
; I
++)
175 if (Inodes
[I
] == Buf
.st_ino
)
180 // Store the inodes weve seen
181 Inodes
[Depth
] = Buf
.st_ino
;
184 if (FindPackages(CD
+ Dir
->d_name
,List
,SList
,SigList
,TransList
,InfoDir
,log
,Depth
+1) == false)
187 if (chdir(CD
.c_str()) != 0)
189 _error
->Errno("chdir","Unable to change to %s", CD
.c_str());
197 return !_error
->PendingError();
200 // Score - We compute a 'score' for a path /*{{{*/
201 // ---------------------------------------------------------------------
202 /* Paths are scored based on how close they come to what I consider
203 normal. That is ones that have 'dist' 'stable' 'testing' will score
204 higher than ones without. */
205 int pkgCdrom::Score(string Path
)
208 if (Path
.find("stable/") != string::npos
)
210 if (Path
.find("/binary-") != string::npos
)
212 if (Path
.find("testing/") != string::npos
)
214 if (Path
.find("unstable/") != string::npos
)
216 if (Path
.find("/dists/") != string::npos
)
218 if (Path
.find("/main/") != string::npos
)
220 if (Path
.find("/contrib/") != string::npos
)
222 if (Path
.find("/non-free/") != string::npos
)
224 if (Path
.find("/non-US/") != string::npos
)
226 if (Path
.find("/source/") != string::npos
)
228 if (Path
.find("/debian/") != string::npos
)
231 // check for symlinks in the patch leading to the actual file
232 // a symlink gets a big penalty
234 string statPath
= flNotFile(Path
);
235 string cdromPath
= _config
->FindDir("Acquire::cdrom::mount");
236 while(statPath
!= cdromPath
&& statPath
!= "./") {
237 statPath
.resize(statPath
.size()-1); // remove the trailing '/'
238 if (lstat(statPath
.c_str(),&Buf
) == 0) {
239 if(S_ISLNK(Buf
.st_mode
)) {
244 statPath
= flNotFile(statPath
); // descent
250 // DropBinaryArch - Dump dirs with a string like /binary-<foo>/ /*{{{*/
251 // ---------------------------------------------------------------------
252 /* Here we drop everything that is not this machines arch */
253 bool pkgCdrom::DropBinaryArch(vector
<string
> &List
)
256 for (unsigned int I
= 0; I
< List
.size(); I
++)
258 const char *Str
= List
[I
].c_str();
259 const char *Start
, *End
;
260 if ((Start
= strstr(Str
,"/binary-")) == 0)
263 // Between Start and End is the architecture
265 if ((End
= strstr(Start
,"/")) != 0 && Start
!= End
&&
266 APT::Configuration::checkArchitecture(string(Start
, End
)) == true)
267 continue; // okay, architecture is accepted
269 // not accepted -> Erase it
270 List
.erase(List
.begin() + I
);
271 --I
; // the next entry is at the same index after the erase
277 // DropTranslation - Dump unwanted Translation-<lang> files /*{{{*/
278 // ---------------------------------------------------------------------
279 /* Here we drop everything that is not configured in Acquire::Languages */
280 bool pkgCdrom::DropTranslation(vector
<string
> &List
)
282 for (unsigned int I
= 0; I
< List
.size(); I
++)
285 if ((Start
= strstr(List
[I
].c_str(), "/Translation-")) == NULL
)
287 Start
+= strlen("/Translation-");
289 if (APT::Configuration::checkLanguage(Start
, true) == true)
292 // not accepted -> Erase it
293 List
.erase(List
.begin() + I
);
294 --I
; // the next entry is at the same index after the erase
300 // DropRepeats - Drop repeated files resulting from symlinks /*{{{*/
301 // ---------------------------------------------------------------------
302 /* Here we go and stat every file that we found and strip dup inodes. */
303 bool pkgCdrom::DropRepeats(vector
<string
> &List
,const char *Name
)
305 bool couldFindAllFiles
= true;
306 // Get a list of all the inodes
307 ino_t
*Inodes
= new ino_t
[List
.size()];
308 for (unsigned int I
= 0; I
!= List
.size(); ++I
)
313 std::vector
<APT::Configuration::Compressor
> const compressor
= APT::Configuration::getCompressors();
314 for (std::vector
<APT::Configuration::Compressor
>::const_iterator c
= compressor
.begin();
315 c
!= compressor
.end(); ++c
)
317 std::string filename
= std::string(List
[I
]).append(Name
).append(c
->Extension
);
318 if (stat(filename
.c_str(), &Buf
) != 0)
320 Inodes
[I
] = Buf
.st_ino
;
327 _error
->Errno("stat","Failed to stat %s%s",List
[I
].c_str(), Name
);
328 couldFindAllFiles
= false;
334 for (unsigned int I
= 0; I
!= List
.size(); I
++)
338 for (unsigned int J
= I
+1; J
< List
.size(); J
++)
341 if (Inodes
[J
] == 0 || Inodes
[J
] != Inodes
[I
])
344 // We score the two paths.. and erase one
345 int ScoreA
= Score(List
[I
]);
346 int ScoreB
= Score(List
[J
]);
358 // Wipe erased entries
359 for (unsigned int I
= 0; I
< List
.size();)
361 if (List
[I
].empty() == false)
364 List
.erase(List
.begin()+I
);
367 return couldFindAllFiles
;
370 // ReduceSourceList - Takes the path list and reduces it /*{{{*/
371 // ---------------------------------------------------------------------
372 /* This takes the list of source list expressed entires and collects
373 similar ones to form a single entry for each dist */
374 void pkgCdrom::ReduceSourcelist(string
/*CD*/,vector
<string
> &List
)
376 sort(List
.begin(),List
.end());
378 // Collect similar entries
379 for (vector
<string
>::iterator I
= List
.begin(); I
!= List
.end(); ++I
)
382 string::size_type Space
= (*I
).find(' ');
383 if (Space
== string::npos
)
385 string::size_type SSpace
= (*I
).find(' ',Space
+ 1);
386 if (SSpace
== string::npos
)
389 string Word1
= string(*I
,Space
,SSpace
-Space
);
390 string Prefix
= string(*I
,0,Space
);
391 string Component
= string(*I
,SSpace
);
392 for (vector
<string
>::iterator J
= List
.begin(); J
!= I
; ++J
)
395 string::size_type Space2
= (*J
).find(' ');
396 if (Space2
== string::npos
)
398 string::size_type SSpace2
= (*J
).find(' ',Space2
+ 1);
399 if (SSpace2
== string::npos
)
402 if (string(*J
,0,Space2
) != Prefix
)
404 if (string(*J
,Space2
,SSpace2
-Space2
) != Word1
)
407 string Component2
= string(*J
, SSpace2
) + " ";
408 if (Component2
.find(Component
+ " ") == std::string::npos
)
414 // Wipe erased entries
415 for (unsigned int I
= 0; I
< List
.size();)
417 if (List
[I
].empty() == false)
420 List
.erase(List
.begin()+I
);
424 // WriteDatabase - Write the CDROM Database file /*{{{*/
425 // ---------------------------------------------------------------------
426 /* We rewrite the configuration class associated with the cdrom database. */
427 bool pkgCdrom::WriteDatabase(Configuration
&Cnf
)
429 string DFile
= _config
->FindFile("Dir::State::cdroms");
430 string NewFile
= DFile
+ ".new";
432 unlink(NewFile
.c_str());
433 ofstream
Out(NewFile
.c_str());
435 return _error
->Errno("ofstream::ofstream",
436 "Failed to open %s.new",DFile
.c_str());
438 /* Write out all of the configuration directives by walking the
439 configuration tree */
440 Cnf
.Dump(Out
, NULL
, "%f \"%v\";\n", false);
444 if (FileExists(DFile
) == true)
445 rename(DFile
.c_str(), (DFile
+ '~').c_str());
446 if (rename(NewFile
.c_str(),DFile
.c_str()) != 0)
447 return _error
->Errno("rename","Failed to rename %s.new to %s",
448 DFile
.c_str(),DFile
.c_str());
453 // WriteSourceList - Write an updated sourcelist /*{{{*/
454 // ---------------------------------------------------------------------
455 /* This reads the old source list and copies it into the new one. It
456 appends the new CDROM entires just after the first block of comments.
457 This places them first in the file. It also removes any old entries
458 that were the same. */
459 bool pkgCdrom::WriteSourceList(string Name
,vector
<string
> &List
,bool Source
)
461 if (List
.empty() == true)
464 string File
= _config
->FindFile("Dir::Etc::sourcelist");
466 // Open the stream for reading
467 ifstream
F((FileExists(File
)?File
.c_str():"/dev/null"),
470 return _error
->Errno("ifstream::ifstream","Opening %s",File
.c_str());
472 string NewFile
= File
+ ".new";
473 unlink(NewFile
.c_str());
474 ofstream
Out(NewFile
.c_str());
476 return _error
->Errno("ofstream::ofstream",
477 "Failed to open %s.new",File
.c_str());
479 // Create a short uri without the path
480 string ShortURI
= "cdrom:[" + Name
+ "]/";
481 string ShortURI2
= "cdrom:" + Name
+ "/"; // For Compatibility
492 while (F
.eof() == false)
494 F
.getline(Buffer
,sizeof(Buffer
));
496 if (F
.fail() && !F
.eof())
497 return _error
->Error(_("Line %u too long in source list %s."),
498 CurLine
,File
.c_str());
499 _strtabexpand(Buffer
,sizeof(Buffer
));
503 if (Buffer
[0] == '#' || Buffer
[0] == 0)
505 Out
<< Buffer
<< endl
;
511 for (vector
<string
>::iterator I
= List
.begin(); I
!= List
.end(); ++I
)
513 string::size_type Space
= (*I
).find(' ');
514 if (Space
== string::npos
)
515 return _error
->Error("Internal error");
516 Out
<< Type
<< " cdrom:[" << Name
<< "]/" << string(*I
,0,Space
) <<
517 " " << string(*I
,Space
+1) << endl
;
525 const char *C
= Buffer
;
526 if (ParseQuoteWord(C
,cType
) == false ||
527 ParseQuoteWord(C
,URI
) == false)
529 Out
<< Buffer
<< endl
;
533 // Emit lines like this one
534 if (cType
!= Type
|| (string(URI
,0,ShortURI
.length()) != ShortURI
&&
535 string(URI
,0,ShortURI
.length()) != ShortURI2
))
537 Out
<< Buffer
<< endl
;
542 // Just in case the file was empty
545 for (vector
<string
>::iterator I
= List
.begin(); I
!= List
.end(); ++I
)
547 string::size_type Space
= (*I
).find(' ');
548 if (Space
== string::npos
)
549 return _error
->Error("Internal error");
551 Out
<< "deb cdrom:[" << Name
<< "]/" << string(*I
,0,Space
) <<
552 " " << string(*I
,Space
+1) << endl
;
558 rename(File
.c_str(), (File
+ '~').c_str());
559 if (rename(NewFile
.c_str(),File
.c_str()) != 0)
560 return _error
->Errno("rename","Failed to rename %s.new to %s",
561 File
.c_str(),File
.c_str());
566 bool pkgCdrom::MountAndIdentCDROM(Configuration
&Database
, std::string
&CDROM
, std::string
&ident
, pkgCdromStatus
* const log
, bool const interactive
)/*{{{*/
569 CDROM
= _config
->FindDir("Acquire::cdrom::mount");
571 CDROM
= SafeGetCWD() + '/' + CDROM
;
576 log
->SetTotal(STEP_LAST
);
577 strprintf(msg
, _("Using CD-ROM mount point %s\n"), CDROM
.c_str());
578 log
->Update(msg
, STEP_PREPARE
);
581 // Unmount the CD and get the user to put in the one they want
582 if (_config
->FindB("APT::CDROM::NoMount", false) == false)
584 if (interactive
== true)
587 log
->Update(_("Unmounting CD-ROM...\n"), STEP_LAST
);
592 log
->Update(_("Waiting for disc...\n"), STEP_WAIT
);
593 if(!log
->ChangeCdrom()) {
600 // Mount the new CDROM
602 log
->Update(_("Mounting CD-ROM...\n"), STEP_MOUNT
);
604 if (MountCdrom(CDROM
) == false)
605 return _error
->Error("Failed to mount the cdrom.");
608 // Hash the CD to get an ID
610 log
->Update(_("Identifying... "), STEP_IDENT
);
612 if (IdentCdrom(CDROM
,ident
) == false)
623 strprintf(msg
, "[%s]\n", ident
.c_str());
628 string DFile
= _config
->FindFile("Dir::State::cdroms");
629 if (FileExists(DFile
) == true)
631 if (ReadConfigFile(Database
,DFile
) == false)
632 return _error
->Error("Unable to read the cdrom database %s",
638 bool pkgCdrom::Ident(string
&ident
, pkgCdromStatus
*log
) /*{{{*/
640 Configuration Database
;
642 if (MountAndIdentCDROM(Database
, CDROM
, ident
, log
, false) == false)
648 strprintf(msg
, _("Stored label: %s\n"),
649 Database
.Find("CD::"+ident
).c_str());
653 // Unmount and finish
654 if (_config
->FindB("APT::CDROM::NoMount",false) == false)
657 log
->Update(_("Unmounting CD-ROM...\n"), STEP_LAST
);
664 bool pkgCdrom::Add(pkgCdromStatus
*log
) /*{{{*/
666 Configuration Database
;
667 std::string ID
, CDROM
;
668 if (MountAndIdentCDROM(Database
, CDROM
, ID
, log
, true) == false)
672 log
->Update(_("Scanning disc for index files...\n"),STEP_SCAN
);
674 // Get the CD structure
676 vector
<string
> SourceList
;
677 vector
<string
> SigList
;
678 vector
<string
> TransList
;
679 string StartDir
= SafeGetCWD();
681 if (FindPackages(CDROM
,List
,SourceList
, SigList
,TransList
,InfoDir
,log
) == false)
688 if (chdir(StartDir
.c_str()) != 0)
689 return _error
->Errno("chdir","Unable to change to %s", StartDir
.c_str());
691 if (_config
->FindB("Debug::aptcdrom",false) == true)
693 cout
<< "I found (binary):" << endl
;
694 for (vector
<string
>::iterator I
= List
.begin(); I
!= List
.end(); ++I
)
696 cout
<< "I found (source):" << endl
;
697 for (vector
<string
>::iterator I
= SourceList
.begin(); I
!= SourceList
.end(); ++I
)
699 cout
<< "I found (Signatures):" << endl
;
700 for (vector
<string
>::iterator I
= SigList
.begin(); I
!= SigList
.end(); ++I
)
704 //log->Update(_("Cleaning package lists..."), STEP_CLEAN);
707 DropBinaryArch(List
);
708 DropRepeats(List
,"Packages");
709 DropRepeats(SourceList
,"Sources");
710 // FIXME: We ignore stat() errors here as we usually have only one of those in use
711 // This has little potencial to drop 'valid' stat() errors as we know that one of these
712 // files need to exist, but it would be better if we would check it here
713 _error
->PushToStack();
714 DropRepeats(SigList
,"Release.gpg");
715 DropRepeats(SigList
,"InRelease");
716 _error
->RevertToStack();
717 DropRepeats(TransList
,"");
718 if (_config
->FindB("APT::CDROM::DropTranslation", true) == true)
719 DropTranslation(TransList
);
722 strprintf(msg
, _("Found %zu package indexes, %zu source indexes, "
723 "%zu translation indexes and %zu signatures\n"),
724 List
.size(), SourceList
.size(), TransList
.size(),
726 log
->Update(msg
, STEP_SCAN
);
729 if (List
.empty() == true && SourceList
.empty() == true)
731 if (_config
->FindB("APT::CDROM::NoMount",false) == false)
733 return _error
->Error(_("Unable to locate any package files, perhaps this is not a Debian Disc or the wrong architecture?"));
736 // Check if the CD is in the database
738 if (Database
.Exists("CD::" + ID
) == false ||
739 _config
->FindB("APT::CDROM::Rename",false) == true)
741 // Try to use the CDs label if at all possible
742 if (InfoDir
.empty() == false &&
743 FileExists(InfoDir
+ "/info") == true)
745 ifstream
F((InfoDir
+ "/info").c_str());
749 if (Name
.empty() == false)
751 // Escape special characters
752 string::iterator J
= Name
.begin();
753 for (; J
!= Name
.end(); ++J
)
754 if (*J
== '"' || *J
== ']' || *J
== '[')
760 strprintf(msg
, _("Found label '%s'\n"), Name
.c_str());
763 Database
.Set("CD::" + ID
+ "::Label",Name
);
767 if (_config
->FindB("APT::CDROM::Rename",false) == true ||
768 Name
.empty() == true)
772 if (_config
->FindB("APT::CDROM::NoMount",false) == false)
774 return _error
->Error("No disc name found and no way to ask for it");
778 if(!log
->AskCdromName(Name
)) {
782 cout
<< "Name: '" << Name
<< "'" << endl
;
784 if (Name
.empty() == false &&
785 Name
.find('"') == string::npos
&&
786 Name
.find('[') == string::npos
&&
787 Name
.find(']') == string::npos
)
789 log
->Update(_("That is not a valid name, try again.\n"));
794 Name
= Database
.Find("CD::" + ID
);
796 // Escape special characters
797 string::iterator J
= Name
.begin();
798 for (; J
!= Name
.end(); ++J
)
799 if (*J
== '"' || *J
== ']' || *J
== '[')
802 Database
.Set("CD::" + ID
,Name
);
806 strprintf(msg
, _("This disc is called: \n'%s'\n"), Name
.c_str());
808 log
->Update(_("Copying package lists..."), STEP_COPY
);
811 // check for existence and possibly create state directory for copying
812 string
const listDir
= _config
->FindDir("Dir::State::lists");
813 string
const partialListDir
= listDir
+ "partial/";
814 if (CreateAPTDirectoryIfNeeded(_config
->FindDir("Dir::State"), partialListDir
) == false &&
815 CreateAPTDirectoryIfNeeded(listDir
, partialListDir
) == false)
816 return _error
->Errno("cdrom", _("List directory %spartial is missing."), listDir
.c_str());
818 // take care of the signatures and copy them if they are ok
819 // (we do this before PackageCopy as it modifies "List" and "SourceList")
820 SigVerify SignVerify
;
821 SignVerify
.CopyAndVerify(CDROM
, Name
, SigList
, List
, SourceList
);
823 // Copy the package files to the state directory
826 TranslationsCopy TransCopy
;
827 if (Copy
.CopyPackages(CDROM
,Name
,List
, log
) == false ||
828 SrcCopy
.CopyPackages(CDROM
,Name
,SourceList
, log
) == false ||
829 TransCopy
.CopyTranslations(CDROM
,Name
,TransList
, log
) == false)
832 // reduce the List so that it takes less space in sources.list
833 ReduceSourcelist(CDROM
,List
);
834 ReduceSourcelist(CDROM
,SourceList
);
836 // Write the database and sourcelist
837 if (_config
->FindB("APT::cdrom::NoAct",false) == false)
839 if (WriteDatabase(Database
) == false)
843 log
->Update(_("Writing new source list\n"), STEP_WRITE
);
844 if (WriteSourceList(Name
,List
,false) == false ||
845 WriteSourceList(Name
,SourceList
,true) == false)
849 // Print the sourcelist entries
851 log
->Update(_("Source list entries for this disc are:\n"));
853 for (vector
<string
>::iterator I
= List
.begin(); I
!= List
.end(); ++I
)
855 string::size_type Space
= (*I
).find(' ');
856 if (Space
== string::npos
)
858 if (_config
->FindB("APT::CDROM::NoMount",false) == false)
860 return _error
->Error("Internal error");
866 msg
<< "deb cdrom:[" << Name
<< "]/" << string(*I
,0,Space
) <<
867 " " << string(*I
,Space
+1) << endl
;
868 log
->Update(msg
.str());
872 for (vector
<string
>::iterator I
= SourceList
.begin(); I
!= SourceList
.end(); ++I
)
874 string::size_type Space
= (*I
).find(' ');
875 if (Space
== string::npos
)
877 if (_config
->FindB("APT::CDROM::NoMount",false) == false)
879 return _error
->Error("Internal error");
884 msg
<< "deb-src cdrom:[" << Name
<< "]/" << string(*I
,0,Space
) <<
885 " " << string(*I
,Space
+1) << endl
;
886 log
->Update(msg
.str());
890 // Unmount and finish
891 if (_config
->FindB("APT::CDROM::NoMount",false) == false) {
893 log
->Update(_("Unmounting CD-ROM...\n"), STEP_LAST
);
900 pkgUdevCdromDevices::pkgUdevCdromDevices() /*{{{*/
901 : libudev_handle(NULL
)
908 pkgUdevCdromDevices::Dlopen() /*{{{*/
911 if(libudev_handle
!= NULL
)
914 // see if we can get libudev
915 void *h
= ::dlopen("libudev.so.0", RTLD_LAZY
);
919 // get the pointers to the udev structs
921 udev_new
= (udev
* (*)(void)) dlsym(h
, "udev_new");
922 udev_enumerate_add_match_property
= (int (*)(udev_enumerate
*, const char*, const char*))dlsym(h
, "udev_enumerate_add_match_property");
923 udev_enumerate_add_match_sysattr
= (int (*)(udev_enumerate
*, const char*, const char*))dlsym(h
, "udev_enumerate_add_match_sysattr");
924 udev_enumerate_scan_devices
= (int (*)(udev_enumerate
*))dlsym(h
, "udev_enumerate_scan_devices");
925 udev_enumerate_get_list_entry
= (udev_list_entry
* (*)(udev_enumerate
*))dlsym(h
, "udev_enumerate_get_list_entry");
926 udev_device_new_from_syspath
= (udev_device
* (*)(udev
*, const char*))dlsym(h
, "udev_device_new_from_syspath");
927 udev_enumerate_get_udev
= (udev
* (*)(udev_enumerate
*))dlsym(h
, "udev_enumerate_get_udev");
928 udev_list_entry_get_name
= (const char* (*)(udev_list_entry
*))dlsym(h
, "udev_list_entry_get_name");
929 udev_device_get_devnode
= (const char* (*)(udev_device
*))dlsym(h
, "udev_device_get_devnode");
930 udev_enumerate_new
= (udev_enumerate
* (*)(udev
*))dlsym(h
, "udev_enumerate_new");
931 udev_list_entry_get_next
= (udev_list_entry
* (*)(udev_list_entry
*))dlsym(h
, "udev_list_entry_get_next");
932 udev_device_get_property_value
= (const char* (*)(udev_device
*, const char *))dlsym(h
, "udev_device_get_property_value");
938 // convenience interface, this will just call ScanForRemovable
940 pkgUdevCdromDevices::Scan()
942 bool CdromOnly
= _config
->FindB("APT::cdrom::CdromOnly", true);
943 return ScanForRemovable(CdromOnly
);
948 pkgUdevCdromDevices::ScanForRemovable(bool CdromOnly
)
950 vector
<CdromDevice
> cdrom_devices
;
951 struct udev_enumerate
*enumerate
;
952 struct udev_list_entry
*l
, *devices
;
953 struct udev
*udev_ctx
;
955 if(libudev_handle
== NULL
)
956 return cdrom_devices
;
958 udev_ctx
= udev_new();
959 enumerate
= udev_enumerate_new (udev_ctx
);
961 udev_enumerate_add_match_property(enumerate
, "ID_CDROM", "1");
963 udev_enumerate_add_match_sysattr(enumerate
, "removable", "1");
966 udev_enumerate_scan_devices (enumerate
);
967 devices
= udev_enumerate_get_list_entry (enumerate
);
968 for (l
= devices
; l
!= NULL
; l
= udev_list_entry_get_next (l
))
971 struct udev_device
*udevice
;
972 udevice
= udev_device_new_from_syspath (udev_enumerate_get_udev (enumerate
), udev_list_entry_get_name (l
));
975 const char* devnode
= udev_device_get_devnode(udevice
);
977 // try fstab_dir first
979 const char* mp
= udev_device_get_property_value(udevice
, "FSTAB_DIR");
981 mountpath
= string(mp
);
983 mountpath
= FindMountPointForDevice(devnode
);
985 // fill in the struct
986 cdrom
.DeviceName
= string(devnode
);
987 if (mountpath
!= "") {
988 cdrom
.MountPath
= mountpath
;
989 string s
= mountpath
;
990 cdrom
.Mounted
= IsMounted(s
);
992 cdrom
.Mounted
= false;
993 cdrom
.MountPath
= "";
995 cdrom_devices
.push_back(cdrom
);
997 return cdrom_devices
;
1001 pkgUdevCdromDevices::~pkgUdevCdromDevices() /*{{{*/
1003 if (libudev_handle
!= NULL
)
1004 dlclose(libudev_handle
);