[Insight-users] Image (memory) management question

Luis Ibanez luis.ibanez@kitware.com
Mon, 17 Mar 2003 23:06:00 -0500


Hi Gheorghe,

You may find interesting to take a look at the plug-ins
that we have been writing for integrating ITK into Volview.

They are basically following the approach that you just
described

1) Importing a C-like data buffer with the ImportImageFilter.
2) Using the CastImageFilter to convert types.
3) Hiding most of the filter parameters, and exposing a few ones.
4) Taking the output data and writing it on an external
   C-like array.

Until very recently ITK didn't offer much support for processing
data "in-place".  The pipelin architecture is always assuming that
input data is passed to a filter and that from it, independently
stored output data will be generated.

Jim Miller recently added the "itkInPlaceImageFilter" into the
Code/Common directory. This is still work in progress and has
so far affected a reduced number of filters. The population of
in-place filter is expected to grow steadily.

Probably Jim could provide a better advice about how to write
portions of the output data into the input data. He knows the
internals of the pipeline better that anyone of us.



  Regards


     Luis


---------------------------------------------
Iordanescu, Gheorghe (NIH/CC/DRD) wrote:
> Hello everybody,
> 
> I have a question about image memory management.
> 
> It seems the way to use ITK is by cascading different filters, where each
> filter generates a new image.  I would like to wrap the ITK in simple
> objects, i.e. to have an object that has a setInput() method (which should
> hide/take care about all the settings), a process(some parameters,...)
> method, a getResult() method and so on, an I would like the methods to
> affect the same data, not to generate new data. For this I’ve built a class
> (code is attached below). 
> 
> I am using as input an array style "cxLattice" structure (parent class
> BasicFilter  extracts CTWidth, CTHeight, NoOfCTSlices and data pointer
> CTPixels from it). As you can see I am using an ImportImageFilter (to import
> 3D data) and an ExtractImageFilter (to extract 2D slices and be able to save
> them on the disk). I am trying to use CastImageFilter as an image holder (as
> a pointer to an itk image). I would like to be able to process the 2D slices
> in the current 3D data, i.e. modify this data. For example I would like to
> apply a morphological filter which should affect the existing 3D data and
> not generate new 3D/2D data. In other words I would like my methods to
> affect current data, not to generate new one. 
> 
> As you can see - since I don't know any better, I am using
> importFilter->GetOutput() method to retrieve and return a pointer to the 3D
> data in the getWholeImageFromCXLattice() wrapping method, hoping it will not
> trigger an Update() request through the whole pipe-line. Now I am quite
> stuck because I would like to be able to process a certain slice (in 2D) or
> portions of the 3D data. A possible solution would be to copy the 2D data
> back to the 3D data using either an ImportImageFilter or some code like that
> suggested by Dr. Ibanez (:)) - Luis), i.e create a new 3D image and copy in
> it (mix) parts of the old 3D data and new 2D data.
> 
> new(2Dor3D)Image->SetRegions(old(2Dor3D)Image ->GetLargestPossibleRegion()
> );
> new(2Dor3D)Image ->Allocate();
> new(2Dor3D)Image ->CopyInformation(old(2Dor3D)Image );
> 
> Is there a better solution?
> 
> Thanks,
> 
> George
> 
> 
> 
> //header file START --����
> 
> // BasicFilterITK.h: interface for the BasicFilterITK class.
> //
> //////////////////////////////////////////////////////////////////////
> 
> #if
> !defined(AFX_BASICFILTERITK_H__119E5013_4EAB_41A2_A3B2_079BD6CC583E__INCLUDE
> D_)
> #define
> AFX_BASICFILTERITK_H__119E5013_4EAB_41A2_A3B2_079BD6CC583E__INCLUDED_
> 
> #if _MSC_VER > 1000
> #pragma once
> #endif // _MSC_VER > 1000
> 
> #include "BasicFilter.h"
> #include "itkImportImageFilter.h"
> #include "itkExtractImageFilter.h"
> #include "itkCastImageFilter.h"
> 
> 
> class BasicFilterITK : public BasicFilter  
> {
> public:
> 	typedef BasicFilter  Superclass;
> 	typedef Superclass::basicFilterDataType inputPixelType;
> 	typedef inputPixelType slicePixelType;
> 	enum { inputImageDimension = 3 , sliceImageDimension = 2 };
>     typedef	itk::Image< slicePixelType, sliceImageDimension >
> outputSliceImageType;
> 	typedef itk::Image< inputPixelType, inputImageDimension >
> imageFromCXLatticeType;
> 
> 	imageFromCXLatticeType::Pointer getWholeImageFromCXLattice();
> 	outputSliceImageType::Pointer	getSliceImageFromCXLattice(unsigned
> int SliceNo);
> 	
> void							saveSliceToFile
> (unsigned int SliceNo, char* SliceName=0);
> 	
> virtual	int						setInputCxLattice
> (cxLattice *lat16BitCTVolume);
> 	
> //virtual	int						setInputITKI
> mage (imageFromCXLatticeType::Pointer 
> 	
> //									
> input3DITKImage);
> 
> 	BasicFilterITK();
> 	virtual ~BasicFilterITK();
> 
> protected:
> 	typedef itk::ImportImageFilter< inputPixelType, inputImageDimension
> 
>>ImportFilterType;
> 
>     ImportFilterType::Pointer importFilter;      
> 	typedef itk::CastImageFilter< imageFromCXLatticeType,
> imageFromCXLatticeType > CastFilterType;
>     CastFilterType::Pointer imageHolderFilter;      
> 	typedef itk::ExtractImageFilter< imageFromCXLatticeType,
> outputSliceImageType > SliceExtractorFilterType;
> 	SliceExtractorFilterType::Pointer slicer ;
> 	void prepareSlicer ();
> 	//imageFromCXLatticeType::IndexType sliceIndex;
> 	//imageFromCXLatticeType::SizeType sliceSize;
> 	imageFromCXLatticeType::RegionType sliceWantedRegion;
> 
> };
> 
> #endif //
> !defined(AFX_BASICFILTERITK_H__119E5013_4EAB_41A2_A3B2_079BD6CC583E__INCLUDE
> D_)
> 
> //<- ---header file END 
> 
> ____________________________________________________________
> ____________________________________________________________
> 
> 
> 
> ////cpp file START --����
> // BasicFilterITK.cpp: implementation of the BasicFilterITK class.
> //
> //////////////////////////////////////////////////////////////////////
> 
> #include "BasicFilterITK.h"
> //#include "itkImportImageFilter.h"
> #include "itkImageFileWriter.h"
> 
> //////////////////////////////////////////////////////////////////////
> // Construction/Destruction
> //////////////////////////////////////////////////////////////////////
> BasicFilterITK::BasicFilterITK()
> {
> 	importFilter = ImportFilterType::New();
> 	slicer = SliceExtractorFilterType::New();
> 	imageHolderFilter = CastFilterType::New();      
> 
> }
> 
> BasicFilterITK::~BasicFilterITK()
> {
> }
> 
> /**/
> BasicFilterITK::imageFromCXLatticeType::Pointer
> BasicFilterITK::getWholeImageFromCXLattice()
> {
> 	return importFilter->GetOutput();
> }
> 
> BasicFilterITK::outputSliceImageType::Pointer 
> BasicFilterITK::getSliceImageFromCXLattice(unsigned int SliceNo)
> {
> 	imageFromCXLatticeType::Pointer imageFromCXLattice = 
> 		getWholeImageFromCXLattice();
> 	sliceWantedRegion.SetIndex(2, SliceNo);
> 	slicer->SetExtractionRegion  ( sliceWantedRegion) ;
> 	try{
> 		slicer->Update();
> 	}
> 	catch( itk::ExceptionObject & exp ){
> 		std::cerr << "Exception caught !" << std::endl;
> 		std::cerr << exp << std::endl;
> 	}
> 	return slicer->GetOutput();
> }
> 
> 
> int BasicFilterITK::setInputCxLattice (cxLattice *latt)
> {
> 	int returnedByParentClass = Superclass::setInput(latt);
> 	//CTSlice.setHeight(CTHeight);	//
> or	CTSlice.setHeight(getYsize(latt));
> 	
> //CTSlice.setWidth(CTWidth);	//or	CTSlice.setWidth(getXsize(latt));
> 	//if (imageFromCXLattice) imageFromCXLattice->UnRegister();
> 
> 	//import data from the cx lattice
> 	//typedef itk::ImportImageFilter< inputPixelType,
> inputImageDimension > ImportFilterType;
>     //ImportFilterType::Pointer importFilter = ImportFilterType::New();
> 
> 	ImportFilterType::SizeType  size;
> 		size[0]  = CTWidth;  // size along X
> 		size[1]  = CTHeight;  // size along Y
> 		size[2]  = NoOfCTSlices;  // size along Z
> 
> 	ImportFilterType::IndexType start;
> 	start.Fill( 0 );
> 
> 	ImportFilterType::RegionType region;
> 		region.SetIndex( start );
> 		region.SetSize(  size  );
> 	
> 	importFilter->SetRegion( region );
> 
> 	double origin[ inputImageDimension ];
> 		origin[0] = 0.0;    // X coordinate 
> 		origin[1] = 0.0;    // Y coordinate
> 		origin[2] = 0.0;    // Z coordinate
> 	
> 	importFilter->SetOrigin( origin );
> 
> 	double spacing[ inputImageDimension ];
> 		spacing[0] = 1.0;    // along X direction 
> 		spacing[1] = 1.0;    // along Y direction
> 		spacing[2] = 1.0;    // along Z direction
> 
> 	importFilter->SetSpacing( spacing );
> 
> 	const bool userPromiseToDeleteTheBuffer = true;
> 	unsigned int numberOfPixels =  size[0] * size[1] * size[2];
> 
> 	importFilter->SetImportPointer( CTPixels, 
>                                   numberOfPixels, 
>                                   userPromiseToDeleteTheBuffer );
> 
> 	//imageFromCXLattice = importFilter->GetOutput();
> 
> 	try{
> 		importFilter->Update();
> 	}
> 	catch( itk::ExceptionObject & exp ){
> 		std::cerr << "Exception caught !" << std::endl;
> 		std::cerr << exp << std::endl;
> 	}
> 
> 	prepareSlicer ();
> 
> 	return returnedByParentClass;
> }
> 
> 
> void BasicFilterITK::prepareSlicer (){
> 	imageFromCXLatticeType::Pointer imageFromCXLattice = 
> 		getWholeImageFromCXLattice();
> 	imageFromCXLatticeType::IndexType sliceIndex;
> 		sliceIndex[0] = 0; //atoi( 0 );
> 		sliceIndex[1] = 0;//atoi( 0 );
> 		sliceIndex[2] = 0;				// this is
> the Number of slice to be saved
> 
> 
> 	imageFromCXLatticeType::RegionType regionOfImageFromCxLattice = 
> 		imageFromCXLattice->GetLargestPossibleRegion ();
> 	imageFromCXLatticeType::SizeType sizeOfImageFromCxLattice = 
> 		regionOfImageFromCxLattice.GetSize ();
> 	imageFromCXLatticeType::SizeType sliceSize;
> 		sliceSize[0] = sizeOfImageFromCxLattice[0];
> 		sliceSize[1] = sizeOfImageFromCxLattice[1];
> 		sliceSize[2] = 0;
> 	//imageFromCXLatticeType::RegionType sliceWantedRegion;
> 		sliceWantedRegion.SetSize(  sliceSize  );
> 		sliceWantedRegion.SetIndex( sliceIndex );
> 	try
> 	{
> 	  slicer->SetExtractionRegion( sliceWantedRegion );
> 	}
> 	catch( itk::ExceptionObject & exp ) 
> 	{
> 		std::cerr << "Exception caught !" << std::endl;
> 		std::cerr << exp << std::endl;
> 	}
> 
> 	slicer->SetInput(  imageFromCXLattice );
> 
> }
> 
> 
> void BasicFilterITK::saveSliceToFile (unsigned int SliceNo, char* SliceName)
> {
> 	//compute the output file name
> 	char fileName[200];
> 	if (SliceName) strcpy(fileName, SliceName);
> 	else strcpy(fileName, "Slice");
> 	char buffer[20];
> 	_itoa( SliceNo, buffer, 10 );
> 	strcat(fileName, buffer);
> 	strcat(fileName, ".png");
> 
> 	//and now write the slice
> 	typedef itk::ImageFileWriter< outputSliceImageType > WriterType;
> 	WriterType::Pointer writer = WriterType::New();
> 
> 	writer->SetFileName( fileName );
> 	writer->SetInput(  getSliceImageFromCXLattice(SliceNo)  );
> 	try{
> 		writer->Update();
> 	}
> 	catch( itk::ExceptionObject & exp ){
> 		std::cerr << "Exception caught !" << std::endl;
> 		std::cerr << exp << std::endl;
> 	}
> 
> }
> 
> //<- ---CPP file END
> 
> 
> _______________________________________________
> Insight-users mailing list
> Insight-users@public.kitware.com
> http://public.kitware.com/mailman/listinfo/insight-users
>