This article is based mostly on the work of Grzegorz Tworek (@0gtweet)
I recently saw this tweet from Grzegorz Tworek (@0gtweet – who if you aren’t following you really should be!) come across my timeline
data:image/s3,"s3://crabby-images/e3bc2/e3bc227755748485e75e4d36f0e718ab29210d8b" alt=""
I had seen previous tweets referencing the AppLocker hash/signature cache and having a CPD day I thought I would take a closer look at see what did work and what didn’t. Probably fair to say if it didn’t work – that would be on me, rather than the source material
Having a look at the https://github.com/gtworek/PSBits/tree/master/CopyEAs repository there isn’t a huge amount of material to go off (for someone new to it like me – once you get your head around it, then it actually is everything you need to know).
data:image/s3,"s3://crabby-images/c4cca/c4cca0124c5fafbbf3526bf44db25ff2de906977" alt=""
Righty then. Let’s dig down and see what we can find. Let’s start with NTFS EA – Wikipedia helpfully tells us Extended Attributes (EA) are file system features that enable users to associate computer files with metadata not interpreted by the filesystem, whereas regular attributes have a purpose strictly defined by the filesystem (such as permissions or records of creation and modification times). General documentation on EAs is actually quite sparse – the best resource I found giving an overview is the ever dependable SpecterOps : https://posts.specterops.io/host-based-threat-modeling-indicator-design-a9dbbb53d5ea
Like Alternative Data Streams (ADS) but with a data limit of ~65k on NTFS (varies according to file system but that limit is from the Linux implementation of EAs)
data:image/s3,"s3://crabby-images/debf0/debf00d0414ecd24e932316b6f976862dd9a0d64" alt=""
So how does AppLocker use these EAs and how do we abuse them to bypass it?
In my test environment I set up an AppLocker rule to allow a file with a certain hash
data:image/s3,"s3://crabby-images/486cf/486cf07264ca944201c87af5e0702dadfdbaa890" alt=""
It is worth noting that this is a specific AppLocker hash, not a file hash
data:image/s3,"s3://crabby-images/e77e3/e77e314ffbb6b43d32fdd670aa226b1d8e5a4358" alt=""
Querying the EAs of the file using fsutil shows that the AppLocker Hash is stored in $KERNEL.PURGE.APPID.HASHINFO
From what I can gather from the tools that Grzegorz released, we can write EAs but we can’t overwrite the $ prefixed entries. That is why his CopyEAs toolkit creates entries prefixed with a # and direct disk access is required to rename them.
data:image/s3,"s3://crabby-images/2bc32/2bc32e4f272b5c2fefb2910a67707aa97fa8b108" alt=""
So let’s PoC this up and see what we can do. I created a 20Mb VHD and mounted it as a test user. I placed a file, imaginatively called Malware.exe on the mounted drive.
Prior to execution, no attributes were visible
data:image/s3,"s3://crabby-images/8915d/8915dbbd893efe85033e276c7b36f5e537e3917d" alt=""
Running it was prohibited via AppLocker
data:image/s3,"s3://crabby-images/2a557/2a5570c7ea7885833b3f35c0991957332d1f5f60" alt=""
After running it we could see that some EAs had been populated
data:image/s3,"s3://crabby-images/9cff5/9cff5f9ba408d67edaee35632e51786548d69077" alt=""
Our hash did not match the AppLocker rule according to the EA value on KERNEL.PURGE.APPID.HASHINFO
data:image/s3,"s3://crabby-images/7d79a/7d79a474a8bdc9cda77105f10552725ee6b8aeae" alt=""
Which can be confirmed from the command line
data:image/s3,"s3://crabby-images/2bda1/2bda14cee92bcff8c3b9f5bdc8c49dc0f234ec76" alt=""
Now we have a couple of options at this point – we can add a #KERNEL.PURGE.APPID.HASHINFO
with a ‘good’ hash value using the SetApplockerCache.exe that is part of the CopyEAS tool suite as below :
data:image/s3,"s3://crabby-images/f9d44/f9d44ea6cff69f2c2e766a17c7f714906e48ce63" alt=""
Or we can just search and replace for the original hash value with the ‘good’ value. Unmounting the VHD and popping it into a hex editor we can search for the values we are looking for.
data:image/s3,"s3://crabby-images/04460/04460b8ef47300b62ed31f402567f53bae0931f7" alt=""
Replace those hash values with the ‘good’ value and after remounting the VHD and re-querying the values shows that the AppLocker hash cache now contains the ‘good’ values.
data:image/s3,"s3://crabby-images/afe1c/afe1cd831eeb3ee3e284d23b32c0b39d4f40aeef" alt=""
So what happens if we run it?
data:image/s3,"s3://crabby-images/c373c/c373c7efada2f93706f7b8aac4e925c0010b8b6c" alt=""
data:image/s3,"s3://crabby-images/9638f/9638f234822cecb0eed0a2ce2d39a2fbc0e0e8e3" alt=""
We get our CS_Is_Dead_Sliver_Is_The_New_Hotness callback 🙂
This also worked for me on a USB stick, or any NTFS aware filesystem.
Big shout out to Grzegorz Tworek for https://github.com/gtworek/PSBits – you can literally lose days of your life digging a little deeper into the stuff he uncovers!
Hope you found this useful. There is mention of getting it to work with Microsoft signed AppLocker rules using the CopyEAs tool but I couldn’t get that working, not sure if that has been patched since the tool release. If you get it working, please let me know!