#include<bits/stdc++.h>
using namespace std;
#define F first
#define S second
#define pb push_back
#define vll vector<ll>
#define pll pair<ll, ll>
typedef int ll;
const ll mxN=5e4+5;
const ll LOG=21;
ll dsu[mxN];
ll ans[mxN];
ll svid[mxN][LOG];
ll find_set(ll tar){
if(dsu[tar]<0) return tar;
return dsu[tar]=find_set(dsu[tar]);
}
void unite(ll a, ll b){
ll f=find_set(a);
ll s=find_set(b);
assert(f!=s);
if(abs(dsu[f])<abs(dsu[s])) swap(f, s);
dsu[f]+=dsu[s];
ans[f] += ans[s];
dsu[s]=f;
}
struct segtree1{
vector<pll> tree;
ll treelen;
void init(ll siz){
treelen=siz+1;
while(__builtin_popcount(treelen)!=1) treelen++;
tree=vector<pll> (2*treelen, make_pair(0, 0));
}
void add1(ll idx, ll low, ll high, ll qlow, ll qhigh, ll id){
if(low>=qlow && high<=qhigh){
if(tree[idx].S!=0){
if(find_set(tree[idx].F)!=find_set(id)){
unite(tree[idx].F, id);
}
}
tree[idx].F=id;
tree[idx].S++;
return;
}
if(low>qhigh || high<qlow){
return;
}
ll mid=(low+high)/2;
add1(2*idx, low, mid, qlow, qhigh, id);
add1(2*idx+1, mid+1, high, qlow, qhigh, id);
}
void add(ll qlow, ll qhigh, ll id){
add1(1, 0, treelen-1, qlow, qhigh, id);
}
void rem1(ll idx, ll low, ll high, ll qlow, ll qhigh, ll id){
if(low>=qlow && high<=qhigh){
tree[idx].S--;
return;
}
if(low>qhigh || high<qlow){
return;
}
ll mid=(low+high)/2;
rem1(2*idx, low, mid, qlow, qhigh, id);
rem1(2*idx+1, mid+1, high, qlow, qhigh, id);
}
void rem(ll qlow, ll qhigh, ll id){
rem1(1, 0, treelen-1, qlow, qhigh, id);
}
void qry(ll idx, ll id){
ll tar=idx+treelen;
while(tar>0){
if(tree[tar].S>0){
if(find_set(tree[tar].F)!=find_set(id)){
unite(tree[tar].F, id);
}
}
tar/=2;
}
}
};
struct segtree2{
vector<vector<pll>> tree;
ll treelen;
ll d;
void init(ll siz){
treelen=siz+1;
while(__builtin_popcount(treelen)!=1) treelen++;
d=__builtin_ctz(treelen);
tree=vector<vector<pll>> (2*treelen, vector<pll>());
}
void add(ll idx, ll id){
// cout<<"adding "<<idx<<' '<<id<<'\n';
ll tar=idx+treelen;
ll dep=d;
while(tar>0){
svid[id][dep]=(ll) tree[tar].size();
// cout<<"dep: "<<dep<<'\n';
// cout<<"svid["<<id<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
tree[tar].pb({id, 1});
tar/=2;
dep--;
}
assert(dep==-1);
}
void rem(ll idx, ll id){
ll tar=idx+treelen;
ll dep=d;
while(tar>0){
assert(svid[id][dep]<tree[tar].size());
// cout<<"svid["<<id<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
tree[tar][svid[id][dep]].S--;
tar/=2;
dep--;
}
assert(dep==-1);
}
void qry1(ll idx, ll low, ll high, ll qlow, ll qhigh, ll id, ll dep){
if(low>=qlow && high<=qhigh){
ll siz=0;
ll dumb=0;
for(auto &[x, y]:tree[idx]){
if(y>0){
dumb=x;
svid[x][dep]=0;
// cout<<"svid["<<x<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
if(find_set(x)!=find_set(id)){
unite(x, id);
}
}
siz+=y;
}
tree[idx].clear();
if(siz>0) tree[idx].pb({dumb, siz});
// svid[id][dep]=0;
// cout<<"svid["<<id<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
return;
}
if(low>qhigh || high<qlow){
return;
}
ll mid=(low+high)/2;
qry1(2*idx, low, mid, qlow, qhigh, id, dep+1);
qry1(2*idx+1, mid+1, high, qlow, qhigh, id, dep+1);
}
void qry(ll qlow, ll qhigh, ll id){
qry1(1, 0, treelen-1, qlow, qhigh, id, 0);
}
};
ll n;
vll con;
segtree1 seg1;
segtree2 seg2;
vector<array<ll, 3>> v;
ll siz;
ll id(ll tar){
return lower_bound(con.begin(), con.end(), tar)-con.begin();
}
bool compare(array<ll, 3> a, array<ll, 3> b){
if(a[0]!=b[0]){
return a[0]<b[0];
}
if(a[1]!=b[1]){
return a[1]>b[1];
}
return a[2]<b[2];
}
int solve(int N){
memset(dsu, -1, sizeof(dsu));
for(int i = 0; i < mxN; i++){
ans[i] = 0;
}
memset(svid, 0, sizeof(svid));
n=N;
vector<int>a(n), b(n), c(n), d(n);
for(ll i=0;i<n;i++){
int x, y, h, w;
cin >> x >> y >> h >> w;
a[i] = x;
c[i] = x + h;
b[i] = y;
d[i] = y + w;
con.pb(b[i]);
con.pb(d[i]);
v.pb({a[i], 1, i});
v.pb({c[i], -1, i});
ans[i] = h * w;
}
sort(con.begin(), con.end());
con.erase(unique(con.begin(), con.end()), con.end());
sort(v.begin(), v.end(), compare);
siz=(ll) con.size();
seg1.init(siz);
seg2.init(siz);
for(auto &it:v){
ll cur=it[2];
if(it[1]==1){
seg1.qry(id(b[cur]), cur);
seg1.qry(id(d[cur]), cur);
seg2.qry(id(b[cur]), id(d[cur]), cur);
seg1.add(id(b[cur]), id(d[cur]), cur);
seg2.add(id(b[cur]), cur);
// seg2.add(id(d[cur]), cur);
}
else{
assert(it[1]==-1);
seg1.rem(id(b[cur]), id(d[cur]), cur);
seg2.rem(id(b[cur]), cur);
// seg2.rem(id(d[cur]), cur);
}
}
con.clear();
for(ll i=0;i<n;i++){
con.pb(find_set(i));
}
sort(con.begin(), con.end());
con.erase(unique(con.begin(), con.end()), con.end());
vll re(n);
int an = 0;
for(ll i=0;i<n;i++){
an = max(an, ans[find_set(i)]);
}
return an;
}
int main(){
int n;
while(cin >> n){
cout << solve(n) << '\n';
}
}
#include<bits/stdc++.h>
using namespace std;

#define F first
#define S second
#define pb push_back
#define vll vector<ll>
#define pll pair<ll, ll>

typedef int ll;

const ll mxN=5e4+5;
const ll LOG=21;

ll dsu[mxN];
ll ans[mxN];
ll svid[mxN][LOG];

ll find_set(ll tar){
    if(dsu[tar]<0) return tar;
    return dsu[tar]=find_set(dsu[tar]);
}

void unite(ll a, ll b){
    ll f=find_set(a);
    ll s=find_set(b);
    assert(f!=s);
    if(abs(dsu[f])<abs(dsu[s])) swap(f, s);
    dsu[f]+=dsu[s];
    ans[f] += ans[s];
    dsu[s]=f;
}

struct segtree1{
    vector<pll> tree;
    ll treelen;

    void init(ll siz){
        treelen=siz+1;
        while(__builtin_popcount(treelen)!=1) treelen++;
        tree=vector<pll> (2*treelen, make_pair(0, 0));
    }

    void add1(ll idx, ll low, ll high, ll qlow, ll qhigh, ll id){
        if(low>=qlow && high<=qhigh){
            if(tree[idx].S!=0){
                if(find_set(tree[idx].F)!=find_set(id)){
                    unite(tree[idx].F, id);
                }
            }
            tree[idx].F=id;
            tree[idx].S++;
            return;
        }
        if(low>qhigh || high<qlow){
            return;
        }
        ll mid=(low+high)/2;
        add1(2*idx, low, mid, qlow, qhigh, id);
        add1(2*idx+1, mid+1, high, qlow, qhigh, id);
    }

    void add(ll qlow, ll qhigh, ll id){
        add1(1, 0, treelen-1, qlow, qhigh, id);
    }

    void rem1(ll idx, ll low, ll high, ll qlow, ll qhigh, ll id){
        if(low>=qlow && high<=qhigh){
            tree[idx].S--;
            return;
        }
        if(low>qhigh || high<qlow){
            return;
        }
        ll mid=(low+high)/2;
        rem1(2*idx, low, mid, qlow, qhigh, id);
        rem1(2*idx+1, mid+1, high, qlow, qhigh, id);
    }

    void rem(ll qlow, ll qhigh, ll id){
        rem1(1, 0, treelen-1, qlow, qhigh, id);
    }

    void qry(ll idx, ll id){
        ll tar=idx+treelen;
        while(tar>0){
            if(tree[tar].S>0){
                if(find_set(tree[tar].F)!=find_set(id)){
                    unite(tree[tar].F, id);
                }
            }
            tar/=2;
        }
    }
};

struct segtree2{
    vector<vector<pll>> tree;
    ll treelen;
    ll d;

    void init(ll siz){
        treelen=siz+1;
        while(__builtin_popcount(treelen)!=1) treelen++;
        d=__builtin_ctz(treelen);
        tree=vector<vector<pll>> (2*treelen, vector<pll>());
    }

    void add(ll idx, ll id){
        // cout<<"adding "<<idx<<' '<<id<<'\n';
        ll tar=idx+treelen;
        ll dep=d;
        while(tar>0){
            svid[id][dep]=(ll) tree[tar].size();
            // cout<<"dep: "<<dep<<'\n';
            // cout<<"svid["<<id<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n'; 
            tree[tar].pb({id, 1});
            tar/=2;
            dep--;
        }
        assert(dep==-1);
    }

    void rem(ll idx, ll id){
        ll tar=idx+treelen;
        ll dep=d;
        while(tar>0){
            assert(svid[id][dep]<tree[tar].size());
            // cout<<"svid["<<id<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
            tree[tar][svid[id][dep]].S--;
            tar/=2;
            dep--;
        }
        assert(dep==-1);
    }

    void qry1(ll idx, ll low, ll high, ll qlow, ll qhigh, ll id, ll dep){
        if(low>=qlow && high<=qhigh){
            ll siz=0;
            ll dumb=0;
            for(auto &[x, y]:tree[idx]){
                if(y>0){
                    dumb=x;
                    svid[x][dep]=0;
                    // cout<<"svid["<<x<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
                    if(find_set(x)!=find_set(id)){
                        unite(x, id);
                    }
                }
                siz+=y;
            }
            tree[idx].clear();
            if(siz>0) tree[idx].pb({dumb, siz});
            // svid[id][dep]=0;
            // cout<<"svid["<<id<<"]["<<dep<<"] : "<<svid[id][dep]<<'\n';
            return;
        }
        if(low>qhigh || high<qlow){
            return;
        }
        ll mid=(low+high)/2;
        qry1(2*idx, low, mid, qlow, qhigh, id, dep+1);
        qry1(2*idx+1, mid+1, high, qlow, qhigh, id, dep+1);
    }

    void qry(ll qlow, ll qhigh, ll id){
        qry1(1, 0, treelen-1, qlow, qhigh, id, 0);
    }
};

ll n;
vll con;
segtree1 seg1;
segtree2 seg2;
vector<array<ll, 3>> v;
ll siz;

ll id(ll tar){
    return lower_bound(con.begin(), con.end(), tar)-con.begin();
}

bool compare(array<ll, 3> a, array<ll, 3> b){
    if(a[0]!=b[0]){
        return a[0]<b[0];
    }
    if(a[1]!=b[1]){
        return a[1]>b[1];
    }
    return a[2]<b[2];
}

int solve(int N){
    memset(dsu, -1, sizeof(dsu));
    for(int i = 0; i < mxN; i++){
        ans[i] = 0;
    }
    memset(svid, 0, sizeof(svid));
    n=N;
    vector<int>a(n), b(n), c(n), d(n);
    for(ll i=0;i<n;i++){
        int x, y, h, w;
        cin >> x >> y >> h >> w;
        a[i] = x;
        c[i] = x + h;
        b[i] = y;
        d[i] = y + w;
        con.pb(b[i]);
        con.pb(d[i]);
        v.pb({a[i], 1, i});
        v.pb({c[i], -1, i});
        ans[i] = h * w;
    }
    sort(con.begin(), con.end());
    con.erase(unique(con.begin(), con.end()), con.end());
    sort(v.begin(), v.end(), compare);
    siz=(ll) con.size();
    seg1.init(siz);
    seg2.init(siz);
    for(auto &it:v){
        ll cur=it[2];
        if(it[1]==1){
            seg1.qry(id(b[cur]), cur);
            seg1.qry(id(d[cur]), cur);
            seg2.qry(id(b[cur]), id(d[cur]), cur);
            seg1.add(id(b[cur]), id(d[cur]), cur);
            seg2.add(id(b[cur]), cur);
            // seg2.add(id(d[cur]), cur);
        }
        else{
            assert(it[1]==-1);
            seg1.rem(id(b[cur]), id(d[cur]), cur);
            seg2.rem(id(b[cur]), cur);
            // seg2.rem(id(d[cur]), cur);
        }
    }
    con.clear();
    for(ll i=0;i<n;i++){
        con.pb(find_set(i));
    }
    sort(con.begin(), con.end());
    con.erase(unique(con.begin(), con.end()), con.end());
    vll re(n);
    int an = 0;
    for(ll i=0;i<n;i++){
        an = max(an, ans[find_set(i)]);
    }
    return an;
}
int main(){
    int n;
    while(cin >> n){
        cout << solve(n) << '\n';
    }
}