#include <algorithm>
#include <iostream>
#include <cstring>
using namespace std;
const int LIM = 3030;
template<typename T> void maximize(T &res, const T &val) { if (res < val) res = val; }
template<typename T> void minimize(T &res, const T &val) { if (res > val) res = val; }
int r, c, h, w;
int a[LIM][LIM];
int subtask1()
{
return r * c;
}
int subtask2()
{
return (r * c + 1) / 2;
}
int value[LIM * LIM];
int subtask3()
{
int res = 0;
for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
{
for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
{
int p = 0;
for (int x = lx; x <= rx; ++x)
for (int y = ly; y <= ry; ++y)
value[++p] = a[x][y];
sort(value + 1, value + p + 1);
maximize(res, value[(p + 1) / 2]);
}
}
return res;
}
/// ====*====*====*====*====*====*====*====*====*====*====*====*====*====*====*====*====*====
int used[LIM * LIM];
int subtask4()
{
int timer = 0;
memset(used, 0, sizeof(used));
int res = 0;
for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
{
for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
{
int mn = r * c;
int mx = 1;
++timer;
for (int x = lx; x <= rx; ++x)
{
for (int y = ly; y <= ry; ++y)
{
minimize(mn, a[x][y]);
maximize(mx, a[x][y]);
used[a[x][y]] = timer;
}
}
int cnt = (h * w + 1) / 2;
for (int v = mn; v <= mx; ++v)
{
if (used[v] == timer)
{
if (--cnt == 0)
{
maximize(res, v);
break;
}
}
}
}
}
return res;
}
int bit[LIM * LIM];
void bit_construct()
{
memset(bit, 0, sizeof(bit[0]) * (r * c + 1));
}
void bit_update(int p, int v)
{
for (; p <= r * c; p += p & -p)
bit[p] += v;
}
int bit_query(int p)
{
int res = 0;
for (; p >= 1; p -= p & -p)
res += bit[p];
return res;
}
int bit_median()
{
int expected = (h * w + 1) / 2;
int upper = r * c - expected + 1;
int lower = expected;
int res = -1;
for (int l = lower, r = upper; l <= r; )
{
int m = (l + r) >> 1;
if (bit_query(m) >= expected)
{
res = m;
r = m - 1;
}
else
{
l = m + 1;
}
}
return res;
}
int subtask5()
{
int res = 0;
for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
{
bit_construct();
for (int x = lx; x <= rx; ++x)
for (int y = 1; y <= w; ++y)
bit_update(a[x][y], +1);
for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
{
maximize(res, bit_median());
if (ry == c) break;
for (int x = lx; x <= rx; ++x)
{
bit_update(a[x][ly + 0], -1);
bit_update(a[x][ry + 1], +1);
}
}
}
return res;
}
int bit2d[LIM][LIM];
void bit2d_construct()
{
for (int x = 1; x <= r; ++x)
memset(bit2d[x], 0, sizeof(bit2d[x][0]) * (c + 1));
}
void bit2d_update(int x, int y, int v)
{
for (int p = x; p <= r; p += p & -p)
for (int q = y; q <= c; q += q & -q)
bit2d[p][q] += v;
}
int bit2d_query(int x, int y)
{
int res = 0;
for (int p = x; p >= 1; p -= p & -p)
for (int q = y; q >= 1; q -= q & -q)
res += bit2d[p][q];
return res;
}
void bit2d_area_update(int x, int y, int u, int v, int k)
{
bit2d_update(u, v, +k);
bit2d_update(u, y - 1, -k);
bit2d_update(x - 1, v, -k);
bit2d_update(x - 1, y - 1, +k);
}
int bit2d_area_query(int x, int y, int u, int v)
{
int res = 0;
res += bit2d_query(u, v);
res -= bit2d_query(u, y - 1);
res -= bit2d_query(x - 1, v);
res += bit2d_query(x - 1, y - 1);
return res;
}
int testing(int value, int target)
{
bit2d_construct();
for (int i = 1; i <= r; ++i)
{
for (int j = 1; j <= c; ++j)
{
if (a[i][j] >= value)
bit2d_update(i, j, +1);
if (i >= h && j >= w)
{
int cnt = bit2d_area_query(i - h + 1, j - w + 1, i, j);
if (cnt >= target)
return true;
}
}
}
return false;
}
int subtask6()
{
int expected = (h * w + 1) / 2;
int upper = r * c - expected + 1;
int lower = expected;
int res = -1;
for (int l = lower, r = upper; l <= r; )
{
int m = (l + r) >> 1;
if (testing(m, expected))
{
maximize(res, m);
l = m + 1;
}
else
{
r = m - 1;
}
}
return res;
}
int b[LIM][LIM];
int check(int value, int target)
{
for (int i = 1; i <= r; ++i)
{
for (int j = 1; j <= c; ++j)
{
b[i][j] = b[i - 1][j] + b[i][j - 1] - b[i - 1][j - 1];
b[i][j] += (a[i][j] >= value);
}
}
for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
{
for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
{
int cnt = b[rx][ry] - b[lx - 1][ry] - b[rx][ly - 1] + b[lx - 1][ly - 1];
if (cnt >= target)
return true;
}
}
return false;
}
int subtask7()
{
int expected = (h * w + 1) / 2;
int upper = r * c - expected + 1;
int lower = expected;
int res = -1;
for (int l = lower, r = upper; l <= r; )
{
int m = (l + r) >> 1;
if (check(m, expected))
{
maximize(res, m);
l = m + 1;
}
else
{
r = m - 1;
}
}
return res;
}
int main()
{
ios::sync_with_stdio(NULL);
cin.tie(NULL);
cin >> r >> c >> h >> w;
if (h == 1 && w == 1) return cout << subtask1(), 0;
if (h == r && w == c) return cout << subtask2(), 0;
for (int i = 1; i <= r; ++i)
for (int j = 1; j <= c; ++j)
cin >> a[i][j];
if (r <= 30 && c <= 30) return cout << subtask3(), 0;
if (r <= 100 && c <= 100) return cout << subtask4(), 0;
if (r <= 300 && c <= 300) return cout << subtask5(), 0;
if (r <= 1000 && c <= 1000) return cout << subtask6(), 0;
if (r <= 3000 && c <= 3000) return cout << subtask7(), 0;
return 0;
}
#include <algorithm>
#include <iostream>
#include <cstring>

using namespace std;

const int LIM = 3030;

template<typename T> void maximize(T &res, const T &val) { if (res < val) res = val; }
template<typename T> void minimize(T &res, const T &val) { if (res > val) res = val; }

int r, c, h, w;
int a[LIM][LIM];


int subtask1() 
{ 
    return r * c; 
}
    

int subtask2()
{ 
    return (r * c + 1) / 2;
}


int value[LIM * LIM];
int subtask3()
{
    int res = 0;
    for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
    {
        for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
        {
            int p = 0;
            for (int x = lx; x <= rx; ++x)
                for (int y = ly; y <= ry; ++y)
                    value[++p] = a[x][y];

            sort(value + 1, value + p + 1);
            maximize(res, value[(p + 1) / 2]);
        }
    }

    return res;
}

/// ====*====*====*====*====*====*====*====*====*====*====*====*====*====*====*====*====*====

int used[LIM * LIM];
int subtask4()
{
    int timer = 0;
    memset(used, 0, sizeof(used));

    int res = 0;
    for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
    {
        for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
        {
            int mn = r * c;
            int mx = 1;
            ++timer;
            for (int x = lx; x <= rx; ++x)
            {
                for (int y = ly; y <= ry; ++y)
                {
                    minimize(mn, a[x][y]);
                    maximize(mx, a[x][y]);
                    used[a[x][y]] = timer;
                }
            }

            int cnt = (h * w + 1) / 2;
            for (int v = mn; v <= mx; ++v)
            {
                if (used[v] == timer)
                {
                    if (--cnt == 0)
                    {
                        maximize(res, v);
                        break;
                    }
                }
            }
        }
    }

    return res;
}


int bit[LIM * LIM];
void bit_construct()
{
    memset(bit, 0, sizeof(bit[0]) * (r * c + 1));
}

void bit_update(int p, int v)
{
    for (; p <= r * c; p += p & -p)
        bit[p] += v;
}

int bit_query(int p)
{
    int res = 0;
    for (; p >= 1; p -= p & -p)
        res += bit[p];

    return res;
}

int bit_median()
{
    int expected = (h * w + 1) / 2;
    int upper = r * c - expected + 1;
    int lower = expected;

    int res = -1;
    for (int l = lower, r = upper; l <= r; )
    {
        int m = (l + r) >> 1;
        if (bit_query(m) >= expected)
        {
            res = m;
            r = m - 1;
        }
        else 
        {
            l = m + 1;
        }
    }

    return res;
}

int subtask5()
{
    int res = 0;
    for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
    {
        bit_construct();
        for (int x = lx; x <= rx; ++x)
            for (int y = 1; y <= w; ++y)
                bit_update(a[x][y], +1);   

        for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
        {
            maximize(res, bit_median());

            if (ry == c) break;
            for (int x = lx; x <= rx; ++x)
            {
                bit_update(a[x][ly + 0], -1);
                bit_update(a[x][ry + 1], +1);
            }
        }
    }

    return res;
}


int bit2d[LIM][LIM];
void bit2d_construct()
{
    for (int x = 1; x <= r; ++x)
        memset(bit2d[x], 0, sizeof(bit2d[x][0]) * (c + 1));
}

void bit2d_update(int x, int y, int v)
{
    for (int p = x; p <= r; p += p & -p)
        for (int q = y; q <= c; q += q & -q)
            bit2d[p][q] += v;
}

int bit2d_query(int x, int y)
{
    int res = 0;
    for (int p = x; p >= 1; p -= p & -p)
        for (int q = y; q >= 1; q -= q & -q)
            res += bit2d[p][q];

    return res;
}

void bit2d_area_update(int x, int y, int u, int v, int k)
{
    bit2d_update(u, v, +k);
    bit2d_update(u, y - 1, -k);
    bit2d_update(x - 1, v, -k);
    bit2d_update(x - 1, y - 1, +k);
}

int bit2d_area_query(int x, int y, int u, int v)
{
    int res = 0;
    res += bit2d_query(u, v);
    res -= bit2d_query(u, y - 1);
    res -= bit2d_query(x - 1, v);
    res += bit2d_query(x - 1, y - 1);
    return res;
}

int testing(int value, int target)
{
    bit2d_construct();
    for (int i = 1; i <= r; ++i)
    {
        for (int j = 1; j <= c; ++j)
        {
            if (a[i][j] >= value)
                bit2d_update(i, j, +1);
            
            if (i >= h && j >= w)
            {
                int cnt = bit2d_area_query(i - h + 1, j - w + 1, i, j);
                if (cnt >= target)
                    return true;
            }
        }
    }

    return false;
}

int subtask6()
{
    int expected = (h * w + 1) / 2;
    int upper = r * c - expected + 1;
    int lower = expected;

    int res = -1;
    for (int l = lower, r = upper; l <= r; )
    {
        int m = (l + r) >> 1;
        if (testing(m, expected))
        {
            maximize(res, m);
            l = m + 1;
        }
        else 
        {
            r = m - 1;
        }
    }
    
    return res;
}

int b[LIM][LIM];
int check(int value, int target)
{
    for (int i = 1; i <= r; ++i)
    {
        for (int j = 1; j <= c; ++j)
        {
            b[i][j] = b[i - 1][j] + b[i][j - 1] - b[i - 1][j - 1];
            b[i][j] += (a[i][j] >= value);
        }
    }

    for (int lx = 1, rx = h; rx <= r; ++lx, ++rx)
    {
        for (int ly = 1, ry = w; ry <= c; ++ly, ++ry)
        {
            int cnt = b[rx][ry] - b[lx - 1][ry] - b[rx][ly - 1] + b[lx - 1][ly - 1];
            if (cnt >= target)
                return true;
        }
    }

    return false;
}

int subtask7()
{
    int expected = (h * w + 1) / 2;
    int upper = r * c - expected + 1;
    int lower = expected;

    int res = -1;

    for (int l = lower, r = upper; l <= r; )
    {
        int m = (l + r) >> 1;
        if (check(m, expected))
        {
            maximize(res, m);
            l = m + 1;
        }
        else 
        {
            r = m - 1;
        }
    }
    
    return res;
}

int main()
{
    ios::sync_with_stdio(NULL);
    cin.tie(NULL);

    cin >> r >> c >> h >> w;
    if (h == 1 && w == 1) return cout << subtask1(), 0;
    if (h == r && w == c) return cout << subtask2(), 0;
    for (int i = 1; i <= r; ++i)
        for (int j = 1; j <= c; ++j)
            cin >> a[i][j];

    if (r <=   30 && c <=   30) return cout << subtask3(), 0;
    if (r <=  100 && c <=  100) return cout << subtask4(), 0;
    if (r <=  300 && c <=  300) return cout << subtask5(), 0;
    if (r <= 1000 && c <= 1000) return cout << subtask6(), 0;
    if (r <= 3000 && c <= 3000) return cout << subtask7(), 0;
    return 0;
}