Set stack size with setrlimit() and provoke a stack overflow/segfault

前端 未结 2 678
没有蜡笔的小新
没有蜡笔的小新 2021-01-31 06:24

In the given example below I try to set the stacksize to 1kb.

Why is it now possible to allocate an array of ints on the stack with size 8kb in foo()

相关标签:
2条回答
  • 2021-01-31 06:41

    I think setrlimit moves the "resource pointers" but doesn't apply the new limits until you exec a new copy of the program.

    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/resource.h>
    
    void foo(int chk) {
      unsigned ints[2048];
      ints[2047] = 42;
      printf("foo %d: %u\n", chk, ints[2047]);
    }
    
    int main(int argc, char **argv) {
      char *newarg[] = { "argv[0]", "one", "two" };
      char *newenv[] = { NULL };
      struct rlimit lim;
    
      newarg[0] = argv[0];
      getrlimit(RLIMIT_STACK, &lim);
      printf("lim: %d / %d\n", (int)lim.rlim_cur, (int)lim.rlim_max);
      switch (argc) {
        case 1: /* first call from command line */
          lim.rlim_cur = 65536;
          lim.rlim_max = 65536;
          if (setrlimit(RLIMIT_STACK, &lim) == -1) return EXIT_FAILURE;
          newarg[2] = NULL;
          foo(1);
          execve(argv[0], newarg, newenv);
          break;
        case 2: /* second call */
          lim.rlim_cur = 1024;
          lim.rlim_max = 1024;
          if (setrlimit(RLIMIT_STACK, &lim) == -1) return EXIT_FAILURE;
          foo(2);
          execve(argv[0], newarg, newenv);
          break;
        default: /* third call */
          foo(3);
          break;
      }
      return 0;
    }
    

    And a test run:

    $ ./a.out 
    lim: 8388608 / -1
    foo 1: 42
    lim: 65536 / 65536
    foo 2: 42
    Killed
    

    Why the process gets killed before printing the limits (and before calling foo), I don't know.

    0 讨论(0)
  • 2021-01-31 06:57

    The limit is set immediately but only checked when trying to allocate a new stack or trying to grow the existing stack. A grep for RLIMIT_STACK (or a LXR identifier search) on the kernel sources should tell.

    Apparently, the initial size of the stack is whatever is needed to the filename + env strings + arg strings plus some extra pages allocated on setup_arg_pages (20 pages in 2.6.33 1,2, 128 Kb on 2.6.34 3).

    In summary:

    initial stack size = MIN(size for filename + arg strings + env strings + extra pages, MAX(size for filename + arg strings + env strings, RLIMIT_STACK))
    

    where

    size for filename + arg strings + env strings <= MAX(ARG_MAX(32 pages), RLIMIT_STACK/4)
    

    Additionally, kernels with Ingo Molnar's exec-shield patch (Fedora, Ubuntu, ...) have an additional EXEC_STACK_BIAS "(2MB more to cover randomization effects.)", see the call to the new function over_stack_limit() from acct_stack_growth() ([Ubuntu1], [Ubuntu2], [Ubuntu3]).

    I've edited the original program to show this:

    #include <stdio.h>
    #include <sys/resource.h>
    
    void foo(void);
    
    int main(int argc, char *argv[]) {
            struct rlimit lim = {1, 1};
    
    
            if (argc > 1 && argv[1][0] == '-' && argv[1][8]=='l') {
                    printf("limiting stack size\n");
                    if (setrlimit(RLIMIT_STACK, &lim) == -1) {
                            printf("rlimit failed\n");
                            return 1;
                    }
            }
    
            foo();
    
            return 0;
    }
    
    void foo() {
            unsigned ints[32768];
    
            printf("foo: %u\n", ints[2047]=42);
    }
    

    Which results in:

    $./rl
    foo: 42
    $./rl -l
    limiting stack size
    Segmentation fault
    $  
    
    0 讨论(0)
提交回复
热议问题