#include <bits/stdc++.h>
using namespace std;
mt19937_64 rd( chrono:: steady_clock :: now ( ) .time_since_epoch ( ) .count ( ) ) ;
#define rf if(fopen(name".inp", "r")) {freopen(name".inp", "r", stdin); freopen(name".out", "w", stdout);}
#define rd ({int x = 0; int c = getchar(), n = 0; for(; !isdigit(c); c = getchar()) n = (c == '-'); for(; isdigit(c); c = getchar()) x = x * 10 + c - '0'; n ? -x : x;})
#define bit(i, mask) (((mask) >> (i)) & 1)
#define on(i, mask) ((mask) | (1LL << (i)))
#define off(i, mask) ((mask) & (~(1LL << (i))))
#define ll long long
#define fi first
#define se second
#define pii pair<int, int>
#define vii vector<int>
#define all(a) (a).begin(), (a).end()
#define len(x) ((int)(x).size())
#define pb push_back
#define endl '\n'
#define TIME (1.0 * clock() / CLOCKS_PER_SEC)
#define name ""
template < typename T1, typename T2> bool mini( T1 & a, T2 b)
{ if ( a > b) a = b; else return 0 ; return 1 ; }
template < typename T1, typename T2> bool maxi( T1 & a, T2 b)
{ if ( a < b) a = b; else return 0 ; return 1 ; }
const int mod = 1e9 + 7 ;
const int inf = 1e9 + 9 ;
const ll oo = 1e18l+ 7 ;
const int M = 5e5 + 6 ;
const int N = 2468 ;
const int LOG = 31 - __builtin_clz( N) ;
int n, m, s[ N] [ N] ;
struct Edge{
int v, flow, cap;
} ;
struct Dinic{
int n, s, t, work[ N] , d[ N] ;
vector< Edge> e;
vii a[ N] ;
Dinic( int _n, int _s, int _t) {
n = _n, s = _s, t = _t;
}
void Add( int u, int v, int c) {
e.pb ( { v, 0 , c} ) ;
a[ u] .pb ( e.size ( ) - 1 ) ;
e.pb ( { u, 0 , 0 } ) ;
a[ v] .pb ( e.size ( ) - 1 ) ;
}
bool bfs( ) {
for ( int i = 1 ; i <= n; i++ ) d[ i] = 0 ;
d[ s] = 1 ;
queue< int > q;
q.push ( s) ;
while ( ! q.empty ( ) ) {
int k = q.front ( ) ;
q.pop ( ) ;
for ( int u : a[ k] ) {
int v = e[ u] .v ;
if ( e[ u] .cap > e[ u] .flow && d[ v] == 0 ) {
d[ v] = d[ k] + 1 ;
q.push ( v) ;
}
}
}
return d[ t] > 0 ;
}
int dfs( int u, int flow) {
if ( u == t) return flow;
for ( int & z = work[ u] ; z < ( int ) a[ u] .size ( ) ; z++ ) {
int i = a[ u] [ z] ;
int v = e[ i] .v ;
if ( e[ i] .cap > e[ i] .flow && d[ v] == d[ u] + 1 ) {
int tmp = dfs( v, min( flow, e[ i] .cap - e[ i] .flow ) ) ;
if ( tmp) {
e[ i] .flow + = tmp;
e[ i ^ 1 ] .flow - = tmp;
return tmp;
}
}
}
return 0 ;
}
int getFlow( ) {
int res = 0 ;
while ( bfs( ) ) {
for ( int i = 1 ; i <= n; i++ ) work[ i] = 0 ;
while ( int delta = dfs( s, inf) ) res + = delta;
}
//for(int i = 0; i < e.size(); i += 2) cout << e[i].flow << endl;
return res;
}
} ;
void in( ) {
cin >> n >> m;
for ( int i = 1 ; i <= n; i++ ) {
for ( int j = 1 ; j <= m; j++ ) cin >> s[ i] [ j] ;
}
}
namespace subtask1{
int dp[ 15 ] [ 1 << 12 ] , trace[ 15 ] [ 1 << 12 ] , cost[ 15 ] [ 1 << 12 ] ;
void solve( ) {
for ( int i = 1 ; i <= n; i++ ) {
for ( int mask = 0 ; mask < ( 1 << m) ; mask++ ) {
for ( int j = 0 ; j < m; j++ ) if ( bit( j, mask) ) cost[ i] [ mask] + = s[ i] [ j + 1 ] ;
}
}
memset ( dp, - 1 , sizeof dp) ;
dp[ 0 ] [ 0 ] = inf;
for ( int mask = 0 ; mask < ( 1 << m) ; mask++ ) {
for ( int i = 0 ; i < n; i++ ) {
if ( dp[ i] [ mask] == - 1 ) continue ;
int tmp = ( ( 1 << m) - 1 ) ^ mask;
for ( int submask = tmp; submask > 0 ; submask = ( submask - 1 ) & tmp) {
if ( maxi( dp[ i + 1 ] [ mask ^ submask] , min( cost[ i + 1 ] [ submask] , dp[ i] [ mask] ) ) )
trace[ i + 1 ] [ mask ^ submask] = submask;
}
}
}
vii ans;
int mask = ( 1 << m) - 1 ;
while ( n > 0 ) {
ans.pb ( trace[ n] [ mask] ) ;
mask ^ = trace[ n] [ mask] ;
n-- ;
}
reverse( all( ans) ) ;
for ( int it : ans) {
cout << __builtin_popcount( it) << " " ;
for ( int i = 1 ; i <= m; i++ ) if ( bit( i - 1 , it) ) cout << i << " " ;
cout << endl;
}
}
}
namespace subtask2{
int dp[ 2 ] [ 100005 ] , trace[ 1234 ] [ 100005 ] ;
// dp[i][j]: maxi 1, j la delta 1 va 2 => max = dp - j
// y tuong tu bai treasure (th tre)
void solve( ) {
memset ( dp[ 0 ] , - 0x3f , sizeof dp) ;
int w = 50000 ;
dp[ 0 ] [ w] = 0 ;
for ( int i = 1 ; i <= m; i++ ) {
int cur = i & 1 , pre = cur ^ 1 ;
for ( int j = - w; j <= w; j++ ) dp[ cur] [ j + w] = - inf;
for ( int j = - w; j <= w; j++ ) {
if ( dp[ pre] [ j + w] == - inf) continue ;
if ( j + s[ 1 ] [ i] <= w && maxi( dp[ cur] [ j + w + s[ 1 ] [ i] ] , dp[ pre] [ j + w] + s[ 1 ] [ i] ) )
trace[ i] [ j + w + s[ 1 ] [ i] ] = 1 ;
if ( j - s[ 2 ] [ i] >= - w && maxi( dp[ cur] [ j + w - s[ 2 ] [ i] ] , dp[ pre] [ j + w] ) )
trace[ i] [ j + w - s[ 2 ] [ i] ] = 2 ;
}
}
int val = 0 , id = 0 ;
for ( int i = - w; i <= w; i++ ) {
if ( dp[ m & 1 ] [ i + w] == - inf) continue ;
if ( maxi( val, dp[ m & 1 ] [ i + w] - ( i > 0 ? i : 0 ) ) ) id = i + w;
}
vii va, vb;
while ( m) {
if ( trace[ m] [ id] == 1 ) {
va.pb ( m) ;
id - = s[ 1 ] [ m] ;
}
else {
vb.pb ( m) ;
id + = s[ 2 ] [ m] ;
}
m-- ;
}
reverse( all( va) ) ; reverse( all( vb) ) ;
cout << len( va) << ' ' ;
for ( int it : va) cout << it << ' ' ;
cout << endl;
cout << len( vb) << ' ' ;
for ( int it : vb) cout << it << ' ' ;
}
}
namespace subtask3{
int st, en, cnt, ng[ N] , gift[ N] ;
bool check( int x) {
Dinic mxFlow( cnt, st, en) ;
for ( int i = 1 ; i <= n; i++ ) {
for ( int j = 1 ; j <= m; j++ ) if ( s[ i] [ j] >= x) mxFlow.Add ( ng[ i] , gift[ j] , 1 ) ;
}
for ( int i = 1 ; i <= n; i++ ) mxFlow.Add ( st, ng[ i] , 1 ) ;
for ( int i = 1 ; i <= m; i++ ) mxFlow.Add ( gift[ i] , en, 1 ) ;
return ( mxFlow.getFlow ( ) == n) ;
}
void output( int x) {
Dinic mxFlow( cnt, st, en) ;
for ( int i = 1 ; i <= n; i++ ) {
for ( int j = 1 ; j <= m; j++ ) if ( s[ i] [ j] >= x) mxFlow.Add ( ng[ i] , gift[ j] , 1 ) ;
}
for ( int i = 1 ; i <= n; i++ ) mxFlow.Add ( st, ng[ i] , 1 ) ;
for ( int i = 1 ; i <= m; i++ ) mxFlow.Add ( gift[ i] , en, 1 ) ;
mxFlow.getFlow ( ) ;
for ( int i = 1 ; i <= n; i++ ) {
int u = ng[ i] ;
for ( auto it : mxFlow.a [ u] ) {
int v = mxFlow.e [ it] .v ;
if ( mxFlow.e [ it] .flow == 1 ) {
cout << 1 << ' ' << v - n << endl;
break ;
}
}
}
}
void solve( ) {
for ( int i = 1 ; i <= n; i++ ) ng[ i] = ++ cnt;
for ( int i = 1 ; i <= m; i++ ) gift[ i] = ++ cnt;
st = ++ cnt, en = ++ cnt;
int l = 1 , r = 1000 , res = 1 ;
while ( l <= r) {
int mid = ( l + r) >> 1 ;
if ( check( mid) ) {
res = mid;
l = mid + 1 ;
}
else r = mid - 1 ;
}
output( res) ;
}
}
void proc( ) {
if ( n <= 12 && m <= 12 ) {
subtask1:: solve ( ) ;
return ;
}
if ( n == 2 ) {
subtask2:: solve ( ) ;
return ;
}
subtask3:: solve ( ) ;
}
int main( ) {
cin .tie ( nullptr) - > sync_with_stdio( false ) ;
rf
int test = 1 ;
//cin >> test;
while ( test-- ) {
in( ) ;
proc( ) ;
}
cerr << "Time elapsed: " << TIME << "s" << endl;
return 0 ;
}
#include <bits/stdc++.h>

using namespace std;

mt19937_64 rd(chrono::steady_clock::now().time_since_epoch().count());

#define rf if(fopen(name".inp", "r")) {freopen(name".inp", "r", stdin); freopen(name".out", "w", stdout);}
#define rd ({int x = 0; int c = getchar(), n = 0; for(; !isdigit(c); c = getchar()) n = (c == '-'); for(; isdigit(c); c = getchar()) x = x * 10 + c - '0'; n ? -x : x;})

#define bit(i, mask) (((mask) >> (i)) & 1)
#define on(i, mask) ((mask) | (1LL << (i)))
#define off(i, mask) ((mask) & (~(1LL << (i))))

#define ll long long
#define fi first
#define se second
#define pii pair<int, int>
#define vii vector<int>
#define all(a) (a).begin(), (a).end()
#define len(x) ((int)(x).size())
#define pb push_back
#define endl '\n'
#define TIME (1.0 * clock() / CLOCKS_PER_SEC)

#define name ""

template<typename T1, typename T2> bool mini(T1 &a, T2 b)
    {if(a > b) a = b; else return 0; return 1;}
template<typename T1, typename T2> bool maxi(T1 &a, T2 b)
    {if(a < b) a = b; else return 0; return 1;}

const int mod = 1e9+7;
const int inf = 1e9+9;
const ll oo = 1e18l+7;
const int M = 5e5+6;
const int N = 2468;
const int LOG = 31 - __builtin_clz(N);

int n, m, s[N][N];

struct Edge{
    int v, flow, cap;
};

struct Dinic{
    int n, s, t, work[N], d[N];
    vector<Edge> e;
    vii a[N];
    Dinic(int _n, int _s, int _t){
        n = _n, s = _s, t = _t;
    }

    void Add(int u, int v, int c){
        e.pb({v, 0, c});
        a[u].pb(e.size() - 1);
        e.pb({u, 0, 0});
        a[v].pb(e.size() - 1);
    }

    bool bfs(){
        for(int i = 1; i <= n; i++) d[i] = 0;
        d[s] = 1;
        queue<int> q;
        q.push(s);
        while(!q.empty()){
            int k = q.front();
            q.pop();
            for(int u : a[k]){
                int v = e[u].v;
                if(e[u].cap > e[u].flow && d[v] == 0){
                    d[v] = d[k] + 1;
                    q.push(v);
                }
            }
        }
        return d[t] > 0;
    }

    int dfs(int u, int flow){
        if(u == t) return flow;
        for(int &z = work[u]; z < (int)a[u].size(); z++){
            int i = a[u][z];
            int v = e[i].v;
            if(e[i].cap > e[i].flow && d[v] == d[u] + 1){
                int tmp = dfs(v, min(flow, e[i].cap - e[i].flow));
                if(tmp){
                    e[i].flow += tmp;
                    e[i ^ 1].flow -= tmp;
                    return tmp;
                }
            }
        }
        return 0;
    }

    int getFlow(){
        int res = 0;
        while(bfs()){
            for(int i = 1; i <= n; i++) work[i] = 0;
            while(int delta = dfs(s, inf)) res += delta;
        }
        //for(int i = 0; i < e.size(); i += 2) cout << e[i].flow << endl;
        return res;
    }
};

void in(){
    cin >> n >> m;
    for(int i = 1; i <= n; i++){
        for(int j = 1; j <= m; j++) cin >> s[i][j];
    }
}

namespace subtask1{
    int dp[15][1 << 12], trace[15][1 << 12], cost[15][1 << 12];

    void solve(){
        for(int i = 1; i <= n; i++){
            for(int mask = 0; mask < (1 << m); mask++){
                for(int j = 0; j < m; j++) if(bit(j, mask)) cost[i][mask] += s[i][j + 1];
            }
        }
        memset(dp, -1, sizeof dp);
        dp[0][0] = inf;
        for(int mask = 0; mask < (1 << m); mask++){
            for(int i = 0; i < n; i++){
                if(dp[i][mask] == -1) continue;
                int tmp = ((1 << m) - 1) ^ mask;
                for(int submask = tmp; submask > 0; submask = (submask - 1) & tmp){
                    if(maxi(dp[i + 1][mask ^ submask], min(cost[i + 1][submask], dp[i][mask])))
                        trace[i + 1][mask ^ submask] = submask;
                }
            }
        }
        vii ans;
        int mask = (1 << m) - 1;
        while(n > 0){
            ans.pb(trace[n][mask]);
            mask ^= trace[n][mask];
            n--;
        }
        reverse(all(ans));
        for(int it : ans){
            cout << __builtin_popcount(it) << " ";
            for(int i = 1; i <= m; i++) if(bit(i - 1, it)) cout << i << " ";
            cout << endl;
        }
    }
}

namespace subtask2{
    int dp[2][100005], trace[1234][100005];
    // dp[i][j]: maxi 1, j la delta 1 va 2 => max = dp - j
    // y tuong tu bai treasure (th tre)

    void solve(){
        memset(dp[0], -0x3f, sizeof dp);
        int w = 50000;
        dp[0][w] = 0;
        for(int i = 1; i <= m; i++){
            int cur = i & 1, pre = cur ^ 1;
            for(int j = -w; j <= w; j++) dp[cur][j + w] = -inf;
            for(int j = -w; j <= w; j++){
                if(dp[pre][j + w] == -inf) continue;
                if(j + s[1][i] <= w && maxi(dp[cur][j + w + s[1][i]], dp[pre][j + w] + s[1][i]))
                    trace[i][j + w + s[1][i]] = 1;
                if(j - s[2][i] >= -w && maxi(dp[cur][j + w - s[2][i]], dp[pre][j + w]))
                    trace[i][j + w - s[2][i]] = 2;
            }
        }

        int val = 0, id = 0;
        for(int i = -w; i <= w; i++){
            if(dp[m & 1][i + w] == -inf) continue;
            if(maxi(val, dp[m & 1][i + w] - (i > 0 ? i : 0))) id = i + w;
        }

        vii va, vb;
        while(m){
            if(trace[m][id] == 1){
                va.pb(m);
                id -= s[1][m];
            }
            else{
                vb.pb(m);
                id += s[2][m];
            }
            m--;
        }
        reverse(all(va)); reverse(all(vb));
        cout << len(va) << ' ';
        for(int it : va) cout << it << ' ';
        cout << endl;
        cout << len(vb) << ' ';
        for(int it : vb) cout << it << ' ';
    }
}

namespace subtask3{
    int st, en, cnt, ng[N], gift[N];

    bool check(int x){
        Dinic mxFlow(cnt, st, en);
        for(int i = 1; i <= n; i++){
            for(int j = 1; j <= m; j++) if(s[i][j] >= x) mxFlow.Add(ng[i], gift[j], 1);
        }
        for(int i = 1; i <= n; i++) mxFlow.Add(st, ng[i], 1);
        for(int i = 1; i <= m; i++) mxFlow.Add(gift[i], en, 1);
        return (mxFlow.getFlow() == n);
    }

    void output(int x){
        Dinic mxFlow(cnt, st, en);
        for(int i = 1; i <= n; i++){
            for(int j = 1; j <= m; j++) if(s[i][j] >= x) mxFlow.Add(ng[i], gift[j], 1);
        }
        for(int i = 1; i <= n; i++) mxFlow.Add(st, ng[i], 1);
        for(int i = 1; i <= m; i++) mxFlow.Add(gift[i], en, 1);
        mxFlow.getFlow();
        for(int i = 1; i <= n; i++){
            int u = ng[i];
            for(auto it : mxFlow.a[u]){
                int v = mxFlow.e[it].v;
                if(mxFlow.e[it].flow == 1){
                    cout << 1 << ' ' << v - n << endl;
                    break;
                }
            }
        }
    }

    void solve(){
        for(int i = 1; i <= n; i++) ng[i] = ++cnt;
        for(int i = 1; i <= m; i++) gift[i] = ++cnt;
        st = ++cnt, en = ++cnt;
        int l = 1, r = 1000, res = 1;
        while(l <= r){
            int mid = (l + r) >> 1;
            if(check(mid)){
                res = mid;
                l = mid + 1;
            }
            else r = mid - 1;
        }
        output(res);
    }
}

void proc(){
    if(n <= 12 && m <= 12){
        subtask1::solve();
        return;
    }
    if(n == 2){
        subtask2::solve();
        return;
    }
    subtask3::solve();
}

int main(){
    cin.tie(nullptr)->sync_with_stdio(false);

    rf

    int test = 1;
    //cin >> test;

    while(test--){
        in();
        proc();
    }

    cerr << "Time elapsed: " << TIME << "s" << endl;
    return 0;
}
