-
Notifications
You must be signed in to change notification settings - Fork 2
TestBloodFlowTutorial
This tutorial is automatically generated from the file test/tutorials/TestBloodFlowTutorial.hpp. Note that the code is given in full at the bottom of the page.
This tutorial demonstrates functionality for modelling blood flow, structural adaptation and vessel regression in a vessel network.
This tutorial covers:
- Managing parameter values
- Running a minimal Poiseuille flow simulation and looking at results
- Adding haematocrit
- Adding structural adaptation in response to flow
- Adding vessel regression in low flow regions
Start by introducing the necessary header files, explained in previous tutorials.
#include <vector>
#include <cxxtest/TestSuite.h>
#include "Owen11Parameters.hpp"
#include "AbstractCellBasedWithTimingsTestSuite.hpp"
#include "SmartPointers.hpp"
#include "OutputFileHandler.hpp"
#include "VesselNode.hpp"
#include "VesselSegment.hpp"
#include "Vessel.hpp"
#include "VesselNetwork.hpp"
#include "VesselNetworkGenerator.hpp"
#include "UnitCollection.hpp"
#include "BaseUnits.hpp"
A container for flow information (boundary conditions, pressure values) for nodes.
#include "NodeFlowProperties.hpp"
A collection of useful literature parameter values and a way to dump values to file after use.
#include "Owen11Parameters.hpp"
#include "GenericParameters.hpp"
#include "ParameterCollection.hpp"
The flow and haematocrit solvers, along with neccessary calculators.
#include "VesselImpedanceCalculator.hpp"
#include "FlowSolver.hpp"
#include "AlarconHaematocritSolver.hpp"
#include "ConstantHaematocritSolver.hpp"
#include "StructuralAdaptationSolver.hpp"
#include "WallShearStressCalculator.hpp"
#include "MechanicalStimulusCalculator.hpp"
The vessel regression solver and a generic solver to collect all the flow solvers.
#include "WallShearStressBasedRegressionSolver.hpp"
#include "MicrovesselSolver.hpp"
Keep this last.
#include "PetscSetupAndFinalize.hpp"
class TestBloodFlowLiteratePaper : public AbstractCellBasedWithTimingsTestSuite
{
public:
In the first test we will simulate blood flow in a simple bifurcating vessel network. Subsequent tests will add detail in the form of more complex networks, structural adaptation and vessel regression.
void TestSimpleFlowProblem() throw (Exception)
{
We will work in microns
units::quantity<unit::length> reference_length(1.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(reference_length);
First make the network using a generator. Start with a simple unit.
units::quantity<unit::length> vessel_length(100.0*unit::microns);
DimensionalChastePoint<2> start_point(0.0, 0.0);
VesselNetworkGenerator<2> network_generator;
boost::shared_ptr<VesselNetwork<2> > p_network = network_generator.GenerateBifurcationUnit(vessel_length, start_point);
Next, pattern it to make a larger network
std::vector<unsigned> num_units_per_direction;
num_units_per_direction.push_back(2);
num_units_per_direction.push_back(0);
network_generator.PatternUnitByTranslation(p_network, num_units_per_direction);
Specify which nodes will be the inlets and outlets of the network for the flow problem. This information, as well
as all other info related to the flow problem, is contained in a NodeFlowProperties
instance. Also, set the inlet and
outlet pressures in Pa or mmHg. We can mix and match these thanks to the dimensional analysis functionality. Because
the network is simple, we can figure out which node is which just from their index. For more complicated networks
we need to use spatial locators and NearestNode
type methods.
units::quantity<unit::pressure> inlet_pressure(50.0 * unit::mmHg);
p_network->GetNode(0)->GetFlowProperties()->SetIsInputNode(true);
p_network->GetNode(0)->GetFlowProperties()->SetPressure(inlet_pressure);
It would be useful if we had a record of which parameter values we have used in the simulation and where they are sourced in the
literature. Instead of manually entering parameter values like above we can use a ParameterCollection
singleton which allows for
some extra metadata storage. We will take some parameter values from a paper by Owen et al. (2011).
p_network->GetNode(p_network->GetNumberOfNodes()-1)->GetFlowProperties()->SetIsOutputNode(true);
p_network->GetNode(p_network->GetNumberOfNodes()-1)->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
Now set the segment radii and viscosity values.
units::quantity<unit::length> vessel_radius(GenericParameters::mpCapillaryRadius->GetValue());
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
We use a calculator to work out the impedance of each vessel based on assumptions of Poiseuille flow and cylindrical vessels. This updates the value of the impedance in the vessel.
VesselImpedanceCalculator<2> impedance_calculator = VesselImpedanceCalculator<2>();
impedance_calculator.SetVesselNetwork(p_network);
impedance_calculator.Calculate();
Check that the impedance is as expected in one of the vessels
units::quantity<unit::flow_impedance> expected_impedance = 8.0 * viscosity* vessel_length/(M_PI*units::pow<4>(vessel_radius));
TS_ASSERT_DELTA(p_network->GetVessel(0)->GetSegment(0)->GetFlowProperties()->GetImpedance().value(), expected_impedance.value(), 1.e-6);
Now we can solve for the flow rates in each vessel based on the inlet and outlet pressures and impedances. The solver updates the value of pressures and flow rates in each vessel and node in the network.
FlowSolver<2> flow_solver = FlowSolver<2>();
flow_solver.SetVesselNetwork(p_network);
flow_solver.Solve();
Check the pressure, it is expected to drop linearly so should be the average of the input and output half way along the network.
units::quantity<unit::pressure> expected_pressure = (inlet_pressure + Owen11Parameters::mpOutletPressure->GetValue())/2.0;
TS_ASSERT_DELTA(p_network->GetNode(7)->GetFlowProperties()->GetPressure().value(), expected_pressure.value(), 1.e-6);
Next we write out the network, including updated flow data, to file.
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestSimpleFlowProblem"));
p_network->Write(p_handler->GetOutputDirectoryFullPath() + "bifurcating_network_results.vtp");
Now we can visualize the results in Paraview. To view the network import the file
TestBloodFlowLiteratePaper\bifurcating_network.vtp
into Paraview. For a nicer rendering you can do Filters->Alphabetical->Tube
.
Finally, dump our parameter collection to an xml file and, importantly, clear it for the next test.
ParameterCollection::Instance()->DumpToFile(p_handler->GetOutputDirectoryFullPath() + "parameter_collection.xml");
ParameterCollection::Instance()->Destroy();
BaseUnits::Instance()->Destroy();
}
In this test we will simulate haematocrit transport in a 3d vessel network.
void TestFlowProblemWithHaematocrit() throw (Exception)
{
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestFlowProblemWithHaematocrit", false));
This time we solve a flow problem and then use the solution to calculate the haematocrit distribution, assuming it has no effect on the flow. Set up the network as before
units::quantity<unit::length> cell_width(25.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(cell_width);
units::quantity<unit::length> target_width = 100.0 * cell_width;
units::quantity<unit::length> target_height = 30.0 * cell_width;
units::quantity<unit::length> vessel_length = 4.0 * cell_width;
VesselNetworkGenerator<3> network_generator;
boost::shared_ptr<VesselNetwork<3> > p_network = network_generator.GenerateHexagonalNetwork(target_width,
target_height,
vessel_length);
We will use a locator to mark the bottom left and top right nodes as respective inlets and outlets
DimensionalChastePoint<3> inlet_locator(0.0, 0.0, 0.0, cell_width);
DimensionalChastePoint<3> outlet_locator(target_width/cell_width, target_height/cell_width, 0.0, cell_width);
boost::shared_ptr<VesselNode<3> > p_inlet_node = p_network->GetNearestNode(inlet_locator);
boost::shared_ptr<VesselNode<3> > p_outlet_node = p_network->GetNearestNode(outlet_locator);
p_inlet_node->GetFlowProperties()->SetIsInputNode(true);
p_inlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpInletPressure->GetValue());
p_outlet_node->GetFlowProperties()->SetIsOutputNode(true);
p_outlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
Now set the segment radii and viscosity values.
units::quantity<unit::length> vessel_radius(GenericParameters::mpCapillaryRadius->GetValue());
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
Next some simple extra functionality is demonstrated by mapping the network onto a hemisphere
units::quantity<unit::length> sphere_radius = 400.0 * cell_width;
units::quantity<unit::length> sphere_thickess = 1.0 * cell_width;
double sphere_azimuth = M_PI;
double sphere_polar = M_PI/2.0;
network_generator.MapToSphere(p_network, sphere_radius, sphere_thickess, sphere_azimuth, sphere_polar);
Get the impedance
VesselImpedanceCalculator<3> impedance_calculator = VesselImpedanceCalculator<3>();
impedance_calculator.SetVesselNetwork(p_network);
impedance_calculator.Calculate();
Solve the flow problem
FlowSolver<3> flow_solver = FlowSolver<3>();
flow_solver.SetVesselNetwork(p_network);
flow_solver.Solve();
Solve the haematocrit problem
AlarconHaematocritSolver<3> haematocrit_solver = AlarconHaematocritSolver<3>();
haematocrit_solver.SetVesselNetwork(p_network);
haematocrit_solver.SetHaematocrit(0.45);
haematocrit_solver.Calculate();
Next we write out the network, including updated flow data, to file.
BaseUnits::Instance()->SetReferenceLengthScale(1.e-6*unit::metres);
p_network->Write(p_handler->GetOutputDirectoryFullPath() + "network_haematocrit.vtp");
Now we can visualize the results in Paraview. To view the network import the file
TestBloodFlowLiteratePaper\bifurcating_network.vtp
into Paraview. For a nicer rendering you can do Filters->Alphabetical->Tube
.
ParameterCollection::Instance()->DumpToFile(p_handler->GetOutputDirectoryFullPath() + "parameter_collection.xml");
ParameterCollection::Instance()->Destroy();
}
In this test the vessel network will adapt over time as a result of flow conditions.
void TestFlowProblemStucturalAdaptation() throw (Exception)
{
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestFlowProblemStucturalAdaptation", false));
We will work in microns
units::quantity<unit::length> reference_length(1.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(reference_length);
Set up a hexagonal vessel network
units::quantity<unit::length> target_width(8000*unit::microns);
units::quantity<unit::length> target_height(2000*unit::microns);
units::quantity<unit::length> vessel_length(300.0*unit::microns);
VesselNetworkGenerator<3> network_generator;
boost::shared_ptr<VesselNetwork<3> > p_network = network_generator.GenerateHexagonalNetwork(target_width,
target_height,
vessel_length);
We will use a locator to mark the bottom left and top right nodes as respective inlets and outlets as before.
DimensionalChastePoint<3> inlet_locator(0.0, 0.0, 0.0, reference_length);
DimensionalChastePoint<3> outlet_locator(target_width/reference_length, target_height/reference_length, 0.0, reference_length);
boost::shared_ptr<VesselNode<3> > p_inlet_node = p_network->GetNearestNode(inlet_locator);
boost::shared_ptr<VesselNode<3> > p_outlet_node = p_network->GetNearestNode(outlet_locator);
p_inlet_node->GetFlowProperties()->SetIsInputNode(true);
p_inlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpInletPressure->GetValue());
p_outlet_node->GetFlowProperties()->SetIsOutputNode(true);
p_outlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
Set the radius and viscosity and write the initial network to file.
units::quantity<unit::length> vessel_radius(40.0*unit::microns);
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
p_network->Write(p_handler->GetOutputDirectoryFullPath()+"initial_network.vtp");
Set up the timer and time scale. The structural adaptation alrogithm works by calcualting growth or shrinkage stimuli from several flow derived sources. We can specify how the network adapts as a function of flow by adding a collection of calculators for wall shear stress, haematocrit etc.
BaseUnits::Instance()->SetReferenceTimeScale(60.0*unit::seconds);
SimulationTime::Instance()->SetEndTimeAndNumberOfTimeSteps(30, 1);
boost::shared_ptr<VesselImpedanceCalculator<3> > p_impedance_calculator = VesselImpedanceCalculator<3>::Create();
boost::shared_ptr<ConstantHaematocritSolver<3> > p_haematocrit_calculator = ConstantHaematocritSolver<3>::Create();
p_haematocrit_calculator->SetHaematocrit(0.45);
boost::shared_ptr<WallShearStressCalculator<3> > p_wss_calculator = WallShearStressCalculator<3>::Create();
boost::shared_ptr<MechanicalStimulusCalculator<3> > p_mech_stimulus_calculator = MechanicalStimulusCalculator<3>::Create();
Set up the structural adaptation solver, which manages iterations over a flow solve and executes each calculator in the order it has been added.
StructuralAdaptationSolver<3> structural_adaptation_solver;
structural_adaptation_solver.SetVesselNetwork(p_network);
structural_adaptation_solver.SetWriteOutput(true);
structural_adaptation_solver.SetOutputFileName(p_handler->GetOutputDirectoryFullPath()+"adaptation_data.dat");
structural_adaptation_solver.SetTolerance(0.0001);
structural_adaptation_solver.SetMaxIterations(10);
structural_adaptation_solver.AddPreFlowSolveCalculator(p_impedance_calculator);
structural_adaptation_solver.AddPostFlowSolveCalculator(p_haematocrit_calculator);
structural_adaptation_solver.AddPostFlowSolveCalculator(p_wss_calculator);
structural_adaptation_solver.AddPostFlowSolveCalculator(p_mech_stimulus_calculator);
structural_adaptation_solver.SetTimeIncrement(0.01 * unit::seconds);
Do the solve and write the network to file.
structural_adaptation_solver.Solve();
p_network->Write(p_handler->GetOutputDirectoryFullPath()+"network_initial_sa.vtp");
}
In this test the vessel network will adapt over time as a result of flow conditions and also vessels will be removed to regression in low wall shear stress regions.
void TestFlowProblemStucturalAdaptationWithRegression() throw (Exception)
{
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestFlowProblemStucturalAdaptationWithRegression", false));
Set up the problem as before.
units::quantity<unit::length> reference_length(1.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(reference_length);
units::quantity<unit::length> target_width(8000*unit::microns);
units::quantity<unit::length> target_height(2000*unit::microns);
units::quantity<unit::length> vessel_length(300.0*unit::microns);
VesselNetworkGenerator<3> network_generator;
boost::shared_ptr<VesselNetwork<3> > p_network = network_generator.GenerateHexagonalNetwork(target_width,
target_height,
vessel_length);
DimensionalChastePoint<3> inlet_locator(0.0, 0.0, 0.0, reference_length);
DimensionalChastePoint<3> outlet_locator(target_width/reference_length, target_height/reference_length, 0.0, reference_length);
boost::shared_ptr<VesselNode<3> > p_inlet_node = p_network->GetNearestNode(inlet_locator);
boost::shared_ptr<VesselNode<3> > p_outlet_node = p_network->GetNearestNode(outlet_locator);
p_inlet_node->GetFlowProperties()->SetIsInputNode(true);
p_inlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpInletPressure->GetValue());
p_outlet_node->GetFlowProperties()->SetIsOutputNode(true);
p_outlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
units::quantity<unit::length> vessel_radius(40.0*unit::microns);
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
p_network->Write(p_handler->GetOutputDirectoryFullPath()+"initial_network.vtp");
BaseUnits::Instance()->SetReferenceTimeScale(60.0*unit::seconds);
SimulationTime::Instance()->SetEndTimeAndNumberOfTimeSteps(30, 1);
boost::shared_ptr<VesselImpedanceCalculator<3> > p_impedance_calculator = VesselImpedanceCalculator<3>::Create();
boost::shared_ptr<ConstantHaematocritSolver<3> > p_haematocrit_calculator = ConstantHaematocritSolver<3>::Create();
p_haematocrit_calculator->SetHaematocrit(0.45);
boost::shared_ptr<WallShearStressCalculator<3> > p_wss_calculator = WallShearStressCalculator<3>::Create();
boost::shared_ptr<MechanicalStimulusCalculator<3> > p_mech_stimulus_calculator = MechanicalStimulusCalculator<3>::Create();
boost::shared_ptr<StructuralAdaptationSolver<3> > p_structural_adaptation_solver = StructuralAdaptationSolver<3>::Create();
p_structural_adaptation_solver->SetVesselNetwork(p_network);
p_structural_adaptation_solver->SetWriteOutput(true);
p_structural_adaptation_solver->SetOutputFileName(p_handler->GetOutputDirectoryFullPath()+"adaptation_data.dat");
p_structural_adaptation_solver->SetTolerance(0.0001);
p_structural_adaptation_solver->SetMaxIterations(10);
p_structural_adaptation_solver->AddPreFlowSolveCalculator(p_impedance_calculator);
p_structural_adaptation_solver->AddPostFlowSolveCalculator(p_haematocrit_calculator);
p_structural_adaptation_solver->AddPostFlowSolveCalculator(p_wss_calculator);
p_structural_adaptation_solver->AddPostFlowSolveCalculator(p_mech_stimulus_calculator);
p_structural_adaptation_solver->SetTimeIncrement(0.01 * unit::seconds);
Set up a regression solver
boost::shared_ptr<WallShearStressBasedRegressionSolver<3> > p_regression_solver = WallShearStressBasedRegressionSolver<3>::Create();
p_regression_solver->SetMaximumTimeWithLowWallShearStress(2.0*3600.0*unit::seconds);
p_regression_solver->SetLowWallShearStressThreshold(1.e-06*unit::pascals);
p_regression_solver->SetVesselNetwork(p_network);
Set up a MicrovesselSolver
to manage all solves.
MicrovesselSolver<3> microvessel_solver;
microvessel_solver.SetRegressionSolver(p_regression_solver);
microvessel_solver.SetStructuralAdaptationSolver(p_structural_adaptation_solver);
microvessel_solver.SetVesselNetwork(p_network);
microvessel_solver.SetOutputFileHandler(p_handler);
microvessel_solver.SetOutputFrequency(1);
Run the solver
microvessel_solver.Run();
}
};
The full code is given below
#include <vector>
#include <cxxtest/TestSuite.h>
#include "Owen11Parameters.hpp"
#include "AbstractCellBasedWithTimingsTestSuite.hpp"
#include "SmartPointers.hpp"
#include "OutputFileHandler.hpp"
#include "VesselNode.hpp"
#include "VesselSegment.hpp"
#include "Vessel.hpp"
#include "VesselNetwork.hpp"
#include "VesselNetworkGenerator.hpp"
#include "UnitCollection.hpp"
#include "BaseUnits.hpp"
#include "NodeFlowProperties.hpp"
#include "Owen11Parameters.hpp"
#include "GenericParameters.hpp"
#include "ParameterCollection.hpp"
#include "VesselImpedanceCalculator.hpp"
#include "FlowSolver.hpp"
#include "AlarconHaematocritSolver.hpp"
#include "ConstantHaematocritSolver.hpp"
#include "StructuralAdaptationSolver.hpp"
#include "WallShearStressCalculator.hpp"
#include "MechanicalStimulusCalculator.hpp"
#include "WallShearStressBasedRegressionSolver.hpp"
#include "MicrovesselSolver.hpp"
#include "PetscSetupAndFinalize.hpp"
class TestBloodFlowLiteratePaper : public AbstractCellBasedWithTimingsTestSuite
{
public:
void TestSimpleFlowProblem() throw (Exception)
{
units::quantity<unit::length> reference_length(1.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(reference_length);
units::quantity<unit::length> vessel_length(100.0*unit::microns);
DimensionalChastePoint<2> start_point(0.0, 0.0);
VesselNetworkGenerator<2> network_generator;
boost::shared_ptr<VesselNetwork<2> > p_network = network_generator.GenerateBifurcationUnit(vessel_length, start_point);
std::vector<unsigned> num_units_per_direction;
num_units_per_direction.push_back(2);
num_units_per_direction.push_back(0);
network_generator.PatternUnitByTranslation(p_network, num_units_per_direction);
units::quantity<unit::pressure> inlet_pressure(50.0 * unit::mmHg);
p_network->GetNode(0)->GetFlowProperties()->SetIsInputNode(true);
p_network->GetNode(0)->GetFlowProperties()->SetPressure(inlet_pressure);
p_network->GetNode(p_network->GetNumberOfNodes()-1)->GetFlowProperties()->SetIsOutputNode(true);
p_network->GetNode(p_network->GetNumberOfNodes()-1)->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
units::quantity<unit::length> vessel_radius(GenericParameters::mpCapillaryRadius->GetValue());
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
VesselImpedanceCalculator<2> impedance_calculator = VesselImpedanceCalculator<2>();
impedance_calculator.SetVesselNetwork(p_network);
impedance_calculator.Calculate();
units::quantity<unit::flow_impedance> expected_impedance = 8.0 * viscosity* vessel_length/(M_PI*units::pow<4>(vessel_radius));
TS_ASSERT_DELTA(p_network->GetVessel(0)->GetSegment(0)->GetFlowProperties()->GetImpedance().value(), expected_impedance.value(), 1.e-6);
FlowSolver<2> flow_solver = FlowSolver<2>();
flow_solver.SetVesselNetwork(p_network);
flow_solver.Solve();
units::quantity<unit::pressure> expected_pressure = (inlet_pressure + Owen11Parameters::mpOutletPressure->GetValue())/2.0;
TS_ASSERT_DELTA(p_network->GetNode(7)->GetFlowProperties()->GetPressure().value(), expected_pressure.value(), 1.e-6);
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestSimpleFlowProblem"));
p_network->Write(p_handler->GetOutputDirectoryFullPath() + "bifurcating_network_results.vtp");
ParameterCollection::Instance()->DumpToFile(p_handler->GetOutputDirectoryFullPath() + "parameter_collection.xml");
ParameterCollection::Instance()->Destroy();
BaseUnits::Instance()->Destroy();
}
void TestFlowProblemWithHaematocrit() throw (Exception)
{
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestFlowProblemWithHaematocrit", false));
units::quantity<unit::length> cell_width(25.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(cell_width);
units::quantity<unit::length> target_width = 100.0 * cell_width;
units::quantity<unit::length> target_height = 30.0 * cell_width;
units::quantity<unit::length> vessel_length = 4.0 * cell_width;
VesselNetworkGenerator<3> network_generator;
boost::shared_ptr<VesselNetwork<3> > p_network = network_generator.GenerateHexagonalNetwork(target_width,
target_height,
vessel_length);
DimensionalChastePoint<3> inlet_locator(0.0, 0.0, 0.0, cell_width);
DimensionalChastePoint<3> outlet_locator(target_width/cell_width, target_height/cell_width, 0.0, cell_width);
boost::shared_ptr<VesselNode<3> > p_inlet_node = p_network->GetNearestNode(inlet_locator);
boost::shared_ptr<VesselNode<3> > p_outlet_node = p_network->GetNearestNode(outlet_locator);
p_inlet_node->GetFlowProperties()->SetIsInputNode(true);
p_inlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpInletPressure->GetValue());
p_outlet_node->GetFlowProperties()->SetIsOutputNode(true);
p_outlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
units::quantity<unit::length> vessel_radius(GenericParameters::mpCapillaryRadius->GetValue());
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
units::quantity<unit::length> sphere_radius = 400.0 * cell_width;
units::quantity<unit::length> sphere_thickess = 1.0 * cell_width;
double sphere_azimuth = M_PI;
double sphere_polar = M_PI/2.0;
network_generator.MapToSphere(p_network, sphere_radius, sphere_thickess, sphere_azimuth, sphere_polar);
VesselImpedanceCalculator<3> impedance_calculator = VesselImpedanceCalculator<3>();
impedance_calculator.SetVesselNetwork(p_network);
impedance_calculator.Calculate();
FlowSolver<3> flow_solver = FlowSolver<3>();
flow_solver.SetVesselNetwork(p_network);
flow_solver.Solve();
AlarconHaematocritSolver<3> haematocrit_solver = AlarconHaematocritSolver<3>();
haematocrit_solver.SetVesselNetwork(p_network);
haematocrit_solver.SetHaematocrit(0.45);
haematocrit_solver.Calculate();
BaseUnits::Instance()->SetReferenceLengthScale(1.e-6*unit::metres);
p_network->Write(p_handler->GetOutputDirectoryFullPath() + "network_haematocrit.vtp");
ParameterCollection::Instance()->DumpToFile(p_handler->GetOutputDirectoryFullPath() + "parameter_collection.xml");
ParameterCollection::Instance()->Destroy();
}
void TestFlowProblemStucturalAdaptation() throw (Exception)
{
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestFlowProblemStucturalAdaptation", false));
units::quantity<unit::length> reference_length(1.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(reference_length);
units::quantity<unit::length> target_width(8000*unit::microns);
units::quantity<unit::length> target_height(2000*unit::microns);
units::quantity<unit::length> vessel_length(300.0*unit::microns);
VesselNetworkGenerator<3> network_generator;
boost::shared_ptr<VesselNetwork<3> > p_network = network_generator.GenerateHexagonalNetwork(target_width,
target_height,
vessel_length);
DimensionalChastePoint<3> inlet_locator(0.0, 0.0, 0.0, reference_length);
DimensionalChastePoint<3> outlet_locator(target_width/reference_length, target_height/reference_length, 0.0, reference_length);
boost::shared_ptr<VesselNode<3> > p_inlet_node = p_network->GetNearestNode(inlet_locator);
boost::shared_ptr<VesselNode<3> > p_outlet_node = p_network->GetNearestNode(outlet_locator);
p_inlet_node->GetFlowProperties()->SetIsInputNode(true);
p_inlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpInletPressure->GetValue());
p_outlet_node->GetFlowProperties()->SetIsOutputNode(true);
p_outlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
units::quantity<unit::length> vessel_radius(40.0*unit::microns);
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
p_network->Write(p_handler->GetOutputDirectoryFullPath()+"initial_network.vtp");
BaseUnits::Instance()->SetReferenceTimeScale(60.0*unit::seconds);
SimulationTime::Instance()->SetEndTimeAndNumberOfTimeSteps(30, 1);
boost::shared_ptr<VesselImpedanceCalculator<3> > p_impedance_calculator = VesselImpedanceCalculator<3>::Create();
boost::shared_ptr<ConstantHaematocritSolver<3> > p_haematocrit_calculator = ConstantHaematocritSolver<3>::Create();
p_haematocrit_calculator->SetHaematocrit(0.45);
boost::shared_ptr<WallShearStressCalculator<3> > p_wss_calculator = WallShearStressCalculator<3>::Create();
boost::shared_ptr<MechanicalStimulusCalculator<3> > p_mech_stimulus_calculator = MechanicalStimulusCalculator<3>::Create();
StructuralAdaptationSolver<3> structural_adaptation_solver;
structural_adaptation_solver.SetVesselNetwork(p_network);
structural_adaptation_solver.SetWriteOutput(true);
structural_adaptation_solver.SetOutputFileName(p_handler->GetOutputDirectoryFullPath()+"adaptation_data.dat");
structural_adaptation_solver.SetTolerance(0.0001);
structural_adaptation_solver.SetMaxIterations(10);
structural_adaptation_solver.AddPreFlowSolveCalculator(p_impedance_calculator);
structural_adaptation_solver.AddPostFlowSolveCalculator(p_haematocrit_calculator);
structural_adaptation_solver.AddPostFlowSolveCalculator(p_wss_calculator);
structural_adaptation_solver.AddPostFlowSolveCalculator(p_mech_stimulus_calculator);
structural_adaptation_solver.SetTimeIncrement(0.01 * unit::seconds);
structural_adaptation_solver.Solve();
p_network->Write(p_handler->GetOutputDirectoryFullPath()+"network_initial_sa.vtp");
}
void TestFlowProblemStucturalAdaptationWithRegression() throw (Exception)
{
MAKE_PTR_ARGS(OutputFileHandler, p_handler, ("TestBloodFlowLiteratePaper/TestFlowProblemStucturalAdaptationWithRegression", false));
units::quantity<unit::length> reference_length(1.0 * unit::microns);
BaseUnits::Instance()->SetReferenceLengthScale(reference_length);
units::quantity<unit::length> target_width(8000*unit::microns);
units::quantity<unit::length> target_height(2000*unit::microns);
units::quantity<unit::length> vessel_length(300.0*unit::microns);
VesselNetworkGenerator<3> network_generator;
boost::shared_ptr<VesselNetwork<3> > p_network = network_generator.GenerateHexagonalNetwork(target_width,
target_height,
vessel_length);
DimensionalChastePoint<3> inlet_locator(0.0, 0.0, 0.0, reference_length);
DimensionalChastePoint<3> outlet_locator(target_width/reference_length, target_height/reference_length, 0.0, reference_length);
boost::shared_ptr<VesselNode<3> > p_inlet_node = p_network->GetNearestNode(inlet_locator);
boost::shared_ptr<VesselNode<3> > p_outlet_node = p_network->GetNearestNode(outlet_locator);
p_inlet_node->GetFlowProperties()->SetIsInputNode(true);
p_inlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpInletPressure->GetValue());
p_outlet_node->GetFlowProperties()->SetIsOutputNode(true);
p_outlet_node->GetFlowProperties()->SetPressure(Owen11Parameters::mpOutletPressure->GetValue());
units::quantity<unit::length> vessel_radius(40.0*unit::microns);
p_network->SetSegmentRadii(vessel_radius);
units::quantity<unit::dynamic_viscosity> viscosity = Owen11Parameters::mpPlasmaViscosity->GetValue();
p_network->SetSegmentViscosity(viscosity);
p_network->Write(p_handler->GetOutputDirectoryFullPath()+"initial_network.vtp");
BaseUnits::Instance()->SetReferenceTimeScale(60.0*unit::seconds);
SimulationTime::Instance()->SetEndTimeAndNumberOfTimeSteps(30, 1);
boost::shared_ptr<VesselImpedanceCalculator<3> > p_impedance_calculator = VesselImpedanceCalculator<3>::Create();
boost::shared_ptr<ConstantHaematocritSolver<3> > p_haematocrit_calculator = ConstantHaematocritSolver<3>::Create();
p_haematocrit_calculator->SetHaematocrit(0.45);
boost::shared_ptr<WallShearStressCalculator<3> > p_wss_calculator = WallShearStressCalculator<3>::Create();
boost::shared_ptr<MechanicalStimulusCalculator<3> > p_mech_stimulus_calculator = MechanicalStimulusCalculator<3>::Create();
boost::shared_ptr<StructuralAdaptationSolver<3> > p_structural_adaptation_solver = StructuralAdaptationSolver<3>::Create();
p_structural_adaptation_solver->SetVesselNetwork(p_network);
p_structural_adaptation_solver->SetWriteOutput(true);
p_structural_adaptation_solver->SetOutputFileName(p_handler->GetOutputDirectoryFullPath()+"adaptation_data.dat");
p_structural_adaptation_solver->SetTolerance(0.0001);
p_structural_adaptation_solver->SetMaxIterations(10);
p_structural_adaptation_solver->AddPreFlowSolveCalculator(p_impedance_calculator);
p_structural_adaptation_solver->AddPostFlowSolveCalculator(p_haematocrit_calculator);
p_structural_adaptation_solver->AddPostFlowSolveCalculator(p_wss_calculator);
p_structural_adaptation_solver->AddPostFlowSolveCalculator(p_mech_stimulus_calculator);
p_structural_adaptation_solver->SetTimeIncrement(0.01 * unit::seconds);
boost::shared_ptr<WallShearStressBasedRegressionSolver<3> > p_regression_solver = WallShearStressBasedRegressionSolver<3>::Create();
p_regression_solver->SetMaximumTimeWithLowWallShearStress(2.0*3600.0*unit::seconds);
p_regression_solver->SetLowWallShearStressThreshold(1.e-06*unit::pascals);
p_regression_solver->SetVesselNetwork(p_network);
MicrovesselSolver<3> microvessel_solver;
microvessel_solver.SetRegressionSolver(p_regression_solver);
microvessel_solver.SetStructuralAdaptationSolver(p_structural_adaptation_solver);
microvessel_solver.SetVesselNetwork(p_network);
microvessel_solver.SetOutputFileHandler(p_handler);
microvessel_solver.SetOutputFrequency(1);
microvessel_solver.Run();
}
};