//
// main.cpp
// xrqrs
//
// Created by Saras Gupta on 06/01/15.
// Copyright (c) 2015 sarasgupta. All rights reserved.
//
#include <iostream>
#include <algorithm>
#include <cmath>
#include <vector>
#define N 524288
//2^19=524288
#define mp(x,y) make_pair(x,y)
using namespace std;
typedef struct node {
int lp[ 2 ] ,rp[ 2 ] ,l,r,ct,mx;
//lp and rp point to the node's left and right child. [0] refers to the segment tree no. and [1] refers to the node number
//l and r are the left and right limits of the node
//ct is the counter for that node
//mx stores the number of elements <=r
} node;
vector< vector< node> > v( 500005 ) ; //array of segment trees
int ctr= 1 ,bin[ 20 ] ;
//ctr is the number of segment trees which are valid
//bin[i]=2^i
//make a base segment tree when there are no elements in the array
void init_tree( int it, int l, int r) {
v[ 0 ] [ it] .l = l;
v[ 0 ] [ it] .r = r;
v[ 0 ] [ it] .ct = 0 ;
v[ 0 ] [ it] .mx = 0 ;
if ( l! = r) {
int mid= ( l+ r) / 2 ;
init_tree( ( 2 * it) + 1 , l, mid) ;
init_tree( ( 2 * it) + 2 , mid+ 1 , r) ;
v[ 0 ] [ it] .lp [ 0 ] = 0 ;
v[ 0 ] [ it] .lp [ 1 ] = ( 2 * it) + 1 ;
v[ 0 ] [ it] .rp [ 0 ] = 0 ;
v[ 0 ] [ it] .rp [ 1 ] = ( 2 * it) + 2 ;
//add the index of the node's left and right child.
}
}
void q0( int it, int l, int r, pair< int , int > itm) {
if ( v[ ctr] [ it] .l == l&& v[ ctr] [ it] .r == r) {
v[ ctr] [ it] .lp [ 0 ] = v[ itm.first ] [ itm.second ] .lp [ 0 ] ;
v[ ctr] [ it] .lp [ 1 ] = v[ itm.first ] [ itm.second ] .lp [ 1 ] ;
v[ ctr] [ it] .rp [ 0 ] = v[ itm.first ] [ itm.second ] .rp [ 0 ] ;
v[ ctr] [ it] .rp [ 1 ] = v[ itm.first ] [ itm.second ] .rp [ 1 ] ;
//since there are no more changes below this node, we can easily point to the children of the previous segment tree as this node's children
v[ ctr] [ it] .l = l;
v[ ctr] [ it] .r = r;
v[ ctr] [ it] .ct = v[ itm.first ] [ itm.second ] .ct + 1 ;
v[ ctr] [ it] .mx = 0 ;
if ( l! = r) {
v[ ctr] [ it] .mx = v[ v[ ctr] [ it] .rp [ 0 ] ] [ v[ ctr] [ it] .rp [ 1 ] ] .mx + v[ v[ ctr] [ it] .rp [ 0 ] ] [ v[ ctr] [ it] .rp [ 1 ] ] .ct ;
}
}
else {
int mid= ( v[ ctr] [ it] .l + v[ ctr] [ it] .r ) / 2 ;
v[ ctr] [ it] .ct = v[ itm.first ] [ itm.second ] .ct ;
if ( l<= mid) {
//this means that we have to add a new node as the left child of this node and update this node accordingly
node tp;
v[ ctr] .push_back ( tp) ;
v[ ctr] [ it] .lp [ 0 ] = ctr;
v[ ctr] [ it] .lp [ 1 ] = v[ ctr] .size ( ) - 1 ;
v[ ctr] [ v[ ctr] .size ( ) - 1 ] .l = v[ ctr] [ it] .l ;
v[ ctr] [ v[ ctr] .size ( ) - 1 ] .r = mid;
q0( v[ ctr] .size ( ) - 1 , l, min( mid, r) , mp( v[ itm.first ] [ itm.second ] .lp [ 0 ] , v[ itm.first ] [ itm.second ] .lp [ 1 ] ) ) ;
}
else {
//else we can just point to the children of the prevous segment tree's node
v[ ctr] [ it] .lp [ 0 ] = v[ itm.first ] [ itm.second ] .lp [ 0 ] ;
v[ ctr] [ it] .lp [ 1 ] = v[ itm.first ] [ itm.second ] .lp [ 1 ] ;
}
if ( r> mid) {
//this means that we have to add a new node as the right child of this node and update this node accordingly
node tp;
tp.l = mid+ 1 ;
tp.r = r;
v[ ctr] .push_back ( tp) ;
v[ ctr] [ it] .rp [ 0 ] = ctr;
v[ ctr] [ it] .rp [ 1 ] = v[ ctr] .size ( ) - 1 ;
v[ ctr] [ v[ ctr] .size ( ) - 1 ] .l = mid+ 1 ;
v[ ctr] [ v[ ctr] .size ( ) - 1 ] .r = v[ ctr] [ it] .r ;
q0( v[ ctr] .size ( ) - 1 , max( mid+ 1 , l) , r, mp( v[ itm.first ] [ itm.second ] .rp [ 0 ] , v[ itm.first ] [ itm.second ] .rp [ 1 ] ) ) ;
}
else {
//else we can just point to the children of the prevous segment tree's node
v[ ctr] [ it] .rp [ 0 ] = v[ itm.first ] [ itm.second ] .rp [ 0 ] ;
v[ ctr] [ it] .rp [ 1 ] = v[ itm.first ] [ itm.second ] .rp [ 1 ] ;
}
v[ ctr] [ it] .mx = v[ v[ ctr] [ it] .rp [ 0 ] ] [ v[ ctr] [ it] .rp [ 1 ] ] .mx + v[ v[ ctr] [ it] .rp [ 0 ] ] [ v[ ctr] [ it] .rp [ 1 ] ] .ct ;
//updating mx of this node
}
}
int q3( pair< int , int > it, int r, int sum) {
if ( v[ it.first ] [ it.second ] .l == r&& v[ it.first ] [ it.second ] .r == r) {
return v[ it.first ] [ it.second ] .ct + sum;
}
else {
int mid= ( v[ it.first ] [ it.second ] .l + v[ it.first ] [ it.second ] .r ) / 2 ;
sum+ = v[ it.first ] [ it.second ] .ct ;
if ( mid>= r) {
return q3( mp( v[ it.first ] [ it.second ] .lp [ 0 ] , v[ it.first ] [ it.second ] .lp [ 1 ] ) , r, sum) ;
}
else {
return q3( mp( v[ it.first ] [ it.second ] .rp [ 0 ] , v[ it.first ] [ it.second ] .rp [ 1 ] ) , r, sum) ;
}
}
}
int q3f( int l, int r, int x) {
//ans(l,r,x)=ans(0, r, x)-ans(0, l-1, x)
int ans= q3( mp( r, 0 ) , x, 0 ) ;
//{r, 0} refers to the index of the root of the segment tree
if ( l> 1 ) {
ans- = q3( mp( l- 1 , 0 ) , x, 0 ) ;
}
return ans;
}
int q4t( int l, int r, int k) {
//Res ipsa loquitur
pair< int , int > il( l- 1 , 0 ) ;
pair< int , int > ir( r, 0 ) ;
int p1= 0 ,p2= 0 ;
while ( v[ ir.first ] [ ir.second ] .r - v[ ir.first ] [ ir.second ] .l >= 1 ) {
p1+ = v[ ir.first ] [ ir.second ] .ct ;
p2+ = v[ il.first ] [ il.second ] .ct ;
int p= v[ v[ ir.first ] [ ir.second ] .lp [ 0 ] ] [ v[ ir.first ] [ ir.second ] .lp [ 1 ] ] .mx + v[ v[ ir.first ] [ ir.second ] .lp [ 0 ] ] [ v[ ir.first ] [ ir.second ] .lp [ 1 ] ] .ct + p1;
int q= v[ v[ il.first ] [ il.second ] .lp [ 0 ] ] [ v[ il.first ] [ il.second ] .lp [ 1 ] ] .mx + v[ v[ il.first ] [ il.second ] .lp [ 0 ] ] [ v[ il.first ] [ il.second ] .lp [ 1 ] ] .ct + p2;
if ( p- q>= k) {
ir= mp( v[ ir.first ] [ ir.second ] .lp [ 0 ] , v[ ir.first ] [ ir.second ] .lp [ 1 ] ) ;
il= mp( v[ il.first ] [ il.second ] .lp [ 0 ] , v[ il.first ] [ il.second ] .lp [ 1 ] ) ;
}
else {
ir= mp( v[ ir.first ] [ ir.second ] .rp [ 0 ] , v[ ir.first ] [ ir.second ] .rp [ 1 ] ) ;
il= mp( v[ il.first ] [ il.second ] .rp [ 0 ] , v[ il.first ] [ il.second ] .rp [ 1 ] ) ;
}
}
return v[ ir.first ] [ ir.second ] .r ;
}
int q1t( int l, int r, int k) {
//if you understand q4t() than you can make this yourself
pair< int , int > il( l- 1 , 0 ) ;
pair< int , int > ir( r, 0 ) ;
int p1= 0 ,p2= 0 ;
int ctrr= r- l+ 1 ;
for ( int i= 18 ; i>= 0 ; i-- ) {
int p= v[ v[ ir.first ] [ ir.second ] .rp [ 0 ] ] [ v[ ir.first ] [ ir.second ] .rp [ 1 ] ] .mx + v[ v[ ir.first ] [ ir.second ] .rp [ 0 ] ] [ v[ ir.first ] [ ir.second ] .rp [ 1 ] ] .ct - ( v[ v[ ir.first ] [ ir.second ] .lp [ 0 ] ] [ v[ ir.first ] [ ir.second ] .lp [ 1 ] ] .mx + v[ v[ ir.first ] [ ir.second ] .lp [ 0 ] ] [ v[ ir.first ] [ ir.second ] .lp [ 1 ] ] .ct ) ;
int q= v[ v[ il.first ] [ il.second ] .rp [ 0 ] ] [ v[ il.first ] [ il.second ] .rp [ 1 ] ] .mx + v[ v[ il.first ] [ il.second ] .rp [ 0 ] ] [ v[ il.first ] [ il.second ] .rp [ 1 ] ] .ct - ( v[ v[ il.first ] [ il.second ] .lp [ 0 ] ] [ v[ il.first ] [ il.second ] .lp [ 1 ] ] .mx + v[ v[ il.first ] [ il.second ] .lp [ 0 ] ] [ v[ il.first ] [ il.second ] .lp [ 1 ] ] .ct ) ;
if ( bin[ i] & k) {
if ( p- q< ctrr) {
ir= mp( v[ ir.first ] [ ir.second ] .lp [ 0 ] , v[ ir.first ] [ ir.second ] .lp [ 1 ] ) ;
il= mp( v[ il.first ] [ il.second ] .lp [ 0 ] , v[ il.first ] [ il.second ] .lp [ 1 ] ) ;
ctrr- = p- q;
}
else {
ir= mp( v[ ir.first ] [ ir.second ] .rp [ 0 ] , v[ ir.first ] [ ir.second ] .rp [ 1 ] ) ;
il= mp( v[ il.first ] [ il.second ] .rp [ 0 ] , v[ il.first ] [ il.second ] .rp [ 1 ] ) ;
ctrr- = ctrr- ( p- q) ;
}
}
else {
if ( p- q== 0 ) {
ir= mp( v[ ir.first ] [ ir.second ] .lp [ 0 ] , v[ ir.first ] [ ir.second ] .lp [ 1 ] ) ;
il= mp( v[ il.first ] [ il.second ] .lp [ 0 ] , v[ il.first ] [ il.second ] .lp [ 1 ] ) ;
ctrr- = p- q;
}
else {
ir= mp( v[ ir.first ] [ ir.second ] .rp [ 0 ] , v[ ir.first ] [ ir.second ] .rp [ 1 ] ) ;
il= mp( v[ il.first ] [ il.second ] .rp [ 0 ] , v[ il.first ] [ il.second ] .rp [ 1 ] ) ;
ctrr- = ctrr- ( p- q) ;
}
}
}
return v[ ir.first ] [ ir.second ] .r ;
}
int main( int argc, const char * argv[ ] ) {
int m,type,l,r,x,cctt= 0 ;
cin >> m;
for ( int i= 0 ; i< 20 ; i++ ) {
bin[ i] = 1 << i;
}
v[ 0 ] .resize ( ( long long int ) ( ( 2 * ( pow ( 2 , ceil ( log2( N) ) ) ) ) - 1 ) ) ;
init_tree( 0 , 0 , N- 1 ) ;
for ( int i= 0 ; i< m; i++ ) {
cin >> type;
if ( type== 0 ) {
cctt++ ;
cin >> x;
v[ ctr] .clear ( ) ;
node tp;
tp.l = 0 ;
tp.r = N- 1 ;
v[ ctr] .push_back ( tp) ;
//initializing the root for this segment tree
q0( 0 , x, N- 1 , mp( ctr- 1 , 0 ) ) ;
ctr++ ;
}
else if ( type== 1 ) {
cin >> l>> r>> x;
cout << q1t( l, r, x) << endl;
}
else if ( type== 2 ) {
cin >> x;
ctr- = x;
//for deleting the last k elements just forget that the last k elements ever existed, it doesn't affect the trees before it.
}
else if ( type== 3 ) {
cin >> l>> r>> x;
cout << q3f( l, r, x) << endl;
}
else {
cin >> l>> r>> x;
cout << q4t( l, r, x) << endl;
}
}
return 0 ;
}
//
//  main.cpp
//  xrqrs
//
//  Created by Saras Gupta on 06/01/15.
//  Copyright (c) 2015 sarasgupta. All rights reserved.
//

#include <iostream>
#include <algorithm>
#include <cmath>
#include <vector>

#define N 524288
//2^19=524288
#define mp(x,y) make_pair(x,y)

using namespace std;

typedef struct node {
    int lp[2],rp[2],l,r,ct,mx;
    //lp and rp point to the node's left and right child. [0] refers to the segment tree no. and [1] refers to the node number
    //l and r are the left and right limits of the node
    //ct is the counter for that node
    //mx stores the number of elements <=r
}node;

vector<vector<node> > v(500005);//array of segment trees
int ctr=1,bin[20];
//ctr is the number of segment trees which are valid
//bin[i]=2^i

//make a base segment tree when there are no elements in the array
void init_tree(int it, int l, int r) {
    v[0][it].l=l;
    v[0][it].r=r;
    v[0][it].ct=0;
    v[0][it].mx=0;
    if (l!=r) {
        int mid=(l+r)/2;
        init_tree((2*it)+1, l, mid);
        init_tree((2*it)+2, mid+1, r);
        v[0][it].lp[0]=0;
        v[0][it].lp[1]=(2*it)+1;
        v[0][it].rp[0]=0;
        v[0][it].rp[1]=(2*it)+2;
        //add the index of the node's left and right child.
    }
}

void q0(int it, int l, int r, pair<int, int> itm) {
    if (v[ctr][it].l==l&&v[ctr][it].r==r) {
        v[ctr][it].lp[0]=v[itm.first][itm.second].lp[0];
        v[ctr][it].lp[1]=v[itm.first][itm.second].lp[1];
        v[ctr][it].rp[0]=v[itm.first][itm.second].rp[0];
        v[ctr][it].rp[1]=v[itm.first][itm.second].rp[1];
        //since there are no more changes below this node, we can easily point to the children of the previous segment tree as this node's children
        v[ctr][it].l=l;
        v[ctr][it].r=r;
        v[ctr][it].ct=v[itm.first][itm.second].ct+1;
        v[ctr][it].mx=0;
        if (l!=r) {
            v[ctr][it].mx=v[v[ctr][it].rp[0]][v[ctr][it].rp[1]].mx+v[v[ctr][it].rp[0]][v[ctr][it].rp[1]].ct;
        }
    }
    else {
        int mid=(v[ctr][it].l+v[ctr][it].r)/2;
        v[ctr][it].ct=v[itm.first][itm.second].ct;
        if (l<=mid) {
            //this means that we have to add a new node as the left child of this node and update this node accordingly
            node tp;
            v[ctr].push_back(tp);
            v[ctr][it].lp[0]=ctr;
            v[ctr][it].lp[1]=v[ctr].size()-1;
            v[ctr][v[ctr].size()-1].l=v[ctr][it].l;
            v[ctr][v[ctr].size()-1].r=mid;
            q0(v[ctr].size()-1, l, min(mid, r), mp(v[itm.first][itm.second].lp[0], v[itm.first][itm.second].lp[1]));
        }
        else {
            //else we can just point to the children of the prevous segment tree's node
            v[ctr][it].lp[0]=v[itm.first][itm.second].lp[0];
            v[ctr][it].lp[1]=v[itm.first][itm.second].lp[1];
        }
        if (r>mid) {
            //this means that we have to add a new node as the right child of this node and update this node accordingly
            node tp;
            tp.l=mid+1;
            tp.r=r;
            v[ctr].push_back(tp);
            v[ctr][it].rp[0]=ctr;
            v[ctr][it].rp[1]=v[ctr].size()-1;
            v[ctr][v[ctr].size()-1].l=mid+1;
            v[ctr][v[ctr].size()-1].r=v[ctr][it].r;
            q0(v[ctr].size()-1, max(mid+1, l), r, mp(v[itm.first][itm.second].rp[0], v[itm.first][itm.second].rp[1]));
        }
        else {
            //else we can just point to the children of the prevous segment tree's node
            v[ctr][it].rp[0]=v[itm.first][itm.second].rp[0];
            v[ctr][it].rp[1]=v[itm.first][itm.second].rp[1];
        }
        v[ctr][it].mx=v[v[ctr][it].rp[0]][v[ctr][it].rp[1]].mx+v[v[ctr][it].rp[0]][v[ctr][it].rp[1]].ct;
        //updating mx of this node
    }
}

int q3(pair<int, int> it, int r, int sum) {
    if (v[it.first][it.second].l==r&&v[it.first][it.second].r==r) {
        return v[it.first][it.second].ct+sum;
    }
    else {
        int mid=(v[it.first][it.second].l+v[it.first][it.second].r)/2;
        sum+=v[it.first][it.second].ct;
        if (mid>=r) {
            return q3(mp(v[it.first][it.second].lp[0], v[it.first][it.second].lp[1]), r, sum);
        }
        else {
            return q3(mp(v[it.first][it.second].rp[0], v[it.first][it.second].rp[1]), r, sum);
        }
    }
}

int q3f(int l, int r, int x) {
    //ans(l,r,x)=ans(0, r, x)-ans(0, l-1, x)
    int ans=q3(mp(r, 0), x, 0);
    //{r, 0} refers to the index of the root of the segment tree
    if (l>1) {
        ans-=q3(mp(l-1, 0), x, 0);
    }
    return ans;
}

int q4t(int l, int r, int k) {
    //Res ipsa loquitur
    pair<int, int> il(l-1, 0);
    pair<int, int> ir(r, 0);
    int p1=0,p2=0;
    while (v[ir.first][ir.second].r-v[ir.first][ir.second].l>=1) {
        p1+=v[ir.first][ir.second].ct;
        p2+=v[il.first][il.second].ct;
        int p=v[v[ir.first][ir.second].lp[0]][v[ir.first][ir.second].lp[1]].mx + v[v[ir.first][ir.second].lp[0]][v[ir.first][ir.second].lp[1]].ct+p1;
        int q=v[v[il.first][il.second].lp[0]][v[il.first][il.second].lp[1]].mx + v[v[il.first][il.second].lp[0]][v[il.first][il.second].lp[1]].ct+p2;
        if (p-q>=k) {
            ir=mp(v[ir.first][ir.second].lp[0], v[ir.first][ir.second].lp[1]);
            il=mp(v[il.first][il.second].lp[0], v[il.first][il.second].lp[1]);
        }
        else {
            ir=mp(v[ir.first][ir.second].rp[0], v[ir.first][ir.second].rp[1]);
            il=mp(v[il.first][il.second].rp[0], v[il.first][il.second].rp[1]);
        }
    }
    return v[ir.first][ir.second].r;
}

int q1t(int l, int r, int k) {
    //if you understand q4t() than you can make this yourself
    pair<int, int> il(l-1, 0);
    pair<int, int> ir(r, 0);
    int p1=0,p2=0;
    int ctrr=r-l+1;
    for (int i=18; i>=0; i--) {
        int p=v[v[ir.first][ir.second].rp[0]][v[ir.first][ir.second].rp[1]].mx + v[v[ir.first][ir.second].rp[0]][v[ir.first][ir.second].rp[1]].ct  -  (v[v[ir.first][ir.second].lp[0]][v[ir.first][ir.second].lp[1]].mx + v[v[ir.first][ir.second].lp[0]][v[ir.first][ir.second].lp[1]].ct);
        int q=v[v[il.first][il.second].rp[0]][v[il.first][il.second].rp[1]].mx + v[v[il.first][il.second].rp[0]][v[il.first][il.second].rp[1]].ct  -  (v[v[il.first][il.second].lp[0]][v[il.first][il.second].lp[1]].mx + v[v[il.first][il.second].lp[0]][v[il.first][il.second].lp[1]].ct);
        if (bin[i]&k) {
            if (p-q<ctrr) {
                ir=mp(v[ir.first][ir.second].lp[0], v[ir.first][ir.second].lp[1]);
                il=mp(v[il.first][il.second].lp[0], v[il.first][il.second].lp[1]);
                ctrr-=p-q;
            }
            else {
                ir=mp(v[ir.first][ir.second].rp[0], v[ir.first][ir.second].rp[1]);
                il=mp(v[il.first][il.second].rp[0], v[il.first][il.second].rp[1]);
                ctrr-=ctrr-(p-q);
            }
        }
        else {
            if (p-q==0) {
                ir=mp(v[ir.first][ir.second].lp[0], v[ir.first][ir.second].lp[1]);
                il=mp(v[il.first][il.second].lp[0], v[il.first][il.second].lp[1]);
                ctrr-=p-q;
            }
            else {
                ir=mp(v[ir.first][ir.second].rp[0], v[ir.first][ir.second].rp[1]);
                il=mp(v[il.first][il.second].rp[0], v[il.first][il.second].rp[1]);
                ctrr-=ctrr-(p-q);
            }
        }
    }
    return v[ir.first][ir.second].r;
}

int main(int argc, const char * argv[]) {
    int m,type,l,r,x,cctt=0;
    cin>>m;
    for (int i=0; i<20; i++) {
        bin[i]=1<<i;
    }
    v[0].resize((long long int)((2*(pow(2, ceil(log2(N)))))-1));
    init_tree(0, 0, N-1);
    for (int i=0; i<m; i++) {
        cin>>type;
        if (type==0) {
            cctt++;
            cin>>x;
            v[ctr].clear();
            node tp;
            tp.l=0;
            tp.r=N-1;
            v[ctr].push_back(tp);
            //initializing the root for this segment tree
            q0(0, x, N-1, mp(ctr-1, 0));
            ctr++;
        }
        else if (type==1) {
            cin>>l>>r>>x;
            cout<<q1t(l, r, x)<<endl;
        }
        else if (type==2) {
            cin>>x;
            ctr-=x;
            //for deleting the last k elements just forget that the last k elements ever existed, it doesn't affect the trees before it.
        }
        else if  (type==3) {
            cin>>l>>r>>x;
            cout<<q3f(l, r, x)<<endl;
        }
        else {
            cin>>l>>r>>x;
            cout<<q4t(l, r, x)<<endl;
        }
    }
    return 0;
}
