macOS - AMFI - AppleMobileFileIntegrity
Last updated
Last updated
Learn & practice AWS Hacking:HackTricks Training AWS Red Team Expert (ARTE) Learn & practice GCP Hacking: HackTricks Training GCP Red Team Expert (GRTE)
It focuses on enforcing the integrity of the code running on the system providing the logic behind XNU's code signature verification. It's also able to check entitlements and handle other sensitive tasks such as allowing debugging or obtaining task ports.
Moreover, for some operations, the kext prefers to contact the user space running daemon /usr/libexec/amfid
. This trust relationship has been abused in several jailbreaks.
AMFI uses MACF policies and it registers its hooks the moment it's started. Also, preventing its loading or unloading it could trigger a kernel panic. However, there are some boot arguments that allow to debilitate AMFI:
amfi_unrestricted_task_for_pid
: Allow task_for_pid to be allowed without required entitlements
amfi_allow_any_signature
: Allow any code signature
cs_enforcement_disable
: System-wide argument used to disable code signing enforcement
amfi_prevent_old_entitled_platform_binaries
: Void platform binaries with entitlements
amfi_get_out_of_my_way
: Disables amfi completely
These are some of the MACF policies it registers:
cred_check_label_update_execve:
Label update will be performed and return 1
cred_label_associate
: Update AMFI's mac label slot with label
cred_label_destroy
: Remove AMFI’s mac label slot
cred_label_init
: Move 0 in AMFI's mac label slot
cred_label_update_execve
: It checks the entitlements of the process to see it should be allowed to modify the labels.
file_check_mmap
: It checks if mmap is acquiring memory and setting it as executable. In that case it check if library validation is needed and if so, it calls the library validation function.
file_check_library_validation
: Calls the library validation function which checks among other things if a platform binary is loading another platform binary or if the process and the new loaded file have the same TeamID. Certain entitlements will also allow to load any library.
policy_initbsd
: Sets up trusted NVRAM Keys
policy_syscall
: It checks DYLD policies like if the binary has unrestricted segments, if it should allow env vars... this is also called when a process is started via amfi_check_dyld_policy_self()
.
proc_check_inherit_ipc_ports
: It checks if when a processes executes a new binary other processes with SEND rights over the task port of the process should keep them or not. Platform binaries are allowed, get-task-allow
entitled allows it, task_for_pid-allow
entitles are allowed and binaries with the same TeamID.
proc_check_expose_task
: enforce entitlements
amfi_exc_action_check_exception_send
: An exception message is sent to debugger
amfi_exc_action_label_associate & amfi_exc_action_label_copy/populate & amfi_exc_action_label_destroy & amfi_exc_action_label_init & amfi_exc_action_label_update
: Label lifecycle during exception handling (debugging)
proc_check_get_task
: Checks entitlements like get-task-allow
which allows other processes to get the tasks port and task_for_pid-allow
, which allow the process to get other processes tasks ports. If neither of those, it calls up to amfid permitunrestricteddebugging
to check if it's allowed.
proc_check_mprotect
: Deny if mprotect
is called with the flag VM_PROT_TRUSTED
which indicates that the region must be treated as if it has a valid code signature.
vnode_check_exec
: Gets called when a executable files are loaded in memory and sets cs_hard | cs_kill
which will kill the process if any of the pages becomes invalid
vnode_check_getextattr
: MacOS: Check com.apple.root.installed
and isVnodeQuarantined()
vnode_check_setextattr
: As get + com.apple.private.allow-bless and internal-installer-equivalent entitlement
vnode_check_signature
: Code that calls XNU to check the code signature using entitlements, trust cache and amfid
proc_check_run_cs_invalid
: It intercepts ptrace()
calls (PT_ATTACH
and PT_TRACE_ME
). It checks for any of the entitlements get-task-allow
, run-invalid-allow
and run-unsigned-code
and if none, it checks if debugging is permitted.
proc_check_map_anon
: If mmap is called with the MAP_JIT
flag, AMFI will checks for the dynamic-codesigning
entitlement.
AMFI.kext
also exposes an API for other kernel extensions, and it's possible to find its dependencies with:
This is the user mode running daemon that AMFI.kext
will use to check for code signatures in user mode.
For AMFI.kext
to communicate with the daemon it uses mach messages over the port HOST_AMFID_PORT
which is the special port 18
.
Note that in macOS it's no longer possible for root processes to hijack special ports as they are protected by SIP
and only launchd can get them. In iOS it's checked that the process sending the response back has the CDHash hardcoded of amfid
.
It's possible to see when amfid
is requested to check a binary and the response of it by debugging it and setting a breakpoint in mach_msg
.
Once a message is received via the special port MIG is used to send each function to the function it's calling. The main functions were reversed and explained inside the book.
A provisioning profile can be used to sign code. There are Developer profiles that can be used to sign code and test it, and Enterprise profiles which can be used in all devices.
After an App is submitted to the Apple Store, if approved, it's signed by Apple and the provisioning profile is no longer needed.
A profile usually use the extension .mobileprovision
or .provisionprofile
and can be dumped with:
Although sometimes referred as certificated, these provisioning profiles have more than a certificate:
AppIDName: The Application Identifier
AppleInternalProfile: Designates this as an Apple Internal profile
ApplicationIdentifierPrefix: Prepended to AppIDName (same as TeamIdentifier)
CreationDate: Date in YYYY-MM-DDTHH:mm:ssZ
format
DeveloperCertificates: An array of (usually one) certificate(s), encoded as Base64 data
Entitlements: The entitlements allowed with entitlements for this profile
ExpirationDate: Expiration date in YYYY-MM-DDTHH:mm:ssZ
format
Name: The Application Name, the same as AppIDName
ProvisionedDevices: An array (for developer certificates) of UDIDs this profile is valid for
ProvisionsAllDevices: A boolean (true for enterprise certificates)
TeamIdentifier: An array of (usually one) alphanumeric string(s) used to identify the developer for inter-app interaction purposes
TeamName: A human-readable name used to identify the developer
TimeToLive: Validity (in days) of the certificate
UUID: A Universally Unique Identifier for this profile
Version: Currently set to 1
Note that the entitlements entry will contain a restricted set of entitlements and the provisioning profile will only be able to give those specific entitlements to prevent giving Apple private entitlements.
Note that profiles are usually located in /var/MobileDeviceProvisioningProfiles
and it's possible to check them with security cms -D -i /path/to/profile
This is the external library that amfid
calls i order to ask if it should allow something or not. This has been historically abused in jailbreaking by running a backdoored version of it that would allow everything.
In macOS this is inside MobileDevice.framework
.
iOS AMFI maintains a lost of known hashes which are signed ad-hoc, called the Trust Cache and found in the kext's __TEXT.__const
section. Note that in very specific and sensitive operations It's possible to extend this Trust Cache with an external file.
Learn & practice AWS Hacking:HackTricks Training AWS Red Team Expert (ARTE) Learn & practice GCP Hacking: HackTricks Training GCP Red Team Expert (GRTE)