From ceb486d4970c9080d2974d6852de9e9eb846d990 Mon Sep 17 00:00:00 2001 From: =?utf8?q?Jan=20Dj=C3=A4rv?= Date: Tue, 5 Nov 2013 08:51:55 +0100 Subject: [PATCH] * nsfns.m (ns_get_name_from_ioreg): New function. (ns_screen_name): Don't use deprecated CGDisplayIOServicePort on OSX >= 10.9. Use ns_get_name_from_ioreg. --- src/ChangeLog | 6 ++++ src/nsfns.m | 84 +++++++++++++++++++++++++++++++++++++++++++-------- 2 files changed, 77 insertions(+), 13 deletions(-) diff --git a/src/ChangeLog b/src/ChangeLog index 9ba39e20432..227b86a4cab 100644 --- a/src/ChangeLog +++ b/src/ChangeLog @@ -1,3 +1,9 @@ +2013-11-05 Jan Djärv + + * nsfns.m (ns_get_name_from_ioreg): New function. + (ns_screen_name): Don't use deprecated CGDisplayIOServicePort on + OSX >= 10.9. Use ns_get_name_from_ioreg. + 2013-11-05 Paul Eggert Simplify and port recent bool vector changes. diff --git a/src/nsfns.m b/src/nsfns.m index c6730f41aa7..ee36439b1b4 100644 --- a/src/nsfns.m +++ b/src/nsfns.m @@ -2358,28 +2358,86 @@ each physical monitor, use `display-monitor-attributes-list'. */) } #ifdef NS_IMPL_COCOA -/* Returns the name for the screen that DICT came from, or NULL. + +/* Returns the name for the screen that OBJ represents, or NULL. Caller must free return value. */ static char * -ns_screen_name (CGDirectDisplayID did) +ns_get_name_from_ioreg (io_object_t obj) { char *name = NULL; + NSDictionary *info = (NSDictionary *) - IODisplayCreateInfoDictionary (CGDisplayIOServicePort (did), - kIODisplayOnlyPreferredName); - NSDictionary *names - = [info objectForKey: - [NSString stringWithUTF8String:kDisplayProductName]]; - - if ([names count] > 0) { - NSString *n = [names objectForKey: [[names allKeys] objectAtIndex:0]]; - if (n != nil) - name = xstrdup ([n UTF8String]); - } + IODisplayCreateInfoDictionary (obj, kIODisplayOnlyPreferredName); + NSDictionary *names = [info objectForKey: + [NSString stringWithUTF8String: + kDisplayProductName]]; + + if ([names count] > 0) + { + NSString *n = [names objectForKey: [[names allKeys] + objectAtIndex:0]]; + if (n != nil) name = xstrdup ([n UTF8String]); + } [info release]; + + return name; +} + +/* Returns the name for the screen that DID came from, or NULL. + Caller must free return value. +*/ + +static char * +ns_screen_name (CGDirectDisplayID did) +{ + char *name = NULL; + +#if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_9 + mach_port_t masterPort; + io_iterator_t it; + io_object_t obj; + + // CGDisplayIOServicePort is deprecated. Do it another (harder) way. + + if (IOMasterPort (MACH_PORT_NULL, &masterPort) != kIOReturnSuccess + || IOServiceGetMatchingServices (masterPort, + IOServiceMatching ("IONDRVDevice"), + &it) != kIOReturnSuccess) + return name; + + /* Must loop until we find a name. Many devices can have the same unit + number (represents different GPU parts), but only one has a name. */ + while (! name && (obj = IOIteratorNext (it))) + { + CFMutableDictionaryRef props; + const void *val; + + if (IORegistryEntryCreateCFProperties (obj, + &props, + kCFAllocatorDefault, + kNilOptions) == kIOReturnSuccess + && props != nil + && (val = CFDictionaryGetValue(props, @"IOFBDependentIndex"))) + { + unsigned nr = [(NSNumber *)val unsignedIntegerValue]; + if (nr == CGDisplayUnitNumber (did)) + name = ns_get_name_from_ioreg (obj); + } + + CFRelease (props); + IOObjectRelease (obj); + } + + IOObjectRelease (it); + +#else + + name = ns_get_name_from_ioreg (CGDisplayIOServicePort (did)); + +#endif return name; } #endif -- 2.39.2