macOS XPC Connecting Process Check

Aprende hacking en AWS de cero a héroe con htARTE (HackTricks AWS Red Team Expert)!

Otras formas de apoyar a HackTricks:

Verificación del Proceso de Conexión en XPC

Cuando se establece una conexión con un servicio XPC, el servidor verificará si la conexión está permitida. Estas son las comprobaciones que normalmente realizará:

  1. Verificar si el proceso de conexión está firmado con un certificado firmado por Apple (solo otorgado por Apple).

    • Si esto no se verifica, un atacante podría crear un certificado falso para coincidir con cualquier otra comprobación.

  2. Verificar si el proceso de conexión está firmado con el certificado de la organización, (verificación del ID del equipo).

    • Si esto no se verifica, cualquier certificado de desarrollador de Apple se puede usar para firmar y conectarse al servicio.

  3. Verificar si el proceso de conexión contiene un ID de paquete adecuado.

    • Si esto no se verifica, cualquier herramienta firmada por la misma organización podría usarse para interactuar con el servicio XPC.

  4. (4 o 5) Verificar si el proceso de conexión tiene un número de versión de software adecuado.

    • Si esto no se verifica, clientes antiguos e inseguros, vulnerables a la inyección de procesos, podrían usarse para conectarse al servicio XPC incluso con las otras comprobaciones en su lugar.

  5. (4 o 5) Verificar si el proceso de conexión tiene un tiempo de ejecución reforzado sin permisos peligrosos (como los que permiten cargar bibliotecas arbitrarias o usar variables de entorno DYLD)

    • Si esto no se verifica, el cliente podría ser vulnerable a la inyección de código

  6. Verificar si el proceso de conexión tiene un permiso que le permite conectarse al servicio. Esto es aplicable para binarios de Apple.

  7. La verificación debe basarse en el token de auditoría del cliente que se conecta en lugar de su ID de proceso (PID), ya que el primero previene ataques de reutilización de PID.

    • Los desarrolladores raramente usan la llamada a la API del token de auditoría ya que es privada, por lo que Apple podría cambiarla en cualquier momento. Además, el uso de API privadas no está permitido en las aplicaciones de Mac App Store.

    • Si se utiliza el método processIdentifier, podría ser vulnerable

    • En lugar de xpc_connection_get_audit_token, se debería usar xpc_dictionary_get_audit_token, ya que el último también podría ser vulnerable en ciertas situaciones.

Ataques de Comunicación

Para más información sobre el ataque de reutilización de PID, consulta:

pagemacOS PID Reuse

Para más información sobre el ataque xpc_connection_get_audit_token, consulta:

pagemacOS xpc_connection_get_audit_token Attack

Trustcache - Prevención de Ataques de Degradación

Trustcache es un método defensivo introducido en las máquinas de Apple Silicon que almacena una base de datos de CDHSAH de binarios de Apple para que solo se puedan ejecutar binarios no modificados y permitidos. Esto previene la ejecución de versiones anteriores.

Ejemplos de Código

El servidor implementará esta verificación en una función llamada shouldAcceptNewConnection.

- (BOOL)listener:(NSXPCListener *)listener shouldAcceptNewConnection:(NSXPCConnection *)newConnection {
//Check connection
return YES;
}

El objeto NSXPCConnection tiene una propiedad privada auditToken (la que debería usarse pero podría cambiar) y una propiedad pública processIdentifier (la que no debería usarse).

El proceso de conexión podría verificarse con algo como:

[...]
SecRequirementRef requirementRef = NULL;
NSString requirementString = @"anchor apple generic and identifier \"xyz.hacktricks.service\" and certificate leaf [subject.CN] = \"TEAMID\" and info [CFBundleShortVersionString] >= \"1.0\"";
/* Check:
- Signed by a cert signed by Apple
- Check the bundle ID
- Check the TEAMID of the signing cert
- Check the version used
*/

// Check the requirements with the PID (vulnerable)
SecRequirementCreateWithString(requirementString, kSecCSDefaultFlags, &requirementRef);
SecCodeCheckValidity(code, kSecCSDefaultFlags, requirementRef);

// Check the requirements wuing the auditToken (secure)
SecTaskRef taskRef = SecTaskCreateWithAuditToken(NULL, ((ExtendedNSXPCConnection*)newConnection).auditToken);
SecTaskValidateForRequirement(taskRef, (__bridge CFStringRef)(requirementString))

Si un desarrollador no desea verificar la versión del cliente, podría al menos asegurarse de que el cliente no sea vulnerable a la inyección de procesos:

```objectivec [...] CFDictionaryRef csInfo = NULL; SecCodeCopySigningInformation(code, kSecCSDynamicInformation, &csInfo); uint32_t csFlags = [((__bridge NSDictionary *)csInfo)[(__bridge NSString *)kSecCodeInfoStatus] intValue]; const uint32_t cs_hard = 0x100; // don't load invalid page. const uint32_t cs_kill = 0x200; // Kill process if page is invalid const uint32_t cs_restrict = 0x800; // Prevent debugging const uint32_t cs_require_lv = 0x2000; // Library Validation const uint32_t cs_runtime = 0x10000; // hardened runtime if ((csFlags & (cs_hard | cs_require_lv)) { return Yes; // Accept connection } ```

Aprende hacking en AWS de cero a héroe con htARTE (HackTricks AWS Red Team Expert)!

Otras formas de apoyar a HackTricks:

Última actualización