Vincentqyw
update: features and matchers
437b5f6
raw
history blame
3.64 kB
import torch
import numpy as np
import os
from collections import OrderedDict,namedtuple
import sys
ROOT_DIR = os.path.abspath(os.path.join(os.path.dirname(__file__), ".."))
sys.path.insert(0, ROOT_DIR)
from sgmnet import matcher as SGM_Model
from superglue import matcher as SG_Model
from utils import evaluation_utils
class GNN_Matcher(object):
def __init__(self,config,model_name):
assert model_name=='SGM' or model_name=='SG'
config=namedtuple('config',config.keys())(*config.values())
self.p_th=config.p_th
self.model = SGM_Model(config) if model_name=='SGM' else SG_Model(config)
self.model.cuda(),self.model.eval()
checkpoint = torch.load(os.path.join(config.model_dir, 'model_best.pth'))
#for ddp model
if list(checkpoint['state_dict'].items())[0][0].split('.')[0]=='module':
new_stat_dict=OrderedDict()
for key,value in checkpoint['state_dict'].items():
new_stat_dict[key[7:]]=value
checkpoint['state_dict']=new_stat_dict
self.model.load_state_dict(checkpoint['state_dict'])
def run(self,test_data):
norm_x1,norm_x2=evaluation_utils.normalize_size(test_data['x1'][:,:2],test_data['size1']),\
evaluation_utils.normalize_size(test_data['x2'][:,:2],test_data['size2'])
x1,x2=np.concatenate([norm_x1,test_data['x1'][:,2,np.newaxis]],axis=-1),np.concatenate([norm_x2,test_data['x2'][:,2,np.newaxis]],axis=-1)
feed_data={'x1':torch.from_numpy(x1[np.newaxis]).cuda().float(),
'x2':torch.from_numpy(x2[np.newaxis]).cuda().float(),
'desc1':torch.from_numpy(test_data['desc1'][np.newaxis]).cuda().float(),
'desc2':torch.from_numpy(test_data['desc2'][np.newaxis]).cuda().float()}
with torch.no_grad():
res=self.model(feed_data,test_mode=True)
p=res['p']
index1,index2=self.match_p(p[0,:-1,:-1])
corr1,corr2=test_data['x1'][:,:2][index1.cpu()],test_data['x2'][:,:2][index2.cpu()]
if len(corr1.shape)==1:
corr1,corr2=corr1[np.newaxis],corr2[np.newaxis]
return corr1,corr2
def match_p(self,p):#p N*M
score,index=torch.topk(p,k=1,dim=-1)
_,index2=torch.topk(p,k=1,dim=-2)
mask_th,index,index2=score[:,0]>self.p_th,index[:,0],index2.squeeze(0)
mask_mc=index2[index] == torch.arange(len(p)).cuda()
mask=mask_th&mask_mc
index1,index2=torch.nonzero(mask).squeeze(1),index[mask]
return index1,index2
class NN_Matcher(object):
def __init__(self,config):
config=namedtuple('config',config.keys())(*config.values())
self.mutual_check=config.mutual_check
self.ratio_th=config.ratio_th
def run(self,test_data):
desc1,desc2,x1,x2=test_data['desc1'],test_data['desc2'],test_data['x1'],test_data['x2']
desc_mat=np.sqrt(abs((desc1**2).sum(-1)[:,np.newaxis]+(desc2**2).sum(-1)[np.newaxis]-2*desc1@desc2.T))
nn_index=np.argpartition(desc_mat,kth=(1,2),axis=-1)
dis_value12=np.take_along_axis(desc_mat,nn_index, axis=-1)
ratio_score=dis_value12[:,0]/dis_value12[:,1]
nn_index1=nn_index[:,0]
nn_index2=np.argmin(desc_mat,axis=0)
mask_ratio,mask_mutual=ratio_score<self.ratio_th,np.arange(len(x1))==nn_index2[nn_index1]
corr1,corr2=x1[:,:2],x2[:,:2][nn_index1]
if self.mutual_check:
mask=mask_ratio&mask_mutual
else:
mask=mask_ratio
corr1,corr2=corr1[mask],corr2[mask]
return corr1,corr2