Search code examples
cocoamacossolid-state-driveiokit

How to detect SSD in Mac OS X?


Is there a reliable, quick, deterministic way (i.e. not a benchmark) to check whether the system drive Mac OS X is on is a Solid State Drive?

Is there any other indicator how well disk handles parallel access? I'm trying to adjust number of threads that my program is going to use for disk-bound operations.

I'm not interested in raw speed or seek time, only which type of access – serial or parallel – is faster for the drive. I don't expect users of my program to use iSCSI or RAID. SSD is my focus, anything else is nice-to-have.

Device Characteristics of IOAHCIBlockStorageDevice contains this information. How can I read it programmatically?


So far I've figured out it goes like this: (following is pseudocode)

match = IOBSDNameMatching(kIOMasterPortDefault,0,"disk0s2");
IOServiceGetMatchingServices(kIOMasterPortDefault, match, &iterator);
while(entry = IOIteratorNext(iterator)) {
   do {
     entry = IORegistryEntryGetParentEntry(nextMedia, kIOServicePlane, &entry);
     dict = IORegistryEntryCreateCFProperty(nextMedia, 
            CFSTR(kIOPropertyDeviceCharacteristicsKey), kCFAllocatorDefault, 0);
     [dict objectForKey:CFSTR(kIOPropertyMediumTypeKey)];
   } 
   while(!dict && entry); 
}

Edit: Here's complete source code. I've verified it works with Intel SSD and OCZ Vertex.


Solution

  • If you're trying to get that kind of information, you're best guess is IOKit.

    You can try some of it's functionality using the ioreg command line tool or the IORegistryExplorer.


    Here's some code that might help you. It fetches all hard drives that aren't a RAID and aren't partitions. This isn't what you want, but it might get you started.

    #import "TWDevice.h"
    
    #include <stdio.h>
    #include <string.h>
    #include <unistd.h>
    #include <fcntl.h>
    #include <sys/ioctl.h>
    #include <errno.h>
    #include <paths.h>
    #include <sys/param.h>
    #include <IOKit/IOKitLib.h>
    #include <IOKit/IOBSD.h>
    #include <IOKit/storage/IOMedia.h>
    #include <CoreFoundation/CoreFoundation.h>
    #include <IOKit/Kext/KextManager.h>
    
    
    @implementation TWDevice
    
    @synthesize name, devicePath, size, blockSize, writable, icon;
    
    + (NSArray *)allDevices {
        // create matching dictionary
        CFMutableDictionaryRef classesToMatch;
        classesToMatch = IOServiceMatching(kIOMediaClass);
        if (classesToMatch == NULL) {
            [NSException raise:@"TWError" format:@"Classes to match could not be created"];
        }
    
        // get iterator of matching services
        io_iterator_t mediaIterator;
        kern_return_t kernResult;
        kernResult = IOServiceGetMatchingServices(kIOMasterPortDefault,
                                                                          classesToMatch,
                                                                          &mediaIterator);
    
        if (kernResult != KERN_SUCCESS) {
            [NSException raise:@"TWError" format:@"Matching services did not succed."];
        }
    
        // iterate over all found medias
        io_object_t nextMedia;
        NSMutableArray *detectedDevices = [NSMutableArray array];
        while (nextMedia = IOIteratorNext(mediaIterator)) {
            NSMutableDictionary *properties;
            kernResult = IORegistryEntryCreateCFProperties(nextMedia,
                                                                                      (CFMutableDictionaryRef *)&properties,
                                                                                      kCFAllocatorDefault, 0);
    
            if (kernResult != KERN_SUCCESS) {
                [NSException raise:@"TWError" format:@"Getting properties threw error."];
            }
    
            // is it a whole device or just a partition?
            if ([[properties valueForKey:@"Whole"] boolValue] &&
                ![[properties valueForKey:@"RAID"] boolValue]) {
                TWDevice *device = [[[TWDevice alloc] init] autorelease];
    
                device.devicePath = [NSString stringWithFormat:@"%sr%@", _PATH_DEV, [properties valueForKey:@"BSD Name"]];
                device.blockSize = [[properties valueForKey:@"Preferred Block Size"] unsignedLongLongValue];
                device.writable = [[properties valueForKey:@"Writable"] boolValue];
                device.size = [[properties valueForKey:@"Size"] unsignedLongLongValue];
    
                io_name_t name;
                IORegistryEntryGetName(nextMedia, name);
                device.name = [NSString stringWithCString:name encoding:NSASCIIStringEncoding];
    
                …
    
                [detectedDevices addObject:device];
            }
    
            // tidy up
            IOObjectRelease(nextMedia);
            CFRelease(properties);
        }
        IOObjectRelease(mediaIterator);
    
        return detectedDevices;
    }
    
    @end