macOS - AMFI - AppleMobileFileIntegrity
Reading time: 7 minutes
tip
Learn & practice AWS Hacking:HackTricks Training AWS Red Team Expert (ARTE)
Learn & practice GCP Hacking: HackTricks Training GCP Red Team Expert (GRTE)
Support HackTricks
- Check the subscription plans!
- Join the 💬 Discord group or the telegram group or follow us on Twitter 🐦 @hacktricks_live.
- Share hacking tricks by submitting PRs to the HackTricks and HackTricks Cloud github repos.
AppleMobileFileIntegrity.kext and amfid
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 entitlementsamfi_allow_any_signature
: Allow any code signaturecs_enforcement_disable
: System-wide argument used to disable code signing enforcementamfi_prevent_old_entitled_platform_binaries
: Void platform binaries with entitlementsamfi_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 1cred_label_associate
: Update AMFI's mac label slot with labelcred_label_destroy
: Remove AMFI’s mac label slotcred_label_init
: Move 0 in AMFI's mac label slotcred_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 Keyspolicy_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 viaamfi_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 entitlementsamfi_exc_action_check_exception_send
: An exception message is sent to debuggeramfi_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 likeget-task-allow
which allows other processes to get the tasks port andtask_for_pid-allow
, which allow the process to get other processes tasks ports. If neither of those, it calls up toamfid permitunrestricteddebugging
to check if it's allowed.proc_check_mprotect
: Deny ifmprotect
is called with the flagVM_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 setscs_hard | cs_kill
which will kill the process if any of the pages becomes invalidvnode_check_getextattr
: MacOS: Checkcom.apple.root.installed
andisVnodeQuarantined()
vnode_check_setextattr
: As get + com.apple.private.allow-bless and internal-installer-equivalent entitlementvnode_check_signature
: Code that calls XNU to check the code signature using entitlements, trust cache andamfid
proc_check_run_cs_invalid
: It interceptsptrace()
calls (PT_ATTACH
andPT_TRACE_ME
). It checks for any of the entitlementsget-task-allow
,run-invalid-allow
andrun-unsigned-code
and if none, it checks if debugging is permitted.proc_check_map_anon
: If mmap is called with theMAP_JIT
flag, AMFI will checks for thedynamic-codesigning
entitlement.
AMFI.kext
also exposes an API for other kernel extensions, and it's possible to find its dependencies with:
kextstat | grep " 19 " | cut -c2-5,50- | cut -d '(' -f1
Executing: /usr/bin/kmutil showloaded
No variant specified, falling back to release
8 com.apple.kec.corecrypto
19 com.apple.driver.AppleMobileFileIntegrity
22 com.apple.security.sandbox
24 com.apple.AppleSystemPolicy
67 com.apple.iokit.IOUSBHostFamily
70 com.apple.driver.AppleUSBTDM
71 com.apple.driver.AppleSEPKeyStore
74 com.apple.iokit.EndpointSecurity
81 com.apple.iokit.IOUserEthernet
101 com.apple.iokit.IO80211Family
102 com.apple.driver.AppleBCMWLANCore
118 com.apple.driver.AppleEmbeddedUSBHost
134 com.apple.iokit.IOGPUFamily
135 com.apple.AGXG13X
137 com.apple.iokit.IOMobileGraphicsFamily
138 com.apple.iokit.IOMobileGraphicsFamily-DCP
162 com.apple.iokit.IONVMeFamily
amfid
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.
Provisioning Profiles
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:
openssl asn1parse -inform der -in /path/to/profile
# Or
security cms -D -i /path/to/profile
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
libmis.dyld
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
.
AMFI Trust Caches
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.
References
tip
Learn & practice AWS Hacking:HackTricks Training AWS Red Team Expert (ARTE)
Learn & practice GCP Hacking: HackTricks Training GCP Red Team Expert (GRTE)
Support HackTricks
- Check the subscription plans!
- Join the 💬 Discord group or the telegram group or follow us on Twitter 🐦 @hacktricks_live.
- Share hacking tricks by submitting PRs to the HackTricks and HackTricks Cloud github repos.