// 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 "base/mac/mac_util.h" #import #include #import #include #include #include #include #include #include #include #include "base/files/file_path.h" #include "base/logging.h" #include "base/mac/bundle_locations.h" #include "base/mac/foundation_util.h" #include "base/mac/mac_logging.h" #include "base/mac/scoped_cftyperef.h" #include "base/mac/scoped_ioobject.h" #include "base/mac/scoped_nsobject.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_piece.h" #include "base/strings/string_util.h" #include "base/strings/sys_string_conversions.h" #include "build/build_config.h" namespace base { namespace mac { namespace { class LoginItemsFileList { public: LoginItemsFileList() = default; LoginItemsFileList(const LoginItemsFileList&) = delete; LoginItemsFileList& operator=(const LoginItemsFileList&) = delete; ~LoginItemsFileList() = default; bool Initialize() WARN_UNUSED_RESULT { DCHECK(!login_items_.get()) << __func__ << " called more than once."; // The LSSharedFileList suite of functions has been deprecated. Instead, // a LoginItems helper should be registered with SMLoginItemSetEnabled() // https://crbug.com/1154377. #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wdeprecated-declarations" login_items_.reset(LSSharedFileListCreate( nullptr, kLSSharedFileListSessionLoginItems, nullptr)); #pragma clang diagnostic pop DLOG_IF(ERROR, !login_items_.get()) << "Couldn't get a Login Items list."; return login_items_.get(); } LSSharedFileListRef GetLoginFileList() { DCHECK(login_items_.get()) << "Initialize() failed or not called."; return login_items_; } // Looks into Shared File Lists corresponding to Login Items for the item // representing the specified bundle. If such an item is found, returns a // retained reference to it. Caller is responsible for releasing the // reference. ScopedCFTypeRef GetLoginItemForApp(NSURL* url) { DCHECK(login_items_.get()) << "Initialize() failed or not called."; #pragma clang diagnostic push // https://crbug.com/1154377 #pragma clang diagnostic ignored "-Wdeprecated-declarations" base::scoped_nsobject login_items_array( CFToNSCast(LSSharedFileListCopySnapshot(login_items_, nullptr))); #pragma clang diagnostic pop for (id login_item in login_items_array.get()) { LSSharedFileListItemRef item = reinterpret_cast(login_item); #pragma clang diagnostic push // https://crbug.com/1154377 #pragma clang diagnostic ignored "-Wdeprecated-declarations" ScopedCFTypeRef item_url( LSSharedFileListItemCopyResolvedURL(item, 0, nullptr)); #pragma clang diagnostic pop if (item_url && CFEqual(item_url, url)) { return ScopedCFTypeRef( item, base::scoped_policy::RETAIN); } } return ScopedCFTypeRef(); } ScopedCFTypeRef GetLoginItemForMainApp() { NSURL* url = [NSURL fileURLWithPath:[base::mac::MainBundle() bundlePath]]; return GetLoginItemForApp(url); } private: ScopedCFTypeRef login_items_; }; bool IsHiddenLoginItem(LSSharedFileListItemRef item) { #pragma clang diagnostic push // https://crbug.com/1154377 #pragma clang diagnostic ignored "-Wdeprecated-declarations" ScopedCFTypeRef hidden(reinterpret_cast( LSSharedFileListItemCopyProperty(item, reinterpret_cast(kLSSharedFileListLoginItemHidden)))); #pragma clang diagnostic pop return hidden && hidden == kCFBooleanTrue; } } // namespace CGColorSpaceRef GetGenericRGBColorSpace() { // Leaked. That's OK, it's scoped to the lifetime of the application. static CGColorSpaceRef g_color_space_generic_rgb( CGColorSpaceCreateWithName(kCGColorSpaceGenericRGB)); DLOG_IF(ERROR, !g_color_space_generic_rgb) << "Couldn't get the generic RGB color space"; return g_color_space_generic_rgb; } CGColorSpaceRef GetSRGBColorSpace() { // Leaked. That's OK, it's scoped to the lifetime of the application. static CGColorSpaceRef g_color_space_sRGB = CGColorSpaceCreateWithName(kCGColorSpaceSRGB); DLOG_IF(ERROR, !g_color_space_sRGB) << "Couldn't get the sRGB color space"; return g_color_space_sRGB; } CGColorSpaceRef GetSystemColorSpace() { // Leaked. That's OK, it's scoped to the lifetime of the application. // Try to get the main display's color space. static CGColorSpaceRef g_system_color_space = CGDisplayCopyColorSpace(CGMainDisplayID()); if (!g_system_color_space) { // Use a generic RGB color space. This is better than nothing. g_system_color_space = CGColorSpaceCreateDeviceRGB(); if (g_system_color_space) { DLOG(WARNING) << "Couldn't get the main display's color space, using generic"; } else { DLOG(ERROR) << "Couldn't get any color space"; } } return g_system_color_space; } bool GetFileBackupExclusion(const FilePath& file_path) { return CSBackupIsItemExcluded(FilePathToCFURL(file_path), nullptr); } bool SetFileBackupExclusion(const FilePath& file_path) { // When excludeByPath is true the application must be running with root // privileges (admin for 10.6 and earlier) but the URL does not have to // already exist. When excludeByPath is false the URL must already exist but // can be used in non-root (or admin as above) mode. We use false so that // non-root (or admin) users don't get their TimeMachine drive filled up with // unnecessary backups. OSStatus os_err = CSBackupSetItemExcluded(FilePathToCFURL(file_path), /*exclude=*/TRUE, /*excludeByPath=*/FALSE); if (os_err != noErr) { OSSTATUS_DLOG(WARNING, os_err) << "Failed to set backup exclusion for file '" << file_path.value().c_str() << "'"; } return os_err == noErr; } bool CheckLoginItemStatus(bool* is_hidden) { LoginItemsFileList login_items; if (!login_items.Initialize()) return false; base::ScopedCFTypeRef item( login_items.GetLoginItemForMainApp()); if (!item.get()) return false; if (is_hidden) *is_hidden = IsHiddenLoginItem(item); return true; } void AddToLoginItems(bool hide_on_startup) { AddToLoginItems(base::mac::MainBundlePath(), hide_on_startup); } void AddToLoginItems(const FilePath& app_bundle_file_path, bool hide_on_startup) { LoginItemsFileList login_items; if (!login_items.Initialize()) return; NSURL* app_bundle_url = base::mac::FilePathToNSURL(app_bundle_file_path); base::ScopedCFTypeRef item( login_items.GetLoginItemForApp(app_bundle_url)); if (item.get() && (IsHiddenLoginItem(item) == hide_on_startup)) { return; // Already is a login item with required hide flag. } // Remove the old item, it has wrong hide flag, we'll create a new one. if (item.get()) { #pragma clang diagnostic push // https://crbug.com/1154377 #pragma clang diagnostic ignored "-Wdeprecated-declarations" LSSharedFileListItemRemove(login_items.GetLoginFileList(), item); #pragma clang diagnostic pop } #pragma clang diagnostic push // https://crbug.com/1154377 #pragma clang diagnostic ignored "-Wdeprecated-declarations" BOOL hide = hide_on_startup ? YES : NO; NSDictionary* properties = @{(NSString*)kLSSharedFileListLoginItemHidden : @(hide) }; ScopedCFTypeRef new_item( LSSharedFileListInsertItemURL( login_items.GetLoginFileList(), kLSSharedFileListItemLast, nullptr, nullptr, reinterpret_cast(app_bundle_url), reinterpret_cast(properties), nullptr)); #pragma clang diagnostic pop if (!new_item.get()) { DLOG(ERROR) << "Couldn't insert current app into Login Items list."; } } void RemoveFromLoginItems() { RemoveFromLoginItems(base::mac::MainBundlePath()); } void RemoveFromLoginItems(const FilePath& app_bundle_file_path) { LoginItemsFileList login_items; if (!login_items.Initialize()) return; NSURL* app_bundle_url = base::mac::FilePathToNSURL(app_bundle_file_path); base::ScopedCFTypeRef item( login_items.GetLoginItemForApp(app_bundle_url)); if (!item.get()) return; #pragma clang diagnostic push // https://crbug.com/1154377 #pragma clang diagnostic ignored "-Wdeprecated-declarations" LSSharedFileListItemRemove(login_items.GetLoginFileList(), item); #pragma clang diagnostic pop } bool WasLaunchedAsLoginOrResumeItem() { ProcessSerialNumber psn = { 0, kCurrentProcess }; ProcessInfoRec info = {}; info.processInfoLength = sizeof(info); // GetProcessInformation has been deprecated since macOS 10.9, but there is no // replacement that provides the information we need. See // https://crbug.com/650854. #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wdeprecated-declarations" if (GetProcessInformation(&psn, &info) == noErr) { #pragma clang diagnostic pop ProcessInfoRec parent_info = {}; parent_info.processInfoLength = sizeof(parent_info); #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wdeprecated-declarations" if (GetProcessInformation(&info.processLauncher, &parent_info) == noErr) { #pragma clang diagnostic pop return parent_info.processSignature == 'lgnw'; } } return false; } bool WasLaunchedAsLoginItemRestoreState() { // "Reopen windows..." option was added for Lion. Prior OS versions should // not have this behavior. if (!WasLaunchedAsLoginOrResumeItem()) return false; CFStringRef app = CFSTR("com.apple.loginwindow"); CFStringRef save_state = CFSTR("TALLogoutSavesState"); ScopedCFTypeRef plist( CFPreferencesCopyAppValue(save_state, app)); // According to documentation, com.apple.loginwindow.plist does not exist on a // fresh installation until the user changes a login window setting. The // "reopen windows" option is checked by default, so the plist would exist had // the user unchecked it. // https://developer.apple.com/library/mac/documentation/macosx/conceptual/bpsystemstartup/chapters/CustomLogin.html if (!plist) return true; if (CFBooleanRef restore_state = base::mac::CFCast(plist)) return CFBooleanGetValue(restore_state); return false; } bool WasLaunchedAsHiddenLoginItem() { if (!WasLaunchedAsLoginOrResumeItem()) return false; LoginItemsFileList login_items; if (!login_items.Initialize()) return false; base::ScopedCFTypeRef item( login_items.GetLoginItemForMainApp()); if (!item.get()) { // OS X can launch items for the resume feature. return false; } return IsHiddenLoginItem(item); } bool RemoveQuarantineAttribute(const FilePath& file_path) { const char kQuarantineAttrName[] = "com.apple.quarantine"; int status = removexattr(file_path.value().c_str(), kQuarantineAttrName, 0); return status == 0 || errno == ENOATTR; } namespace { // Returns the running system's Darwin major version. Don't call this, it's an // implementation detail and its result is meant to be cached by // MacOSVersionInternal(). int DarwinMajorVersionInternal() { // base::OperatingSystemVersionNumbers() at one time called Gestalt(), which // was observed to be able to spawn threads (see https://crbug.com/53200). // Nowadays that function calls -[NSProcessInfo operatingSystemVersion], whose // current implementation does things like hit the file system, which is // possibly a blocking operation. Either way, it's overkill for what needs to // be done here. // // uname, on the other hand, is implemented as a simple series of sysctl // system calls to obtain the relevant data from the kernel. The data is // compiled right into the kernel, so no threads or blocking or other // funny business is necessary. struct utsname uname_info; if (uname(&uname_info) != 0) { DPLOG(ERROR) << "uname"; return 0; } if (strcmp(uname_info.sysname, "Darwin") != 0) { DLOG(ERROR) << "unexpected uname sysname " << uname_info.sysname; return 0; } int darwin_major_version = 0; char* dot = strchr(uname_info.release, '.'); if (dot) { if (!base::StringToInt(base::StringPiece(uname_info.release, dot - uname_info.release), &darwin_major_version)) { dot = NULL; } } if (!dot) { DLOG(ERROR) << "could not parse uname release " << uname_info.release; return 0; } return darwin_major_version; } // The implementation of MacOSVersion() as defined in the header. Don't call // this, it's an implementation detail and the result is meant to be cached by // MacOSVersion(). int MacOSVersionInternal() { int darwin_major_version = DarwinMajorVersionInternal(); // Darwin major versions 6 through 19 corresponded to macOS versions 10.2 // through 10.15. CHECK(darwin_major_version >= 6); if (darwin_major_version <= 19) return 1000 + darwin_major_version - 4; // Darwin major version 20 corresponds to macOS version 11.0. Assume a // correspondence between Darwin's major version numbers and macOS major // version numbers. int macos_major_version = darwin_major_version - 9; DLOG_IF(WARNING, darwin_major_version > 20) << "Assuming Darwin " << base::NumberToString(darwin_major_version) << " is macOS " << base::NumberToString(macos_major_version); return macos_major_version * 100; } } // namespace namespace internal { int MacOSVersion() { static int macos_version = MacOSVersionInternal(); return macos_version; } } // namespace internal namespace { #if defined(ARCH_CPU_X86_64) // https://developer.apple.com/documentation/apple_silicon/about_the_rosetta_translation_environment#3616845 bool ProcessIsTranslated() { int ret = 0; size_t size = sizeof(ret); if (sysctlbyname("sysctl.proc_translated", &ret, &size, nullptr, 0) == -1) return false; return ret; } #endif // ARCH_CPU_X86_64 } // namespace CPUType GetCPUType() { #if defined(ARCH_CPU_ARM64) return CPUType::kArm; #elif defined(ARCH_CPU_X86_64) return ProcessIsTranslated() ? CPUType::kTranslatedIntel : CPUType::kIntel; #else #error Time for another chip transition? #endif // ARCH_CPU_* } std::string GetModelIdentifier() { std::string return_string; ScopedIOObject platform_expert( IOServiceGetMatchingService(kIOMasterPortDefault, IOServiceMatching("IOPlatformExpertDevice"))); if (platform_expert) { ScopedCFTypeRef model_data( static_cast(IORegistryEntryCreateCFProperty( platform_expert, CFSTR("model"), kCFAllocatorDefault, 0))); if (model_data) { return_string = reinterpret_cast(CFDataGetBytePtr(model_data)); } } return return_string; } bool ParseModelIdentifier(const std::string& ident, std::string* type, int32_t* major, int32_t* minor) { size_t number_loc = ident.find_first_of("0123456789"); if (number_loc == std::string::npos) return false; size_t comma_loc = ident.find(',', number_loc); if (comma_loc == std::string::npos) return false; int32_t major_tmp, minor_tmp; std::string::const_iterator begin = ident.begin(); if (!StringToInt(MakeStringPiece(begin + number_loc, begin + comma_loc), &major_tmp) || !StringToInt(MakeStringPiece(begin + comma_loc + 1, ident.end()), &minor_tmp)) return false; *type = ident.substr(0, number_loc); *major = major_tmp; *minor = minor_tmp; return true; } std::string GetOSDisplayName() { std::string os_name; if (IsAtMostOS10_11()) os_name = "OS X"; else os_name = "macOS"; std::string version_string = base::SysNSStringToUTF8( [[NSProcessInfo processInfo] operatingSystemVersionString]); return os_name + " " + version_string; } std::string GetPlatformSerialNumber() { base::mac::ScopedIOObject expert_device( IOServiceGetMatchingService(kIOMasterPortDefault, IOServiceMatching("IOPlatformExpertDevice"))); if (!expert_device) { DLOG(ERROR) << "Error retrieving the machine serial number."; return std::string(); } base::ScopedCFTypeRef serial_number( IORegistryEntryCreateCFProperty(expert_device, CFSTR(kIOPlatformSerialNumberKey), kCFAllocatorDefault, 0)); CFStringRef serial_number_cfstring = base::mac::CFCast(serial_number); if (!serial_number_cfstring) { DLOG(ERROR) << "Error retrieving the machine serial number."; return std::string(); } return base::SysCFStringRefToUTF8(serial_number_cfstring); } } // namespace mac } // namespace base