#include "jumps.h"
#include <bits/stdc++.h>
using namespace std;
template<typename T>
using vec = vector<T>;
using ll = long long;
#define cerr if(0) cout
ll n, l;
vec<int> a;
vec<vec<ll>> up, upR, upL;
void init(int N, std::vector<int> H) {
    n = N;
    a = H;
    up.assign(n,vec<ll>(20,-1));
    upR.assign(n,vec<ll>(20,-1));
    upL.assign(n,vec<ll>(20,-1));
    stack<ll> st;
    for(ll i = 0; i < n; i++) {
        while(!st.empty() and a[st.top()] < a[i]) {
            upR[st.top()][0] = i;
            st.pop();
        }
        st.push(i);
    }
    while(!st.empty()) st.pop();
    for(ll i = n - 1; i >= 0; i--) {
        while(!st.empty() and a[st.top()] < a[i]) {
            upL[st.top()][0] = i;
            st.pop();
        }
        st.push(i);
    }
    // for(ll i = 0; i < n; i++) {
    //     if(upR[i][0] == -1) upR[i][0] = i;
    //     if(upL[i][0] == -1) upL[i][0] = i;
    // }
    for(ll i = 0; i < n; i++) {
        if(upL[i][0] == -1 and upR[i][0] == -1) up[i][0] = -1;
        else if(upL[i][0] == -1) up[i][0] = upR[i][0];
        else if(upR[i][0] == -1) up[i][0] = upL[i][0];
        else up[i][0] = (a[upL[i][0]] > a[upR[i][0]] ? upL[i][0] : upR[i][0]);
    }
    for(ll i = 1; i < 20; i++) {
        for(ll v = 0; v < n; v++) {
            if(up[v][i-1] != -1) up[v][i] = up[up[v][i-1]][i-1];
            if(upL[v][i-1] != -1) upL[v][i] = upL[upL[v][i-1]][i-1];
            if(upR[v][i-1] != -1) upR[v][i] = upR[upR[v][i-1]][i-1];
        }
    }
}
int minimum_jumps(int A, int B, int C, int D) {
    ll at = B;
    // edge case: no middle 
    if(C == B+1) {
        if(upR[at][0] > D or upR[at][0] == -1) return -1;
        return 1;
    }
    // find the midpeak
    ll midPeak = B+1;
    for(ll i = 19; i >= 0; i--) {
        if(upR[midPeak][i] == -1) continue;
        if(upR[midPeak][i] < C) {
            midPeak = upR[midPeak][i];
        }
    }
    cerr << "Midpeak: " << midPeak << " w/ " << a[midPeak] << endl;
    // edge case: we are already taller than this
    if(a[B] >= a[midPeak]) {
        cerr << "Taller than mid, returning" << endl;
        if(upR[at][0] == -1 or upR[at][0] > D) return -1;
        return 1;
    }
    // find best starting poll from this
    for(ll i = 19; i >= 0; i--) {
        if(upL[at][i] == -1) continue;
        if(upL[at][i] >= A and a[upL[at][i]] < a[midPeak]) {
            at = upL[at][i];
        }
    }
    cerr << "New starting poll: " << at << " w/ " << a[at] << endl;
    // edge case: next best peak brings us right there
    if(upL[at][0] != -1 and A <= upL[at][0] and upR[ upL[at][0] ][0] <= D and C <= upR[ upL[at][0] ][0]) {
        cerr << "Oh we can go back one and skip" << endl;
        return 1;
    }
    ll ans = 0;
    cerr << "Stupid cases done, binlifting to mid" << endl;
    // binlift us up to mid yay
    for(ll i = 19; i >= 0; i--) {
        if(up[at][i] == -1) continue;
        if(a[up[at][i]] <= a[midPeak]) {
            at = up[at][i];
            ans += (1<<i);
        }
    }
    cerr << "After binlift at is " << at << " with " << a[at] << " spent " << ans << endl;
    // if we are NOT at mid check the going left case
    if(at != midPeak) {
        cerr << "Not at mid so check left " << endl;
        if(upL[at][0] != -1 and upR[ upL[at][0] ][0] <= D) return ans + 2;
        return -1;
    }
    // check if we can make it from mid
    if(upR[at][0] != -1 and upR[at][0] <= D) return ans + 1;
    return -1;
}
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... |