#include <bits/stdc++.h>
using namespace std;
const int MAX_N = 2005;
int n, m, k, nRotate, mRotate;
int arr[MAX_N][MAX_N];
int radius[MAX_N][MAX_N];
int prefixSumUnit[2 * MAX_N][2 * MAX_N];
int prefixSumUnitTriangle[2 * MAX_N][2 * MAX_N];
long long prefixSum[2 * MAX_N][2 * MAX_N],
prefixSumTriangle[2 * MAX_N][2 * MAX_N];
long long answer[MAX_N][MAX_N];
pair<int, int> getPos(int x, int y) { return make_pair(x + y - 1, n - x + y); }
int getSumUnit(int u, int v, int x, int y) {
u = max(u, 1);
v = max(v, 1);
x = min(x, nRotate);
y = min(y, nRotate);
return prefixSumUnit[x][y] - prefixSumUnit[x][v - 1] -
prefixSumUnit[u - 1][y] + prefixSumUnit[u - 1][v - 1];
}
long long getSum(int u, int v, int x, int y) {
u = max(u, 1);
v = max(v, 1);
x = min(x, nRotate);
y = min(y, nRotate);
return prefixSum[x][y] - prefixSum[x][v - 1] - prefixSum[u - 1][y] +
prefixSum[u - 1][v - 1];
}
void calculateRadius() {
int row, col;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
tie(row, col) = getPos(i, j);
prefixSumUnit[row][col] = arr[i][j];
}
}
nRotate = n + m - 1;
for (int i = 1; i <= nRotate; i++) {
for (int j = 1; j <= nRotate; j++) {
prefixSumUnit[i][j] =
prefixSumUnit[i][j] + prefixSumUnit[i - 1][j] +
prefixSumUnit[i][j - 1] - prefixSumUnit[i - 1][j - 1];
}
}
radius[1][1] = 0;
tie(row, col) = getPos(1, 1);
while (getSumUnit(row - radius[1][1], col - radius[1][1],
row + radius[1][1], col + radius[1][1]) < k) {
radius[1][1]++;
}
for (int j = 2; j <= m; j++) {
radius[1][j] = radius[1][j - 1] + 1;
int &tmp = radius[1][j];
tie(row, col) = getPos(1, j);
while (tmp > 0 && getSumUnit(row - tmp + 1, col - tmp + 1,
row + tmp - 1, col + tmp - 1) >= k) {
tmp--;
}
}
for (int i = 2; i <= n; i++) {
for (int j = 1; j <= m; j++) {
radius[i][j] = radius[i - 1][j] + 1;
int &tmp = radius[i][j];
tie(row, col) = getPos(i, j);
while (tmp > 0 && getSumUnit(row - tmp + 1, col - tmp + 1,
row + tmp - 1, col + tmp - 1) >= k) {
tmp--;
}
}
}
}
long long getSumTriangleX(int u, int v, int x, int y) {
long long result = 0;
if (u < 1) {
int tmp = 1 - u;
u += tmp;
v += tmp;
}
if (y > nRotate) {
int tmp = y - nRotate;
x -= tmp;
y -= tmp;
}
if (v < 1) {
int tmp = 1 - v;
result += getSum(u, 1, u + tmp - 1, y);
u += tmp;
v += tmp;
}
if (x > nRotate) {
int tmp = x - nRotate;
result += getSum(u, y - tmp + 1, nRotate, y);
x -= tmp;
y -= tmp;
}
result += prefixSumTriangle[x][y] - prefixSumTriangle[u - 1][v - 1] -
getSum(1, v, u - 1, y);
return result;
}
int getSumUnitTriangleX(int u, int v, int x, int y) {
int result = 0;
if (u < 1) {
int tmp = 1 - u;
u += tmp;
v += tmp;
}
if (y > nRotate) {
int tmp = y - nRotate;
x -= tmp;
y -= tmp;
}
if (v < 1) {
int tmp = 1 - v;
result += getSumUnit(u, 1, u + tmp - 1, y);
u += tmp;
v += tmp;
}
if (x > nRotate) {
int tmp = x - nRotate;
result += getSumUnit(u, y - tmp + 1, nRotate, y);
x -= tmp;
y -= tmp;
}
result += prefixSumUnitTriangle[x][y] -
prefixSumUnitTriangle[u - 1][v - 1] - getSumUnit(1, v, u - 1, y);
return result;
}
void calculateAnswerX() {
int row, col;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
tie(row, col) = getPos(i, j);
prefixSum[row][col] = i * arr[i][j];
prefixSumUnitTriangle[row][col] = arr[i][j];
prefixSumTriangle[row][col] = i * arr[i][j];
}
}
for (int i = 1; i <= nRotate; i++) {
for (int j = 1; j <= nRotate; j++) {
prefixSum[i][j] = prefixSum[i][j] + prefixSum[i - 1][j] +
prefixSum[i][j - 1] - prefixSum[i - 1][j - 1];
}
}
for (int i = 2; i <= nRotate; i++) {
for (int j = 1; j <= nRotate; j++) {
prefixSumUnitTriangle[i][j] = prefixSumUnitTriangle[i][j] +
prefixSumUnitTriangle[i - 1][j] +
prefixSumUnitTriangle[i - 1][j - 1] -
prefixSumUnitTriangle[i - 2][j - 1];
prefixSumTriangle[i][j] = prefixSumTriangle[i][j] +
prefixSumTriangle[i - 1][j] +
prefixSumTriangle[i - 1][j - 1] -
prefixSumTriangle[i - 2][j - 1];
}
}
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
if (radius[i][j] > 0) {
tie(row, col) = getPos(i, j);
int tmp = radius[i][j] - 1;
int sumUnitUpTriangle = getSumUnitTriangleX(
row - tmp, col - tmp, row + tmp, col + tmp);
long long sumUpTriangle =
getSumTriangleX(row - tmp, col - tmp, row + tmp, col + tmp);
int sumUnitArea =
getSumUnit(row - tmp, col - tmp, row + tmp, col + tmp);
long long sumArea =
getSum(row - tmp, col - tmp, row + tmp, col + tmp);
answer[i][j] += i * sumUnitUpTriangle - sumUpTriangle +
(sumArea - sumUpTriangle) -
i * (sumUnitArea - sumUnitUpTriangle);
}
}
}
}
long long getSumTriangleY(int u, int v, int x, int y) {
long long result = 0;
if (u < 1) {
int tmp = 1 - u;
u += tmp;
v -= tmp;
}
if (y < 1) {
int tmp = 1 - y;
x -= tmp;
y += tmp;
}
if (x > nRotate) {
int tmp = x - nRotate;
result += getSum(u, y, nRotate, y + tmp - 1);
x -= tmp;
y += tmp;
}
if (v > nRotate) {
int tmp = v - nRotate;
result += getSum(u, y, u + tmp - 1, nRotate);
u += tmp;
v -= tmp;
}
result += prefixSumTriangle[x][y] - prefixSumTriangle[u - 1][v + 1] -
getSum(1, y, u - 1, v);
return result;
}
int getSumUnitTriangleY(int u, int v, int x, int y) {
int result = 0;
if (u < 1) {
int tmp = 1 - u;
u += tmp;
v -= tmp;
}
if (y < 1) {
int tmp = 1 - y;
x -= tmp;
y += tmp;
}
if (x > nRotate) {
int tmp = x - nRotate;
result += getSumUnit(u, y, nRotate, y + tmp - 1);
x -= tmp;
y += tmp;
}
if (v > nRotate) {
int tmp = v - nRotate;
result += getSumUnit(u, y, u + tmp - 1, nRotate);
u += tmp;
v -= tmp;
}
result += prefixSumUnitTriangle[x][y] -
prefixSumUnitTriangle[u - 1][v + 1] - getSumUnit(1, y, u - 1, v);
return result;
}
void calculateAnswerY() {
memset(prefixSum, 0, sizeof(prefixSum));
memset(prefixSumTriangle, 0, sizeof(prefixSumTriangle));
memset(prefixSumUnitTriangle, 0, sizeof(prefixSumUnitTriangle));
int row, col;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
tie(row, col) = getPos(i, j);
prefixSum[row][col] = j * arr[i][j];
prefixSumUnitTriangle[row][col] = arr[i][j];
prefixSumTriangle[row][col] = j * arr[i][j];
}
}
for (int i = 1; i <= nRotate; i++) {
for (int j = 1; j <= nRotate; j++) {
prefixSum[i][j] = prefixSum[i][j] + prefixSum[i - 1][j] +
prefixSum[i][j - 1] - prefixSum[i - 1][j - 1];
}
}
for (int i = 2; i <= nRotate; i++) {
for (int j = 1; j <= nRotate; j++) {
prefixSumUnitTriangle[i][j] = prefixSumUnitTriangle[i][j] +
prefixSumUnitTriangle[i - 1][j] +
prefixSumUnitTriangle[i - 1][j + 1] -
prefixSumUnitTriangle[i - 2][j + 1];
prefixSumTriangle[i][j] = prefixSumTriangle[i][j] +
prefixSumTriangle[i - 1][j] +
prefixSumTriangle[i - 1][j + 1] -
prefixSumTriangle[i - 2][j + 1];
}
}
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
if (radius[i][j] > 0) {
tie(row, col) = getPos(i, j);
int tmp = radius[i][j] - 1;
int sumUnitUpTriangle = getSumUnitTriangleY(
row - tmp, col + tmp, row + tmp, col - tmp);
long long sumUpTriangle =
getSumTriangleY(row - tmp, col + tmp, row + tmp, col - tmp);
int sumUnitArea =
getSumUnit(row - tmp, col - tmp, row + tmp, col + tmp);
long long sumArea =
getSum(row - tmp, col - tmp, row + tmp, col + tmp);
answer[i][j] += j * sumUnitUpTriangle - sumUpTriangle +
(sumArea - sumUpTriangle) -
j * (sumUnitArea - sumUnitUpTriangle);
}
}
}
}
long long calculateAnswer() {
long long result = 0;
int row, col;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
if (radius[i][j] > 0) {
tie(row, col) = getPos(i, j);
int tmp = radius[i][j] - 1;
result += answer[i][j] +
radius[i][j] * (k - getSumUnit(row - tmp, col - tmp,
row + tmp, col + tmp));
}
}
}
return result;
}
int main() {
ios_base::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
freopen("formation.inp", "r", stdin);
freopen("formation.out", "w", stdout);
cin >> n >> m >> k;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= m; j++) {
cin >> arr[i][j];
}
}
calculateRadius();
calculateAnswerX();
calculateAnswerY();
cout << calculateAnswer();
return 0;
}
#include <bits/stdc++.h>

using namespace std;

const int MAX_N = 2005;

int n, m, k, nRotate, mRotate;
int arr[MAX_N][MAX_N];
int radius[MAX_N][MAX_N];
int prefixSumUnit[2 * MAX_N][2 * MAX_N];
int prefixSumUnitTriangle[2 * MAX_N][2 * MAX_N];
long long prefixSum[2 * MAX_N][2 * MAX_N],
    prefixSumTriangle[2 * MAX_N][2 * MAX_N];
long long answer[MAX_N][MAX_N];

pair<int, int> getPos(int x, int y) { return make_pair(x + y - 1, n - x + y); }

int getSumUnit(int u, int v, int x, int y) {
    u = max(u, 1);
    v = max(v, 1);
    x = min(x, nRotate);
    y = min(y, nRotate);
    return prefixSumUnit[x][y] - prefixSumUnit[x][v - 1] -
           prefixSumUnit[u - 1][y] + prefixSumUnit[u - 1][v - 1];
}

long long getSum(int u, int v, int x, int y) {
    u = max(u, 1);
    v = max(v, 1);
    x = min(x, nRotate);
    y = min(y, nRotate);
    return prefixSum[x][y] - prefixSum[x][v - 1] - prefixSum[u - 1][y] +
           prefixSum[u - 1][v - 1];
}

void calculateRadius() {
    int row, col;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            tie(row, col) = getPos(i, j);
            prefixSumUnit[row][col] = arr[i][j];
        }
    }
    nRotate = n + m - 1;
    for (int i = 1; i <= nRotate; i++) {
        for (int j = 1; j <= nRotate; j++) {
            prefixSumUnit[i][j] =
                prefixSumUnit[i][j] + prefixSumUnit[i - 1][j] +
                prefixSumUnit[i][j - 1] - prefixSumUnit[i - 1][j - 1];
        }
    }
    radius[1][1] = 0;
    tie(row, col) = getPos(1, 1);
    while (getSumUnit(row - radius[1][1], col - radius[1][1],
                      row + radius[1][1], col + radius[1][1]) < k) {
        radius[1][1]++;
    }
    for (int j = 2; j <= m; j++) {
        radius[1][j] = radius[1][j - 1] + 1;
        int &tmp = radius[1][j];
        tie(row, col) = getPos(1, j);
        while (tmp > 0 && getSumUnit(row - tmp + 1, col - tmp + 1,
                                     row + tmp - 1, col + tmp - 1) >= k) {
            tmp--;
        }
    }
    for (int i = 2; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            radius[i][j] = radius[i - 1][j] + 1;
            int &tmp = radius[i][j];
            tie(row, col) = getPos(i, j);
            while (tmp > 0 && getSumUnit(row - tmp + 1, col - tmp + 1,
                                         row + tmp - 1, col + tmp - 1) >= k) {
                tmp--;
            }
        }
    }
}

long long getSumTriangleX(int u, int v, int x, int y) {
    long long result = 0;
    if (u < 1) {
        int tmp = 1 - u;
        u += tmp;
        v += tmp;
    }
    if (y > nRotate) {
        int tmp = y - nRotate;
        x -= tmp;
        y -= tmp;
    }
    if (v < 1) {
        int tmp = 1 - v;
        result += getSum(u, 1, u + tmp - 1, y);
        u += tmp;
        v += tmp;
    }
    if (x > nRotate) {
        int tmp = x - nRotate;
        result += getSum(u, y - tmp + 1, nRotate, y);
        x -= tmp;
        y -= tmp;
    }
    result += prefixSumTriangle[x][y] - prefixSumTriangle[u - 1][v - 1] -
              getSum(1, v, u - 1, y);
    return result;
}

int getSumUnitTriangleX(int u, int v, int x, int y) {
    int result = 0;
    if (u < 1) {
        int tmp = 1 - u;
        u += tmp;
        v += tmp;
    }
    if (y > nRotate) {
        int tmp = y - nRotate;
        x -= tmp;
        y -= tmp;
    }
    if (v < 1) {
        int tmp = 1 - v;
        result += getSumUnit(u, 1, u + tmp - 1, y);
        u += tmp;
        v += tmp;
    }
    if (x > nRotate) {
        int tmp = x - nRotate;
        result += getSumUnit(u, y - tmp + 1, nRotate, y);
        x -= tmp;
        y -= tmp;
    }
    result += prefixSumUnitTriangle[x][y] -
              prefixSumUnitTriangle[u - 1][v - 1] - getSumUnit(1, v, u - 1, y);
    return result;
}

void calculateAnswerX() {
    int row, col;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            tie(row, col) = getPos(i, j);
            prefixSum[row][col] = i * arr[i][j];
            prefixSumUnitTriangle[row][col] = arr[i][j];
            prefixSumTriangle[row][col] = i * arr[i][j];
        }
    }
    for (int i = 1; i <= nRotate; i++) {
        for (int j = 1; j <= nRotate; j++) {
            prefixSum[i][j] = prefixSum[i][j] + prefixSum[i - 1][j] +
                              prefixSum[i][j - 1] - prefixSum[i - 1][j - 1];
        }
    }
    for (int i = 2; i <= nRotate; i++) {
        for (int j = 1; j <= nRotate; j++) {
            prefixSumUnitTriangle[i][j] = prefixSumUnitTriangle[i][j] +
                                          prefixSumUnitTriangle[i - 1][j] +
                                          prefixSumUnitTriangle[i - 1][j - 1] -
                                          prefixSumUnitTriangle[i - 2][j - 1];

            prefixSumTriangle[i][j] = prefixSumTriangle[i][j] +
                                      prefixSumTriangle[i - 1][j] +
                                      prefixSumTriangle[i - 1][j - 1] -
                                      prefixSumTriangle[i - 2][j - 1];
        }
    }
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            if (radius[i][j] > 0) {
                tie(row, col) = getPos(i, j);
                int tmp = radius[i][j] - 1;
                int sumUnitUpTriangle = getSumUnitTriangleX(
                    row - tmp, col - tmp, row + tmp, col + tmp);
                long long sumUpTriangle =
                    getSumTriangleX(row - tmp, col - tmp, row + tmp, col + tmp);
                int sumUnitArea =
                    getSumUnit(row - tmp, col - tmp, row + tmp, col + tmp);
                long long sumArea =
                    getSum(row - tmp, col - tmp, row + tmp, col + tmp);
                answer[i][j] += i * sumUnitUpTriangle - sumUpTriangle +
                                (sumArea - sumUpTriangle) -
                                i * (sumUnitArea - sumUnitUpTriangle);
            }
        }
    }
}

long long getSumTriangleY(int u, int v, int x, int y) {
    long long result = 0;
    if (u < 1) {
        int tmp = 1 - u;
        u += tmp;
        v -= tmp;
    }
    if (y < 1) {
        int tmp = 1 - y;
        x -= tmp;
        y += tmp;
    }
    if (x > nRotate) {
        int tmp = x - nRotate;
        result += getSum(u, y, nRotate, y + tmp - 1);
        x -= tmp;
        y += tmp;
    }
    if (v > nRotate) {
        int tmp = v - nRotate;
        result += getSum(u, y, u + tmp - 1, nRotate);
        u += tmp;
        v -= tmp;
    }
    result += prefixSumTriangle[x][y] - prefixSumTriangle[u - 1][v + 1] -
              getSum(1, y, u - 1, v);
    return result;
}

int getSumUnitTriangleY(int u, int v, int x, int y) {
    int result = 0;
    if (u < 1) {
        int tmp = 1 - u;
        u += tmp;
        v -= tmp;
    }
    if (y < 1) {
        int tmp = 1 - y;
        x -= tmp;
        y += tmp;
    }
    if (x > nRotate) {
        int tmp = x - nRotate;
        result += getSumUnit(u, y, nRotate, y + tmp - 1);
        x -= tmp;
        y += tmp;
    }
    if (v > nRotate) {
        int tmp = v - nRotate;
        result += getSumUnit(u, y, u + tmp - 1, nRotate);
        u += tmp;
        v -= tmp;
    }
    result += prefixSumUnitTriangle[x][y] -
              prefixSumUnitTriangle[u - 1][v + 1] - getSumUnit(1, y, u - 1, v);
    return result;
}

void calculateAnswerY() {
    memset(prefixSum, 0, sizeof(prefixSum));
    memset(prefixSumTriangle, 0, sizeof(prefixSumTriangle));
    memset(prefixSumUnitTriangle, 0, sizeof(prefixSumUnitTriangle));
    int row, col;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            tie(row, col) = getPos(i, j);
            prefixSum[row][col] = j * arr[i][j];
            prefixSumUnitTriangle[row][col] = arr[i][j];
            prefixSumTriangle[row][col] = j * arr[i][j];
        }
    }
    for (int i = 1; i <= nRotate; i++) {
        for (int j = 1; j <= nRotate; j++) {
            prefixSum[i][j] = prefixSum[i][j] + prefixSum[i - 1][j] +
                              prefixSum[i][j - 1] - prefixSum[i - 1][j - 1];
        }
    }
    for (int i = 2; i <= nRotate; i++) {
        for (int j = 1; j <= nRotate; j++) {
            prefixSumUnitTriangle[i][j] = prefixSumUnitTriangle[i][j] +
                                          prefixSumUnitTriangle[i - 1][j] +
                                          prefixSumUnitTriangle[i - 1][j + 1] -
                                          prefixSumUnitTriangle[i - 2][j + 1];

            prefixSumTriangle[i][j] = prefixSumTriangle[i][j] +
                                      prefixSumTriangle[i - 1][j] +
                                      prefixSumTriangle[i - 1][j + 1] -
                                      prefixSumTriangle[i - 2][j + 1];
        }
    }
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            if (radius[i][j] > 0) {
                tie(row, col) = getPos(i, j);
                int tmp = radius[i][j] - 1;
                int sumUnitUpTriangle = getSumUnitTriangleY(
                    row - tmp, col + tmp, row + tmp, col - tmp);
                long long sumUpTriangle =
                    getSumTriangleY(row - tmp, col + tmp, row + tmp, col - tmp);
                int sumUnitArea =
                    getSumUnit(row - tmp, col - tmp, row + tmp, col + tmp);
                long long sumArea =
                    getSum(row - tmp, col - tmp, row + tmp, col + tmp);
                answer[i][j] += j * sumUnitUpTriangle - sumUpTriangle +
                                (sumArea - sumUpTriangle) -
                                j * (sumUnitArea - sumUnitUpTriangle);
            }
        }
    }
}

long long calculateAnswer() {
    long long result = 0;
    int row, col;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            if (radius[i][j] > 0) {
                tie(row, col) = getPos(i, j);
                int tmp = radius[i][j] - 1;
                result += answer[i][j] +
                          radius[i][j] * (k - getSumUnit(row - tmp, col - tmp,
                                                         row + tmp, col + tmp));
            }
        }
    }
    return result;
}

int main() {
    ios_base::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);
    freopen("formation.inp", "r", stdin);
    freopen("formation.out", "w", stdout);
    cin >> n >> m >> k;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= m; j++) {
            cin >> arr[i][j];
        }
    }
    calculateRadius();
    calculateAnswerX();
    calculateAnswerY();
    cout << calculateAnswer();
    return 0;
}
