#include <bits/stdc++.h>
#define int long long
#define endl '\n'
#define PhTrNghia "minhomes"
using namespace std;
const int maxn = 1e5 + 5 ;
const int inf = 1e18 ;
struct DSU{
int n;
vector < int > parent, sz;
DSU( int _n) {
n = _n;
parent.assign ( n + 5 , 0 ) ;
sz.assign ( n + 5 , 0 ) ;
for ( int i = 1 ; i <= n; i++ ) {
parent[ i] = i;
sz[ i] = 1 ;
}
}
int find_parent( int u) {
if ( u == parent[ u] ) return u;
return parent[ u] = find_parent( parent[ u] ) ;
}
bool Union( int u, int v) {
u = find_parent( u) ;
v = find_parent( v) ;
if ( u == v) return 0 ;
if ( sz[ u] < sz[ v] ) swap( u, v) ;
sz[ u] + = sz[ v] ;
parent[ v] = u;
return 1 ;
}
} ;
struct phtrnghia{
int type, x, y;
} ;
bool cmp( phtrnghia x, phtrnghia y) {
return x.x < y.x ;
}
bool in( int l, int pos, int r) {
return ( l <= pos && pos <= r) ;
}
bool overlap( phtrnghia a, phtrnghia b) {
bool c1 = ( a.x <= b.x && b.x <= a.y ) ;
bool c2 = ( a.x <= b.y && b.y <= a.y ) ;
bool c3 = ( b.x <= a.x && a.x <= b.y ) ;
bool c4 = ( b.x <= a.y && a.y <= b.y ) ;
return ( c1 or c2 or c3 or c4) ;
}
int n, m, q, cc;
int ans[ maxn] , pre[ maxn] , pref[ maxn] , res[ maxn] , pointer_col[ maxn] , mark[ maxn] ;
vector < phtrnghia> close_col[ maxn] , row[ maxn] , col[ maxn] , col_close[ maxn] , col_open[ maxn] ;
signed main( ) {
ios_base:: sync_with_stdio ( false ) ;
cin .tie ( 0 ) ; cout .tie ( 0 ) ;
if ( fopen ( PhTrNghia".INP" , "r" ) ) {
freopen ( PhTrNghia".INP" , "r" , stdin ) ;
freopen ( PhTrNghia".OUT" , "w" , stdout ) ;
}
cin >> n >> m >> q;
cc = q;
for ( int i = 1 ; i <= q; i++ ) {
int x1, y1, x2, y2;
cin >> x1 >> y1 >> x2 >> y2; //line
pre[ x1] ++ ;
if ( x1 == x2) {
row[ x1] .push_back ( { i, y1, y2} ) ;
pref[ x1] + = ( y2 - y1 + 1 ) ;
pref[ x1+ 1 ] - = ( y2 - y1 + 1 ) ;
}
if ( x1 == x2 && y1 == y2) continue ;
if ( y1 == y2) {
col_open[ x1] .push_back ( { i, y1, y2} ) ;
col_close[ x2+ 1 ] .push_back ( { i, y1, y2} ) ;
col[ y1] .push_back ( { i, x1, x2} ) ;
close_col[ x2] .push_back ( { i, y1, y2} ) ;
pref[ x1] ++ ;
pref[ x2+ 1 ] -- ;
}
}
for ( int i = 1 ; i <= n; i++ ) {
pre[ i] + = pre[ i- 1 ] ;
pref[ i] + = pref[ i- 1 ] ;
res[ i] = res[ i- 1 ] + pref[ i] ;
sort( row[ i] .begin ( ) , row[ i] .end ( ) , cmp) ;
sort( col_open[ i] .begin ( ) , col_open[ i] .end ( ) , cmp) ;
sort( close_col[ i] .begin ( ) , close_col[ i] .end ( ) , cmp) ;
sort( col[ i] .begin ( ) , col[ i] .end ( ) , cmp) ;
}
DSU dsu( q) ;
for ( int i = 1 ; i <= n; i++ ) {
int idx3 = 0 , idx4 = 0 ;
for ( phtrnghia cur: col_close[ i] ) {
int pos = cur.x ;
mark[ pos] = 0 ;
}
for ( phtrnghia cur: col_open[ i] ) {
int shape = cur.type ;
int pos = cur.x ;
while ( idx3 < row[ i- 1 ] .size ( ) && row[ i- 1 ] [ idx3] .y <= pos) {
if ( in( row[ i- 1 ] [ idx3] .x , pos, row[ i- 1 ] [ idx3] .y ) ) cc - = dsu.Union ( shape, row[ i- 1 ] [ idx3] .type ) ;
idx3++ ;
}
if ( idx3 < row[ i- 1 ] .size ( ) && in( row[ i- 1 ] [ idx3] .x , pos, row[ i- 1 ] [ idx3] .y ) ) cc - = dsu.Union ( shape, row[ i- 1 ] [ idx3] .type ) ;
while ( idx4 < close_col[ i- 1 ] .size ( ) && close_col[ i- 1 ] [ idx4] .y <= pos) {
if ( in( close_col[ i- 1 ] [ idx4] .x , pos, close_col[ i- 1 ] [ idx4] .y ) ) cc - = dsu.Union ( shape, close_col[ i- 1 ] [ idx4] .type ) ;
idx4++ ;
}
if ( idx4 < close_col[ i- 1 ] .size ( ) && in( close_col[ i- 1 ] [ idx4] .x , pos, close_col[ i- 1 ] [ idx4] .y ) ) cc - = dsu.Union ( shape, close_col[ i- 1 ] [ idx4] .type ) ;
if ( mark[ pos- 1 ] ) cc - = dsu.Union ( shape, mark[ pos- 1 ] ) ;
if ( mark[ pos+ 1 ] ) cc - = dsu.Union ( shape, mark[ pos+ 1 ] ) ;
mark[ pos] = shape;
}
int old_shape = inf, old_r = inf, idx1 = 0 , idx2 = 0 ;
for ( phtrnghia cur: row[ i] ) {
int shape = cur.type ;
int L = cur.x ;
int R = cur.y ;
if ( old_r+ 1 == cur.x ) cc - = dsu.Union ( shape, old_shape) ;
while ( idx1 < row[ i- 1 ] .size ( ) && row[ i- 1 ] [ idx1] .y <= cur.y ) {
if ( overlap( cur, row[ i- 1 ] [ idx1] ) ) cc - = dsu.Union ( shape, row[ i- 1 ] [ idx1] .type ) ;
idx1++ ;
}
if ( idx1 < row[ i- 1 ] .size ( ) && overlap( cur, row[ i- 1 ] [ idx1] ) ) cc - = dsu.Union ( shape, row[ i- 1 ] [ idx1] .type ) ;
while ( idx2 < close_col[ i- 1 ] .size ( ) && close_col[ i- 1 ] [ idx2] .y <= cur.y ) {
if ( overlap( cur, close_col[ i- 1 ] [ idx2] ) ) cc - = dsu.Union ( shape, close_col[ i- 1 ] [ idx2] .type ) ;
idx2++ ;
}
if ( idx2 < close_col[ i- 1 ] .size ( ) && overlap( cur, close_col[ i- 1 ] [ idx2] ) ) cc - = dsu.Union ( shape, close_col[ i- 1 ] [ idx2] .type ) ;
while ( pointer_col[ L- 1 ] < col[ L- 1 ] .size ( ) && col[ L- 1 ] [ pointer_col[ L- 1 ] ] .y <= i) {
if ( in( col[ L- 1 ] [ pointer_col[ L- 1 ] ] .x , i, col[ L- 1 ] [ pointer_col[ L- 1 ] ] .y ) ) cc - = dsu.Union ( shape, col[ L- 1 ] [ pointer_col[ L- 1 ] ] .type ) ;
pointer_col[ L- 1 ] ++ ;
}
if ( pointer_col[ L- 1 ] < col[ L- 1 ] .size ( ) && in( col[ L- 1 ] [ pointer_col[ L- 1 ] ] .x , i, col[ L- 1 ] [ pointer_col[ L- 1 ] ] .y ) ) cc - = dsu.Union ( shape, col[ L- 1 ] [ pointer_col[ L- 1 ] ] .type ) ;
while ( pointer_col[ R+ 1 ] < col[ R+ 1 ] .size ( ) && col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .y <= i) {
if ( in( col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .x , i, col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .y ) ) cc - = dsu.Union ( shape, col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .type ) ;
pointer_col[ R+ 1 ] ++ ;
}
if ( pointer_col[ R+ 1 ] < col[ R+ 1 ] .size ( ) && in( col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .x , i, col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .y ) ) cc - = dsu.Union ( shape, col[ R+ 1 ] [ pointer_col[ R+ 1 ] ] .type ) ;
old_shape = shape;
old_r = cur.y ;
}
ans[ i] = cc - ( pre[ n] - pre[ i] ) ;
cout << res[ i] << " " << ans[ i] << endl;
}
return 0 ;
}
/*
4 3 4
1 1 1 2
3 1 3 2
1 3 2 3
4 1 4 3
5 5 7
1 2 1 3
2 1 2 2
1 4 2 4
2 5 3 5
3 3 3 4
4 2 4 3
4 4 4 5
6 3 3
1 1 1 3
2 2 5 2
6 1 6 3
*/
#include <bits/stdc++.h>
#define int long long
#define endl '\n'
#define PhTrNghia "minhomes"

using namespace std;

const int maxn = 1e5 + 5;
const int inf = 1e18;

struct DSU{
    int n;
    vector <int> parent, sz;
    DSU(int _n){
        n =_n;
        parent.assign(n + 5, 0);
        sz.assign(n + 5, 0);
        for (int i = 1; i <= n; i++){
            parent[i] = i;
            sz[i] = 1;
        }
    }

    int find_parent(int u){
        if (u == parent[u]) return u;
        return parent[u] = find_parent(parent[u]);
    }

    bool Union(int u, int v){
        u = find_parent(u);
        v = find_parent(v);
        if (u == v) return 0;
        if (sz[u] < sz[v]) swap(u, v);
        sz[u] += sz[v];
        parent[v] = u;
        return 1;
    }
};

struct phtrnghia{
    int type, x, y;
};

bool cmp(phtrnghia x, phtrnghia y){
    return x.x < y.x;
}

bool in(int l, int pos, int r){
    return (l <= pos && pos <= r);
}

bool overlap(phtrnghia a, phtrnghia b){
    bool c1 = (a.x <= b.x && b.x <= a.y);
    bool c2 = (a.x <= b.y && b.y <= a.y);
    bool c3 = (b.x <= a.x && a.x <= b.y);
    bool c4 = (b.x <= a.y && a.y <= b.y);
    return (c1 or c2 or c3 or c4);
}

int n, m, q, cc;
int ans[maxn], pre[maxn], pref[maxn], res[maxn], pointer_col[maxn], mark[maxn];
vector <phtrnghia> close_col[maxn], row[maxn], col[maxn], col_close[maxn], col_open[maxn];

signed main(){
    ios_base::sync_with_stdio(false);
    cin.tie(0); cout.tie(0);
    if (fopen(PhTrNghia".INP", "r")){
        freopen(PhTrNghia".INP", "r", stdin);
        freopen(PhTrNghia".OUT", "w", stdout);
    }

    cin >> n >> m >> q;
    cc = q;
    for (int i = 1; i <= q; i++){
        int x1, y1, x2, y2;
        cin >> x1 >> y1 >> x2 >> y2; //line
        pre[x1]++;
        if (x1 == x2){
            row[x1].push_back({i, y1, y2});
            pref[x1] += (y2 - y1 + 1);
            pref[x1+1] -= (y2 - y1 + 1);
        }

        if (x1 == x2 && y1 == y2) continue;

        if (y1 == y2){
            col_open[x1].push_back({i, y1, y2});
            col_close[x2+1].push_back({i, y1, y2});

            col[y1].push_back({i, x1, x2});

            close_col[x2].push_back({i, y1, y2});

            pref[x1]++;
            pref[x2+1]--;
        }
    }

    for (int i = 1; i <= n; i++){
        pre[i] += pre[i-1];
        pref[i] += pref[i-1];
        res[i] = res[i-1] + pref[i];
        sort(row[i].begin(), row[i].end(), cmp);
        sort(col_open[i].begin(), col_open[i].end(), cmp);
        sort(close_col[i].begin(), close_col[i].end(), cmp);
        sort(col[i].begin(), col[i].end(), cmp);
    }

    DSU dsu(q);

    for (int i = 1; i <= n; i++){
        
        int idx3 = 0, idx4 = 0;

        for (phtrnghia cur: col_close[i]){
            int pos = cur.x;
            mark[pos] = 0;
        }

        for (phtrnghia cur: col_open[i]){
            int shape = cur.type;
            int pos = cur.x;

            while (idx3 < row[i-1].size() && row[i-1][idx3].y <= pos){
                if (in(row[i-1][idx3].x, pos, row[i-1][idx3].y)) cc -= dsu.Union(shape, row[i-1][idx3].type);
                idx3++;
            }
            if (idx3 < row[i-1].size() && in(row[i-1][idx3].x, pos, row[i-1][idx3].y)) cc -= dsu.Union(shape, row[i-1][idx3].type);

            while (idx4 < close_col[i-1].size() && close_col[i-1][idx4].y <= pos){
                if (in(close_col[i-1][idx4].x, pos, close_col[i-1][idx4].y)) cc -= dsu.Union(shape, close_col[i-1][idx4].type);
                idx4++;
            }
            if (idx4 < close_col[i-1].size() && in(close_col[i-1][idx4].x, pos, close_col[i-1][idx4].y)) cc -= dsu.Union(shape, close_col[i-1][idx4].type);

            if (mark[pos-1]) cc -= dsu.Union(shape, mark[pos-1]);
            if (mark[pos+1]) cc -= dsu.Union(shape, mark[pos+1]);
            mark[pos] = shape;
        }


        int old_shape = inf, old_r = inf, idx1 = 0, idx2 = 0;
        for (phtrnghia cur: row[i]){
            int shape = cur.type;
            int L = cur.x;
            int R = cur.y;
            
            if (old_r+1 == cur.x) cc -= dsu.Union(shape, old_shape);

            while (idx1 < row[i-1].size() && row[i-1][idx1].y <= cur.y){
                if (overlap(cur, row[i-1][idx1])) cc -= dsu.Union(shape, row[i-1][idx1].type);
                idx1++;
            }

            if (idx1 < row[i-1].size() && overlap(cur, row[i-1][idx1])) cc -= dsu.Union(shape, row[i-1][idx1].type);

            while (idx2 < close_col[i-1].size() && close_col[i-1][idx2].y <= cur.y){
                if (overlap(cur, close_col[i-1][idx2])) cc -= dsu.Union(shape, close_col[i-1][idx2].type);
                idx2++;
            }
            if (idx2 < close_col[i-1].size() && overlap(cur, close_col[i-1][idx2])) cc -= dsu.Union(shape, close_col[i-1][idx2].type);

            while (pointer_col[L-1] < col[L-1].size() && col[L-1][pointer_col[L-1]].y <= i){
                if (in(col[L-1][pointer_col[L-1]].x, i, col[L-1][pointer_col[L-1]].y)) cc -= dsu.Union(shape, col[L-1][pointer_col[L-1]].type);
                pointer_col[L-1]++;
            }
            if (pointer_col[L-1] < col[L-1].size() && in(col[L-1][pointer_col[L-1]].x, i, col[L-1][pointer_col[L-1]].y)) cc -= dsu.Union(shape, col[L-1][pointer_col[L-1]].type);

            while (pointer_col[R+1] < col[R+1].size() && col[R+1][pointer_col[R+1]].y <= i){
                if (in(col[R+1][pointer_col[R+1]].x, i, col[R+1][pointer_col[R+1]].y)) cc -= dsu.Union(shape, col[R+1][pointer_col[R+1]].type);
                pointer_col[R+1]++;
            }
            if (pointer_col[R+1] < col[R+1].size() && in(col[R+1][pointer_col[R+1]].x, i, col[R+1][pointer_col[R+1]].y)) cc -= dsu.Union(shape, col[R+1][pointer_col[R+1]].type);

            old_shape = shape;
            old_r = cur.y;
        }

        ans[i] = cc - (pre[n] - pre[i]);
        cout << res[i] << " " << ans[i] << endl;
    }

    return 0;
}
/*
4 3 4
1 1 1 2
3 1 3 2
1 3 2 3
4 1 4 3

5 5 7
1 2 1 3
2 1 2 2
1 4 2 4
2 5 3 5
3 3 3 4
4 2 4 3
4 4 4 5

6 3 3
1 1 1 3
2 2 5 2
6 1 6 3
*/
