changeset 268:382b50754fe4

dep/animone: make osx code a bit less hacky it would be nice if macos actually provided a real API for getting window titles (outside of the accessibility api). the accessibility API is a real mess to work with; the user has to give permission to access it under newer versions.
author Paper <paper@paper.us.eu.org>
date Fri, 12 Apr 2024 05:21:45 -0400 (9 months ago)
parents 1a6a5d3a94cd
children 3efac0541151
files dep/animone/include/animone/util/osx.h dep/animone/src/util/osx.cc dep/animone/src/win/quartz.cc
diffstat 3 files changed, 77 insertions(+), 199 deletions(-) [+]
line wrap: on
line diff
--- a/dep/animone/include/animone/util/osx.h	Thu Apr 11 22:05:41 2024 -0400
+++ b/dep/animone/include/animone/util/osx.h	Fri Apr 12 05:21:45 2024 -0400
@@ -4,35 +4,12 @@
 #include "animone/types.h"
 #include <cstdint>
 #include <string>
+#include <memory>
 
 #include <CoreFoundation/CoreFoundation.h>
 
 namespace animone::internal::osx::util {
 
-template<typename T>
-bool GetCFNumber(CFNumberRef num, T& result) {
-	if (!num)
-		return false;
-
-	int64_t res;
-	if (!CFNumberGetValue(num, static_cast<CFNumberType>(4), &res))
-		return false;
-
-	result = static_cast<T>(res);
-	return true;
-}
-
-template<typename T>
-struct CFDeconstructor {
-	using pointer = T;
-	void operator()(pointer t) const { ::CFRelease(t); };
-};
-
-template<typename T>
-using CFPtr = vector<T, CFDecontructor<T>>; // type-id is vector<T, Alloc<T>>
-
-bool StringFromCFString(CFStringRef string, std::string& result);
-
 bool GetProcessName(pid_t pid, std::string& result);
 
 } // namespace animone::internal::osx::util
--- a/dep/animone/src/util/osx.cc	Thu Apr 11 22:05:41 2024 -0400
+++ b/dep/animone/src/util/osx.cc	Fri Apr 12 05:21:45 2024 -0400
@@ -8,167 +8,30 @@
 
 namespace animone::internal::osx::util {
 
-typedef CFTypeRef (*LSASNCreateWithPidSpec)(CFAllocatorRef, pid_t);
-typedef CFDictionaryRef (*LSCopyApplicationInformationSpec)(int, CFTypeRef, CFArrayRef);
-
-/* retrieved dynamically from launchservices */
-static LSCopyApplicationInformationSpec LSCopyApplicationInformation = nullptr;
-static LSASNCreateWithPidSpec LSASNCreateWithPid = nullptr;
-
-static CFStringRef kLSDisplayNameKey = nullptr;
-static CFStringRef kLSPIDKey = nullptr;
-
-/* retrieved from LaunchServicesSPI.h in WebKit */
-static constexpr int kLSDefaultSessionID = -2;
+static bool GetProcessNameFromProcPidPath(pid_t pid, std::string& result) {
+	result.assign(PROC_PIDPATHINFO_MAXSIZE, '\0');
 
-static const CFStringRef kLaunchServicesBundleID = CFSTR("com.apple.LaunchServices");
-
-static bool GetLaunchServicesPrivateSymbols() {
-	CFBundleRef launch_services_bundle = CFBundleGetBundleWithIdentifier(kLaunchServicesBundleID);
-	if (!launch_services_bundle)
-		return false;
-
-	LSCopyApplicationInformation = reinterpret_cast<LSCopyApplicationInformationSpec>(
-	    CFBundleGetFunctionPointerForName(launch_services_bundle, CFSTR("_LSCopyApplicationInformation")));
-	if (!LSCopyApplicationInformation)
-		return false;
-
-	LSASNCreateWithPid = reinterpret_cast<LSASNCreateWithPidSpec>(
-	    CFBundleGetFunctionPointerForName(launch_services_bundle, CFSTR("_LSASNCreateWithPid")));
-	if (!LSASNCreateWithPid)
+	int ret = proc_pidpath(pid, result.data(), result.size() * sizeof(char));
+	if (ret <= 0)
 		return false;
 
-	CFStringRef* ptr_kLSDisplayNameKey = reinterpret_cast<CFStringRef*>(
-	    CFBundleGetDataPointerForName(launch_services_bundle, CFSTR("_kLSDisplayNameKey")));
-	if (!ptr_kLSDisplayNameKey)
-		return false;
-	kLSDisplayNameKey = *ptr_kLSDisplayNameKey;
-
-	CFStringRef* ptr_kLSPIDKey =
-	    reinterpret_cast<CFStringRef*>(CFBundleGetDataPointerForName(launch_services_bundle, CFSTR("_kLSPIDKey")));
-	if (!ptr_kLSPIDKey)
-		return false;
-	kLSPIDKey = *ptr_kLSPIDKey;
-
-	return true;
-}
-
-static bool LaunchServicesGetProcessName(pid_t pid, std::string& result) {
-	if (!LSCopyApplicationInformation || !LSASNCreateWithPid || !kLSDisplayNameKey || !kLSPIDKey)
-		if (!GetLaunchServicesPrivateSymbols())
-			return false;
+	/* find the last slash, if there's none, we're done here */
+	size_t last_slash = result.rfind('/');
+	if (last_slash == std::string::npos)
+		return true;
 
-	/* what the hell is an `asn`? */
-	CFPtr<CFTypeRef> asn = LSASNCreateWithPid(kCFAllocatorDefault, pid);
-	if (!asn)
-		return false;
-
-	CFPtr<CFArrayRef> request_array = CFArrayCreate(NULL, (const void**)kLSDisplayNameKey, 1, NULL);
-	if (!request_array)
-		return false;
-
-	CFPtr<CFDictionaryRef> dictionary =
-	    LSCopyApplicationInformation(kLSDefaultSessionID, asn.get(), request_array.get());
-	if (!dictionary)
-		return false;
-
-	{
-		/* this doesn't need to be free'd */
-		CFStringRef rstr;
-
-		if (!CFDictionaryGetValueIfPresent(dictionary, kLSDisplayNameKey, (CFTypeRef*)&rstr) || !rstr)
-			return false;
-
-		if (!StringFromCFString(rstr, result))
-			return false;
-	}
-
-	result.resize(result.find('\0'));
-
+	result.erase(0, last_slash + 1);
 	return true;
 }
 
-bool StringFromCFString(CFStringRef string, std::string& result) {
-	if (!string)
-		return false;
-
-	result.resize(CFStringGetMaximumSizeForEncoding(CFStringGetLength(string), kCFStringEncodingUTF8) + 1);
-	if (!CFStringGetCString(string, &result.front(), result.length(), kCFStringEncodingUTF8))
-		return false;
-
-	return true;
-}
-
-static bool GetProcessArgs(pid_t pid, std::string& args) {
-	/* sysctl shouldn't touch these, so we define them as const */
-	int mib[3] = {CTL_KERN, KERN_PROCARGS2, static_cast<int>(pid)};
-	const size_t mib_size = sizeof(mib) / sizeof(*mib);
-
-	/* Get the initial size of the array
-	 *
-	 * NOTE: it IS possible for this value to change inbetween calls to sysctl().
-	 * Unfortunately, I couldn't care less about handling this. :)
-	 *
-	 * is that really true, though? these should be constant values. but are
-	 * argc and argv *really* constant?
-	 */
-	size_t size;
-	{
-		int ret = sysctl((int*)mib, mib_size, nullptr, &size, nullptr, 0);
-		if (ret)
-			return false;
-	}
-
-	/* Reserve the space for it in args */
-	args.resize(size);
-
-	/* Get the contents of argc and argv */
-	{
-		int ret = sysctl((int*)mib, mib_size, &args.front(), &size, NULL, 0);
-		if (ret)
-			return false;
-	}
-
-	/* Is the size big enough to hold at least argc? */
-	if (size < sizeof(int))
-		return false;
-
-	args.resize(size);
-	return true;
-}
-
-static bool GetProcessNameFromArgs(pid_t pid, std::string& result) {
-	if (!GetProcessArgs(pid, result))
-		return false;
-
-	/* Get argc using memcpy */
-	int argc = 0;
-	memcpy(&argc, &result.front(), sizeof(argc));
-
-	/* Do we even have argv[0]? */
-	if (argc < 1)
-		return false;
-
-	/* Find the first null character */
-	size_t null_pos = result.find('\0', sizeof(argc));
-	if (null_pos == std::string::npos)
-		return false;
-
-	/* Find the last slash */
-	size_t last_slash = result.rfind('/', null_pos);
-	if (last_slash == std::string::npos)
-		return false;
-
-	/* Return our result */
-	result = result.substr(last_slash + 1, null_pos - last_slash - 1);
-	return true;
-}
-
-static bool GetProcessNameFromKernel(pid_t pid, std::string& result) {
-	result.resize(2 * MAXCOMLEN);
+static bool GetProcessNameFromProcName(pid_t pid, std::string& result) {
+	result.assign(2 * MAXCOMLEN, '\0');
 
 	int size = proc_name(pid, &result.front(), result.length());
-	if (!size)
+
+	/* if size is MAXCOMLEN or 2 * MAXCOMLEN, assume
+	 * this method won't work and our result is truncated */
+	if (size <= 0 || size == MAXCOMLEN || size == 2 * MAXCOMLEN)
 		return false;
 
 	result.resize(size);
@@ -176,18 +39,10 @@
 }
 
 bool GetProcessName(pid_t pid, std::string& result) {
-	if (LaunchServicesGetProcessName(pid, result))
+	if (GetProcessNameFromProcName(pid, result))
 		return true;
 
-	/* Try parsing the arguments, this prevents the process name being
-	 * cut off to 2*MAXCOMLEN (32 chars) */
-	if (GetProcessNameFromArgs(pid, result))
-		return true;
-
-	/* Then attempt getting it from the kernel, which results in the
-	 * process name being cut to 32 chars (worse, 16 chars if p_name is
-	 * unavailable) */
-	if (GetProcessNameFromKernel(pid, result))
+	if (GetProcessNameFromProcPidPath(pid, result))
 		return true;
 
 	return false;
--- a/dep/animone/src/win/quartz.cc	Thu Apr 11 22:05:41 2024 -0400
+++ b/dep/animone/src/win/quartz.cc	Fri Apr 12 05:21:45 2024 -0400
@@ -17,6 +17,9 @@
 
 namespace animone::internal::quartz {
 
+template<typename T>
+using CFPtr = std::unique_ptr<T, CFDecontructor<T>>;
+
 #if __LP64__
 typedef long NSInteger;
 #else
@@ -58,21 +61,51 @@
 }
 
 template<typename T>
+static bool GetCFNumber(CFNumberRef num, T& result) {
+	if (!num)
+		return false;
+
+	int64_t res;
+	if (!CFNumberGetValue(num, static_cast<CFNumberType>(4), &res))
+		return false;
+
+	result = static_cast<T>(res);
+	return true;
+}
+
+static bool StringFromCFString(CFStringRef string, std::string& result) {
+	if (!string)
+		return false;
+
+	result.resize(CFStringGetMaximumSizeForEncoding(CFStringGetLength(string), kCFStringEncodingUTF8) + 1);
+	if (!CFStringGetCString(string, &result.front(), result.length(), kCFStringEncodingUTF8))
+		return false;
+
+	return true;
+}
+
+template<typename T>
 static bool CFDictionaryGetValue(CFDictionaryRef thedict, CFStringRef key, T& out) {
 	CFTypeRef data = nullptr;
 	if (!CFDictionaryGetValueIfPresent(thedict, key, reinterpret_cast<const void**>(&data)) || !data)
 		return false;
 
 	if constexpr (std::is_arithmetic<T>::value)
-		osx::util::GetCFNumber(reinterpret_cast<CFNumberRef>(data), out);
+		GetCFNumber(reinterpret_cast<CFNumberRef>(data), out);
 	else if constexpr (std::is_same<T, std::string>::value)
-		osx::util::StringFromCFString(reinterpret_cast<CFStringRef>(data), out);
+		StringFromCFString(reinterpret_cast<CFStringRef>(data), out);
 	else
 		return false;
 
 	return true;
 }
 
+template<typename T>
+struct CFDeconstructor {
+	using pointer = T;
+	void operator()(pointer t) const { ::CFRelease(t); };
+};
+
 static bool GetWindowTitleAccessibility(unsigned int wid, pid_t pid, std::string& result) {
 	CGRect bounds = {0};
 	{
@@ -119,24 +152,39 @@
 		    kAXErrorSuccess) {
 			CGPoint point;
 			if (!AXValueGetValue(val, kAXValueTypeCGPoint, reinterpret_cast<CFTypeRef>(&point)) ||
-			    (point.x != bounds.origin.x || point.y != bounds.origin.y))
+			    (point.x != bounds.origin.x || point.y != bounds.origin.y)) {
+				CFRelease(val);
 				continue;
-		} else
+			}
+		} else {
+			CFRelease(val);
 			continue;
+		}
+
+		CFRelease(val);
 
 		if (AXUIElementCopyAttributeValue(window, kAXSizeAttribute, reinterpret_cast<CFTypeRef*>(&val)) ==
 		    kAXErrorSuccess) {
 			CGSize size;
 			if (!AXValueGetValue(val, kAXValueTypeCGSize, reinterpret_cast<CFTypeRef>(&size)) ||
-			    (size.width != bounds.size.width || size.height != bounds.size.height))
+			    (size.width != bounds.size.width || size.height != bounds.size.height)) {
+				CFRelease(val);
 				continue;
-		} else
+			}
+		} else {
+			CFRelease(val);
 			continue;
+		}
+
+		CFRelease(val);
 
 		CFStringRef title;
 		if (AXUIElementCopyAttributeValue(window, kAXTitleAttribute, reinterpret_cast<CFTypeRef*>(&title)) ==
-		    kAXErrorSuccess)
-			return osx::util::StringFromCFString(title, result);
+		    kAXErrorSuccess) {
+			bool success = StringFromCFString(title, result);
+			CFRelease(title);
+			return success;
+		}
 	}
 
 	return false;
@@ -152,7 +200,7 @@
 			title.reset(t);
 		}
 
-		if (title && CFStringGetLength(title.get()) && osx::util::StringFromCFString(title.get(), result))
+		if (title && CFStringGetLength(title.get()) && StringFromCFString(title.get(), result))
 			return true;
 	}
 
@@ -171,8 +219,7 @@
 	if (!bundle_id)
 		return false;
 
-	result = osx::util::StringFromCFString(bundle_id, result);
-	return true;
+	return StringFromCFString(bundle_id, result);
 }
 
 static bool GetProcessBundleIdentifierOld(pid_t pid, std::string& result) {
@@ -189,8 +236,7 @@
 	if (!value)
 		return false;
 
-	result = osx::util::StringFromCFString(value, result);
-	return true;
+	return StringFromCFString(value, result);
 }
 
 static bool GetProcessBundleIdentifier(pid_t pid, std::string& result) {