| 
									
										
										
										
											2015-05-24 06:32:32 +02:00
										 |  |  | // Copyright 2009 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2+
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-28 08:57:34 +00:00
										 |  |  | // Most of the code in this file was shamelessly ripped from libcdio With minor adjustments
 | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-20 04:11:52 +01:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cstdlib>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2014-02-17 05:18:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "Common/CDUtils.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-28 10:57:16 -05:00
										 |  |  | #include "Common/Common.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-07 20:06:58 -05:00
										 |  |  | #include "Common/CommonTypes.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-17 05:18:15 -05:00
										 |  |  | #include "Common/StringUtil.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 19:28:27 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							|  |  |  | #elif __APPLE__
 | 
					
						
							| 
									
										
										
										
											2010-07-12 20:11:19 +00:00
										 |  |  | #include <CoreFoundation/CoreFoundation.h>
 | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #include <IOKit/IOBSD.h>
 | 
					
						
							| 
									
										
										
										
											2010-01-19 19:28:27 +00:00
										 |  |  | #include <IOKit/IOKitLib.h>
 | 
					
						
							|  |  |  | #include <IOKit/storage/IOCDMedia.h>
 | 
					
						
							|  |  |  | #include <IOKit/storage/IOMedia.h>
 | 
					
						
							| 
									
										
										
										
											2010-07-12 20:11:19 +00:00
										 |  |  | #include <paths.h>
 | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2010-01-19 19:28:27 +00:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | #include <sys/ioctl.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | #include <sys/stat.h>
 | 
					
						
							| 
									
										
										
										
											2014-02-20 04:11:52 +01:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | #endif  // WIN32
 | 
					
						
							| 
									
										
										
										
											2010-01-19 19:28:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #ifdef __linux__
 | 
					
						
							|  |  |  | #include <linux/cdrom.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2011-03-12 06:50:20 +00:00
										 |  |  | // takes a root drive path, returns true if it is a cdrom drive
 | 
					
						
							| 
									
										
										
										
											2013-02-27 18:51:02 -06:00
										 |  |  | bool is_cdrom(const TCHAR* drive) | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   return (DRIVE_CDROM == GetDriveType(drive)); | 
					
						
							| 
									
										
										
										
											2011-03-12 06:50:20 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-12 06:50:20 +00:00
										 |  |  | // Returns a vector with the device names
 | 
					
						
							|  |  |  | std::vector<std::string> cdio_get_devices() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   std::vector<std::string> drives; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const DWORD buffsize = GetLogicalDriveStrings(0, nullptr); | 
					
						
							|  |  |  |   std::vector<TCHAR> buff(buffsize); | 
					
						
							|  |  |  |   if (GetLogicalDriveStrings(buffsize, buff.data()) == buffsize - 1) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     auto drive = buff.data(); | 
					
						
							|  |  |  |     while (*drive) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (is_cdrom(drive)) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         std::string str(TStrToUTF8(drive)); | 
					
						
							|  |  |  |         str.pop_back();  // we don't want the final backslash
 | 
					
						
							|  |  |  |         drives.push_back(std::move(str)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // advance to next drive
 | 
					
						
							|  |  |  |       while (*drive++) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return drives; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | #elif defined __APPLE__
 | 
					
						
							|  |  |  | // Returns a pointer to an array of strings with the device names
 | 
					
						
							|  |  |  | std::vector<std::string> cdio_get_devices() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   io_object_t next_media; | 
					
						
							|  |  |  |   mach_port_t master_port; | 
					
						
							|  |  |  |   kern_return_t kern_result; | 
					
						
							|  |  |  |   io_iterator_t media_iterator; | 
					
						
							|  |  |  |   CFMutableDictionaryRef classes_to_match; | 
					
						
							|  |  |  |   std::vector<std::string> drives; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   kern_result = IOMasterPort(MACH_PORT_NULL, &master_port); | 
					
						
							|  |  |  |   if (kern_result != KERN_SUCCESS) | 
					
						
							|  |  |  |     return drives; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   classes_to_match = IOServiceMatching(kIOCDMediaClass); | 
					
						
							|  |  |  |   if (classes_to_match == nullptr) | 
					
						
							|  |  |  |     return drives; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   CFDictionarySetValue(classes_to_match, CFSTR(kIOMediaEjectableKey), kCFBooleanTrue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   kern_result = IOServiceGetMatchingServices(master_port, classes_to_match, &media_iterator); | 
					
						
							|  |  |  |   if (kern_result != KERN_SUCCESS) | 
					
						
							|  |  |  |     return drives; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   next_media = IOIteratorNext(media_iterator); | 
					
						
							|  |  |  |   if (next_media != 0) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     CFTypeRef str_bsd_path; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       str_bsd_path = | 
					
						
							|  |  |  |           IORegistryEntryCreateCFProperty(next_media, CFSTR(kIOBSDNameKey), kCFAllocatorDefault, 0); | 
					
						
							|  |  |  |       if (str_bsd_path == nullptr) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         IOObjectRelease(next_media); | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (CFGetTypeID(str_bsd_path) == CFStringGetTypeID()) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         size_t buf_size = CFStringGetLength((CFStringRef)str_bsd_path) * 4 + 1; | 
					
						
							|  |  |  |         char* buf = new char[buf_size]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (CFStringGetCString((CFStringRef)str_bsd_path, buf, buf_size, kCFStringEncodingUTF8)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           // Below, by appending 'r' to the BSD node name, we indicate
 | 
					
						
							|  |  |  |           // a raw disk. Raw disks receive I/O requests directly and
 | 
					
						
							|  |  |  |           // don't go through a buffer cache.
 | 
					
						
							|  |  |  |           drives.push_back(std::string(_PATH_DEV "r") + buf); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         delete[] buf; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       CFRelease(str_bsd_path); | 
					
						
							|  |  |  |       IOObjectRelease(next_media); | 
					
						
							|  |  |  |     } while ((next_media = IOIteratorNext(media_iterator)) != 0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   IOObjectRelease(media_iterator); | 
					
						
							|  |  |  |   return drives; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | // checklist: /dev/cdrom, /dev/dvd /dev/hd?, /dev/scd? /dev/sr?
 | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | static struct | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   const char* format; | 
					
						
							|  |  |  |   unsigned int num_min; | 
					
						
							|  |  |  |   unsigned int num_max; | 
					
						
							|  |  |  | } checklist[] = { | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #ifdef __linux__
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     {"/dev/cdrom", 0, 0},  {"/dev/dvd", 0, 0},   {"/dev/hd%c", 'a', 'z'}, | 
					
						
							|  |  |  |     {"/dev/scd%d", 0, 27}, {"/dev/sr%d", 0, 27}, | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     {"/dev/acd%d", 0, 27}, | 
					
						
							|  |  |  |     {"/dev/cd%d", 0, 27}, | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     {nullptr, 0, 0}}; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | // Returns true if a device is a block or char device and not a symbolic link
 | 
					
						
							| 
									
										
										
										
											2014-07-08 14:29:26 +02:00
										 |  |  | static bool is_device(const std::string& source_name) | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   struct stat buf; | 
					
						
							|  |  |  |   if (0 != lstat(source_name.c_str(), &buf)) | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   return ((S_ISBLK(buf.st_mode) || S_ISCHR(buf.st_mode)) && !S_ISLNK(buf.st_mode)); | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | // Check a device to see if it is a DVD/CD-ROM drive
 | 
					
						
							| 
									
										
										
										
											2016-01-21 20:46:25 +01:00
										 |  |  | static bool is_cdrom(const std::string& drive, char* mnttype) | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Check if the device exists
 | 
					
						
							|  |  |  |   if (!is_device(drive)) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool is_cd = false; | 
					
						
							|  |  |  |   // If it does exist, verify that it is a cdrom/dvd drive
 | 
					
						
							|  |  |  |   int cdfd = open(drive.c_str(), (O_RDONLY | O_NONBLOCK), 0); | 
					
						
							|  |  |  |   if (cdfd >= 0) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #ifdef __linux__
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     if (ioctl(cdfd, CDROM_GET_CAPABILITY, 0) != -1) | 
					
						
							| 
									
										
										
										
											2010-07-22 03:29:35 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |       is_cd = true; | 
					
						
							|  |  |  |     close(cdfd); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return is_cd; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-28 08:57:34 +00:00
										 |  |  | // Returns a pointer to an array of strings with the device names
 | 
					
						
							| 
									
										
										
										
											2015-02-15 14:43:31 -05:00
										 |  |  | std::vector<std::string> cdio_get_devices() | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   std::vector<std::string> drives; | 
					
						
							|  |  |  |   // Scan the system for DVD/CD-ROM drives.
 | 
					
						
							|  |  |  |   for (unsigned int i = 0; checklist[i].format; ++i) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     for (unsigned int j = checklist[i].num_min; j <= checklist[i].num_max; ++j) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       std::string drive = StringFromFormat(checklist[i].format, j); | 
					
						
							|  |  |  |       if (is_cdrom(drive, nullptr)) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         drives.push_back(std::move(drive)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return drives; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-07 19:47:45 +00:00
										 |  |  | // Returns true if device is a cdrom/dvd drive
 | 
					
						
							|  |  |  | bool cdio_is_cdrom(std::string device) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-10-29 01:09:01 -04:00
										 |  |  | #ifndef _WIN32
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Resolve symbolic links. This allows symbolic links to valid
 | 
					
						
							|  |  |  |   // drives to be passed from the command line with the -e flag.
 | 
					
						
							|  |  |  |   char resolved_path[MAX_PATH]; | 
					
						
							|  |  |  |   char* devname = realpath(device.c_str(), resolved_path); | 
					
						
							|  |  |  |   if (!devname) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   device = devname; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-03-28 08:57:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   std::vector<std::string> devices = cdio_get_devices(); | 
					
						
							|  |  |  |   for (const std::string& d : devices) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (d == device) | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2009-02-24 15:06:52 +00:00
										 |  |  | } |