Home | History | Annotate | Download | only in lib
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include <CoreFoundation/CoreFoundation.h>
      6 #include <IOKit/IOKitLib.h>
      7 #include <IOKit/network/IOEthernetController.h>
      8 #include <IOKit/network/IOEthernetInterface.h>
      9 #include <IOKit/network/IONetworkInterface.h>
     10 
     11 #include "base/logging.h"
     12 #include "base/mac/foundation_util.h"
     13 #include "base/mac/scoped_cftyperef.h"
     14 #include "base/mac/scoped_ioobject.h"
     15 #include "base/strings/string16.h"
     16 #include "base/strings/stringprintf.h"
     17 #include "base/strings/sys_string_conversions.h"
     18 #include "base/strings/utf_string_conversions.h"
     19 
     20 namespace rlz_lib {
     21 
     22 namespace {
     23 
     24 // See http://developer.apple.com/library/mac/#technotes/tn1103/_index.html
     25 
     26 // The caller is responsible for freeing |matching_services|.
     27 bool FindEthernetInterfaces(io_iterator_t* matching_services) {
     28   base::ScopedCFTypeRef<CFMutableDictionaryRef> matching_dict(
     29       IOServiceMatching(kIOEthernetInterfaceClass));
     30   if (!matching_dict)
     31     return false;
     32 
     33   base::ScopedCFTypeRef<CFMutableDictionaryRef> primary_interface(
     34       CFDictionaryCreateMutable(kCFAllocatorDefault,
     35                                 0,
     36                                 &kCFTypeDictionaryKeyCallBacks,
     37                                 &kCFTypeDictionaryValueCallBacks));
     38   if (!primary_interface)
     39     return false;
     40 
     41   CFDictionarySetValue(
     42       primary_interface, CFSTR(kIOPrimaryInterface), kCFBooleanTrue);
     43   CFDictionarySetValue(
     44       matching_dict, CFSTR(kIOPropertyMatchKey), primary_interface);
     45 
     46   kern_return_t kern_result = IOServiceGetMatchingServices(
     47       kIOMasterPortDefault, matching_dict.release(), matching_services);
     48 
     49   return kern_result == KERN_SUCCESS;
     50 }
     51 
     52 bool GetMACAddressFromIterator(io_iterator_t primary_interface_iterator,
     53                                uint8_t* buffer, size_t buffer_size) {
     54   if (buffer_size < kIOEthernetAddressSize)
     55     return false;
     56 
     57   bool success = false;
     58 
     59   bzero(buffer, buffer_size);
     60   base::mac::ScopedIOObject<io_object_t> primary_interface;
     61   while (primary_interface.reset(IOIteratorNext(primary_interface_iterator)),
     62          primary_interface) {
     63     io_object_t primary_interface_parent;
     64     kern_return_t kern_result = IORegistryEntryGetParentEntry(
     65         primary_interface, kIOServicePlane, &primary_interface_parent);
     66     base::mac::ScopedIOObject<io_object_t> primary_interface_parent_deleter(
     67         primary_interface_parent);
     68     success = kern_result == KERN_SUCCESS;
     69 
     70     if (!success)
     71       continue;
     72 
     73     base::ScopedCFTypeRef<CFTypeRef> mac_data(
     74         IORegistryEntryCreateCFProperty(primary_interface_parent,
     75                                         CFSTR(kIOMACAddress),
     76                                         kCFAllocatorDefault,
     77                                         0));
     78     CFDataRef mac_data_data = base::mac::CFCast<CFDataRef>(mac_data);
     79     if (mac_data_data) {
     80       CFDataGetBytes(
     81           mac_data_data, CFRangeMake(0, kIOEthernetAddressSize), buffer);
     82     }
     83   }
     84 
     85   return success;
     86 }
     87 
     88 bool GetMacAddress(unsigned char* buffer, size_t size) {
     89   io_iterator_t primary_interface_iterator;
     90   if (!FindEthernetInterfaces(&primary_interface_iterator))
     91     return false;
     92   bool result = GetMACAddressFromIterator(
     93       primary_interface_iterator, buffer, size);
     94   IOObjectRelease(primary_interface_iterator);
     95   return result;
     96 }
     97 
     98 CFStringRef CopySerialNumber() {
     99   base::mac::ScopedIOObject<io_service_t> expert_device(
    100       IOServiceGetMatchingService(kIOMasterPortDefault,
    101           IOServiceMatching("IOPlatformExpertDevice")));
    102   if (!expert_device)
    103     return NULL;
    104 
    105   base::ScopedCFTypeRef<CFTypeRef> serial_number(
    106       IORegistryEntryCreateCFProperty(expert_device,
    107                                       CFSTR(kIOPlatformSerialNumberKey),
    108                                       kCFAllocatorDefault,
    109                                       0));
    110   CFStringRef serial_number_cfstring =
    111       base::mac::CFCast<CFStringRef>(serial_number);
    112   if (!serial_number_cfstring)
    113     return NULL;
    114 
    115   ignore_result(serial_number.release());
    116   return serial_number_cfstring;
    117 }
    118 
    119 }  // namespace
    120 
    121 bool GetRawMachineId(string16* data, int* more_data) {
    122   uint8_t mac_address[kIOEthernetAddressSize];
    123 
    124   data->clear();
    125   if (GetMacAddress(mac_address, sizeof(mac_address))) {
    126     *data += ASCIIToUTF16(base::StringPrintf("mac:%02x%02x%02x%02x%02x%02x",
    127         mac_address[0], mac_address[1], mac_address[2],
    128         mac_address[3], mac_address[4], mac_address[5]));
    129   }
    130 
    131   // A MAC address is enough to uniquely identify a machine, but it's only 6
    132   // bytes, 3 of which are manufacturer-determined. To make brute-forcing the
    133   // SHA1 of this harder, also append the system's serial number.
    134   CFStringRef serial = CopySerialNumber();
    135   if (serial) {
    136     if (!data->empty())
    137       *data += UTF8ToUTF16(" ");
    138     *data += UTF8ToUTF16("serial:") + base::SysCFStringRefToUTF16(serial);
    139     CFRelease(serial);
    140   }
    141 
    142   // On windows, this is set to the volume id. Since it's not scrambled before
    143   // being sent, just set it to 1.
    144   *more_data = 1;
    145   return true;
    146 }
    147 
    148 }  // namespace rlz_lib
    149