# WWW2Exec - atexit(), TLS Storage और अन्य मैंगल्ड पॉइंटर्स
{% hint style="success" %}
AWS हैकिंग सीखें और प्रैक्टिस करें:[**HackTricks प्रशिक्षण AWS रेड टीम एक्सपर्ट (ARTE)**](https://training.hacktricks.xyz/courses/arte)\
GCP हैकिंग सीखें और प्रैक्टिस करें: [**HackTricks प्रशिक्षण GCP रेड टीम एक्सपर्ट (GRTE)**](https://training.hacktricks.xyz/courses/grte)
HackTricks का समर्थन करें
* [**सदस्यता योजनाएं**](https://github.com/sponsors/carlospolop) की जाँच करें!
* **शामिल हों** 💬 [**डिस्कॉर्ड समूह**](https://discord.gg/hRep4RUj7f) या [**टेलीग्राम समूह**](https://t.me/peass) और हमें **ट्विटर** 🐦 [**@hacktricks\_live**](https://twitter.com/hacktricks\_live)** पर फॉलो** करें।
* **हैकिंग ट्रिक्स साझा करें, PRs सबमिट करके** [**HackTricks**](https://github.com/carlospolop/hacktricks) और [**HackTricks Cloud**](https://github.com/carlospolop/hacktricks-cloud) github रेपो में।
{% endhint %}
## **\_\_atexit संरचनाएँ**
{% hint style="danger" %}
आजकल इसे एक्सप्लॉइट करना बहुत **अजीब** है!
{% endhint %}
**`atexit()`** एक **कार्य** है जिसमें **अन्य कार्यों को पैरामीटर के रूप में पास** किया जाता है। ये **कार्य** एक **`exit()`** या **मुख्य** के **वापसी** को **चलाए**गा।\
यदि आप किसी भी इन **कार्यों** के **पते** को एक शैलकोड पर पॉइंट करने के लिए **संशोधित** कर सकते हैं, तो आप **प्रक्रिया** का **नियंत्रण** प्राप्त करेंगे, लेकिन यह वर्तमान में अधिक जटिल है।\
वर्तमान में **कार्यों के पतों** को निष्क्रिय करने के लिए कई संरचनाओं के पीछे **छिपा हुआ है** और अंत में जिस पते पर यह पॉइंट करता है, वह कार्यों के पतों के पते नहीं हैं, बल्कि **XOR** और एक **रैंडम कुंजी** के साथ **एन्क्रिप्टेड** हैं। इसलिए वर्तमान में यह हमला वेक्टर **कम उपयोगी है कम से कम x86** और **x64\_86** पर।\
**एन्क्रिप्शन कार्य** है **`PTR_MANGLE`**। **अन्य आर्किटेक्चर** जैसे m68k, mips32, mips64, aarch64, arm, hppa... **इस एन्क्रिप्शन** कार्य को **लागू नहीं करते** क्योंकि यह **वही वापस** उसे प्राप्त हुआ जैसा इनपुट था। इसलिए इन आर्किटेक्चर्स पर यह वेक्टर से हमला किया जा सकता है।
आप इसके काम करने की गहराई में व्याख्या [https://m101.github.io/binholic/2017/05/20/notes-on-abusing-exit-handlers.html](https://m101.github.io/binholic/2017/05/20/notes-on-abusing-exit-handlers.html) में पा सकते हैं।
## link\_map
जैसा कि [**इस पोस्ट में**](https://github.com/nobodyisnobody/docs/blob/main/code.execution.on.last.libc/README.md#2---targetting-ldso-link\_map-structure) स्पष्ट किया गया है, यदि कार्यक्रम `return` या `exit()` का उपयोग करके बाहर निकलता है, तो यह `__run_exit_handlers()` को चलाएगा जो पंजीकृत नाशकों को बुलाएगा।
{% hint style="danger" %}
यदि कार्यक्रम **`_exit()`** फ़ंक्शन के माध्यम से बाहर निकलता है, तो यह **`exit` सिस्टम कॉल** को बुलाएगा और बाहर निकास हैंडलर नहीं चलाएगा। इसलिए, `__run_exit_handlers()` को चलाया गया है यह सुनिश्चित करने के लिए आप इस पर एक ब्रेकपॉइंट सेट कर सकते हैं।
{% endhint %}
महत्वपूर्ण कोड है ([स्रोत](https://elixir.bootlin.com/glibc/glibc-2.32/source/elf/dl-fini.c#L131)):
```c
ElfW(Dyn) *fini_array = map->l_info[DT_FINI_ARRAY];
if (fini_array != NULL)
{
ElfW(Addr) *array = (ElfW(Addr) *) (map->l_addr + fini_array->d_un.d_ptr);
size_t sz = (map->l_info[DT_FINI_ARRAYSZ]->d_un.d_val / sizeof (ElfW(Addr)));
while (sz-- > 0)
((fini_t) array[sz]) ();
}
[...]
// This is the d_un structure
ptype l->l_info[DT_FINI_ARRAY]->d_un
type = union {
Elf64_Xword d_val; // address of function that will be called, we put our onegadget here
Elf64_Addr d_ptr; // offset from l->l_addr of our structure
}
```
नोट करें कि `map -> l_addr + fini_array -> d_un.d_ptr` का उपयोग **गणना** के लिए किया जाता है जिससे **कॉल करने के लिए फ़ंक्शन का एरे** का स्थान पता लगाया जा सकता है।
कुछ **विकल्प** हैं:
* `map->l_addr` के मान को ओवरराइट करें ताकि यह एक **नकली `fini_array`** की ओर प्वाइंट करे जिसमें अर्बिट्रेरी कोड को निष्पादित करने के निर्देश हों
* `l_info[DT_FINI_ARRAY]` और `l_info[DT_FINI_ARRAYSZ]` एंट्री के मान को ओवरराइट करें (जो स्मृति में अधिकांशत: एक साथ होते हैं), ताकि वे फिर से एक जाली `Elf64_Dyn` संरचना की ओर प्वाइंट करें जो फिर से **`array` को एक स्मृति** क्षेत्र पर प्वाइंट करेगी जिसे हमलावर नियंत्रित करता है।
* [**इस व्रिटअप**](https://github.com/nobodyisnobody/write-ups/tree/main/DanteCTF.2023/pwn/Sentence.To.Hell) `l_info[DT_FINI_ARRAY]` को एक नियंत्रित स्मृति में स्थित `.bss` में एक जाली `fini_array` के पते के साथ ओवरराइट करता है। यह जाली अर्रे में **पहले एक** [**वन गैजेट**](../rop-return-oriented-programing/ret2lib/one-gadget.md) **पता** शामिल है जो निष्पादित किया जाएगा और फिर **इस जाली अर्रे** और `map->l_addr` के मान के बीच का **अंतर** जिससे `*array` नकली अर्रे पर प्वाइंट करेगा।
* इस तकनीक के मुख्य पोस्ट और [**इस व्रिटअप**](https://activities.tjhsst.edu/csc/writeups/angstromctf-2021-wallstreet) के अनुसार ld.so एक पॉइंटर को स्टैक पर छोड़ देता है जो ld.so में बाइनरी `link_map` की ओर पॉइंट करता है। एक अर्बिट्रेरी राइट के साथ इसे ओवरराइट करना संभव है और इसे एक हमलावर द्वारा नियंत्रित नकली `fini_array` की ओर पॉइंट करना है जिसमें एक [**वन गैजेट**](../rop-return-oriented-programing/ret2lib/one-gadget.md) के पते को उदाहरण के लिए।
पिछले कोड के अनुसरण में आप एक और दिलचस्प खंड पा सकते हैं जिसमें कोड है:
```c
/* Next try the old-style destructor. */
ElfW(Dyn) *fini = map->l_info[DT_FINI];
if (fini != NULL)
DL_CALL_DT_FINI (map, ((void *) map->l_addr + fini->d_un.d_ptr));
}
```
इस मामले में `map->l_info[DT_FINI]` के मान को ओवरराइट करना संभव होगा जो एक जाली `ElfW(Dyn)` संरचना को दर्शाता है। [**यहाँ अधिक जानकारी प्राप्त करें**](https://github.com/nobodyisnobody/docs/blob/main/code.execution.on.last.libc/README.md#2---targetting-ldso-link\_map-structure).
## TLS-Storage dtor\_list overwrite in **`__run_exit_handlers`**
जैसा [**यहाँ स्पष्ट किया गया है**](https://github.com/nobodyisnobody/docs/blob/main/code.execution.on.last.libc/README.md#5---code-execution-via-tls-storage-dtor\_list-overwrite), यदि कोई प्रोग्राम `return` या `exit()` के माध्यम से बंद होता है, तो यह **`__run_exit_handlers()`** को निष्पादित करेगा जो किसी भी नाशक कार्य को कॉल करेगा जो पंजीकृत है।
कोड `_run_exit_handlers()` से:
```c
/* Call all functions registered with `atexit' and `on_exit',
in the reverse of the order in which they were registered
perform stdio cleanup, and terminate program execution with STATUS. */
void
attribute_hidden
__run_exit_handlers (int status, struct exit_function_list **listp,
bool run_list_atexit, bool run_dtors)
{
/* First, call the TLS destructors. */
#ifndef SHARED
if (&__call_tls_dtors != NULL)
#endif
if (run_dtors)
__call_tls_dtors ();
```
कोड से **`__call_tls_dtors()`**:
```c
typedef void (*dtor_func) (void *);
struct dtor_list //struct added
{
dtor_func func;
void *obj;
struct link_map *map;
struct dtor_list *next;
};
[...]
/* Call the destructors. This is called either when a thread returns from the
initial function or when the process exits via the exit function. */
void
__call_tls_dtors (void)
{
while (tls_dtor_list) // parse the dtor_list chained structures
{
struct dtor_list *cur = tls_dtor_list; // cur point to tls-storage dtor_list
dtor_func func = cur->func;
PTR_DEMANGLE (func); // demangle the function ptr
tls_dtor_list = tls_dtor_list->next; // next dtor_list structure
func (cur->obj);
[...]
}
}
```
हर **`tls_dtor_list`** में पंजीकृत फ़ंक्शन के लिए, यह **`cur->func`** से पॉइंटर को डीमैंगल करेगा और इसे **`cur->obj`** के साथ कॉल करेगा।
इस [**GEF के fork**](https://github.com/bata24/gef) से **`tls`** फ़ंक्शन का उपयोग करके, यह देखना संभव है कि वास्तव में **`dtor_list`** बहुत **करीब** है **stack canary** और **PTR\_MANGLE cookie** के। इसलिए, इस पर ओवरफ़्लो होने पर **cookie** और **stack canary** को **ओवरराइट** किया जा सकता है।\
PTR\_MANGLE कुकी को ओवरराइट करने से, इसे 0x00 पर सेट करके **`PTR_DEMANLE` फ़ंक्शन** को **बाईपास** किया जा सकता है, जिससे कि वास्तविक पता प्राप्त करने के लिए उपयोग किया गया **`xor`** केवल पता कॉन्फ़िगर किया गया हो। फिर, **`dtor_list`** पर लिखकर यह संभव है कि फ़ंक्शन **पता** और इसका **विवाद** किया जा सकता है।
अंत में ध्यान दें कि स्टोर किया गया पॉइंटर केवल कुकी के साथ xored होने वाला है बल्कि 17 बिट भी घुमाया जाएगा:
```armasm
0x00007fc390444dd4 <+36>: mov rax,QWORD PTR [rbx] --> mangled ptr
0x00007fc390444dd7 <+39>: ror rax,0x11 --> rotate of 17 bits
0x00007fc390444ddb <+43>: xor rax,QWORD PTR fs:0x30 --> xor with PTR_MANGLE
```
So you need to take this into account before adding a new address.
Find an example in the [**original post**](https://github.com/nobodyisnobody/docs/blob/main/code.execution.on.last.libc/README.md#5---code-execution-via-tls-storage-dtor\_list-overwrite).
## **`__run_exit_handlers`** में अन्य मैंगल्ड पॉइंटर
यह तकनीक [**यहाँ समझाई गई है**](https://github.com/nobodyisnobody/docs/blob/main/code.execution.on.last.libc/README.md#5---code-execution-via-tls-storage-dtor\_list-overwrite) और फिर से कार्यक्रम **`return` या `exit()` को बुलाकर बाहर निकलने पर निर्भर है** ताकि **`__run_exit_handlers()`** को बुलाया जाए।
इस फ़ंक्शन को और अधिक कोड की जाँच करें:
```c
while (true)
{
struct exit_function_list *cur;
restart:
cur = *listp;
if (cur == NULL)
{
/* Exit processing complete. We will not allow any more
atexit/on_exit registrations. */
__exit_funcs_done = true;
break;
}
while (cur->idx > 0)
{
struct exit_function *const f = &cur->fns[--cur->idx];
const uint64_t new_exitfn_called = __new_exitfn_called;
switch (f->flavor)
{
void (*atfct) (void);
void (*onfct) (int status, void *arg);
void (*cxafct) (void *arg, int status);
void *arg;
case ef_free:
case ef_us:
break;
case ef_on:
onfct = f->func.on.fn;
arg = f->func.on.arg;
PTR_DEMANGLE (onfct);
/* Unlock the list while we call a foreign function. */
__libc_lock_unlock (__exit_funcs_lock);
onfct (status, arg);
__libc_lock_lock (__exit_funcs_lock);
break;
case ef_at:
atfct = f->func.at;
PTR_DEMANGLE (atfct);
/* Unlock the list while we call a foreign function. */
__libc_lock_unlock (__exit_funcs_lock);
atfct ();
__libc_lock_lock (__exit_funcs_lock);
break;
case ef_cxa:
/* To avoid dlclose/exit race calling cxafct twice (BZ 22180),
we must mark this function as ef_free. */
f->flavor = ef_free;
cxafct = f->func.cxa.fn;
arg = f->func.cxa.arg;
PTR_DEMANGLE (cxafct);
/* Unlock the list while we call a foreign function. */
__libc_lock_unlock (__exit_funcs_lock);
cxafct (arg, status);
__libc_lock_lock (__exit_funcs_lock);
break;
}
if (__glibc_unlikely (new_exitfn_called != __new_exitfn_called))
/* The last exit function, or another thread, has registered
more exit functions. Start the loop over. */
goto restart;
}
*listp = cur->next;
if (*listp != NULL)
/* Don't free the last element in the chain, this is the statically
allocate element. */
free (cur);
}
__libc_lock_unlock (__exit_funcs_lock);
```
चर `f` **`initial`** संरचना को दर्शाता है और `f->flavor` के मान के आधार पर विभिन्न फ़ंक्शन को कॉल किया जाएगा।
मान के आधार पर, कॉल करने के लिए फ़ंक्शन का पता एक विभिन्न स्थान पर होगा, लेकिन यह हमेशा **demangled** होगा।
इसके अतिरिक्त, विकल्पों **`ef_on`** और **`ef_cxa`** में एक **विधान** को भी नियंत्रित किया जा सकता है।
डीबगिंग सत्र में **GEF** चलाते समय **`gef> p initial`** के साथ **`initial` संरचना** की जांच की जा सकती है।
इसे उपयोग करने के लिए आपको या तो **`PTR_MANGLE` कुकी को लीक करना होगा या मिटाना होगा** और फिर `system('/bin/sh')` के साथ `cxa` प्रविष्टि को ओवरराइट करना होगा।\
आप इसका एक उदाहरण [**तकनीक के बारे में मूल ब्लॉग पोस्ट में**](https://github.com/nobodyisnobody/docs/blob/main/code.execution.on.last.libc/README.md#6---code-execution-via-other-mangled-pointers-in-initial-structure) देख सकते हैं।