#include <bits/stdc++.h>
#define SPED \
ios_base::sync_with_stdio(false); \
cin.tie(0); \
cout.tie(0);
#define endl "\n"
#define fi first
#define se second
#define lint long long
#define fami signed
#define lore main
#define freefire freopen
#define rect tuple<lint, lint, lint, lint>
#define Data tuple<lint, bool, int>
const lint INF = 0x1f1f1f1f1f1f1f1f;
const lint NEG = 0xE1E1E1E1E1E1E1E1;
using namespace std;
struct Button
{
lint X, Y, CX, CY, ID; // tọa độ và tạo độ đc nén,
int idH, idC; // idhàng là id của cột Y trên hàng CX
// idcột là id của hàng X trên cột CY
};
int t, p, q, dp[2005][2005], hpref[2005][2005], cpref[2005][2005];
lint m, n, dist[2][100005];
Button a[100005];
rect b[100005];
vector<pair<lint, int>> adjh[200005],
adjc[200005]; // adjhang, adjcot, nghia la cac tk chung hang thi chung 1 vector, chug cot thi chung 1 vector
void reset()
{
for (int i = 1; i <= p + 2; i++)
{
adjh[i].clear();
adjc[i].clear();
}
}
void reset1()
{
for (int i = 1; i <= p + 2; i++)
{
adjh[i].clear();
adjc[i].clear();
}
for (int i = 0; i <= m + 2; i++)
for (int j = 0; j <= n + 2; j++)
dp[i][j] = 0;
}
/*
quy ước : 0 là UD
1 là LR
*/
inline bool bR(lint X, lint yA, lint yB)
{
lint L = min(yA, yB), R = max(yA, yB);
for (int j = 1; j <= q; ++j)
{
auto [x1, y1, x2, y2] = b[j];
if (x1 <= X and X <= x2)
{
if (!(R < y1 or y2 < L))
return true;
}
}
return false;
}
inline bool bC(lint Y, lint xA, lint xB)
{
lint L = min(xA, xB), R = max(xA, xB);
for (int j = 1; j <= q; ++j)
{
auto [x1, y1, x2, y2] = b[j];
if (y1 <= Y and Y <= y2)
{
if (!(R < x1 or x2 < L))
return true;
}
}
return false;
}
void dijkstra()
{
fill(dist[0], dist[0] + p + 2, INF);
fill(dist[1], dist[1] + p + 2, INF);
dist[0][0] = 0;
priority_queue<Data, vector<Data>, greater<Data>> pq;
pq.emplace(0, 0, 0);
while (!pq.empty())
{
auto [du, mode, u] = pq.top();
pq.pop();
if (du > dist[mode][u])
continue;
auto [idx, idy, cidx, cidy, id, idh, idc] = a[u];
if (1 <= u and u <= p)
{
if (dist[!mode][u] > dist[mode][u] + 1)
{
dist[!mode][u] = dist[mode][u] + 1;
pq.emplace(dist[!mode][u], !mode, u);
}
}
if (mode == 1)
{
if (idh > 0)
{
auto [w_adj, id_adj] = adjh[cidx][idh - 1];
if (!bR(idx, idy, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
if (idh < adjh[cidx].size() - 1)
{
auto [w_adj, id_adj] = adjh[cidx][idh + 1];
if (!bR(idx, idy, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
}
else
{
if (idc > 0)
{
auto [w_adj, id_adj] = adjc[cidy][idc - 1];
if (!bC(idy, idx, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
if (idc < adjc[cidy].size() - 1)
{
auto [w_adj, id_adj] = adjc[cidy][idc + 1];
if (!bC(idy, idx, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
}
}
}
void dijkstra1()
{
fill(dist[0], dist[0] + p + 2, INF);
fill(dist[1], dist[1] + p + 2, INF);
dist[0][0] = 0;
priority_queue<Data, vector<Data>, greater<Data>> pq;
pq.emplace(0, 0, 0);
while (!pq.empty())
{
auto [du, mode, u] = pq.top();
pq.pop();
if (du > dist[mode][u])
continue;
auto [idx, idy, cidx, cidy, id, idh, idc] = a[u];
if (1 <= u and u <= p)
{
if (dist[!mode][u] > dist[mode][u] + 1)
{
dist[!mode][u] = dist[mode][u] + 1;
pq.emplace(dist[!mode][u], !mode, u);
}
}
if (mode == 1)
{
if (idh > 0)
{
auto [w_adj, id_adj] = adjh[cidx][idh - 1];
lint XX = a[id_adj].X;
lint YY = a[id_adj].Y;
if (!(hpref[XX][idy] - hpref[XX][YY - 1]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
if (idh < adjh[cidx].size() - 1)
{
auto [w_adj, id_adj] = adjh[cidx][idh + 1];
lint XX = a[id_adj].X;
lint YY = a[id_adj].Y;
if (!(hpref[XX][YY] - hpref[XX][idy - 1]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
}
else
{
if (idc > 0)
{
auto [w_adj, id_adj] = adjc[cidy][idc - 1];
lint XX = a[id_adj].X;
lint YY = a[id_adj].Y;
if (!(cpref[idx][YY] - cpref[XX - 1][YY]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
if (idc < adjc[cidy].size() - 1)
{
auto [w_adj, id_adj] = adjc[cidy][idc + 1];
lint XX = a[id_adj].X;
lint YY = a[id_adj].Y;
if (!(cpref[XX][YY] - cpref[idx - 1][YY]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
{
dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
pq.emplace(dist[mode][id_adj], mode, id_adj);
}
}
}
}
}
void compress()
{
vector<pair<lint, int>> vx, vy;
for (int i = 0; i <= p + 1; i++)
vx.emplace_back(a[i].X, i);
for (int i = 0; i <= p + 1; i++)
vy.emplace_back(a[i].Y, i);
sort(vx.begin(), vx.end());
sort(vy.begin(), vy.end());
int cx = 0, cy = 0;
for (int i = 0; i < vx.size(); i++)
{
if (i == 0 or vx[i].fi != vx[i - 1].fi)
++cx;
a[vx[i].se].CX = cx;
}
for (int i = 0; i < vy.size(); i++)
{
if (i == 0 or vy[i].fi != vy[i - 1].fi)
++cy;
a[vy[i].se].CY = cy;
}
}
fami lore()
{
if (fopen("starofhope.inp", "r"))
{
freefire("starofhope.inp", "r", stdin);
freefire("starofhope.out", "w", stdout);
}
SPED;
cin >> t;
while (t--)
{
cin >> m >> n >> p >> q;
a[0] = {1, 1, 0, 0, 0, 0, 0};
for (int i = 1; i <= p; i++)
{
int N, E;
cin >> N >> E;
a[i] = {N, E, 0, 0, i, 0, 0};
}
a[p + 1] = {m, n, 0, 0, p + 1, 0, 0};
for (int i = 1; i <= q; i++)
{
lint N, I, G, A;
cin >> N >> I >> G >> A;
b[i] = {N, I, G, A};
}
compress();
for (int i = 0; i <= p + 1; i++)
{
auto [idx, idy, cidx, cidy, id, idh, idc] = a[i];
adjh[cidx].emplace_back(idy, id);
adjc[cidy].emplace_back(idx, id);
}
for (int i = 1; i <= p + 2; i++)
{
sort(adjh[i].begin(), adjh[i].end());
sort(adjc[i].begin(), adjc[i].end());
}
for (int id = 1; id <= p + 2; id++)
{
for (int i = 0; i < adjh[id].size(); i++)
{
auto [wc, idx] = adjh[id][i];
a[idx].idH = i;
}
for (int i = 0; i < adjc[id].size(); i++)
{
auto [wh, idx] = adjc[id][i];
a[idx].idC = i;
}
}
if (m <= 2000 and n <= 2000 and q > 10)
{
for (int i = 1; i <= q; ++i)
{
auto [x1, y1, x2, y2] = b[i];
++dp[x1][y1];
--dp[x2 + 1][y1];
--dp[x1][y2 + 1];
++dp[x2 + 1][y2 + 1];
}
for (int i = 1; i <= m + 1; ++i)
for (int j = 1; j <= n + 1; ++j)
dp[i][j] += dp[i - 1][j] + dp[i][j - 1] - dp[i - 1][j - 1];
for (int i = 1; i <= m; ++i)
for (int j = 1; j <= n; ++j)
dp[i][j] = (dp[i][j] > 0);
for (int i = 1; i <= m; ++i)
{
hpref[i][0] = 0;
for (int j = 1; j <= n; ++j)
hpref[i][j] = hpref[i][j - 1] + dp[i][j];
}
for (int j = 1; j <= n; ++j)
{
cpref[0][j] = 0;
for (int i = 1; i <= m; ++i)
cpref[i][j] = cpref[i - 1][j] + dp[i][j];
}
dijkstra1();
lint res = min(dist[0][p + 1], dist[1][p + 1]);
cout << (res == INF ? -1 : res) << endl;
reset1();
}
else
{
dijkstra();
lint res = min(dist[0][p + 1], dist[1][p + 1]);
cout << (res == INF ? -1 : res) << endl;
reset();
}
}
}
// Let your soul wander where dreams are born.
#include <bits/stdc++.h>

#define SPED                                                                                                           \
    ios_base::sync_with_stdio(false);                                                                                  \
    cin.tie(0);                                                                                                        \
    cout.tie(0);

#define endl "\n"
#define fi first
#define se second
#define lint long long
#define fami signed
#define lore main
#define freefire freopen
#define rect tuple<lint, lint, lint, lint>
#define Data tuple<lint, bool, int>

const lint INF = 0x1f1f1f1f1f1f1f1f;
const lint NEG = 0xE1E1E1E1E1E1E1E1;

using namespace std;

struct Button
{
    lint X, Y, CX, CY, ID; // tọa độ và tạo độ đc nén,
    int idH, idC;          // idhàng là id của cột Y trên hàng CX
                           // idcột là id của hàng X trên cột CY
};

int t, p, q, dp[2005][2005], hpref[2005][2005], cpref[2005][2005];
lint m, n, dist[2][100005];
Button a[100005];
rect b[100005];
vector<pair<lint, int>> adjh[200005],
    adjc[200005]; // adjhang, adjcot, nghia la cac tk chung hang thi chung 1 vector, chug cot thi chung 1 vector

void reset()
{
    for (int i = 1; i <= p + 2; i++)
    {
        adjh[i].clear();
        adjc[i].clear();
    }
}

void reset1()
{
    for (int i = 1; i <= p + 2; i++)
    {
        adjh[i].clear();
        adjc[i].clear();
    }
    for (int i = 0; i <= m + 2; i++)
        for (int j = 0; j <= n + 2; j++)
            dp[i][j] = 0;
}

/*
    quy ước : 0 là UD
         1 là LR
*/

inline bool bR(lint X, lint yA, lint yB)
{
    lint L = min(yA, yB), R = max(yA, yB);
    for (int j = 1; j <= q; ++j)
    {
        auto [x1, y1, x2, y2] = b[j];
        if (x1 <= X and X <= x2)
        {
            if (!(R < y1 or y2 < L))
                return true;
        }
    }
    return false;
}

inline bool bC(lint Y, lint xA, lint xB)
{
    lint L = min(xA, xB), R = max(xA, xB);
    for (int j = 1; j <= q; ++j)
    {
        auto [x1, y1, x2, y2] = b[j];
        if (y1 <= Y and Y <= y2)
        {
            if (!(R < x1 or x2 < L))
                return true;
        }
    }
    return false;
}

void dijkstra()
{
    fill(dist[0], dist[0] + p + 2, INF);
    fill(dist[1], dist[1] + p + 2, INF);

    dist[0][0] = 0;

    priority_queue<Data, vector<Data>, greater<Data>> pq;

    pq.emplace(0, 0, 0);

    while (!pq.empty())
    {
        auto [du, mode, u] = pq.top();
        pq.pop();
        if (du > dist[mode][u])
            continue;

        auto [idx, idy, cidx, cidy, id, idh, idc] = a[u];

        if (1 <= u and u <= p)
        {
            if (dist[!mode][u] > dist[mode][u] + 1)
            {
                dist[!mode][u] = dist[mode][u] + 1;
                pq.emplace(dist[!mode][u], !mode, u);
            }
        }

        if (mode == 1)
        {
            if (idh > 0)
            {
                auto [w_adj, id_adj] = adjh[cidx][idh - 1];
                if (!bR(idx, idy, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
            if (idh < adjh[cidx].size() - 1)
            {
                auto [w_adj, id_adj] = adjh[cidx][idh + 1];
                if (!bR(idx, idy, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
        }
        else
        {
            if (idc > 0)
            {
                auto [w_adj, id_adj] = adjc[cidy][idc - 1];
                if (!bC(idy, idx, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
            if (idc < adjc[cidy].size() - 1)
            {
                auto [w_adj, id_adj] = adjc[cidy][idc + 1];
                if (!bC(idy, idx, w_adj) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
        }
    }
}

void dijkstra1()
{
    fill(dist[0], dist[0] + p + 2, INF);
    fill(dist[1], dist[1] + p + 2, INF);

    dist[0][0] = 0;

    priority_queue<Data, vector<Data>, greater<Data>> pq;

    pq.emplace(0, 0, 0);

    while (!pq.empty())
    {
        auto [du, mode, u] = pq.top();
        pq.pop();
        if (du > dist[mode][u])
            continue;

        auto [idx, idy, cidx, cidy, id, idh, idc] = a[u];

        if (1 <= u and u <= p)
        {
            if (dist[!mode][u] > dist[mode][u] + 1)
            {
                dist[!mode][u] = dist[mode][u] + 1;
                pq.emplace(dist[!mode][u], !mode, u);
            }
        }

        if (mode == 1)
        {
            if (idh > 0)
            {
                auto [w_adj, id_adj] = adjh[cidx][idh - 1];
                lint XX = a[id_adj].X;
                lint YY = a[id_adj].Y;
                if (!(hpref[XX][idy] - hpref[XX][YY - 1]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
            if (idh < adjh[cidx].size() - 1)
            {
                auto [w_adj, id_adj] = adjh[cidx][idh + 1];
                lint XX = a[id_adj].X;
                lint YY = a[id_adj].Y;
                if (!(hpref[XX][YY] - hpref[XX][idy - 1]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idy))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idy);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
        }
        else
        {
            if (idc > 0)
            {
                auto [w_adj, id_adj] = adjc[cidy][idc - 1];
                lint XX = a[id_adj].X;
                lint YY = a[id_adj].Y;
                if (!(cpref[idx][YY] - cpref[XX - 1][YY]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
            if (idc < adjc[cidy].size() - 1)
            {
                auto [w_adj, id_adj] = adjc[cidy][idc + 1];
                lint XX = a[id_adj].X;
                lint YY = a[id_adj].Y;
                if (!(cpref[XX][YY] - cpref[idx - 1][YY]) and dist[mode][id_adj] > dist[mode][u] + abs(w_adj - idx))
                {
                    dist[mode][id_adj] = dist[mode][u] + abs(w_adj - idx);
                    pq.emplace(dist[mode][id_adj], mode, id_adj);
                }
            }
        }
    }
}

void compress()

{
    vector<pair<lint, int>> vx, vy;
    for (int i = 0; i <= p + 1; i++)
        vx.emplace_back(a[i].X, i);
    for (int i = 0; i <= p + 1; i++)
        vy.emplace_back(a[i].Y, i);
    sort(vx.begin(), vx.end());
    sort(vy.begin(), vy.end());
    int cx = 0, cy = 0;
    for (int i = 0; i < vx.size(); i++)
    {
        if (i == 0 or vx[i].fi != vx[i - 1].fi)
            ++cx;
        a[vx[i].se].CX = cx;
    }
    for (int i = 0; i < vy.size(); i++)
    {
        if (i == 0 or vy[i].fi != vy[i - 1].fi)
            ++cy;
        a[vy[i].se].CY = cy;
    }
}

fami lore()
{
    if (fopen("starofhope.inp", "r"))
    {
        freefire("starofhope.inp", "r", stdin);
        freefire("starofhope.out", "w", stdout);
    }
    SPED;
    cin >> t;
    while (t--)
    {
        cin >> m >> n >> p >> q;

        a[0] = {1, 1, 0, 0, 0, 0, 0};
        for (int i = 1; i <= p; i++)
        {
            int N, E;
            cin >> N >> E;
            a[i] = {N, E, 0, 0, i, 0, 0};
        }
        a[p + 1] = {m, n, 0, 0, p + 1, 0, 0};

        for (int i = 1; i <= q; i++)
        {
            lint N, I, G, A;
            cin >> N >> I >> G >> A;
            b[i] = {N, I, G, A};
        }

        compress();

        for (int i = 0; i <= p + 1; i++)
        {
            auto [idx, idy, cidx, cidy, id, idh, idc] = a[i];
            adjh[cidx].emplace_back(idy, id);
            adjc[cidy].emplace_back(idx, id);
        }

        for (int i = 1; i <= p + 2; i++)
        {
            sort(adjh[i].begin(), adjh[i].end());
            sort(adjc[i].begin(), adjc[i].end());
        }

        for (int id = 1; id <= p + 2; id++)
        {
            for (int i = 0; i < adjh[id].size(); i++)
            {
                auto [wc, idx] = adjh[id][i];
                a[idx].idH = i;
            }
            for (int i = 0; i < adjc[id].size(); i++)
            {
                auto [wh, idx] = adjc[id][i];
                a[idx].idC = i;
            }
        }

        if (m <= 2000 and n <= 2000 and q > 10)
        {
            for (int i = 1; i <= q; ++i)
            {
                auto [x1, y1, x2, y2] = b[i];
                ++dp[x1][y1];
                --dp[x2 + 1][y1];
                --dp[x1][y2 + 1];
                ++dp[x2 + 1][y2 + 1];
            }

            for (int i = 1; i <= m + 1; ++i)
                for (int j = 1; j <= n + 1; ++j)
                    dp[i][j] += dp[i - 1][j] + dp[i][j - 1] - dp[i - 1][j - 1];

            for (int i = 1; i <= m; ++i)
                for (int j = 1; j <= n; ++j)
                    dp[i][j] = (dp[i][j] > 0);

            for (int i = 1; i <= m; ++i)
            {
                hpref[i][0] = 0;
                for (int j = 1; j <= n; ++j)
                    hpref[i][j] = hpref[i][j - 1] + dp[i][j];
            }

            for (int j = 1; j <= n; ++j)
            {
                cpref[0][j] = 0;
                for (int i = 1; i <= m; ++i)
                    cpref[i][j] = cpref[i - 1][j] + dp[i][j];
            }

            dijkstra1();

            lint res = min(dist[0][p + 1], dist[1][p + 1]);

            cout << (res == INF ? -1 : res) << endl;

            reset1();
        }

        else
        {
            dijkstra();

            lint res = min(dist[0][p + 1], dist[1][p + 1]);

            cout << (res == INF ? -1 : res) << endl;

            reset();
        }
    }
}
// Let your soul wander where dreams are born.
