From fd1539af9f636a4ad272315d5cfba562288d0858 Mon Sep 17 00:00:00 2001
From: scott snyder <sss@karma>
Date: Sat, 27 Jul 2019 16:23:56 +0200
Subject: [PATCH] TrkVertexFitters: Add some basic unit tests.

Add some basic unit tests, in preparation for constness fixes.
---
 .../TrkVertexFitters/CMakeLists.txt           |  15 +
 .../share/AdaptiveMultiVertexFitter_test.py   |  64 ++
 .../share/AdaptiveMultiVertexFitter_test.ref  |   2 +
 .../share/AdaptiveVertexFitter_test.py        |  67 ++
 .../share/AdaptiveVertexFitter_test.ref       |   2 +
 .../src/AdaptiveMultiVertexFitter.cxx         |   4 +-
 .../src/AdaptiveMultiVertexFitterTestAlg.cxx  | 364 +++++++++
 .../src/AdaptiveMultiVertexFitterTestAlg.h    |  52 ++
 .../src/AdaptiveVertexFitter.cxx              |  54 +-
 .../src/AdaptiveVertexFitterTestAlg.cxx       | 755 ++++++++++++++++++
 .../src/AdaptiveVertexFitterTestAlg.h         |  57 ++
 .../components/TrkVertexFitters_entries.cxx   |   4 +
 12 files changed, 1410 insertions(+), 30 deletions(-)
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.py
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.ref
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.py
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.ref
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.cxx
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.h
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.cxx
 create mode 100644 Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.h

diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/CMakeLists.txt b/Tracking/TrkVertexFitter/TrkVertexFitters/CMakeLists.txt
index 9cbb8c70af47..cf3e86f316b6 100644
--- a/Tracking/TrkVertexFitter/TrkVertexFitters/CMakeLists.txt
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/CMakeLists.txt
@@ -33,3 +33,18 @@ atlas_add_component( TrkVertexFitters
                      src/components/*.cxx
                      LINK_LIBRARIES AthenaBaseComps xAODTracking GaudiKernel TrkParameters TrkParametersBase TrkParticleBase VxVertex TrkVertexFitterInterfaces TrkSurfaces TrkLinks TrkTrack VxMultiVertex TrkExInterfaces TrkVertexFittersLib )
 
+
+# Install files from the package:
+atlas_install_joboptions( share/*.py )
+
+
+atlas_add_test( AdaptiveVertexFitter_test
+                SCRIPT athena.py TrkVertexFitters/AdaptiveVertexFitter_test.py
+                PROPERTIES TIMEOUT 300
+                EXTRA_PATTERNS " INFO |WARNING |found service|Adding private|^ +[+]|HepPDT Version|^indet|^pixel|^phi|^eta|^GEOPIXEL|in material|no dictionary|^subdet|^part|^barrel_endcap|^layer|^sct|^bec|^side0|^strip0|^lay_disk|TrackingGeometrySvc|^PixelID" )
+
+
+atlas_add_test( AdaptiveMultiVertexFitter_test
+                SCRIPT athena.py TrkVertexFitters/AdaptiveMultiVertexFitter_test.py
+                PROPERTIES TIMEOUT 300
+                EXTRA_PATTERNS " INFO |WARNING |found service|Adding private|^ +[+]|HepPDT Version|^indet|^pixel|^phi|^eta|^GEOPIXEL|in material|no dictionary|^subdet|^part|^barrel_endcap|^layer|^sct|^bec|^side0|^strip0|^lay_disk|TrackingGeometrySvc|^PixelID" )
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.py b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.py
new file mode 100644
index 000000000000..b5527ad49615
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.py
@@ -0,0 +1,64 @@
+#
+# Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration.
+#
+# File: TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.py
+# Author: scott snyder <snyder@bnl.gov>
+# Data: Jul, 2019
+# Brief: Unit test for AdaptiveMultiVertexFitter.  Incomplete!
+#
+
+
+from __future__ import print_function
+
+
+from AthenaCommon.DetFlags      import DetFlags
+DetFlags.detdescr.ID_setOn()
+
+RunNumber = 284500
+
+import sys
+import string
+import ROOT
+import math
+from AtlasGeoModel import SetGeometryVersion
+from AtlasGeoModel import GeoModelInit
+from AtlasGeoModel import SetupRecoGeometry
+
+from GeoModelSvc.GeoModelSvcConf import GeoModelSvc
+ServiceMgr += GeoModelSvc()
+theApp.CreateSvc += [ "GeoModelSvc"]
+from AtlasGeoModel import InDetGM
+
+from IOVDbSvc.IOVDbSvcConf import IOVDbSvc
+IOVDbSvc().GlobalTag = 'OFLCOND-RUN12-SDR-35'
+
+import MagFieldServices.SetupField
+
+from TrkDetDescrSvc.AtlasTrackingGeometrySvc import AtlasTrackingGeometrySvc
+
+from AthenaCommon.AlgSequence import AlgSequence
+topSequence = AlgSequence()
+
+theApp.EvtMax=2
+
+from xAODEventInfoCnv.xAODEventInfoCnvConf import xAODMaker__EventInfoCnvAlg
+eialg = xAODMaker__EventInfoCnvAlg (DoBeginRun = False)
+topSequence += eialg
+
+
+
+# Suppress useless GeoModelSvc messages.
+from AthenaCommon import Constants
+GeoModelSvc().OutputLevel=Constants.WARNING
+
+from TrkExTools.AtlasExtrapolator import AtlasExtrapolator
+
+
+from TrkVertexFitters.TrkVertexFittersConf import \
+    Trk__AdaptiveMultiVertexFitterTestAlg, Trk__AdaptiveMultiVertexFitter
+fitter = Trk__AdaptiveMultiVertexFitter ('AdaptiveMultiVertexFitter',
+                                         OutputLevel = INFO)
+testalg1 = Trk__AdaptiveMultiVertexFitterTestAlg ('testalg1',
+                                                  OutputLevel = VERBOSE,
+                                                  Tool = fitter)
+topSequence += testalg1
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.ref b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.ref
new file mode 100644
index 000000000000..973203c71168
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveMultiVertexFitter_test.ref
@@ -0,0 +1,2 @@
+testalg1          VERBOSE execute
+testalg1          VERBOSE execute
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.py b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.py
new file mode 100644
index 000000000000..083ac972875a
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.py
@@ -0,0 +1,67 @@
+#
+# Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration.
+#
+# File: TrkVertexFitters/share/AdaptiveVertexFitter_test.py
+# Author: scott snyder <snyder@bnl.gov>
+# Data: Jul, 2019
+# Brief: Unit test for AdaptiveVertexFitter.  Incomplete!
+#
+
+
+from __future__ import print_function
+
+
+from AthenaCommon.DetFlags      import DetFlags
+DetFlags.detdescr.ID_setOn()
+
+RunNumber = 284500
+
+import sys
+import string
+import ROOT
+import math
+from AtlasGeoModel import SetGeometryVersion
+from AtlasGeoModel import GeoModelInit
+from AtlasGeoModel import SetupRecoGeometry
+
+from GeoModelSvc.GeoModelSvcConf import GeoModelSvc
+ServiceMgr += GeoModelSvc()
+theApp.CreateSvc += [ "GeoModelSvc"]
+from AtlasGeoModel import InDetGM
+
+from IOVDbSvc.IOVDbSvcConf import IOVDbSvc
+IOVDbSvc().GlobalTag = 'OFLCOND-RUN12-SDR-35'
+
+import MagFieldServices.SetupField
+
+from TrkDetDescrSvc.AtlasTrackingGeometrySvc import AtlasTrackingGeometrySvc
+
+from AthenaCommon.AlgSequence import AlgSequence
+topSequence = AlgSequence()
+
+theApp.EvtMax=2
+
+from xAODEventInfoCnv.xAODEventInfoCnvConf import xAODMaker__EventInfoCnvAlg
+eialg = xAODMaker__EventInfoCnvAlg (DoBeginRun = False)
+topSequence += eialg
+
+
+
+# Suppress useless GeoModelSvc messages.
+from AthenaCommon import Constants
+GeoModelSvc().OutputLevel=Constants.WARNING
+
+#from AthenaCommon.GlobalFlags import globalflags
+#from InDetRecExample.InDetJobProperties import InDetFlags
+#include ('InDetRecExample/InDetRecLoadTools.py')
+from TrkExTools.AtlasExtrapolator import AtlasExtrapolator
+
+
+from TrkVertexFitters.TrkVertexFittersConf import \
+    Trk__AdaptiveVertexFitterTestAlg, Trk__AdaptiveVertexFitter
+fitter = Trk__AdaptiveVertexFitter ('AdaptiveVertexFitter',
+                                    OutputLevel = INFO)
+testalg1 = Trk__AdaptiveVertexFitterTestAlg ('testalg1',
+                                             OutputLevel = VERBOSE,
+                                             Tool = fitter)
+topSequence += testalg1
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.ref b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.ref
new file mode 100644
index 000000000000..973203c71168
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/share/AdaptiveVertexFitter_test.ref
@@ -0,0 +1,2 @@
+testalg1          VERBOSE execute
+testalg1          VERBOSE execute
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitter.cxx b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitter.cxx
index ec90d1b95be7..fb3eec7bed9c 100755
--- a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitter.cxx
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitter.cxx
@@ -1,5 +1,5 @@
 /*
-  Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration
+  Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration
 */
 
 
@@ -371,7 +371,7 @@ namespace Trk
 	
         // std::cout << " step: " << num_steps << " considering shift of: " << *iter << " vtx z: " <<  (*iter)->recVertex().position().z() <<std::endl;
 
-        Amg::Vector3D vrtpos(3,0);
+        Amg::Vector3D vrtpos;
 
         vrtpos[0]=oldpositions[*iter][0]-(*iter)->position()[0];
         vrtpos[1]=oldpositions[*iter][1]-(*iter)->position()[1];
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.cxx b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.cxx
new file mode 100644
index 000000000000..223ca0c249fa
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.cxx
@@ -0,0 +1,364 @@
+/*
+ * Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration.
+ */
+/**
+ * @file TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.cxx
+ * @author scott snyder <snyder@bnl.gov>
+ * @date Jul, 2019
+ * @brief Algorithm for testing AdaptiveMultiVertexFitter.
+ */
+
+
+#undef NDEBUG
+#include "AdaptiveMultiVertexFitterTestAlg.h"
+#include "xAODTracking/Vertex.h"
+#include "xAODTracking/TrackParticle.h"
+#include "VxMultiVertex/MvfFitInfo.h"
+#include "VxMultiVertex/MVFVxTrackAtVertex.h"
+#include "VxMultiVertex/TrackToVtxLink.h"
+#include "TrkTrack/Track.h"
+#include "TrkParticleBase/TrackParticleBase.h"
+#include "EventPrimitives/EventPrimitivesHelpers.h"
+#include "TestTools/FLOATassert.h"
+#include "GaudiKernel/SystemOfUnits.h"
+#include <cassert>
+
+
+#include "CLHEP/Vector/LorentzVector.h"
+
+using Gaudi::Units::mm;
+using Gaudi::Units::MeV;
+using Gaudi::Units::GeV;
+
+
+namespace {
+
+
+std::unique_ptr<AmgSymMatrix(5)> cov5()
+{
+  auto m = std::make_unique<AmgSymMatrix(5)>();
+  m->setIdentity();
+  return m;
+}
+
+
+std::unique_ptr<AmgSymMatrix(5)> cov5a()
+{
+  auto m = std::make_unique<AmgSymMatrix(5)>();
+  m->setZero();
+  for (int i=0; i < 5; i++) {
+    (*m)(i,i) = 1e-2;
+  }
+  (*m)(1,1)=1;
+  return m;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::Perigee> > PerigeeUVec_t;
+PerigeeUVec_t makePerigees1()
+{
+  Amg::Vector3D pos0 { 0, 0, 0 };
+
+  Amg::Vector3D pos1a { 2*mm, 1*mm, -10*mm };
+  Amg::Vector3D mom1a { 400*MeV, 600*MeV, 200*MeV };
+  Amg::Vector3D pos1b { 1*mm, 2*mm, -3*mm };
+  Amg::Vector3D mom1b { 600*MeV, 400*MeV, -200*MeV };
+  Amg::Vector3D pos1c { 1.2*mm, 1.3*mm, -7*mm };
+  Amg::Vector3D mom1c { 300*MeV, 1000*MeV, 100*MeV };
+
+  PerigeeUVec_t ret;
+
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1a, mom1a,  1, pos1a, cov5().release()));
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1b, mom1b, -1, pos1b, cov5().release()));
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1c, mom1c, -1, pos1c, cov5().release()));
+
+  return ret;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::Perigee> > PerigeeUVec_t;
+PerigeeUVec_t makePerigees2()
+{
+  Amg::Vector3D pos1a { 10*mm,   0*mm, -5*mm };
+  Amg::Vector3D mom1a { 1000*MeV, 0*MeV, 0*MeV };
+  Amg::Vector3D pos1b { 10.5*mm, 0.5*mm, -5.5*mm };
+  Amg::Vector3D mom1b { 800*MeV, 200*MeV, 200*MeV };
+  Amg::Vector3D pos1c { 9.5*mm, -0.5*mm, -4.5*mm };
+  Amg::Vector3D mom1c { 700*MeV, -300*MeV, -200*MeV };
+
+  PerigeeUVec_t ret;
+
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1a, mom1a,  1, pos1a, cov5a().release()));
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1b, mom1b, -1, pos1a, cov5a().release()));
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1c, mom1c, -1, pos1a, cov5a().release()));
+
+  return ret;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::Track> > TrackUVec_t;
+TrackUVec_t makeTracks (const PerigeeUVec_t& perigees)
+{
+  TrackUVec_t tracks;
+
+  for (const std::unique_ptr<Trk::Perigee>& p : perigees) {
+    Trk::TrackInfo info (Trk::TrackInfo::Unknown, Trk::undefined);
+    auto fqual = std::make_unique<Trk::FitQuality> (0, 0);
+    auto tsos = std::make_unique<DataVector<const Trk::TrackStateOnSurface> >();
+    std::bitset<Trk::TrackStateOnSurface::NumberOfTrackStateOnSurfaceTypes> typePattern(0);
+    typePattern.set(Trk::TrackStateOnSurface::Perigee);
+    tsos->push_back (std::make_unique<Trk::TrackStateOnSurface>
+                     (nullptr, p->clone(), nullptr, nullptr, typePattern));
+    tracks.push_back (std::make_unique<Trk::Track> (info,
+                                                    tsos.release(),
+                                                    fqual.release()));
+  }
+
+  return tracks;
+}
+
+
+void dumpCovariance (const AmgSymMatrix(5)& m)
+{
+  for (int i=0; i < 5; i++) {
+    for (int j=0; j < 5; j++) {
+      std::cout << m(i,j) << ", ";
+    }
+  }
+}
+
+
+[[maybe_unused]]
+void dumpVertex (const xAOD::Vertex& v)
+{
+  std::cout << "vvv\n";
+  std::cout << v.x() << ", " << v.y() << ", " << v.z() << "\n";
+  if (v.isAvailable<short> ("vertexType")) {
+    std::cout << "vertexType " << v.vertexType() << "\n";
+  }
+  std::cout << "chi2/ndof " << v.chiSquared() << ", " << v.numberDoF() << "\n";
+
+  std::cout << "cov ";
+  for (float f : v.covariance()) {
+    std::cout << f << ", ";
+  }
+  std::cout << "\n";
+
+  if (v.isAvailable<std::vector<ElementLink<xAOD::TrackParticleContainer> > > ("trackParticleLinks")) {
+    std::cout << "tplinks ";
+    for (const ElementLink< xAOD::TrackParticleContainer >& l : v.trackParticleLinks()) {
+      std::cout << l.dataID() << "/" << l.index() << " ";
+    }
+    std::cout << "\n";
+  }
+
+  if (v.isAvailable<std::vector<float> > ("trackWeights")) {
+    std::cout << "wt ";
+    for (float f : v.trackWeights()) {
+      std::cout << f << " ";
+    }
+    std::cout << "\n";
+  }
+
+  if (v.isAvailable<std::vector<ElementLink<xAOD::NeutralParticleContainer> > > ("neutralParticleLinks")) {
+    std::cout << "nplinks ";
+    for (const ElementLink< xAOD::NeutralParticleContainer >& l : v.neutralParticleLinks()) {
+      std::cout << l.dataID() << "/" << l.index() << " ";
+    }
+    std::cout << "\n";
+  }
+
+  if (v.isAvailable<std::vector<float> > ("neutralWeights")) {
+    std::cout << "wt ";
+    for (float f : v.neutralWeights()) {
+      std::cout << f << " ";
+    }
+    std::cout << "\n";
+  }
+
+  std::cout << v.vxTrackAtVertexAvailable() << "\n";
+  for (const Trk::VxTrackAtVertex& vv : v.vxTrackAtVertex()) {
+    vv.dump (std::cout);
+    std::cout << "cov ";
+    if (vv.perigeeAtVertex()) {
+      dumpCovariance (*vv.perigeeAtVertex()->covariance());
+    }
+    else {
+      std::cout << "(null)";
+    }
+    std::cout << "\n";
+  }
+}
+
+
+void assertVec3D (const char* which,
+                  const Amg::Vector3D& a,
+                  const Amg::Vector3D& b,
+                  double thresh = 2e-5)
+{
+  if ( ! Athena_test::isEqual (a.x(), b.x(), thresh) ||
+       ! Athena_test::isEqual (a.y(), b.y(), thresh) ||
+       ! Athena_test::isEqual (a.z(), b.z(), thresh) )
+  {
+    std::cerr << "TrkVKalVrtFitterTestAlg::assertVec3D mismatch " << which
+              << ": ["
+              << a.x() << ", "
+              << a.y() << ", "
+              << a.z() << "] / ["
+              << b.x() << ", "
+              << b.y() << ", "
+              << b.z() << "]\n";
+    std::abort();
+  }
+}
+
+
+void compareVertex (const xAOD::Vertex& a, const xAOD::Vertex& b)
+{
+  assertVec3D ("vertex pos", a.position(), b.position(), 5e-4);
+  assert (Athena_test::isEqual (a.chiSquared(), b.chiSquared(), 5e-3) );
+  assert (Athena_test::isEqual (a.numberDoF(), b.numberDoF(), 5e-3) );
+
+  assert (a.covariance().size() == b.covariance().size());
+  for (unsigned int i = 0; i < a.covariance().size(); i++) {
+    if (isinf(a.covariance()[i]) && isinf(b.covariance()[i])) continue;
+    assert (Athena_test::isEqual (a.covariance()[i], b.covariance()[i], 2e-2) );
+  }
+}
+
+
+void setInitialPerigee (xAOD::Vertex& v, unsigned i, const Trk::Perigee* p)
+{
+  std::vector< Trk::VxTrackAtVertex >& vec = v.vxTrackAtVertex();
+  if (vec.size() <= i) vec.resize(i+1);
+  vec[i].setInitialPerigee (p);
+}
+
+
+void setInitialPerigees (xAOD::Vertex& v, const TrackUVec_t& tracks)
+{
+  int i = 0;
+  for (const std::unique_ptr<Trk::Track>& t : tracks) {
+    setInitialPerigee (v, i, t->perigeeParameters());
+    ++i;
+  }
+}
+
+
+struct VertexInfo
+{
+  xAOD::Vertex v;
+  TrackUVec_t tracks;
+  PerigeeUVec_t perigees;
+  std::unique_ptr<Trk::MvfFitInfo> fi;
+  std::vector<Trk::VxTrackAtVertex*> vtracks;
+  std::vector<Trk::TrackToVtxLink> links;
+};
+
+
+void initVertex (VertexInfo& vi,
+                 const Amg::Vector3D& pos,
+                 PerigeeUVec_t&& perigees)
+{
+  xAOD::Vertex::Decorator< Trk::MvfFitInfo* > MvfFitInfo("MvfFitInfo");
+  xAOD::Vertex::Decorator< bool > isInitialized("isInitialized");
+  xAOD::Vertex::Decorator< std::vector<Trk::VxTrackAtVertex*> > VTAV("VTAV");
+
+  vi.perigees = std::move (perigees);
+
+  vi.v.makePrivateStore();
+  vi.v.setPosition (pos);
+
+  vi.tracks = makeTracks (vi.perigees);
+  setInitialPerigees (vi.v, vi.tracks);
+
+  auto cv = std::make_unique<xAOD::Vertex>();
+  cv->makePrivateStore();
+  cv->setPosition (pos);
+  Amg::MatrixX looseConstraintCovariance(3, 3);
+  looseConstraintCovariance.setIdentity();
+  looseConstraintCovariance = looseConstraintCovariance * 1e+8;
+  cv->setCovariancePosition (looseConstraintCovariance);
+  cv->setFitQuality (0, -3);
+  vi.fi = std::make_unique<Trk::MvfFitInfo>  (cv.release(),
+                                              new Amg::Vector3D(pos),
+                                              new Amg::Vector3D(pos));
+  MvfFitInfo(vi.v) = vi.fi.get();
+  isInitialized(vi.v) = false;
+  
+  vi.links.reserve (vi.perigees.size());
+  for (std::unique_ptr<Trk::Perigee>& p : vi.perigees) {
+    vi.links.push_back (new std::vector<xAOD::Vertex*> {&vi.v});
+    auto tav = std::make_unique<Trk::MVFVxTrackAtVertex>(1.5, p->clone(), p.get());
+    tav->setLinkToVertices (&vi.links.back());
+    vi.vtracks.push_back (tav.release());
+  }
+  VTAV(vi.v) = vi.vtracks;
+}
+
+
+} // anonymous namespace
+
+
+namespace Trk {
+
+
+/**
+ * @brief Standard Gaudi initialize method.
+ */
+StatusCode AdaptiveMultiVertexFitterTestAlg::initialize()
+{
+  ATH_CHECK( m_fitter.retrieve() );
+  return StatusCode::SUCCESS;
+}
+
+
+/**
+ * @brief Standard Gaudi execute method.
+ */
+StatusCode AdaptiveMultiVertexFitterTestAlg::execute()
+{
+  ATH_MSG_VERBOSE ("execute");
+
+  ATH_CHECK( test1() );
+
+  return StatusCode::SUCCESS;
+}
+
+
+StatusCode AdaptiveMultiVertexFitterTestAlg::test1()
+{
+  VertexInfo v1;
+  initVertex (v1, {1.5*mm, 1.7*mm, -6*mm}, makePerigees1());
+
+                       
+  VertexInfo v2;
+  initVertex (v2, {9.8*mm, 0.2*mm, -4.8*mm}, makePerigees2());
+
+  std::vector<xAOD::Vertex*> verts {&v1.v, &v2.v};
+  m_fitter->fit (verts);
+
+  xAOD::Vertex exp_v1;
+  exp_v1.makePrivateStore();
+  exp_v1.setPosition ({8.67978, 10.3569, -6.33368});
+  exp_v1.setFitQuality (0.0141012, 0.797808);
+  exp_v1.setCovariance (std::vector<float>
+                        {751.563, 142.738, 950.953,
+                           -7.62904, 47.5311, 67.4557});
+
+  xAOD::Vertex exp_v2;
+  exp_v2.makePrivateStore();
+  exp_v2.setPosition ({7.97404, 0.106101, -4.97333});
+  exp_v2.setFitQuality (1.74029, 1.6738);
+  exp_v2.setCovariance (std::vector<float>
+                        {0.348487, -0.0194744, 0.0290909,
+                           -0.00362647, 0.000706365, 0.445415});
+
+  compareVertex (v1.v, exp_v1);
+  compareVertex (v2.v, exp_v2);
+
+  return StatusCode::SUCCESS;
+}
+
+
+} // namespace Trk
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.h b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.h
new file mode 100644
index 000000000000..b8d77a2e620e
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveMultiVertexFitterTestAlg.h
@@ -0,0 +1,52 @@
+// This file's extension implies that it's C, but it's really -*- C++ -*-.
+/*
+ * Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration.
+ */
+/**
+ * @file TrkVertexFitters/AdaptiveMultiVertexFitterTestAlg.h
+ * @author scott snyder <snyder@bnl.gov>
+ * @date Jul, 2019
+ * @brief Algorithm for testing AdaptiveMultiVertexFitter.
+ */
+
+
+#ifndef TRKVERTEXFITTERS_ADAPTIVEMULTIVERTEXFITTERTESTALG_H
+#define TRKVERTEXFITTERS_ADAPTIVEMULTIVERTEXFITTERTESTALG_H
+
+
+#include "AthenaBaseComps/AthAlgorithm.h"
+#include "TrkVertexFitters/AdaptiveMultiVertexFitter.h"
+#include "GaudiKernel/ToolHandle.h"
+
+
+
+namespace Trk {
+
+
+class AdaptiveMultiVertexFitterTestAlg
+  : public AthAlgorithm
+{
+public:
+  using AthAlgorithm::AthAlgorithm;
+
+
+  /// Standard Gaudi initialize method.
+  virtual StatusCode initialize() override;
+
+  /// Execute the algorithm.
+  virtual StatusCode execute() override;
+
+
+private:
+  StatusCode test1();
+
+
+  ToolHandle<Trk::AdaptiveMultiVertexFitter> m_fitter
+  { this, "Tool", "Trk::AdaptiveMultiVertexFitter", "Tool to test." };
+};
+
+
+} // namespace Trk
+
+
+#endif // not TRKVERTEXFITTERS_ADAPTIVEMULTIVERTEXFITTERTESTALG_H
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitter.cxx b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitter.cxx
index fb80726f59c9..787fd74322c0 100755
--- a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitter.cxx
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitter.cxx
@@ -1,5 +1,5 @@
 /*
-  Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration
+  Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration
 */
 
 
@@ -157,25 +157,29 @@ namespace Trk
     }
 
 
-    Amg::Vector3D* SeedPoint(0);
+    Amg::Vector3D SeedPoint;
     //now find the best point for seeding and linearization of the tracks
     if (IsStartingPoint) {
-      SeedPoint=new Amg::Vector3D(startingPoint);
+      SeedPoint=startingPoint;
     } else {
       if (perigeeList.size()>1) {
         if (IsConstraint) {
-          SeedPoint=new Amg::Vector3D(m_SeedFinder->findSeed(perigeeList,&constraint));
+          SeedPoint=m_SeedFinder->findSeed(perigeeList,&constraint);
         } else {
-          SeedPoint=new Amg::Vector3D(m_SeedFinder->findSeed(perigeeList));
+          SeedPoint=m_SeedFinder->findSeed(perigeeList);
         }
-      } else {
-        SeedPoint=new Amg::Vector3D(constraint.position().x(),constraint.position().y(),constraint.position().z());
+      }
+      else if (IsConstraint) {
+        SeedPoint=constraint.position();
+      }
+      else {
+        SeedPoint.setZero();
       }
     }
 
     //in case m_onlyzseed is on, just use only the z component given by the seed finder
     if (m_onlyzseed&&IsConstraint) {
-      *SeedPoint=Amg::Vector3D(constraint.position().x(),constraint.position().y(),SeedPoint->z());
+      SeedPoint=constraint.position();
     }
     
 
@@ -189,7 +193,7 @@ namespace Trk
     std::vector<const Trk::TrackParameters*>::const_iterator perigeesEnd=perigeeList.end();
 
     ATH_MSG_DEBUG("Inside fitter with track perigee parameters.");
-    ATH_MSG_DEBUG("Seed point: " << *SeedPoint);
+    ATH_MSG_DEBUG("Seed point: " << SeedPoint);
     int myDebugNTrk(0);
 
     for (std::vector<const Trk::TrackParameters*>::const_iterator perigeesIter=perigeesBegin;perigeesIter!=perigeesEnd;++perigeesIter) {
@@ -203,8 +207,8 @@ namespace Trk
      
       VxTrackAtVertex* LinTrackToAdd = new VxTrackAtVertex(0., 0, NULL, (*perigeesIter), NULL); //TODO: Must think now about when to delete this memory! -David S.
       
-      //m_LinearizedTrackFactory->linearize(*LinTrackToAdd,*SeedPoint); why linearize it? maybe you don't need it at all!!!!! <19-05-2006>
-      bool success=m_ImpactPoint3dEstimator->addIP3dAtaPlane(*LinTrackToAdd,*SeedPoint);
+      //m_LinearizedTrackFactory->linearize(*LinTrackToAdd,SeedPoint); why linearize it? maybe you don't need it at all!!!!! <19-05-2006>
+      bool success=m_ImpactPoint3dEstimator->addIP3dAtaPlane(*LinTrackToAdd,SeedPoint);
       if (!success)
       {
 	msg(MSG::WARNING) << "Adding compatibility to vertex information failed. Newton distance finder didn't converge..." << endmsg;
@@ -220,7 +224,7 @@ namespace Trk
     std::vector<const Trk::NeutralParameters*>::const_iterator neutralPerigeesEnd  =neutralPerigeeList.end();
 
     ATH_MSG_DEBUG("Inside fitter with neutral perigee parameters.");
-    ATH_MSG_DEBUG("Seed point: " << *SeedPoint);
+    ATH_MSG_DEBUG("Seed point: " << SeedPoint);
     int myDebugNNeutral(0);
     for (std::vector<const Trk::NeutralParameters*>::const_iterator neutralPerigeesIter=neutralPerigeesBegin;neutralPerigeesIter!=neutralPerigeesEnd;++neutralPerigeesIter) {
 
@@ -230,7 +234,7 @@ namespace Trk
 
       VxTrackAtVertex* LinTrackToAdd = new VxTrackAtVertex(0., 0, NULL, NULL, (*neutralPerigeesIter) ); //TODO: Must think now about when to delete this memory! -David S.
      
-      bool success = m_ImpactPoint3dEstimator->addIP3dAtaPlane(*LinTrackToAdd,*SeedPoint);
+      bool success = m_ImpactPoint3dEstimator->addIP3dAtaPlane(*LinTrackToAdd,SeedPoint);
       if (!success)
       {
 	msg(MSG::WARNING) << "Adding compatibility to vertex information failed. Newton distance finder didn't converge..." << endmsg;
@@ -253,28 +257,25 @@ namespace Trk
 
 
    
-    xAOD::Vertex* ConstraintVertex(0);
+    xAOD::Vertex ConstraintVertex;
+    ConstraintVertex.makePrivateStore();
     //use the previous prior vertex info for the initial vertex if there
     if (IsConstraint ) {
-      ConstraintVertex=new xAOD::Vertex();
-      ConstraintVertex->makePrivateStore();
-      ConstraintVertex->setPosition( constraint.position() );
-      ConstraintVertex->setCovariancePosition( constraint.covariancePosition() );
-      ConstraintVertex->setFitQuality( 0., 0. );
+      ConstraintVertex.setPosition( constraint.position() );
+      ConstraintVertex.setCovariancePosition( constraint.covariancePosition() );
+      ConstraintVertex.setFitQuality( 0., 0. );
     } else {
       AmgSymMatrix(3) startingCovMatrix;
       startingCovMatrix.setIdentity();
       startingCovMatrix = startingCovMatrix / m_initialError;
       //now initialize with starting position and covariance matrix
-      ConstraintVertex=new xAOD::Vertex();
-      ConstraintVertex->makePrivateStore();
-      ConstraintVertex->setPosition( *SeedPoint );
-      ConstraintVertex->setCovariancePosition( startingCovMatrix );
-      ConstraintVertex->setFitQuality( 0., -3. );
+      ConstraintVertex.setPosition( SeedPoint );
+      ConstraintVertex.setCovariancePosition( startingCovMatrix );
+      ConstraintVertex.setFitQuality( 0., -3. );
     }
 
     //now put the linearizedtracks into VxTrackAtVertex
-    return dothefit(*ConstraintVertex,*SeedPoint,theLinTracks);
+    return dothefit(ConstraintVertex,SeedPoint,theLinTracks);
 
 
   }
@@ -599,9 +600,6 @@ namespace Trk
                       << "the ndf of the vertex is at fit end: " << ActualVertex->numberDoF() << endmsg;
     }
 
-    delete &ConstraintVertex;
-    delete &SeedVertex;
-
     // TODO: get rid of following line
     //std::cout << "Number of steps: " << num_steps << ". Number of relinearizations: " << num_relinearizations << "." << std::endl << std::endl;
 
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.cxx b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.cxx
new file mode 100644
index 000000000000..b1a69f7731d5
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.cxx
@@ -0,0 +1,755 @@
+/*
+ * Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration.
+ */
+/**
+ * @file TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.cxx
+ * @author scott snyder <snyder@bnl.gov>
+ * @date Jul, 2019
+ * @brief Algorithm for testing AdaptiveVertexFitter.
+ */
+
+
+#undef NDEBUG
+#include "AdaptiveVertexFitterTestAlg.h"
+#include "xAODTracking/Vertex.h"
+#include "xAODTracking/TrackParticle.h"
+#include "TrkTrack/Track.h"
+#include "TrkParticleBase/TrackParticleBase.h"
+#include "EventPrimitives/EventPrimitivesHelpers.h"
+#include "TestTools/FLOATassert.h"
+#include "GaudiKernel/SystemOfUnits.h"
+#include <cassert>
+
+
+#include "CLHEP/Vector/LorentzVector.h"
+
+using Gaudi::Units::mm;
+using Gaudi::Units::MeV;
+using Gaudi::Units::GeV;
+
+
+namespace {
+
+
+template <class T>
+std::vector<const T*> asVec (const std::vector<std::unique_ptr<T> >& v)
+{
+  std::vector<const T*> ret;
+  for (const std::unique_ptr<T>& p : v) {
+    ret.push_back (p.get());
+  }
+  return ret;
+}
+
+
+std::vector<const Trk::TrackParameters*>
+asVec (const std::vector<std::unique_ptr<Trk::Perigee> >& v)
+{
+  std::vector<const Trk::TrackParameters*> ret;
+  for (const std::unique_ptr<Trk::Perigee>& p : v) {
+    ret.push_back (p.get());
+  }
+  return ret;
+}
+
+
+std::vector<const Trk::NeutralParameters*>
+asVec (const std::vector<std::unique_ptr<Trk::NeutralPerigee> >& v)
+{
+  std::vector<const Trk::NeutralParameters*> ret;
+  for (const std::unique_ptr<Trk::NeutralPerigee>& p : v) {
+    ret.push_back (p.get());
+  }
+  return ret;
+}
+
+
+std::unique_ptr<AmgSymMatrix(5)> cov5()
+{
+  auto m = std::make_unique<AmgSymMatrix(5)>();
+  m->setIdentity();
+  return m;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::Perigee> > PerigeeUVec_t;
+PerigeeUVec_t makePerigees1()
+{
+  Amg::Vector3D pos0 { 0, 0, 0 };
+
+  Amg::Vector3D pos1a { 2*mm, 1*mm, -10*mm };
+  Amg::Vector3D mom1a { 400*MeV, 600*MeV, 200*MeV };
+  Amg::Vector3D pos1b { 1*mm, 2*mm, -3*mm };
+  Amg::Vector3D mom1b { 600*MeV, 400*MeV, -200*MeV };
+  Amg::Vector3D pos1c { 1.2*mm, 1.3*mm, -7*mm };
+  Amg::Vector3D mom1c { 300*MeV, 1000*MeV, 100*MeV };
+
+  PerigeeUVec_t ret;
+
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1a, mom1a,  1, pos1a, cov5().release()));
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1b, mom1b, -1, pos1b, cov5().release()));
+  ret.emplace_back (std::make_unique<Trk::Perigee>(pos1c, mom1c, -1, pos1c, cov5().release()));
+
+  return ret;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::NeutralPerigee> > NeutralUVec_t;
+NeutralUVec_t makeNeutrals1()
+{
+  Amg::Vector3D pos0 { 0, 0, 0 };
+
+  Amg::Vector3D pos1a { 3*mm, 0.5*mm, -7*mm };
+  Amg::Vector3D mom1a { 1000*MeV, 900*MeV, 2000*MeV };
+  Amg::Vector3D pos1b { -1*mm, 2.5*mm, 1*mm };
+  Amg::Vector3D mom1b { 800*MeV, 1000*MeV, 300*MeV };
+  Amg::Vector3D pos1c { -1.5*mm, 1*mm, -3*mm };
+  Amg::Vector3D mom1c { 500*MeV, 4000*MeV, 800*MeV };
+
+  NeutralUVec_t ret;
+
+  ret.emplace_back (std::make_unique<Trk::NeutralPerigee>(pos1a, mom1a,  1, pos1a, cov5().release()));
+  ret.emplace_back (std::make_unique<Trk::NeutralPerigee>(pos1b, mom1b,  1, pos1b, cov5().release()));
+  ret.emplace_back (std::make_unique<Trk::NeutralPerigee>(pos1c, mom1c,  1, pos1c, cov5().release()));
+
+  return ret;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::Track> > TrackUVec_t;
+TrackUVec_t makeTracks (PerigeeUVec_t&& perigees)
+{
+  TrackUVec_t tracks;
+
+  for (std::unique_ptr<Trk::Perigee>& p : perigees) {
+    Trk::TrackInfo info (Trk::TrackInfo::Unknown, Trk::undefined);
+    auto fqual = std::make_unique<Trk::FitQuality> (0, 0);
+    auto tsos = std::make_unique<DataVector<const Trk::TrackStateOnSurface> >();
+    std::bitset<Trk::TrackStateOnSurface::NumberOfTrackStateOnSurfaceTypes> typePattern(0);
+    typePattern.set(Trk::TrackStateOnSurface::Perigee);
+    tsos->push_back (std::make_unique<Trk::TrackStateOnSurface>
+                     (nullptr, p.release(), nullptr, nullptr, typePattern));
+    tracks.push_back (std::make_unique<Trk::Track> (info,
+                                                    tsos.release(),
+                                                    fqual.release()));
+  }
+
+  return tracks;
+}
+
+
+typedef std::vector<std::unique_ptr<Trk::TrackParticleBase> > TrackParticleUVec_t;
+TrackParticleUVec_t makeTrackParticles (PerigeeUVec_t&& perigees)
+{
+  TrackParticleUVec_t tracks;
+
+  std::vector<const Trk::TrackParameters*> pvec;
+  for (std::unique_ptr<Trk::Perigee>& p : perigees) {
+    tracks.push_back (std::make_unique<Trk::TrackParticleBase> (nullptr,
+                                                                Trk::NoVtx,
+                                                                nullptr,
+                                                                nullptr,
+                                                                pvec,
+                                                                p.release(),
+                                                                nullptr));
+  }
+
+  return tracks;
+}
+
+
+// Copied from TrackParticleCreatorTool.
+void setDefiningParameters( xAOD::TrackParticle& tp,
+                            const Trk::Perigee& perigee )
+{
+  tp.setDefiningParameters(perigee.parameters()[Trk::d0],
+    perigee.parameters()[Trk::z0],
+    perigee.parameters()[Trk::phi0],
+    perigee.parameters()[Trk::theta],
+    perigee.parameters()[Trk::qOverP]);
+  const AmgSymMatrix(5)* covMatrix = perigee.covariance();
+  // see https://its.cern.ch/jira/browse/ATLASRECTS-645 for justification to comment out the following line 
+  // assert(covMatrix && covMatrix->rows()==5&& covMatrix->cols()==5); 
+  std::vector<float> covMatrixVec;
+  if( !covMatrix ) ;//ATH_MSG_WARNING("Setting Defining parameters without error matrix");
+  else Amg::compress(*covMatrix,covMatrixVec);
+  tp.setDefiningParametersCovMatrixVec(covMatrixVec);
+  const Amg::Vector3D& surfaceCenter = perigee.associatedSurface().center(); 
+  tp.setParametersOrigin(surfaceCenter.x(), surfaceCenter.y(), surfaceCenter.z() );
+}
+void setDefiningParameters( xAOD::NeutralParticle& tp,
+                            const Trk::NeutralPerigee& perigee )
+{
+  tp.setDefiningParameters(perigee.parameters()[Trk::d0],
+    perigee.parameters()[Trk::z0],
+    perigee.parameters()[Trk::phi0],
+    perigee.parameters()[Trk::theta],
+    perigee.parameters()[Trk::qOverP]);
+  const AmgSymMatrix(5)* covMatrix = perigee.covariance();
+  // see https://its.cern.ch/jira/browse/ATLASRECTS-645 for justification to comment out the following line 
+  // assert(covMatrix && covMatrix->rows()==5&& covMatrix->cols()==5); 
+  std::vector<float> covMatrixVec;
+  if( !covMatrix ) ;//ATH_MSG_WARNING("Setting Defining parameters without error matrix");
+  else Amg::compress(*covMatrix,covMatrixVec);
+  tp.setDefiningParametersCovMatrixVec(covMatrixVec);
+  const Amg::Vector3D& surfaceCenter = perigee.associatedSurface().center(); 
+  tp.setParametersOrigin(surfaceCenter.x(), surfaceCenter.y(), surfaceCenter.z() );
+}
+
+
+typedef std::vector<std::unique_ptr<xAOD::TrackParticle> > xAODTPUVec_t;
+xAODTPUVec_t makexAODTP (PerigeeUVec_t&& perigees)
+{
+  xAODTPUVec_t tracks;
+
+  for (std::unique_ptr<Trk::Perigee>& p : perigees) {
+    auto tp = std::make_unique<xAOD::TrackParticle>();
+    tp->makePrivateStore();
+    setDefiningParameters (*tp, *p);
+    tracks.push_back (std::move (tp));
+  }
+
+  return tracks;
+}
+
+
+typedef std::vector<std::unique_ptr<xAOD::NeutralParticle> > xAODNPUVec_t;
+xAODNPUVec_t makexAODNP (NeutralUVec_t&& perigees)
+{
+  xAODNPUVec_t tracks;
+
+  for (std::unique_ptr<Trk::NeutralPerigee>& p : perigees) {
+    auto tp = std::make_unique<xAOD::NeutralParticle>();
+    tp->makePrivateStore();
+    setDefiningParameters (*tp, *p);
+    tracks.push_back (std::move (tp));
+  }
+
+  return tracks;
+}
+
+
+void dumpCovariance (const AmgSymMatrix(5)& m)
+{
+  for (int i=0; i < 5; i++) {
+    for (int j=0; j < 5; j++) {
+      std::cout << m(i,j) << ", ";
+    }
+  }
+}
+
+
+[[maybe_unused]]
+void dumpVertex (const xAOD::Vertex& v)
+{
+  std::cout << "vvv\n";
+  std::cout << v.x() << ", " << v.y() << ", " << v.z() << "\n";
+  if (v.isAvailable<short> ("vertexType")) {
+    std::cout << "vertexType " << v.vertexType() << "\n";
+  }
+  std::cout << "chi2/ndof " << v.chiSquared() << ", " << v.numberDoF() << "\n";
+
+  std::cout << "cov ";
+  for (float f : v.covariance()) {
+    std::cout << f << ", ";
+  }
+  std::cout << "\n";
+
+  if (v.isAvailable<std::vector<ElementLink<xAOD::TrackParticleContainer> > > ("trackParticleLinks")) {
+    std::cout << "tplinks ";
+    for (const ElementLink< xAOD::TrackParticleContainer >& l : v.trackParticleLinks()) {
+      std::cout << l.dataID() << "/" << l.index() << " ";
+    }
+    std::cout << "\n";
+  }
+
+  if (v.isAvailable<std::vector<float> > ("trackWeights")) {
+    std::cout << "wt ";
+    for (float f : v.trackWeights()) {
+      std::cout << f << " ";
+    }
+    std::cout << "\n";
+  }
+
+  if (v.isAvailable<std::vector<ElementLink<xAOD::NeutralParticleContainer> > > ("neutralParticleLinks")) {
+    std::cout << "nplinks ";
+    for (const ElementLink< xAOD::NeutralParticleContainer >& l : v.neutralParticleLinks()) {
+      std::cout << l.dataID() << "/" << l.index() << " ";
+    }
+    std::cout << "\n";
+  }
+
+  if (v.isAvailable<std::vector<float> > ("neutralWeights")) {
+    std::cout << "wt ";
+    for (float f : v.neutralWeights()) {
+      std::cout << f << " ";
+    }
+    std::cout << "\n";
+  }
+
+  std::cout << v.vxTrackAtVertexAvailable() << "\n";
+  for (const Trk::VxTrackAtVertex& vv : v.vxTrackAtVertex()) {
+    vv.dump (std::cout);
+    std::cout << "cov ";
+    if (vv.perigeeAtVertex()) {
+      dumpCovariance (*vv.perigeeAtVertex()->covariance());
+    }
+    else {
+      std::cout << "(null)";
+    }
+    std::cout << "\n";
+  }
+}
+
+
+void assertVec3D (const char* which,
+                  const Amg::Vector3D& a,
+                  const Amg::Vector3D& b)
+{
+  if ( ! Athena_test::isEqual (a.x(), b.x(), 2e-5) ||
+       ! Athena_test::isEqual (a.y(), b.y(), 2e-5) ||
+       ! Athena_test::isEqual (a.z(), b.z(), 2e-5) )
+  {
+    std::cerr << "TrkVKalVrtFitterTestAlg::assertVec3D mismatch " << which
+              << ": ["
+              << a.x() << ", "
+              << a.y() << ", "
+              << a.z() << "] / ["
+              << b.x() << ", "
+              << b.y() << ", "
+              << b.z() << "]\n";
+    std::abort();
+  }
+}
+
+
+void comparePerigee (const Trk::TrackParameters* a,
+                     const Trk::TrackParameters* b)
+{
+  if (!a && !b) return;
+  if (!a || !b) std::abort();
+  assertVec3D ("perigee pos", a->position(), b->position());
+  assertVec3D ("perigee mom", a->momentum(), b->momentum());
+  assert (a->charge() == b->charge());
+  assertVec3D ("perigee surf",
+               a->associatedSurface().center(),
+               b->associatedSurface().center());
+}
+
+
+void compareVertex (const xAOD::Vertex& a, const xAOD::Vertex& b)
+{
+  assertVec3D ("vertex pos", a.position(), b.position());
+  assert (Athena_test::isEqual (a.chiSquared(), b.chiSquared(), 5e-5) );
+  assert (Athena_test::isEqual (a.numberDoF(), b.numberDoF(), 1e-5) );
+
+  assert (a.covariance().size() == b.covariance().size());
+  for (unsigned int i = 0; i < a.covariance().size(); i++) {
+    if (isinf(a.covariance()[i]) && isinf(b.covariance()[i])) continue;
+    assert (Athena_test::isEqual (a.covariance()[i], b.covariance()[i], 2e-2) );
+  }
+
+  assert (a.vxTrackAtVertexAvailable() == b.vxTrackAtVertexAvailable());
+  if (a.vxTrackAtVertexAvailable()) {
+    const std::vector< Trk::VxTrackAtVertex >& avec = a.vxTrackAtVertex();
+    const std::vector< Trk::VxTrackAtVertex >& bvec = b.vxTrackAtVertex();
+    assert (avec.size() == bvec.size());
+    for (unsigned int i = 0; i < avec.size(); i++) {
+      comparePerigee (avec[i].initialPerigee(), bvec[i].initialPerigee());
+      comparePerigee (avec[i].perigeeAtVertex(), bvec[i].perigeeAtVertex());
+      assert (Athena_test::isEqual (avec[i].trackQuality().chiSquared(),
+                                    bvec[i].trackQuality().chiSquared(),
+                                    3e-2));
+      assert (avec[i].trackQuality().numberDoF() ==
+              bvec[i].trackQuality().numberDoF());
+    }
+  }
+}
+
+
+void setInitialPerigee (xAOD::Vertex& v, unsigned i, const Trk::Perigee* p)
+{
+  std::vector< Trk::VxTrackAtVertex >& vec = v.vxTrackAtVertex();
+  if (vec.size() <= i) vec.resize(i+1);
+  vec[i].setInitialPerigee (p);
+}
+
+
+void setInitialPerigees (xAOD::Vertex& v, const TrackUVec_t& tracks)
+{
+  int i = 0;
+  for (const std::unique_ptr<Trk::Track>& t : tracks) {
+    setInitialPerigee (v, i, t->perigeeParameters());
+    ++i;
+  }
+}
+
+
+void clearInitialPerigees (xAOD::Vertex& v)
+{
+  for (Trk::VxTrackAtVertex& v : v.vxTrackAtVertex()) {
+    v.setInitialPerigee (static_cast<const Trk::Perigee*>(nullptr));
+  }
+}
+
+
+void setRefittedPerigee (xAOD::Vertex& v, unsigned i,
+                         float charge,
+                         const Amg::Vector3D& pos,
+                         const Amg::Vector3D& mom,
+                         const std::vector<float>& c)
+{
+  std::vector< Trk::VxTrackAtVertex >& vec = v.vxTrackAtVertex();
+  if (vec.size() <= i) vec.resize(i+1);
+
+  std::unique_ptr<AmgSymMatrix(5)> cov = cov5();
+  for (int i=0; i < 5; i++) {
+    for (int j=0; j < 5; j++) {
+      unsigned ipos = i*5 + j;
+      (*cov)(i,j) = ipos < c.size() ? c[ipos] : 0;
+    }
+  }
+
+  const Amg::Vector3D& vpos = v.position();
+  auto p = std::make_unique<Trk::Perigee> (pos, mom, charge, vpos,
+                                           cov.release());
+  vec[i].setPerigeeAtVertex (p.release());
+}
+
+
+void setFitQuality (xAOD::Vertex& v, unsigned i, float chi2, int ndof)
+{
+  std::vector< Trk::VxTrackAtVertex >& vec = v.vxTrackAtVertex();
+  if (vec.size() <= i) vec.resize(i+1);
+  vec[i].setTrackQuality (Trk::FitQuality (chi2, ndof));
+}
+
+
+} // anonymous namespace
+
+
+namespace Trk {
+
+
+/**
+ * @brief Standard Gaudi initialize method.
+ */
+StatusCode AdaptiveVertexFitterTestAlg::initialize()
+{
+  ATH_CHECK( m_fitter.retrieve() );
+  return StatusCode::SUCCESS;
+}
+
+
+/**
+ * @brief Standard Gaudi execute method.
+ */
+StatusCode AdaptiveVertexFitterTestAlg::execute()
+{
+  ATH_MSG_VERBOSE ("execute");
+
+  ATH_CHECK( test1() );
+  ATH_CHECK( test2() );
+  ATH_CHECK( test3() );
+  ATH_CHECK( test4() );
+  ATH_CHECK( test5() );
+  ATH_CHECK( test6() );
+
+  return StatusCode::SUCCESS;
+}
+
+
+// Charged, no constraint.
+StatusCode AdaptiveVertexFitterTestAlg::test1()
+{
+  xAOD::Vertex exp_v0;
+  exp_v0.makePrivateStore();
+  exp_v0.setPosition ({8.43429, 9.21288, -6.60683});
+  exp_v0.setFitQuality (0.235948, 1.83137);
+  exp_v0.setCovariance (std::vector<float>
+                        {35.7266, 30.7883, 47.0715, 3.14669, 4.52157, 6.74599 });
+  setRefittedPerigee (exp_v0, 0, 1,
+                      {7.8113830, 9.6359772, -7.1130181},
+                      {405.1715584, 596.5212045, 199.9961520},
+                      {1239.56, -45.9167, 227.862, -0.121956, -33.6206, -45.9167, 128.234, -8.63731, -11.1976, 1.30089, 227.862, -8.63731, 42.8345, -0.0234628, -6.46787, -0.121956, -11.1976, -0.0234628, 1.00001, 0.00399819, -33.6206, 1.30089, -6.46787, 0.00399819, 1}
+                      );
+  setFitQuality (exp_v0, 0, 0, 2);
+  setRefittedPerigee (exp_v0, 1, -1,
+                      {9.4647993, 7.6951354, -5.8295872},
+                      {596.5909112, 405.0689507, -199.9960745},
+                      {1150.81, -107.955, 215.525, -0.120009, -32.3376, -107.955, 132.06, -20.6979, -10.9741, 3.17765, 215.525, -20.6979, 41.3112, -0.0235635, -6.34888, -0.120009, -10.9741, -0.0235635, 1.00001, 0.00408136, -32.3376, 3.17765, -6.34888, 0.00408136, 1}
+                      );
+  setFitQuality (exp_v0, 1, 0.175, 2);
+  setRefittedPerigee (exp_v0, 2, -1,
+                      {3.9422703, 10.5334868, -6.0773998},
+                      {294.4712678, 1001.6416761, 100.0031255},
+                      {872.333, -76.5329, 170.831, -0.0897349, -27.9026, -76.5329, 102.388, -15.3986, -9.71236, 2.58803, 170.831, -15.3986, 34.3815, -0.0185874, -5.77814, -0.0897349, -9.71236, -0.0185874, 1.00001, 0.00312552, -27.9026, 2.58803, -5.77814, 0.00312552, 1}
+                      );
+  setFitQuality (exp_v0, 2, 0.027, 2);
+
+  TrackUVec_t tracks = makeTracks (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v1 (m_fitter->fit (asVec (tracks)));
+  setInitialPerigees (exp_v0, tracks);
+  compareVertex (*v1, exp_v0);
+
+  PerigeeUVec_t perigees = makePerigees1();
+  std::unique_ptr<xAOD::Vertex> v2 (m_fitter->fit (asVec (perigees)));
+  compareVertex (*v2, exp_v0);
+
+  return StatusCode::SUCCESS;
+}
+
+
+// Neutral, no constraint.
+StatusCode AdaptiveVertexFitterTestAlg::test2()
+{
+  xAOD::Vertex exp_v0;
+  exp_v0.makePrivateStore();
+  exp_v0.setPosition ({4.27611, 4.35683, 1.62467});
+  exp_v0.setFitQuality (1.69878, 0.79376);
+  exp_v0.setCovariance (std::vector<float>
+                        {28.318, 26.913, 37.8531, 17.3323, 17.3041, 23.6152});
+  setFitQuality (exp_v0, 0, 0.936, 1);
+  setFitQuality (exp_v0, 1, 0.000, 2);
+  setFitQuality (exp_v0, 2, 0.000, 2);
+
+  NeutralUVec_t neutrals = makeNeutrals1();
+  std::unique_ptr<xAOD::Vertex> v1 (m_fitter->fit (std::vector<const Trk::TrackParameters*>(),
+                                                   asVec (neutrals)));
+  compareVertex (*v1, exp_v0);
+
+  return StatusCode::SUCCESS;
+}
+
+
+// Charged + Vector3D constraint.
+StatusCode AdaptiveVertexFitterTestAlg::test3()
+{
+  xAOD::Vertex exp_v0;
+  exp_v0.makePrivateStore();
+  exp_v0.setPosition ({8.82019, 9.90325, -6.4356});
+  exp_v0.setFitQuality (0.0430536, 1.85853);
+  exp_v0.setCovariance (std::vector<float>
+                        {198.998, 111.025, 256.182, 7.89219, 18.6016, 25.5699});
+  setRefittedPerigee (exp_v0, 0, 1,
+                      {8.2545569, 10.2879887, -6.8943690},
+                      {405.5620128, 596.2559088, 199.9958625},
+                      {1640.25, -51.1682, 281.924, -0.152864, -38.9104, -51.1682, 147.868, -8.97462, -12.0465, 1.26021, 281.924, -8.97462, 49.4107, -0.0273351, -6.9577, -0.152864, -12.0465, -0.0273351, 1.00002, 0.00429935, -38.9104, 1.26021, -6.9577, 0.00429935, 1}
+                      );
+  setFitQuality (exp_v0, 0, 0.000, 2);
+  setRefittedPerigee (exp_v0, 1, -1,
+                      {10.0507183, 8.0932102, -6.0260413},
+                      {596.3526270, 405.4198090, -199.9958068},
+                      {1487.84, -156.737, 262.111, -0.147597, -36.9838, -156.737, 155.885, -28.189, -11.7322, 4.05867, 262.111, -28.189, 47.1302, -0.0271073, -6.79166, -0.147597, -11.7322, -0.0271073, 1.00002, 0.00436106, -36.9838, 4.05867, -6.79166, 0.00436106, 1 }
+                      );
+  setFitQuality (exp_v0, 1, 0.017, 2);
+  setRefittedPerigee (exp_v0, 2, -1,
+                      {4.1590180, 11.2713536, -6.0037370},
+                      {294.0294524, 1001.7714343, 100.0033730},
+                      {1167.78, -99.1794, 213.359, -0.112662, -32.5361, -99.1794, 119.804, -18.5507, -10.4868, 2.89953, 213.359, -18.5507, 39.9175, -0.0216083, -6.23884, -0.112662, -10.4868, -0.0216083, 1.00001, 0.00337217, -32.5361, 2.89953, -6.23884, 0.00337217, 1 }
+                      );
+  setFitQuality (exp_v0, 2, 0.020, 2);
+
+  Amg::Vector3D pnt1 (5, 6, -3);
+
+  TrackUVec_t tracks = makeTracks (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v1 (m_fitter->fit (asVec (tracks), pnt1));
+  setInitialPerigees (exp_v0, tracks);
+  compareVertex (*v1, exp_v0);
+
+  PerigeeUVec_t perigees = makePerigees1();
+  std::unique_ptr<xAOD::Vertex> v2 (m_fitter->fit (asVec (perigees), pnt1));
+  compareVertex (*v2, exp_v0);
+
+  TrackParticleUVec_t tps = makeTrackParticles (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v3 (m_fitter->fit (asVec (tps), pnt1));
+  compareVertex (*v3, exp_v0);
+
+  xAODTPUVec_t xtps = makexAODTP (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v4 (m_fitter->fit (asVec (xtps), pnt1));
+  clearInitialPerigees (exp_v0);
+  compareVertex (*v4, exp_v0);
+
+  return StatusCode::SUCCESS;
+}
+
+
+// Charged + Vertex constraint.
+StatusCode AdaptiveVertexFitterTestAlg::test4()
+{
+  xAOD::Vertex exp_v0;
+  exp_v0.makePrivateStore();
+  exp_v0.setPosition ({5.00265, 6.02993, -3.2627});
+  exp_v0.setFitQuality (0.904414, 4.84469);
+  exp_v0.setCovariance (std::vector<float>
+                        {1.24242, 0.00898474, 1.25033,
+                           -0.0018625, 0.00503271, 1.16608});
+  setRefittedPerigee (exp_v0, 0, 1,
+                      {5.2563912, 5.8589792, -8.3777169},
+                      {402.9097324, 598.0506755, 199.9978428},
+                      { 148.334, 3.90191, 44.5024, -0.019887, -10.635, 3.90191, 40.8102, 1.24424, -6.30148, -0.32025, 44.5024, 1.24424, 14.2092, -0.00679499, -3.63453, -0.019887, -6.30148, -0.00679499, 1, 0.00224026, -10.635, -0.32025, -3.63453, 0.00224026, 1 }
+                      );
+  setFitQuality (exp_v0, 0, 0.682, 2);
+  setRefittedPerigee (exp_v0, 1, -1,
+                      {5.6281930, 5.1009479, -4.5451087},
+                      {598.1437842, 402.7714914, -199.9978479},
+                      {124.605, -11.9636, 38.8859, -0.0179296, -9.62427, -11.9636, 38.1524, -3.99608, -5.99156, 1.06341, 38.8859, -3.99608, 12.9843, -0.00645023, -3.46157, -0.0179296, -5.99156, -0.00645023, 1, 0.00223385, -9.62427, 1.06341, -3.46157, 0.00223385, 1 }
+                      );
+  setFitQuality (exp_v0, 1, 0.061, 2);
+  setRefittedPerigee (exp_v0, 2, -1,
+                      {2.8050119, 6.6815112, -6.4621037},
+                      {296.7777220, 1000.9608467, 100.0018360},
+                      { 123.052, -8.25619, 37.7378, -0.0183804, -9.50958, -8.25619, 33.7962, -2.7124, -5.67336, 0.738344, 37.7378, -2.7124, 12.4073, -0.0065315, -3.37786, -0.0183804, -5.67336, -0.0065315, 1, 0.0018425, -9.50958, 0.738344, -3.37786, 0.0018425, 1 }
+                      );
+  setFitQuality (exp_v0, 2, 0.352, 2);
+
+  Amg::Vector3D pnt1 (5, 6, -3);
+  xAOD::Vertex pnt2;
+  pnt2.makePrivateStore();
+  pnt2.setPosition (pnt1);
+  AmgSymMatrix(3) pnt2covar;
+  pnt2covar.setIdentity();
+  pnt2.setCovariancePosition (pnt2covar);
+
+  TrackUVec_t tracks = makeTracks (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v1 (m_fitter->fit (asVec (tracks), pnt2));
+  setInitialPerigees (exp_v0, tracks);
+  compareVertex (*v1, exp_v0);
+
+  PerigeeUVec_t perigees = makePerigees1();
+  std::unique_ptr<xAOD::Vertex> v2 (m_fitter->fit (asVec (perigees), pnt2));
+  compareVertex (*v2, exp_v0);
+
+  TrackParticleUVec_t tps = makeTrackParticles (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v3 (m_fitter->fit (asVec (tps), pnt2));
+  compareVertex (*v3, exp_v0);
+
+  xAODTPUVec_t xtps = makexAODTP (makePerigees1());
+  std::unique_ptr<xAOD::Vertex> v4 (m_fitter->fit (asVec (xtps), pnt2));
+  clearInitialPerigees (exp_v0);
+  compareVertex (*v4, exp_v0);
+
+  return StatusCode::SUCCESS;
+}
+
+
+// Charged + Neutral + Vector3D constraint
+StatusCode AdaptiveVertexFitterTestAlg::test5()
+{
+  xAOD::Vertex exp_v0;
+  exp_v0.makePrivateStore();
+  exp_v0.setPosition ({-0.664619, 1.88247, -4.28452});
+  exp_v0.setFitQuality (1.36786, 6.60783);
+  exp_v0.setCovariance (std::vector<float>
+                        {25.8826, 26.6122, 91.0458, 6.34189, 14.6174, 13.9884});
+  setRefittedPerigee (exp_v0, 0, 1,
+                      {1.5885284, 0.3823804, -10.2058300},
+                      {399.6301520, 600.2463141, 200.0002601},
+                      { 1.58278, -0.618193, -0.821391, 0.00010879, -0.171884, -0.618193, 2.32566, 0.908938, 0.800925, 0.346876, -0.821391, 0.908938, 1.20816, -0.000294073, 0.461137, 0.00010879, 0.800925, -0.000294073, 1, -0.000269915, -0.171884, 0.346876, 0.461137, -0.000269915, 1 }
+                      );
+  setFitQuality (exp_v0, 0, 0.000, 2);
+  setRefittedPerigee (exp_v0, 1, -1,
+                      {-0.2079830, 1.1957289, -2.5975002},
+                      {600.4814296, 399.2766328, -200.0005578},
+                      { 3.53014, 0.466334, -2.04043, 0.000621337, -0.653413, 0.466334, 3.53405, -0.415368, 1.56192, -0.206492, -2.04043, -0.415368, 1.81448, -0.000856625, 0.901728, 0.000621337, 1.56192, -0.000856625, 1, -0.000578862, -0.653413, -0.206492, 0.901728, -0.000578862, 1 }
+                      );
+  setFitQuality (exp_v0, 1, 0.017, 2);
+  setRefittedPerigee (exp_v0, 2, -1,
+                      {1.2063615, 1.3212056, -6.9978818},
+                      {299.9873170, 1000.0038041, 100.0000072},
+                      { 1.00049, 0.00413671, 0.0221412, -2.7918e-08, -0.000147081, 0.00413671, 1.03551, 0.18734, -0.0223173, -0.00248881, 0.0221412, 0.18734, 1.00242, -1.91988e-06, -0.0133167, -2.7918e-08, -0.0223173, -1.91988e-06, 1, 7.24261e-06, -0.000147081, -0.00248881, -0.0133167, 7.24261e-06, 1 }
+                      );
+  setFitQuality (exp_v0, 2, 0.020, 2);
+  setFitQuality (exp_v0, 3, 0.136, 2);
+  setFitQuality (exp_v0, 4, 0.000, 2);
+  setFitQuality (exp_v0, 5, 0.000, 2);
+
+  Amg::Vector3D pnt1 (5, 6, -3);
+
+  TrackUVec_t tracks = makeTracks (makePerigees1());
+  setInitialPerigees (exp_v0, tracks);
+
+  PerigeeUVec_t perigees = makePerigees1();
+  NeutralUVec_t neutrals = makeNeutrals1();
+  std::unique_ptr<xAOD::Vertex> v1 (m_fitter->fit (asVec (perigees),
+                                                   asVec (neutrals),
+                                                   pnt1));
+  compareVertex (*v1, exp_v0);
+
+  xAODTPUVec_t xtps = makexAODTP (makePerigees1());
+  xAODNPUVec_t xaodnp = makexAODNP (makeNeutrals1());
+  std::unique_ptr<xAOD::Vertex> v2 (m_fitter->fit (asVec (xtps),
+                                                   asVec (xaodnp),
+                                                   pnt1));
+  clearInitialPerigees (exp_v0);
+  compareVertex (*v2, exp_v0);
+
+  return StatusCode::SUCCESS;
+}
+
+
+// Charged+Neutral + Vertex constraint
+StatusCode AdaptiveVertexFitterTestAlg::test6()
+{
+  xAOD::Vertex exp_v0;
+  exp_v0.makePrivateStore();
+  exp_v0.setPosition ({4.85215, 5.94893, -3.13472});
+  exp_v0.setFitQuality (2.38431, 8.54327);
+  exp_v0.setCovariance (std::vector<float>
+                        {1.183, 0.0323074, 1.21271,
+                           0.00903037, 0.0167373, 1.12584});
+  setRefittedPerigee (exp_v0, 0, 1,
+                      {5.1719083, 5.7335618, -8.4196568},
+                      {402.8346276, 598.1012479, 199.9979000},
+                      { 135.368, 4.54292, 41.4349, -0.0182748, -10.0936, 4.54292, 38.8427, 1.48244, -6.13913, -0.389733, 41.4349, 1.48244, 13.5349, -0.00640884, -3.54057, -0.0182748, -6.13913, -0.00640884, 1, 0.00218082, -10.0936, -0.389733, -3.54057, 0.00218082, 1 }
+                      );
+  setFitQuality (exp_v0, 0, 0.682, 2);
+
+  setRefittedPerigee (exp_v0, 1, -1,
+                      {5.4869777, 5.0058724, -4.4978936},
+                      {598.2006961, 402.6869274, -199.9979142},
+                      { 111.922, -11.1715, 35.7532, -0.0162277, -9.04482, -11.1715, 35.9577, -3.83303, -5.80842, 1.04702, 35.7532, -3.83303, 12.2632, -0.0060222, -3.35579, -0.0162277, -5.80842, -0.0060222, 1, 0.00216502, -9.04482, 1.04702, -3.35579, 0.00216502, 1 }
+                      );
+  setFitQuality (exp_v0, 1, 0.061, 2);
+
+  setRefittedPerigee (exp_v0, 2, -1,
+                      {2.7708142, 6.5661849, -6.4736255},
+                      {296.8467752, 1000.9403742, 100.0017963},
+                      { 114.181, -7.37341, 35.593, -0.0172367, -9.10529, -7.37341, 32.345, -2.46816, -5.55155, 0.684107, 35.593, -2.46816, 11.9239, -0.00626001, -3.30551, -0.0172367, -5.55155, -0.00626001, 1, 0.00180269, -9.10529, 0.684107, -3.30551, 0.00180269, 1 }
+                      );
+  setFitQuality (exp_v0, 2, 0.352, 2);
+  setFitQuality (exp_v0, 3, 1.119, 1);
+  setFitQuality (exp_v0, 4, 0.000, 2);
+  setFitQuality (exp_v0, 5, 0.000, 2);
+
+  Amg::Vector3D pnt1 (5, 6, -3);
+  xAOD::Vertex pnt2;
+  pnt2.makePrivateStore();
+  pnt2.setPosition (pnt1);
+  AmgSymMatrix(3) pnt2covar;
+  pnt2covar.setIdentity();
+  pnt2.setCovariancePosition (pnt2covar);
+
+  TrackUVec_t tracks = makeTracks (makePerigees1());
+  setInitialPerigees (exp_v0, tracks);
+
+  PerigeeUVec_t perigees = makePerigees1();
+  NeutralUVec_t neutrals = makeNeutrals1();
+  std::unique_ptr<xAOD::Vertex> v1 (m_fitter->fit (asVec (perigees),
+                                                   asVec (neutrals),
+                                                   pnt2));
+  compareVertex (*v1, exp_v0);
+
+  xAODTPUVec_t xtps = makexAODTP (makePerigees1());
+  xAODNPUVec_t xaodnp = makexAODNP (makeNeutrals1());
+  std::unique_ptr<xAOD::Vertex> v2 (m_fitter->fit (asVec (xtps),
+                                                   asVec (xaodnp),
+                                                   pnt2));
+  clearInitialPerigees (exp_v0);
+  compareVertex (*v2, exp_v0);
+
+  return StatusCode::SUCCESS;
+}
+
+
+} // namespace Trk
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.h b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.h
new file mode 100644
index 000000000000..359d24b3728c
--- /dev/null
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.h
@@ -0,0 +1,57 @@
+// This file's extension implies that it's C, but it's really -*- C++ -*-.
+/*
+ * Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration.
+ */
+/**
+ * @file TrkVertexFitters/src/AdaptiveVertexFitterTestAlg.h
+ * @author scott snyder <snyder@bnl.gov>
+ * @date Jul, 2019
+ * @brief Algorithm for testing AdaptiveVertexFitter.
+ */
+
+
+#ifndef TRKVERTEXFITTERS_ADAPTIVEVERTEXFITTERTESTALG_H
+#define TRKVERTEXFITTERS_ADAPTIVEVERTEXFITTERTESTALG_H
+
+
+#include "AthenaBaseComps/AthAlgorithm.h"
+#include "TrkVertexFitterInterfaces/IVertexFitter.h"
+#include "GaudiKernel/ToolHandle.h"
+
+
+
+namespace Trk {
+
+
+class AdaptiveVertexFitterTestAlg
+  : public AthAlgorithm
+{
+public:
+  using AthAlgorithm::AthAlgorithm;
+
+
+  /// Standard Gaudi initialize method.
+  virtual StatusCode initialize() override;
+
+  /// Execute the algorithm.
+  virtual StatusCode execute() override;
+
+
+private:
+  StatusCode test1();
+  StatusCode test2();
+  StatusCode test3();
+  StatusCode test4();
+  StatusCode test5();
+  StatusCode test6();
+
+
+  ToolHandle<Trk::IVertexFitter> m_fitter
+  { this, "Tool", "Trk::AdaptiveVertexFitter", "Tool to test." };
+};
+
+
+} // namespace Trk
+
+
+#endif // not TRKVERTEXFITTERS_ADAPTIVEVERTEXFITTERTESTALG_H
diff --git a/Tracking/TrkVertexFitter/TrkVertexFitters/src/components/TrkVertexFitters_entries.cxx b/Tracking/TrkVertexFitter/TrkVertexFitters/src/components/TrkVertexFitters_entries.cxx
index 3306f166b690..901907d71ee1 100644
--- a/Tracking/TrkVertexFitter/TrkVertexFitters/src/components/TrkVertexFitters_entries.cxx
+++ b/Tracking/TrkVertexFitter/TrkVertexFitters/src/components/TrkVertexFitters_entries.cxx
@@ -3,6 +3,8 @@
 #include "TrkVertexFitters/AdaptiveVertexFitter.h"
 #include "TrkVertexFitters/SequentialVertexSmoother.h"
 #include "TrkVertexFitters/DummyVertexSmoother.h"
+#include "../AdaptiveVertexFitterTestAlg.h"
+#include "../AdaptiveMultiVertexFitterTestAlg.h"
 
 using namespace Trk;
 
@@ -11,4 +13,6 @@ DECLARE_COMPONENT( AdaptiveMultiVertexFitter )
 DECLARE_COMPONENT( SequentialVertexSmoother )
 DECLARE_COMPONENT( AdaptiveVertexFitter )
 DECLARE_COMPONENT( DummyVertexSmoother )
+DECLARE_COMPONENT( AdaptiveVertexFitterTestAlg )
+DECLARE_COMPONENT( AdaptiveMultiVertexFitterTestAlg )
 
-- 
GitLab