#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<fstream>
#include<map>
#include<ctime>
#include<set>
#include<queue>
#include<cmath>
#include<vector>
#include<bitset>
#include<functional>
#define x first
#define y second
#define mp make_pair
#define pb push_back
#define REP(i,l,r) for((i)=(l);(i)<=(r);++(i))
#define REP2(i,l,r) for((i)=(l);(i)!=(r);++(i))
using namespace std;
typedef long long LL;
typedef double ld;
const int NUM=9;
const int NUMM=22;
const int NUMK=20;
const int P[]={1,1<<2,1<<4,1<<6,1<<8,1<<10,1<<12,1<<14,1<<16,1<<18};
int n,m,K;
char g[NUMM][NUMM];
inline int get(int a,int k)
{
return a/P[k]%4;
}
inline int revise(int a,int k,int d)
{
return a-get(a,k)*P[k]+d*P[k];
}
int ma[ 1<<16 ][NUM];
int hash[1<<16];//判断状态合法
void update(int& a,int p)//更新
{
if(p>0 && ( a==0 || p>a ) )
a=p;
}
void show(int a)//显示状态
{
int i;
REP2(i,0,n+1)
cout<<get(a,i);
cout<<endl;
}
int change(string a)//将字符串变成数
{
int ans=0;
for(int i=n;i>=0;--i)
ans=ans*4+a[i]-'0';
return ans;
}
void pre()//事先计算出所有可能的状态
{
int i,j;
int st[NUM],top;
REP2(i,0,P[n+1])
{
int num=0;
int flag=1;
top=0;
REP2(j,0,n+1)
{
int p=get(i,j);
if(p==3)
++num;
else if(p==0)
;
else if(p==2)
{
if(top<=0)
{
flag=0;
break;
}
ma[i][j]=st[--top];
ma[i][ma[i][j]]=j;
}
else st[top++]=j;
}
if(flag && num<=2 && top==0)
hash[i]=1;
}
}
vector< pair<int,int> > zhuanyi(int i,int j,int oldp,int house,int is)//看拼音知程序
{
//并没有考虑地图,仅仅返回所有的转移
//Oh No,这道题目比较神,必须保证若两个格子相连,那么这两个点之间必须连边。。。
//你只能在S或者T产生两条独立路径或者将路径封闭 用is表示
//要求独立插头在端点产生,且端点必须有独立插头
int p=oldp,np;
int p1=get(p,j-1);
int p2=get(p,j);
p=revise(p,j-1,0);
p=revise(p,j,0);
vector< pair<int,int> > ans;
int left=j>=2?get(house,j-2):3;
int up=get(house,j);
if( ( left==0 && !p1 ) || (up==0 && !p2))//有格子但不相连,否决
{
if(!p1 && !p2)//如果本身不填还是没事的
ans.pb(mp(p,0));
return ans;
}
if(p1==0 && p2==0)
{
//新增左右匹配的插头
np=revise(p,j-1,1);
np=revise(np,j,2);
if(!is)
ans.pb(mp(np,1));
//跳过
np=p;
if(!is)
ans.pb(mp(np,0));
//新增独立向下的独立插头
np=revise(p,j-1,3);
if(is)
ans.pb(mp(np,1));
//新增独立向右的独立插头
np=revise(p,j,3);
if(is)
ans.pb(mp(np,1));
}
else if(p1!=3 && p2!=3)
{
if(p1 && p2)//不可能有独立插头
{
if(p1==p2 && !is)//只能连起来
{
if(p1==2)
{
np=revise(p,ma[oldp][j-1],2);
ans.pb(mp(np,1));
}
else if(p1==1)
{
np=revise(p,ma[oldp][j],1);
ans.pb(mp(np,1));
}
}
else if(p1==2 && p2==1 && !is)//可以连起来
{
np=p;
ans.pb(mp(np,1));
}
else //1,2的话,那么就形成了回路,题目求的是路径,故否决
{
;
}
}
else
{
int u=p1?p1:p2;//两种情况似乎可以合并
//上面有东西,作为傻叉,我可以将其继续
//往下连
np=revise(p,j-1,u);
if(!is)
ans.pb(mp(np,1));
//拐了个弯
np=revise(p,j,u);
if(!is)
ans.pb(mp(np,1));
//封住
if(p2 && is)//往下的封住了
{
np=revise(p,ma[oldp][j],3);//相连的那个成了可怜的独立插头
ans.pb(mp(np,1));
}
else if(p1 && is)
{
np=revise(p,ma[oldp][j-1],3);//同上
ans.pb(mp(np,1));
}
}
}
else //现在肯定有了个独立插头,yy开始
{
if(p1==3 && p2==3 && !is)//两个独立插头,如果连起来的话就结束算法,找到答案。
{
if(!p)
ans.pb(mp(p,1));
}
else if(!p1 || !p2)//只有一个独立插头
{
//首先似乎可以更新答案
if(!p && is)//封住了
ans.pb(mp(p,1));
//然后可以往下或往右拓展
//往下连
np=revise(p,j-1,3);
if(!is)ans.pb(mp(np,1));
//拐了个弯
np=revise(p,j,3);
if(!is)ans.pb(mp(np,1));
}
else //另一个插头会与独立插头相连,于是修改另一边为独立插头
{
if(p1==3 && !is)
{
np=revise(p,ma[oldp][j],3);
ans.pb(mp(np,1));
}
else if(p2==3 && !is)
{
np=revise(p,ma[oldp][j-1],3);
ans.pb(mp(np,1));
}
}
}
return ans;
}
/*
x,y
y y+1 y+2
y-2 y-1
*/
pair<int,int> getHouse(int x,int y,int old,int what)//更新炮台安放
{
//x,y 格子从1开始标号
//what 0 路线 1 炮台 2 障碍
int num[4]={0,0,0,0};
if(y>1)
{
++num[get(old,y-2)];
++num[get(old,y-1)];
}
++num[get(old,y)];
if(y+1<=n)
++num[get(old,y+1)];
int w=what<2 ? num[!what] : 0;
/*
cout<<x<<" "<<y<<" "<<what<<endl;
show(old);
show(revise(old,y-1,what));
cout<<endl;
*/
return mp(revise(old,y-1,what),w);
}
map< pair<int,int> ,int > G[NUMK],F[NUMK];
int answer=-100000000;
int find(int x)
{
int i;
REP2(i,0,n+1)
if(get(x,i)==0)
return 1;
return 0;
}
void work(int x,int y)
{
char p='#';
if(y)
p=g[x-1][y-1];
map< pair<int,int> ,int >::iterator it;
int i,j,k;
REP(k,0,K)
{
G[k]=F[k];
F[k].clear();
}
int pp=0;
REP2(i,0,n+1)
pp=pp*4+2;
if(p!='S' && p!='T')F[0][mp(0,pp)]=1;
REP(k,0,K)
REP2(it,G[k].begin(),G[k].end())
{
if(!y)
{
if(get(it->x.x,n)==0)
update(F[k][mp(it->x.x*4,revise(it->x.y*4+2,n+1,0))],it->y);
continue;
}
vector< pair<int,int > > ans=zhuanyi(x,y,it->x.x,it->x.y,p=='S' || p=='T');
REP2(i,0,3)
{
if( ( p=='S' || p=='T' ) && i!=0)
continue;
if(p=='#' && i!=2)
continue;
REP2(j,0,ans.size())
{
if(ans[j].y && i!=0)
continue;
if(!ans[j].y && i==0)
continue;
int nk=k+(i==1);
if(nk>K)continue;
int np=ans[j].x;
//状态不合法
if(hash[np]==0)
continue;
pair<int,int> nf=getHouse(x,y,it->x.y,i);
if(!np)
{
answer=max(answer,nf.y+it->y);
if(!find(nf.x))
continue;
}
update(F[nk][mp(np,nf.x)],nf.y+it->y);
}
}
}
/*
int sum=0;
REP(k,0,K)
{
sum+=F[k].size();
REP2(it,F[k].begin(),F[k].end())
{
printf("%d %d %d\n",x,y,k);
show(it->x.x);
show(it->x.y);
printf("%d\n",it->y);
printf("\n");
}
}
*/
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("input.txt","r",stdin);freopen("output.txt","w",stdout);
#endif
int i,j;
scanf("%d%d%d",&m,&n,&K);
REP2(i,0,m)
{
char str[100];
scanf("%s",str);
REP2(j,0,n)
g[i][j]=str[j];
}
if(n>m)
{
int temp[NUMM][NUMM];
REP2(i,0,m)
REP2(j,0,n)
temp[j][i]=g[i][j];
swap(n,m);
REP2(i,0,m)
REP2(j,0,n)
g[i][j]=temp[i][j];
}
pre();
REP(i,1,m)
REP(j,0,n)
work(i,j);
printf("%d\n",answer-1);
return 0;
}
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<fstream>
#include<map>
#include<ctime>
#include<set>
#include<queue>
#include<cmath>
#include<vector>
#include<bitset>
#include<functional>
#define x first
#define y second
#define mp make_pair
#define pb push_back
#define REP(i,l,r) for((i)=(l);(i)<=(r);++(i))
#define REP2(i,l,r) for((i)=(l);(i)!=(r);++(i))
using namespace std;

typedef long long LL;
typedef double ld;

const int NUM=9;
const int NUMM=22;
const int NUMK=20;

const int P[]={1,1<<2,1<<4,1<<6,1<<8,1<<10,1<<12,1<<14,1<<16,1<<18};

int n,m,K;
char g[NUMM][NUMM];

inline int get(int a,int k)
{
	return a/P[k]%4;
}

inline int revise(int a,int k,int d)
{
	return a-get(a,k)*P[k]+d*P[k];
}

int ma[ 1<<16 ][NUM];
int hash[1<<16];//判断状态合法

void update(int& a,int p)//更新
{
	if(p>0 && ( a==0 || p>a ) )
		a=p;
}

void show(int a)//显示状态
{
	int i;
	REP2(i,0,n+1)
		cout<<get(a,i);
	cout<<endl;
}

int change(string a)//将字符串变成数
{
	int ans=0;
	for(int i=n;i>=0;--i)
		ans=ans*4+a[i]-'0';
	return ans;
}

void pre()//事先计算出所有可能的状态
{
	int i,j;
	int st[NUM],top;
	REP2(i,0,P[n+1])
	{
		int num=0;
		int flag=1;
		top=0;
		REP2(j,0,n+1)
		{
			int p=get(i,j);
			if(p==3)
				++num;
			else if(p==0)
				;
			else if(p==2)
			{
				if(top<=0)
				{
					flag=0;
					break;
				}
				ma[i][j]=st[--top];
				ma[i][ma[i][j]]=j;
			}
			else st[top++]=j;
		}
		if(flag && num<=2 && top==0)
			hash[i]=1;
	}
}

vector< pair<int,int> > zhuanyi(int i,int j,int oldp,int house,int is)//看拼音知程序
{
	//并没有考虑地图,仅仅返回所有的转移
	//Oh No,这道题目比较神，必须保证若两个格子相连，那么这两个点之间必须连边。。。
	//你只能在S或者T产生两条独立路径或者将路径封闭 用is表示
	//要求独立插头在端点产生，且端点必须有独立插头
	int p=oldp,np;
	int p1=get(p,j-1);
	int p2=get(p,j);
	p=revise(p,j-1,0);
	p=revise(p,j,0);
	vector< pair<int,int> > ans; 
	int left=j>=2?get(house,j-2):3;
	int up=get(house,j);
	if( ( left==0 && !p1 ) || (up==0 && !p2))//有格子但不相连，否决
	{
		if(!p1 && !p2)//如果本身不填还是没事的
			ans.pb(mp(p,0));
		return ans;
	}
	if(p1==0 && p2==0)
	{
		//新增左右匹配的插头
		np=revise(p,j-1,1);
		np=revise(np,j,2);
		if(!is)
			ans.pb(mp(np,1));
		//跳过
		np=p;
		if(!is)
			ans.pb(mp(np,0));
		//新增独立向下的独立插头
		np=revise(p,j-1,3);
		if(is)
			ans.pb(mp(np,1));
		//新增独立向右的独立插头
		np=revise(p,j,3);
		if(is)
			ans.pb(mp(np,1));
	}
	else if(p1!=3 && p2!=3)
	{
		if(p1 && p2)//不可能有独立插头
		{
			if(p1==p2 && !is)//只能连起来
			{
				if(p1==2)
				{
					np=revise(p,ma[oldp][j-1],2);
					ans.pb(mp(np,1));
				}
				else if(p1==1)
				{
					np=revise(p,ma[oldp][j],1);
					ans.pb(mp(np,1));
				}
			}
			else if(p1==2 && p2==1 && !is)//可以连起来
			{
				np=p;
				ans.pb(mp(np,1));
			}
			else //1,2的话，那么就形成了回路，题目求的是路径,故否决
			{
				;
			}
		}
		else
		{
			int u=p1?p1:p2;//两种情况似乎可以合并
			//上面有东西，作为傻叉，我可以将其继续
			//往下连
			np=revise(p,j-1,u);
			if(!is)
				ans.pb(mp(np,1));
			//拐了个弯
			np=revise(p,j,u);
			if(!is)
				ans.pb(mp(np,1));
			//封住
			if(p2 && is)//往下的封住了
			{
				np=revise(p,ma[oldp][j],3);//相连的那个成了可怜的独立插头
				ans.pb(mp(np,1));
			}
			else if(p1 && is)
			{
				np=revise(p,ma[oldp][j-1],3);//同上
				ans.pb(mp(np,1));
			}
		}
	}
	else //现在肯定有了个独立插头,yy开始
	{
		if(p1==3 && p2==3 && !is)//两个独立插头，如果连起来的话就结束算法，找到答案。
		{
			if(!p)
				ans.pb(mp(p,1));
		}
		else if(!p1 || !p2)//只有一个独立插头
		{
			//首先似乎可以更新答案
			if(!p && is)//封住了
				ans.pb(mp(p,1));
			//然后可以往下或往右拓展
			//往下连
			np=revise(p,j-1,3);
			if(!is)ans.pb(mp(np,1));
			//拐了个弯
			np=revise(p,j,3);
			if(!is)ans.pb(mp(np,1));
		}
		else //另一个插头会与独立插头相连，于是修改另一边为独立插头
		{
			if(p1==3 && !is)
			{
				np=revise(p,ma[oldp][j],3);
				ans.pb(mp(np,1));
			}
			else if(p2==3 && !is)
			{
				np=revise(p,ma[oldp][j-1],3);
				ans.pb(mp(np,1));
			}
		}
	}
	return ans;
}

/*
   x,y
       y   y+1 y+2
   y-2 y-1
   */

pair<int,int> getHouse(int x,int y,int old,int what)//更新炮台安放
{
	//x,y 格子从1开始标号
	//what 0 路线 1 炮台 2 障碍
	int num[4]={0,0,0,0};
	if(y>1)
	{
		++num[get(old,y-2)];
		++num[get(old,y-1)];
	}
	++num[get(old,y)];
	if(y+1<=n)
		++num[get(old,y+1)];
	int w=what<2 ? num[!what] : 0;
	/*
	cout<<x<<" "<<y<<" "<<what<<endl;
	show(old);
	show(revise(old,y-1,what));
	cout<<endl;
	*/
	return mp(revise(old,y-1,what),w);
}

map< pair<int,int> ,int > G[NUMK],F[NUMK];
int answer=-100000000;

int find(int x)
{
	int i;
	REP2(i,0,n+1)
		if(get(x,i)==0)
			return 1;
	return 0;
}

void work(int x,int y)
{
	char p='#';
	if(y)
		p=g[x-1][y-1];
	map< pair<int,int> ,int >::iterator it;
	int i,j,k;
	REP(k,0,K)
	{
		G[k]=F[k];
		F[k].clear();
	}
	int pp=0;
	REP2(i,0,n+1)
		pp=pp*4+2;
	if(p!='S' && p!='T')F[0][mp(0,pp)]=1;
	REP(k,0,K)
		REP2(it,G[k].begin(),G[k].end())
		{
			if(!y)
			{
				if(get(it->x.x,n)==0)
					update(F[k][mp(it->x.x*4,revise(it->x.y*4+2,n+1,0))],it->y);
				continue;
			}
			vector< pair<int,int > > ans=zhuanyi(x,y,it->x.x,it->x.y,p=='S' || p=='T');
			REP2(i,0,3)
			{
				if( ( p=='S' || p=='T' ) && i!=0)
					continue;
				if(p=='#' && i!=2)
					continue;
				REP2(j,0,ans.size())
				{
					if(ans[j].y && i!=0)
						continue;
					if(!ans[j].y && i==0)
						continue;
					int nk=k+(i==1);
					if(nk>K)continue;
					int np=ans[j].x;
					//状态不合法
					if(hash[np]==0)
						continue;
					pair<int,int> nf=getHouse(x,y,it->x.y,i);
					if(!np)
					{
						answer=max(answer,nf.y+it->y);
						if(!find(nf.x))
							continue;
					}
					update(F[nk][mp(np,nf.x)],nf.y+it->y);
				}
			}
		}
	/*
	int sum=0;
	REP(k,0,K)
	{
		sum+=F[k].size();
		REP2(it,F[k].begin(),F[k].end())
		{
			printf("%d %d %d\n",x,y,k);
			show(it->x.x);
			show(it->x.y);
			printf("%d\n",it->y);
			printf("\n");
		}
	}
	*/
}

int main()
{
#ifndef ONLINE_JUDGE
	freopen("input.txt","r",stdin);freopen("output.txt","w",stdout);
#endif
	int i,j;
	scanf("%d%d%d",&m,&n,&K);
	REP2(i,0,m)
	{
		char str[100];
		scanf("%s",str);
		REP2(j,0,n)
			g[i][j]=str[j];
	}
	if(n>m)
	{
		int temp[NUMM][NUMM];
		REP2(i,0,m)
			REP2(j,0,n)
				temp[j][i]=g[i][j];
		swap(n,m);
		REP2(i,0,m)
			REP2(j,0,n)
				g[i][j]=temp[i][j];
	}
	pre();
	REP(i,1,m)
		REP(j,0,n)
			work(i,j);
	printf("%d\n",answer-1);
	return 0;
}
