-is-this-fft-'s blog

By -is-this-fft-, history, 5 weeks ago, In English

This is my in-contest submission for 2048F - Kevin and Math Class: 297343653. It got "runtime error on pretest 3", which I initially thought was some stupid bug. After the contest was over, I was surprised to see "exit code: -1073741571 (STATUS_STACK_OVERFLOW)". Stack overflow happens when the recursion goes too deep, but the stack limits on Codeforces are pretty generous, and recursion 2e5 calls deep is generally not an issue (imagine how annoying all DFS problems would be otherwise).

At first I thought that this must still be a bug that led to infinite recursion. But I verified locally that this is not the case, and I added an assertion to check that no more than 2e5 calls to solve(ll, ll) are performed in any one test case:

ll solve (ll l, ll r) {
  cnt++;
  assert(cnt < maxn);
// .. rest of the function ..

I also moved the references to tree and arr to global variables for good measure. Still a stack overflow: 297352244.

For what it's worth, both versions get accepted under C++20: 297353626, 297353915.

  • Vote: I like it
  • +54
  • Vote: I do not like it

»
5 weeks ago, # |
Rev. 4   Vote: I like it +8 Vote: I do not like it

Oh, that is weird. The same thing happened to me at F: an actual stack overflow. Just like you, I thought it was infinite recursion, but with minimal changes, the code passes. I do not think I ever got stack overflow which was not infinite recursion and was in intended complexity.

I made my function return array of 61 numbers representing dp values. I had similar recursive approach as you. Interestingly, when I replaced those with a vector, it passed.

Submission with returning array 297354881

Submission with returning vector 297353480

I guess it kinda makes sense as I used a lot of memory in my calls, and if I am not mistaken when I call function that returns struct, it gets stored in stack memory. But vector gets stored in heap. I could be totally wrong about that, so please correct me in that case.

  • »
    »
    5 weeks ago, # ^ |
    Rev. 2   Vote: I like it +8 Vote: I do not like it

    In the implementation of libstdc++, the array<_Tp, _Nm>::_M_elems is defined as __array_traits<_Tp, _Nm>::_Type, and the latter is simply _Tp[_Nm].

    So you're indeed piling $$$61$$$ values on the stack which overflows.

    • »
      »
      »
      5 weeks ago, # ^ |
        Vote: I like it 0 Vote: I do not like it

      I used struct actually. But I think it's the same?

      • »
        »
        »
        »
        5 weeks ago, # ^ |
          Vote: I like it 0 Vote: I do not like it

        Oh i don't look at your submission and assume you use std::array. You code nevertheless does the same thing as std::array.

»
5 weeks ago, # |
Rev. 2   Vote: I like it +3 Vote: I do not like it

encountered stack overflow on many 5e5-1e6 recursion calls also on so many dp problems where its not possible to initialize the dp array globally because some states change in an amortized way not sure why cf doesnt increase the stack to 512 or 1024 , it wont cause any issues no ?

another thing about returning structs its so random to me i never understand struct memory or how it works sometimes really heavy structs dont take as much memory as structs that seem alot lighter

»
5 weeks ago, # |
  Vote: I like it +53 Vote: I do not like it

It is simply worth increasing the stack in such situations. For example, if you double the stack, then the solution works: 297356923. C++ on Codeforces uses 256 MiB by default, in my solution I increased my stack to 512 MiB.

  • »
    »
    5 weeks ago, # ^ |
      Vote: I like it +4 Vote: I do not like it

    What? That feels like something that should not be legal...

    • »
      »
      »
      5 weeks ago, # ^ |
        Vote: I like it +26 Vote: I do not like it

      Why? I'm just using the OS's capabilities. I just create a new thread, give 512 MiB stack memory to it, and order the main thread wait until the one I created completes. If the solution with this technique it does not exceed ML, then it should be legal, I think.

      CreateThread documentation.

      • »
        »
        »
        »
        5 weeks ago, # ^ |
          Vote: I like it 0 Vote: I do not like it

        Yeah. But it's crazy that's possible. It kinda gives advantage. Though similar argument could be made about ios_base::sync_with_stdio(false);

        • »
          »
          »
          »
          »
          5 weeks ago, # ^ |
            Vote: I like it 0 Vote: I do not like it

          I also remembered another old technique for stack increasing. People wrote #pragma comment(linker, "/STACK:<Stack size in bytes>") in the source code and submit solution with Microsoft C++. But, Microsoft C++ compiler is generally slower than g++, and now there is no Microsoft C++ compiler on Codeforces.

          In general, the same can be said about pragmas, which sometimes make it possible to get AC with solution which should not fit TL according to the authors of the problem.

          P.S. All of C/C++ default IO is shit. Custom IO will be significantly faster (ofc only in proper implementation).

      • »
        »
        »
        »
        5 weeks ago, # ^ |
        Rev. 2   Vote: I like it -12 Vote: I do not like it

        To my knowledge, in CP history, people have been developing various limitations to C++ itself to intentionally make some code run slower and therefore stress the importance of complexity rather than constant factors. For example, in contests organized by CCF (China Computing Federation), most modern optimization switches are turned off, and it's explicitly not allowed to do multithreading. It's actually likely that Codeforces would disallow this in the future.

        By the way, I think Codeforces should have infinite stack size by default. (However, the stack memory still counts toward the memory limit)

  • »
    »
    5 weeks ago, # ^ |
      Vote: I like it +55 Vote: I do not like it

    How to increase time limit?

»
5 weeks ago, # |
  Vote: I like it +75 Vote: I do not like it

I have an answer but this margin is to small to contain it. So I posted it in a separate blog: https://codeforces.net/blog/entry/137488

TL;DR: GCC 7 (32-bit) bad

»
5 weeks ago, # |
  Vote: I like it 0 Vote: I do not like it

That's... strange? I literally created a static long long dp array of size n * 61, and still did not stack overflow. 297338243