Tags: pwn
Rating: 5.0
# Ghost in the heap - Write up
### Program
data:image/s3,"s3://crabby-images/32561/32561da967a4243fe83be4ed9e7831bd71ed7405" alt=""
+ New heap
+ Allocate a 168 bytes heap.
+ You can only have three heap.
+ Delete heap
+ Delete heap by index.
+ Add ghost
+ Add a ghost with magic
+ You can only have one ghost.
```
struct Ghost {
char desc[0x48] ;
long magic ;
};
```
+ Watch ghost
+ Show the information of ghost if magic is matched.
+ Remove ghost
+ Remove the ghost.
### Vulnerability
+ off-by-one NULL byte in `new_heap`
```
void new_heap(){
int i ;
size_t size = 0;
for(i = 0 ; i < MAX ; i++){
if(!heap[i]){
heap[i] = malloc(HEAPSIZE);
if(!heap[i]){
puts("Alloacte error !");
_exit(-1);
}
printf("Data :");
scanf("%168s",heap[i]);
return ;
}
}
puts("Too many heap !");
}
```
+ The size of heap is 168 bytes, but it use `scanf("%168s",heap[i[)` to read input. It will puts a null byte in the end of input. If your length of input is equal to 168 byte, it would lead to off-byte one overflow.
+ Information leak
+ It use read() to read input without NULL byte which leads to information leak.
```
void read_input(char *buf,unsigned int size){
int ret ;
ret = read(0,buf,size);
if(ret <= 0){
puts("read error");
_exit(1);
}
}
```
+ Exploitation
+ Idea
+ [Fail] Fastbin dup
+ It only allocate a fastbin chunk.
+ [Fail] Unsorted bin attack to overwrite `_IO_list_all`.
+ Because there are some vtable vertify in lastet libc.
+ [Success] Unsoted bin attack to corrupt stdin buffer
+ It's using `scanf` and it would use stdin buffer. So we can overwrite the `_IO_buf_end`,you will have a stdin buffer in libc. After do that you can control the flow.
+ Information leak
+ Heap address
+ It a little hard to get heap address. Because we only have three smallbin and one fastbin chunk.
1. Add a ghost and three heap
2. Remove the first heap and ghost
data:image/s3,"s3://crabby-images/30549/30549f725e49ea089d8bb40bb5f4bc1b99be259f" alt=""
3. Remove the last heap, it would merge with top and trigger `malloc_cosolidate`. The fastbin chunk would merge with unsorted chunk.
data:image/s3,"s3://crabby-images/0c5ac/0c5ac6557e373872a3361d42ab543094b996b729" alt=""
4. Add two heap
data:image/s3,"s3://crabby-images/84938/84938beba743652751b17d6488bebaced959a466" alt=""
5. Remove `Heap 1` so that it can merge with free chunk and return to unsorted bin.
data:image/s3,"s3://crabby-images/bebcc/bebcc4357a452ab02a6bf2a565c104ae3ded1662" alt=""
6. Add one heap, and remove `heap 0`. We can see that we have two chunk in unsoted bin, so we have heap address in the heap.
data:image/s3,"s3://crabby-images/56ea9/56ea9ca781aaea50cbc54ee49965665511830aad" alt=""
7. Add ghost and we can get heap address by watching ghost
+ Libc address
+ It's easier than heap address. Just use unsorted chunk.
+ Create a overlap chunk
1. Remove ghost and all of heap
2. Add a heap, ghost and two heap
3. Remove `heap 0` and ghost
data:image/s3,"s3://crabby-images/9b57e/9b57e463c2a2e66cd01bd2b3449604398e859edf" alt=""
4. Remove `heap 2` and it would trigger `malloc_consolidate`
data:image/s3,"s3://crabby-images/00c75/00c75d8de0a0f7245bf98d2033da97105b50228f" alt=""
5. Add 2 heap
data:image/s3,"s3://crabby-images/bce2b/bce2b55be4a06e220ec2d371e3e9f33fd4bbc630" alt=""
6. Remove `heap 0` and `heap 1`
data:image/s3,"s3://crabby-images/8ac06/8ac06f0707d72df5572c59d64a0ef5da6a7c9ffb" alt=""
7. Trigger the vulnerability
data:image/s3,"s3://crabby-images/caa26/caa2605844fe3b5f103ef977493ef1d82f73fc7e" alt=""
8. Add 1 heap
data:image/s3,"s3://crabby-images/8f24c/8f24c06df69e50a511d09a1dcda7d3de08ab3f0e" alt=""
9. Remove `heap 1` (Let it can remove `heap 0`)
data:image/s3,"s3://crabby-images/739a6/739a6815869d5aefd663f5fab70dd0482c709055" alt=""
10. Remove `heap 0`
data:image/s3,"s3://crabby-images/184d5/184d55354be037ac68a21ed3bb8e3ed1a774a5d1" alt=""
11. Add a ghost
data:image/s3,"s3://crabby-images/1d228/1d228215d15a6c7b479bed0b573bdbfe9d1deb77" alt=""
12. Add a new heap and forge fake chunk
data:image/s3,"s3://crabby-images/7d25e/7d25e27d6df42537f0c03ae9966df59275061a1e" alt=""
13. Add a new heap and remove `heap 2`, then you can unlink success and create overlap chunk.
data:image/s3,"s3://crabby-images/45f2c/45f2c957a8a04b3b0e5a6102d669c0b13db4e6c0" alt=""
14. More detail you can see my [exploit](./exp.py)
+ Unsorted bin attack
+ Use unsorted bin attak to overwrite `_IO_buf_end`
data:image/s3,"s3://crabby-images/55f5a/55f5a7fc521d29a0ea58ea26525e4246b9947ae0" alt=""
+ Trigger `scanf()`
+ It will read data to stdin buffer. You can use it to overwrite `malloc_hook` with one gadget
+ Trigger `malloc` and you will get shell.
+ More detail about FILE structure
+ [Play with FILE Structure](http://4ngelboy.blogspot.tw/2017/11/play-with-file-structure-yet-another.html)
+ Exploit
+ [exp.py](./exp.py)