| // Copyright 2014 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. |
| |
| /* |
| * Copyright (C) 2010 Google Inc. All rights reserved. |
| * Copyright (C) 2012 Apple Inc. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are |
| * met: |
| * |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following disclaimer |
| * in the documentation and/or other materials provided with the |
| * distribution. |
| * * Neither the name of Google Inc. nor the names of its |
| * contributors may be used to endorse or promote products derived from |
| * this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #import <AppKit/AppKit.h> |
| #import <AvailabilityMacros.h> |
| #include <signal.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| |
| // This is a simple helper app that changes the color sync profile to the |
| // generic profile and back when done. This program is managed by the layout |
| // test script, so it can do the job for multiple DumpRenderTree while they are |
| // running layout tests. |
| |
| namespace { |
| |
| #if defined(MAC_OS_X_VERSION_10_7) && \ |
| MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_7 |
| |
| CFURLRef user_color_profile_url; |
| |
| void InstallLayoutTestColorProfile() { |
| // To make sure we get consistent colors (not dependent on the chosen color |
| // space of the main display), we force the generic RGB color profile. |
| // This causes a change the user can see. |
| |
| CFUUIDRef main_display_id = |
| CGDisplayCreateUUIDFromDisplayID(CGMainDisplayID()); |
| |
| if (!user_color_profile_url) { |
| CFDictionaryRef device_info = ColorSyncDeviceCopyDeviceInfo( |
| kColorSyncDisplayDeviceClass, main_display_id); |
| |
| if (!device_info) { |
| NSLog(@"No display attached to system; not setting main display's color " |
| "profile."); |
| CFRelease(main_display_id); |
| return; |
| } |
| |
| CFDictionaryRef profile_info = (CFDictionaryRef)CFDictionaryGetValue( |
| device_info, kColorSyncCustomProfiles); |
| if (profile_info) { |
| user_color_profile_url = |
| (CFURLRef)CFDictionaryGetValue(profile_info, CFSTR("1")); |
| CFRetain(user_color_profile_url); |
| } else { |
| profile_info = (CFDictionaryRef)CFDictionaryGetValue( |
| device_info, kColorSyncFactoryProfiles); |
| CFDictionaryRef factory_profile = |
| (CFDictionaryRef)CFDictionaryGetValue(profile_info, CFSTR("1")); |
| user_color_profile_url = (CFURLRef)CFDictionaryGetValue( |
| factory_profile, kColorSyncDeviceProfileURL); |
| CFRetain(user_color_profile_url); |
| } |
| |
| CFRelease(device_info); |
| } |
| |
| ColorSyncProfileRef generic_rgb_profile = |
| ColorSyncProfileCreateWithName(kColorSyncGenericRGBProfile); |
| CFErrorRef error; |
| CFURLRef profile_url = ColorSyncProfileGetURL(generic_rgb_profile, &error); |
| if (!profile_url) { |
| NSLog(@"Failed to get URL of Generic RGB color profile! Many pixel tests " |
| "may fail as a result. Error: %@", |
| error); |
| |
| if (user_color_profile_url) { |
| CFRelease(user_color_profile_url); |
| user_color_profile_url = 0; |
| } |
| |
| CFRelease(generic_rgb_profile); |
| CFRelease(main_display_id); |
| return; |
| } |
| |
| CFMutableDictionaryRef profile_info = |
| CFDictionaryCreateMutable(kCFAllocatorDefault, |
| 0, |
| &kCFTypeDictionaryKeyCallBacks, |
| &kCFTypeDictionaryValueCallBacks); |
| CFDictionarySetValue( |
| profile_info, kColorSyncDeviceDefaultProfileID, profile_url); |
| |
| if (!ColorSyncDeviceSetCustomProfiles( |
| kColorSyncDisplayDeviceClass, main_display_id, profile_info)) { |
| NSLog(@"Failed to set color profile for main display! Many pixel tests may " |
| "fail as a result."); |
| |
| if (user_color_profile_url) { |
| CFRelease(user_color_profile_url); |
| user_color_profile_url = 0; |
| } |
| } |
| |
| CFRelease(profile_info); |
| CFRelease(generic_rgb_profile); |
| CFRelease(main_display_id); |
| } |
| |
| void RestoreUserColorProfile() { |
| // This is used as a signal handler, and thus the calls into ColorSync are |
| // unsafe. |
| // But we might as well try to restore the user's color profile, we're going |
| // down anyway... |
| |
| if (!user_color_profile_url) |
| return; |
| |
| CFUUIDRef main_display_id = |
| CGDisplayCreateUUIDFromDisplayID(CGMainDisplayID()); |
| CFMutableDictionaryRef profile_info = |
| CFDictionaryCreateMutable(kCFAllocatorDefault, |
| 0, |
| &kCFTypeDictionaryKeyCallBacks, |
| &kCFTypeDictionaryValueCallBacks); |
| CFDictionarySetValue( |
| profile_info, kColorSyncDeviceDefaultProfileID, user_color_profile_url); |
| ColorSyncDeviceSetCustomProfiles( |
| kColorSyncDisplayDeviceClass, main_display_id, profile_info); |
| CFRelease(main_display_id); |
| CFRelease(profile_info); |
| } |
| |
| #else // For Snow Leopard and before, use older CM* API. |
| |
| const char color_profile_path[] = |
| "/System/Library/ColorSync/Profiles/Generic RGB Profile.icc"; |
| |
| // The locType field is initialized to 0 which is the same as cmNoProfileBase. |
| CMProfileLocation initial_color_profile_location; |
| |
| void InstallLayoutTestColorProfile() { |
| // To make sure we get consistent colors (not dependent on the Main display), |
| // we force the generic rgb color profile. This cases a change the user can |
| // see. |
| const CMDeviceScope scope = {kCFPreferencesCurrentUser, |
| kCFPreferencesCurrentHost}; |
| |
| CMProfileRef profile = 0; |
| int error = |
| CMGetProfileByAVID((CMDisplayIDType)kCGDirectMainDisplay, &profile); |
| if (!error) { |
| UInt32 size = sizeof(initial_color_profile_location); |
| error = |
| NCMGetProfileLocation(profile, &initial_color_profile_location, &size); |
| CMCloseProfile(profile); |
| } |
| if (error) { |
| NSLog(@"failed to get the current color profile, pixmaps won't match. " |
| "Error: %d", |
| (int)error); |
| initial_color_profile_location.locType = cmNoProfileBase; |
| return; |
| } |
| |
| CMProfileLocation location; |
| location.locType = cmPathBasedProfile; |
| strncpy(location.u.pathLoc.path, |
| color_profile_path, |
| sizeof(location.u.pathLoc.path)); |
| error = CMSetDeviceProfile(cmDisplayDeviceClass, |
| (CMDeviceID)kCGDirectMainDisplay, |
| &scope, |
| cmDefaultProfileID, |
| &location); |
| if (error) { |
| NSLog(@"failed install the generic color profile, pixmaps won't match. " |
| "Error: %d", |
| (int)error); |
| initial_color_profile_location.locType = cmNoProfileBase; |
| } |
| } |
| |
| void RestoreUserColorProfile() { |
| // This is used as a signal handler, and thus the calls into ColorSync are |
| // unsafe. |
| // But we might as well try to restore the user's color profile, we're going |
| // down anyway... |
| if (initial_color_profile_location.locType != cmNoProfileBase) { |
| const CMDeviceScope scope = {kCFPreferencesCurrentUser, |
| kCFPreferencesCurrentHost}; |
| int error = CMSetDeviceProfile(cmDisplayDeviceClass, |
| (CMDeviceID)kCGDirectMainDisplay, |
| &scope, |
| cmDefaultProfileID, |
| &initial_color_profile_location); |
| if (error) { |
| NSLog(@"Failed to restore color profile, use System Preferences -> " |
| "Displays -> Color to reset. Error: %d", |
| (int)error); |
| } |
| initial_color_profile_location.locType = cmNoProfileBase; |
| } |
| } |
| |
| #endif |
| |
| void SimpleSignalHandler(int sig) { |
| // Try to restore the color profile and try to go down cleanly. |
| RestoreUserColorProfile(); |
| exit(128 + sig); |
| } |
| |
| } // namespace |
| |
| int main(int argc, char* argv[]) { |
| NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]; |
| |
| // Hooks the ways we might get told to clean up... |
| signal(SIGINT, SimpleSignalHandler); |
| signal(SIGHUP, SimpleSignalHandler); |
| signal(SIGTERM, SimpleSignalHandler); |
| |
| // Save off the current profile, and then install the layout test profile. |
| InstallLayoutTestColorProfile(); |
| |
| // Let the script know we're ready. |
| printf("ready\n"); |
| fflush(stdout); |
| |
| // Wait for any key (or signal). |
| getchar(); |
| |
| // Restore the profile. |
| RestoreUserColorProfile(); |
| |
| [pool release]; |
| return 0; |
| } |