diff options
Diffstat (limited to 'rlz/mac')
-rw-r--r-- | rlz/mac/lib/machine_id_mac.cc | 147 | ||||
-rw-r--r-- | rlz/mac/lib/rlz_value_store_mac.h | 80 | ||||
-rw-r--r-- | rlz/mac/lib/rlz_value_store_mac.mm | 447 |
3 files changed, 0 insertions, 674 deletions
diff --git a/rlz/mac/lib/machine_id_mac.cc b/rlz/mac/lib/machine_id_mac.cc deleted file mode 100644 index 2198978..0000000 --- a/rlz/mac/lib/machine_id_mac.cc +++ /dev/null @@ -1,147 +0,0 @@ -// Copyright (c) 2012 The Chromium Authors. All rights reserved. -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#include <CoreFoundation/CoreFoundation.h> -#include <IOKit/IOKitLib.h> -#include <IOKit/network/IOEthernetInterface.h> -#include <IOKit/network/IONetworkInterface.h> -#include <IOKit/network/IOEthernetController.h> - -#include "base/logging.h" -#include "base/mac/foundation_util.h" -#include "base/mac/scoped_cftyperef.h" -#include "base/mac/scoped_ioobject.h" -#include "base/string16.h" -#include "base/stringprintf.h" -#include "base/sys_string_conversions.h" -#include "base/utf_string_conversions.h" - -namespace rlz_lib { - -namespace { - -// See http://developer.apple.com/library/mac/#technotes/tn1103/_index.html - -// The caller is responsible for freeing |matching_services|. -bool FindEthernetInterfaces(io_iterator_t* matching_services) { - base::mac::ScopedCFTypeRef<CFMutableDictionaryRef> matching_dict( - IOServiceMatching(kIOEthernetInterfaceClass)); - if (!matching_dict) - return false; - - base::mac::ScopedCFTypeRef<CFMutableDictionaryRef> primary_interface( - CFDictionaryCreateMutable(kCFAllocatorDefault, 0, - &kCFTypeDictionaryKeyCallBacks, - &kCFTypeDictionaryValueCallBacks)); - if (!primary_interface) - return false; - - CFDictionarySetValue( - primary_interface, CFSTR(kIOPrimaryInterface), kCFBooleanTrue); - CFDictionarySetValue( - matching_dict, CFSTR(kIOPropertyMatchKey), primary_interface); - - kern_return_t kern_result = IOServiceGetMatchingServices( - kIOMasterPortDefault, matching_dict.release(), matching_services); - - return kern_result == KERN_SUCCESS; -} - -bool GetMACAddressFromIterator(io_iterator_t primary_interface_iterator, - uint8_t* buffer, size_t buffer_size) { - if (buffer_size < kIOEthernetAddressSize) - return false; - - bool success = false; - - bzero(buffer, buffer_size); - base::mac::ScopedIOObject<io_object_t> primary_interface; - while (primary_interface.reset(IOIteratorNext(primary_interface_iterator)), - primary_interface) { - io_object_t primary_interface_parent; - kern_return_t kern_result = IORegistryEntryGetParentEntry( - primary_interface, kIOServicePlane, &primary_interface_parent); - base::mac::ScopedIOObject<io_object_t> primary_interface_parent_deleter( - primary_interface_parent); - success = kern_result == KERN_SUCCESS; - - if (!success) - continue; - - base::mac::ScopedCFTypeRef<CFTypeRef> mac_data( - IORegistryEntryCreateCFProperty(primary_interface_parent, - CFSTR(kIOMACAddress), - kCFAllocatorDefault, - 0)); - CFDataRef mac_data_data = base::mac::CFCast<CFDataRef>(mac_data); - if (mac_data_data) { - CFDataGetBytes( - mac_data_data, CFRangeMake(0, kIOEthernetAddressSize), buffer); - } - } - - return success; -} - -bool GetMacAddress(unsigned char* buffer, size_t size) { - io_iterator_t primary_interface_iterator; - if (!FindEthernetInterfaces(&primary_interface_iterator)) - return false; - bool result = GetMACAddressFromIterator( - primary_interface_iterator, buffer, size); - IOObjectRelease(primary_interface_iterator); - return result; -} - -CFStringRef CopySerialNumber() { - base::mac::ScopedIOObject<io_service_t> expert_device( - IOServiceGetMatchingService(kIOMasterPortDefault, - IOServiceMatching("IOPlatformExpertDevice"))); - if (!expert_device) - return NULL; - - base::mac::ScopedCFTypeRef<CFTypeRef> serial_number( - IORegistryEntryCreateCFProperty(expert_device, - CFSTR(kIOPlatformSerialNumberKey), - kCFAllocatorDefault, - 0)); - CFStringRef serial_number_cfstring = - base::mac::CFCast<CFStringRef>(serial_number); - if (!serial_number_cfstring) - return NULL; - - ignore_result(serial_number.release()); - return serial_number_cfstring; -} - -} // namespace - -bool GetRawMachineId(string16* data, int* more_data) { - uint8_t mac_address[kIOEthernetAddressSize]; - - data->clear(); - if (GetMacAddress(mac_address, sizeof(mac_address))) { - *data += ASCIIToUTF16(StringPrintf("mac:%02x%02x%02x%02x%02x%02x", - mac_address[0], mac_address[1], mac_address[2], - mac_address[3], mac_address[4], mac_address[5])); - } - - // A MAC address is enough to uniquely identify a machine, but it's only 6 - // bytes, 3 of which are manufacturer-determined. To make brute-forcing the - // SHA1 of this harder, also append the system's serial number. - CFStringRef serial = CopySerialNumber(); - if (serial) { - if (!data->empty()) - *data += UTF8ToUTF16(" "); - *data += UTF8ToUTF16("serial:") + base::SysCFStringRefToUTF16(serial); - CFRelease(serial); - } - - // On windows, this is set to the volume id. Since it's not scrambled before - // being sent, just set it to 1. - *more_data = 1; - return true; -} - -} // namespace rlz_lib diff --git a/rlz/mac/lib/rlz_value_store_mac.h b/rlz/mac/lib/rlz_value_store_mac.h deleted file mode 100644 index b7ffb4e..0000000 --- a/rlz/mac/lib/rlz_value_store_mac.h +++ /dev/null @@ -1,80 +0,0 @@ -// Copyright (c) 2012 The Chromium Authors. All rights reserved. -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#ifndef RLZ_MAC_LIB_RLZ_VALUE_STORE_MAC_H_ -#define RLZ_MAC_LIB_RLZ_VALUE_STORE_MAC_H_ - -#include "rlz/lib/rlz_value_store.h" -#include "base/compiler_specific.h" -#include "base/memory/scoped_nsobject.h" - -@class NSDictionary; -@class NSMutableDictionary; - -namespace rlz_lib { - -// An implementation of RlzValueStore for mac. It stores information in a -// plist file in the user's Application Support folder. -class RlzValueStoreMac : public RlzValueStore { - public: - virtual bool HasAccess(AccessType type) OVERRIDE; - - virtual bool WritePingTime(Product product, int64 time) OVERRIDE; - virtual bool ReadPingTime(Product product, int64* time) OVERRIDE; - virtual bool ClearPingTime(Product product) OVERRIDE; - - virtual bool WriteAccessPointRlz(AccessPoint access_point, - const char* new_rlz) OVERRIDE; - virtual bool ReadAccessPointRlz(AccessPoint access_point, - char* rlz, - size_t rlz_size) OVERRIDE; - virtual bool ClearAccessPointRlz(AccessPoint access_point) OVERRIDE; - - virtual bool AddProductEvent(Product product, const char* event_rlz) OVERRIDE; - virtual bool ReadProductEvents(Product product, - std::vector<std::string>* events) OVERRIDE; - virtual bool ClearProductEvent(Product product, - const char* event_rlz) OVERRIDE; - virtual bool ClearAllProductEvents(Product product) OVERRIDE; - - virtual bool AddStatefulEvent(Product product, - const char* event_rlz) OVERRIDE; - virtual bool IsStatefulEvent(Product product, - const char* event_rlz) OVERRIDE; - virtual bool ClearAllStatefulEvents(Product product) OVERRIDE; - - virtual void CollectGarbage() OVERRIDE; - - private: - // |dict| is the dictionary that backs all data. plist_path is the name of the - // plist file, used solely for implementing HasAccess(). - RlzValueStoreMac(NSMutableDictionary* dict, NSString* plist_path); - virtual ~RlzValueStoreMac(); - friend class ScopedRlzValueStoreLock; - - // Returns the backing dictionary that should be written to disk. - NSDictionary* dictionary(); - - // Returns the dictionary to which all data should be written. Usually, this - // is just |dictionary()|, but if supplementary branding is used, it's a - // subdirectory at key "brand_<supplementary branding code>". - // Note that windows stores data at - // rlz/name (e.g. "pingtime")/supplementalbranding/productcode - // Mac on the other hand does - // supplementalbranding/productcode/pingtime. - NSMutableDictionary* WorkingDict(); - - // Returns the subdirectory of |WorkingDict()| used to store data for - // product p. - NSMutableDictionary* ProductDict(Product p); - - scoped_nsobject<NSMutableDictionary> dict_; - scoped_nsobject<NSString> plist_path_; - - DISALLOW_COPY_AND_ASSIGN(RlzValueStoreMac); -}; - -} // namespace rlz_lib - -#endif // RLZ_MAC_LIB_RLZ_VALUE_STORE_MAC_H_ diff --git a/rlz/mac/lib/rlz_value_store_mac.mm b/rlz/mac/lib/rlz_value_store_mac.mm deleted file mode 100644 index 5313a30..0000000 --- a/rlz/mac/lib/rlz_value_store_mac.mm +++ /dev/null @@ -1,447 +0,0 @@ -// Copyright (c) 2012 The Chromium Authors. All rights reserved. -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#include "rlz/mac/lib/rlz_value_store_mac.h" - -#include "base/mac/foundation_util.h" -#include "base/file_path.h" -#include "base/logging.h" -#include "base/sys_string_conversions.h" -#include "rlz/lib/assert.h" -#include "rlz/lib/lib_values.h" -#include "rlz/lib/rlz_lib.h" - -#import <Foundation/Foundation.h> -#include <pthread.h> - -using base::mac::ObjCCast; - -namespace rlz_lib { - -// These are written to disk and should not be changed. -NSString* const kPingTimeKey = @"pingTime"; -NSString* const kAccessPointKey = @"accessPoints"; -NSString* const kProductEventKey = @"productEvents"; -NSString* const kStatefulEventKey = @"statefulEvents"; - -namespace { - -NSString* GetNSProductName(Product product) { - return base::SysUTF8ToNSString(GetProductName(product)); -} - -NSString* GetNSAccessPointName(AccessPoint p) { - return base::SysUTF8ToNSString(GetAccessPointName(p)); -} - -// Retrieves a subdictionary in |p| for key |k|, creating it if necessary. -// If the dictionary contains an object for |k| that is not a mutable -// dictionary, that object is replaced with an empty mutable dictinary. -NSMutableDictionary* GetOrCreateDict( - NSMutableDictionary* p, NSString* k) { - NSMutableDictionary* d = ObjCCast<NSMutableDictionary>([p objectForKey:k]); - if (!d) { - d = [NSMutableDictionary dictionaryWithCapacity:0]; - [p setObject:d forKey:k]; - } - return d; -} - -} // namespace - -RlzValueStoreMac::RlzValueStoreMac(NSMutableDictionary* dict, - NSString* plist_path) - : dict_([dict retain]), plist_path_([plist_path retain]) { -} - -RlzValueStoreMac::~RlzValueStoreMac() { -} - -bool RlzValueStoreMac::HasAccess(AccessType type) { - NSFileManager* manager = [NSFileManager defaultManager]; - switch (type) { - case kReadAccess: return [manager isReadableFileAtPath:plist_path_]; - case kWriteAccess: return [manager isWritableFileAtPath:plist_path_]; - } -} - -bool RlzValueStoreMac::WritePingTime(Product product, int64 time) { - NSNumber* n = [NSNumber numberWithLongLong:time]; - [ProductDict(product) setObject:n forKey:kPingTimeKey]; - return true; -} - -bool RlzValueStoreMac::ReadPingTime(Product product, int64* time) { - if (NSNumber* n = - ObjCCast<NSNumber>([ProductDict(product) objectForKey:kPingTimeKey])) { - *time = [n longLongValue]; - return true; - } - return false; -} - -bool RlzValueStoreMac::ClearPingTime(Product product) { - [ProductDict(product) removeObjectForKey:kPingTimeKey]; - return true; -} - - -bool RlzValueStoreMac::WriteAccessPointRlz(AccessPoint access_point, - const char* new_rlz) { - NSMutableDictionary* d = GetOrCreateDict(WorkingDict(), kAccessPointKey); - [d setObject:base::SysUTF8ToNSString(new_rlz) - forKey:GetNSAccessPointName(access_point)]; - return true; -} - -bool RlzValueStoreMac::ReadAccessPointRlz(AccessPoint access_point, - char* rlz, - size_t rlz_size) { - // Reading a non-existent access point counts as success. - if (NSDictionary* d = ObjCCast<NSDictionary>( - [WorkingDict() objectForKey:kAccessPointKey])) { - NSString* val = ObjCCast<NSString>( - [d objectForKey:GetNSAccessPointName(access_point)]); - if (!val) { - if (rlz_size > 0) - rlz[0] = '\0'; - return true; - } - - std::string s = base::SysNSStringToUTF8(val); - if (s.size() >= rlz_size) { - rlz[0] = 0; - ASSERT_STRING("GetAccessPointRlz: Insufficient buffer size"); - return false; - } - strncpy(rlz, s.c_str(), rlz_size); - return true; - } - if (rlz_size > 0) - rlz[0] = '\0'; - return true; -} - -bool RlzValueStoreMac::ClearAccessPointRlz(AccessPoint access_point) { - if (NSMutableDictionary* d = ObjCCast<NSMutableDictionary>( - [WorkingDict() objectForKey:kAccessPointKey])) { - [d removeObjectForKey:GetNSAccessPointName(access_point)]; - } - return true; -} - - -bool RlzValueStoreMac::AddProductEvent(Product product, - const char* event_rlz) { - [GetOrCreateDict(ProductDict(product), kProductEventKey) - setObject:[NSNumber numberWithBool:YES] - forKey:base::SysUTF8ToNSString(event_rlz)]; - return true; -} - -bool RlzValueStoreMac::ReadProductEvents(Product product, - std::vector<std::string>* events) { - if (NSDictionary* d = ObjCCast<NSDictionary>( - [ProductDict(product) objectForKey:kProductEventKey])) { - for (NSString* s in d) - events->push_back(base::SysNSStringToUTF8(s)); - return true; - } - return true; -} - -bool RlzValueStoreMac::ClearProductEvent(Product product, - const char* event_rlz) { - if (NSMutableDictionary* d = ObjCCast<NSMutableDictionary>( - [ProductDict(product) objectForKey:kProductEventKey])) { - [d removeObjectForKey:base::SysUTF8ToNSString(event_rlz)]; - return true; - } - return false; -} - -bool RlzValueStoreMac::ClearAllProductEvents(Product product) { - [ProductDict(product) removeObjectForKey:kProductEventKey]; - return true; -} - - -bool RlzValueStoreMac::AddStatefulEvent(Product product, - const char* event_rlz) { - [GetOrCreateDict(ProductDict(product), kStatefulEventKey) - setObject:[NSNumber numberWithBool:YES] - forKey:base::SysUTF8ToNSString(event_rlz)]; - return true; -} - -bool RlzValueStoreMac::IsStatefulEvent(Product product, - const char* event_rlz) { - if (NSDictionary* d = ObjCCast<NSDictionary>( - [ProductDict(product) objectForKey:kStatefulEventKey])) { - return [d objectForKey:base::SysUTF8ToNSString(event_rlz)] != nil; - } - return false; -} - -bool RlzValueStoreMac::ClearAllStatefulEvents(Product product) { - [ProductDict(product) removeObjectForKey:kStatefulEventKey]; - return true; -} - - -void RlzValueStoreMac::CollectGarbage() { - NOTIMPLEMENTED(); -} - -NSDictionary* RlzValueStoreMac::dictionary() { - return dict_.get(); -} - -NSMutableDictionary* RlzValueStoreMac::WorkingDict() { - std::string brand(SupplementaryBranding::GetBrand()); - if (brand.empty()) - return dict_; - - NSString* brand_ns = - [@"brand_" stringByAppendingString:base::SysUTF8ToNSString(brand)]; - - return GetOrCreateDict(dict_.get(), brand_ns); -} - -NSMutableDictionary* RlzValueStoreMac::ProductDict(Product p) { - return GetOrCreateDict(WorkingDict(), GetNSProductName(p)); -} - - -namespace { - -// Creating a recursive cross-process mutex on windows is one line. On mac, -// there's no primitve for that, so this lock is emulated by an in-process -// mutex to get the recursive part, followed by a cross-process lock for the -// cross-process part. - -// This is a struct so that it doesn't need a static initializer. -struct RecursiveCrossProcessLock { - // Tries to acquire a recursive cross-process lock. Note that this _always_ - // acquires the in-process lock (if it wasn't already acquired). The parent - // directory of |lock_file| must exist. - bool TryGetCrossProcessLock(NSString* lock_filename); - - // Releases the lock. Should always be called, even if - // TryGetCrossProcessLock() returns false. - void ReleaseLock(); - - pthread_mutex_t recursive_lock_; - pthread_t locking_thread_; - - NSDistributedLock* file_lock_; -} g_recursive_lock = { - // PTHREAD_RECURSIVE_MUTEX_INITIALIZER doesn't exist before 10.7 and is buggy - // on 10.7 (http://gcc.gnu.org/bugzilla/show_bug.cgi?id=51906#c34), so emulate - // recursive locking with a normal non-recursive mutex. - PTHREAD_MUTEX_INITIALIZER -}; - -bool RecursiveCrossProcessLock::TryGetCrossProcessLock( - NSString* lock_filename) { - bool just_got_lock = false; - - // Emulate a recursive mutex with a non-recursive one. - if (pthread_mutex_trylock(&recursive_lock_) == EBUSY) { - if (pthread_equal(pthread_self(), locking_thread_) == 0) { - // Some other thread has the lock, wait for it. - pthread_mutex_lock(&recursive_lock_); - CHECK(locking_thread_ == 0); - just_got_lock = true; - } - } else { - just_got_lock = true; - } - - locking_thread_ = pthread_self(); - - // Try to acquire file lock. - if (just_got_lock) { - const int kMaxTimeoutMS = 5000; // Matches windows. - const int kSleepPerTryMS = 200; - - CHECK(!file_lock_); - file_lock_ = [[NSDistributedLock alloc] initWithPath:lock_filename]; - - BOOL got_file_lock = NO; - int elapsedMS = 0; - while (!(got_file_lock = [file_lock_ tryLock]) && - elapsedMS < kMaxTimeoutMS) { - usleep(kSleepPerTryMS * 1000); - elapsedMS += kSleepPerTryMS; - } - - if (!got_file_lock) { - [file_lock_ release]; - file_lock_ = nil; - return false; - } - return true; - } else { - return file_lock_ != nil; - } -} - -void RecursiveCrossProcessLock::ReleaseLock() { - if (file_lock_) { - [file_lock_ unlock]; - [file_lock_ release]; - file_lock_ = nil; - } - - locking_thread_ = 0; - pthread_mutex_unlock(&recursive_lock_); -} - - -// This is set during test execution, to write RLZ files into a temporary -// directory instead of the user's Application Support folder. -NSString* g_test_folder; - -// RlzValueStoreMac keeps its data in memory and only writes it to disk when -// ScopedRlzValueStoreLock goes out of scope. Hence, if several -// ScopedRlzValueStoreLocks are nested, they all need to use the same store -// object. - -// This counts the nesting depth. -int g_lock_depth = 0; - -// This is the store object that might be shared. Only set if g_lock_depth > 0. -RlzValueStoreMac* g_store_object = NULL; - - -NSString* CreateRlzDirectory() { - NSFileManager* manager = [NSFileManager defaultManager]; - NSArray* paths = NSSearchPathForDirectoriesInDomains( - NSApplicationSupportDirectory, NSUserDomainMask, /*expandTilde=*/YES); - NSString* folder = nil; - if ([paths count] > 0) - folder = ObjCCast<NSString>([paths objectAtIndex:0]); - if (!folder) - folder = [@"~/Library/Application Support" stringByStandardizingPath]; - folder = [folder stringByAppendingPathComponent:@"Google/RLZ"]; - - if (g_test_folder) - folder = [g_test_folder stringByAppendingPathComponent:folder]; - - [manager createDirectoryAtPath:folder - withIntermediateDirectories:YES - attributes:nil - error:nil]; - return folder; -} - -// Returns the path of the rlz plist store, also creates the parent directory -// path if it doesn't exist. -NSString* RlzPlistFilename() { - NSString* const kRlzFile = @"RlzStore.plist"; - return [CreateRlzDirectory() stringByAppendingPathComponent:kRlzFile]; -} - -// Returns the path of the rlz lock file, also creates the parent directory -// path if it doesn't exist. -NSString* RlzLockFilename() { - NSString* const kRlzFile = @"lockfile"; - return [CreateRlzDirectory() stringByAppendingPathComponent:kRlzFile]; -} - -} // namespace - -ScopedRlzValueStoreLock::ScopedRlzValueStoreLock() { - bool got_distributed_lock = - g_recursive_lock.TryGetCrossProcessLock(RlzLockFilename()); - // At this point, we hold the in-process lock, no matter the value of - // |got_distributed_lock|. - - ++g_lock_depth; - - if (!got_distributed_lock) { - // Give up. |store_| isn't set, which signals to callers that acquiring - // the lock failed. |g_recursive_lock| will be released by the - // destructor. - CHECK(!g_store_object); - return; - } - - if (g_lock_depth > 1) { - // Reuse the already existing store object. - CHECK(g_store_object); - store_.reset(g_store_object); - return; - } - - CHECK(!g_store_object); - - NSString* plist = RlzPlistFilename(); - - // Create an empty file if none exists yet. - NSFileManager* manager = [NSFileManager defaultManager]; - if (![manager fileExistsAtPath:plist isDirectory:NULL]) - [[NSDictionary dictionary] writeToFile:plist atomically:YES]; - - NSMutableDictionary* dict = - [NSMutableDictionary dictionaryWithContentsOfFile:plist]; - VERIFY(dict); - - if (dict) { - store_.reset(new RlzValueStoreMac(dict, plist)); - g_store_object = (RlzValueStoreMac*)store_.get(); - } -} - -ScopedRlzValueStoreLock::~ScopedRlzValueStoreLock() { - --g_lock_depth; - CHECK(g_lock_depth >= 0); - - if (g_lock_depth > 0) { - // Other locks are still using store_, don't free it yet. - ignore_result(store_.release()); - return; - } - - if (store_.get()) { - g_store_object = NULL; - - NSDictionary* dict = - static_cast<RlzValueStoreMac*>(store_.get())->dictionary(); - VERIFY([dict writeToFile:RlzPlistFilename() atomically:YES]); - } - - // Check that "store_ set" => "file_lock acquired". The converse isn't true, - // for example if the rlz data file can't be read. - if (store_.get()) - CHECK(g_recursive_lock.file_lock_); - if (!g_recursive_lock.file_lock_) - CHECK(!store_.get()); - - g_recursive_lock.ReleaseLock(); -} - -RlzValueStore* ScopedRlzValueStoreLock::GetStore() { - return store_.get(); -} - -namespace testing { - -void SetRlzStoreDirectory(const FilePath& directory) { - base::mac::ScopedNSAutoreleasePool pool; - - [g_test_folder release]; - if (directory.empty()) { - g_test_folder = nil; - } else { - // Not Unsafe on OS X. - g_test_folder = - [[NSString alloc] initWithUTF8String:directory.AsUTF8Unsafe().c_str()]; - } -} - -} // namespace testing - -} // namespace rlz_lib |