# -----------------------------------------------------------------------
# Copyright: 2010-2016, iMinds-Vision Lab, University of Antwerp
# 2013-2016, CWI, Amsterdam
#
# Contact: astra@uantwerpen.be
# Website: http://www.astra-toolbox.com/
#
# This file is part of the ASTRA Toolbox.
#
#
# The ASTRA Toolbox is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# The ASTRA Toolbox is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with the ASTRA Toolbox. If not, see .
#
# -----------------------------------------------------------------------
#
# distutils: language = c++
# distutils: libraries = astra
import six
cimport cython
cimport PyData3DManager
from .PyData3DManager cimport CData3DManager
from .PyIncludes cimport *
import numpy as np
cimport numpy as np
np.import_array()
cimport PyXMLDocument
from .PyXMLDocument cimport XMLDocument
cimport utils
from .utils import wrap_from_bytes
from .pythonutils import geom_size, GPULink
import operator
from six.moves import reduce
include "config.pxi"
cdef extern from "Python.h":
void* PyLong_AsVoidPtr(object)
cdef CData3DManager * man3d = PyData3DManager.getSingletonPtr()
cdef extern from *:
CFloat32Data3DMemory * dynamic_cast_mem "dynamic_cast" (CFloat32Data3D * ) except NULL
cdef extern from "CFloat32CustomPython.h":
cdef cppclass CFloat32CustomPython:
CFloat32CustomPython(arrIn)
def create(datatype,geometry,data=None, link=False):
cdef Config *cfg
cdef CVolumeGeometry3D * pGeometry
cdef CProjectionGeometry3D * ppGeometry
cdef CFloat32Data3D * pDataObject3D
cdef CConeProjectionGeometry3D* pppGeometry
cdef CFloat32CustomMemory * pCustom = NULL
IF HAVE_CUDA==True:
cdef MemHandle3D hnd
if link:
if isinstance(data, GPULink):
s = geom_size(geometry)
if geom_size(geometry) != ( data.z, data.y, data.x ):
raise Exception("The dimensions of the data do not match those specified in the geometry.")
elif data.shape!=geom_size(geometry):
raise Exception("The dimensions of the data do not match those specified in the geometry.")
if datatype == '-vol':
cfg = utils.dictToConfig(six.b('VolumeGeometry'), geometry)
pGeometry = new CVolumeGeometry3D()
if not pGeometry.initialize(cfg[0]):
del cfg
del pGeometry
raise Exception('Geometry class not initialized.')
if link:
if isinstance(data, GPULink):
IF HAVE_CUDA==True:
s = geom_size(geometry)
hnd = wrapHandle(PyLong_AsVoidPtr(data.ptr), data.x, data.y, data.z, data.pitch/4)
pDataObject3D = new CFloat32VolumeData3DGPU(pGeometry, hnd)
ELSE:
raise NotImplementedError("CUDA support is not enabled in ASTRA")
else:
pCustom = new CFloat32CustomPython(data)
pDataObject3D = new CFloat32VolumeData3DMemory(pGeometry, pCustom)
else:
pDataObject3D = new CFloat32VolumeData3DMemory(pGeometry)
del cfg
del pGeometry
elif datatype == '-sino' or datatype == '-proj3d' or datatype == '-sinocone':
cfg = utils.dictToConfig(six.b('ProjectionGeometry'), geometry)
tpe = wrap_from_bytes(cfg.self.getAttribute(six.b('type')))
if (tpe == "parallel3d"):
ppGeometry = new CParallelProjectionGeometry3D();
elif (tpe == "parallel3d_vec"):
ppGeometry = new CParallelVecProjectionGeometry3D();
elif (tpe == "cone"):
ppGeometry = new CConeProjectionGeometry3D();
elif (tpe == "cone_vec"):
ppGeometry = new CConeVecProjectionGeometry3D();
else:
raise Exception("Invalid geometry type.")
if not ppGeometry.initialize(cfg[0]):
del cfg
del ppGeometry
raise Exception('Geometry class not initialized.')
if link:
if isinstance(data, GPULink):
IF HAVE_CUDA==True:
s = geom_size(geometry)
hnd = wrapHandle(PyLong_AsVoidPtr(data.ptr), data.x, data.y, data.z, data.pitch/4)
pDataObject3D = new CFloat32ProjectionData3DGPU(ppGeometry, hnd)
ELSE:
raise NotImplementedError("CUDA support is not enabled in ASTRA")
else:
pCustom = new CFloat32CustomPython(data)
pDataObject3D = new CFloat32ProjectionData3DMemory(ppGeometry, pCustom)
else:
pDataObject3D = new CFloat32ProjectionData3DMemory(ppGeometry)
del ppGeometry
del cfg
else:
raise Exception("Invalid datatype. Please specify '-vol' or '-proj3d'.")
if not pDataObject3D.isInitialized():
del pDataObject3D
raise Exception("Couldn't initialize data object.")
if not link: fillDataObject(dynamic_cast_mem(pDataObject3D), data)
return man3d.store(pDataObject3D)
def get_geometry(i):
cdef CFloat32Data3DMemory * pDataObject = dynamic_cast_mem(getObject(i))
cdef CFloat32ProjectionData3DMemory * pDataObject2
cdef CFloat32VolumeData3DMemory * pDataObject3
if pDataObject.getType() == THREEPROJECTION:
pDataObject2 = pDataObject
geom = utils.configToDict(pDataObject2.getGeometry().getConfiguration())
elif pDataObject.getType() == THREEVOLUME:
pDataObject3 = pDataObject
geom = utils.configToDict(pDataObject3.getGeometry().getConfiguration())
else:
raise Exception("Not a known data object")
return geom
def change_geometry(i, geom):
cdef CFloat32Data3DMemory * pDataObject = dynamic_cast_mem(getObject(i))
cdef CFloat32ProjectionData3DMemory * pDataObject2
cdef CFloat32VolumeData3DMemory * pDataObject3
if pDataObject.getType() == THREEPROJECTION:
pDataObject2 = pDataObject
# TODO: Reduce code duplication here
cfg = utils.dictToConfig(six.b('ProjectionGeometry'), geom)
tpe = wrap_from_bytes(cfg.self.getAttribute(six.b('type')))
if (tpe == "parallel3d"):
ppGeometry = new CParallelProjectionGeometry3D();
elif (tpe == "parallel3d_vec"):
ppGeometry = new CParallelVecProjectionGeometry3D();
elif (tpe == "cone"):
ppGeometry = new CConeProjectionGeometry3D();
elif (tpe == "cone_vec"):
ppGeometry = new CConeVecProjectionGeometry3D();
else:
raise Exception("Invalid geometry type.")
if not ppGeometry.initialize(cfg[0]):
del cfg
del ppGeometry
raise Exception('Geometry class not initialized.')
del cfg
if (ppGeometry.getDetectorColCount() != pDataObject2.getDetectorColCount() or \
ppGeometry.getProjectionCount() != pDataObject2.getAngleCount() or \
ppGeometry.getDetectorRowCount() != pDataObject2.getDetectorRowCount()):
del ppGeometry
raise Exception(
"The dimensions of the data do not match those specified in the geometry.")
pDataObject2.changeGeometry(ppGeometry)
del ppGeometry
elif pDataObject.getType() == THREEVOLUME:
pDataObject3 = pDataObject
cfg = utils.dictToConfig(six.b('VolumeGeometry'), geom)
pGeometry = new CVolumeGeometry3D()
if not pGeometry.initialize(cfg[0]):
del cfg
del pGeometry
raise Exception('Geometry class not initialized.')
del cfg
if (pGeometry.getGridColCount() != pDataObject3.getColCount() or \
pGeometry.getGridRowCount() != pDataObject3.getRowCount() or \
pGeometry.getGridSliceCount() != pDataObject3.getSliceCount()):
del pGeometry
raise Exception(
"The dimensions of the data do not match those specified in the geometry.")
pDataObject3.changeGeometry(pGeometry)
del pGeometry
else:
raise Exception("Not a known data object")
cdef fillDataObject(CFloat32Data3DMemory * obj, data):
if data is None:
fillDataObjectScalar(obj, 0)
else:
if isinstance(data, np.ndarray):
fillDataObjectArray(obj, np.ascontiguousarray(data,dtype=np.float32))
else:
fillDataObjectScalar(obj, np.float32(data))
cdef fillDataObjectScalar(CFloat32Data3DMemory * obj, float s):
cdef int i
for i in range(obj.getSize()):
obj.getData()[i] = s
@cython.boundscheck(False)
@cython.wraparound(False)
cdef fillDataObjectArray(CFloat32Data3DMemory * obj, float [:,:,::1] data):
if (not data.shape[0] == obj.getDepth()) or (not data.shape[1] == obj.getHeight()) or (not data.shape[2] == obj.getWidth()):
raise Exception(
"The dimensions of the data do not match those specified in the geometry.")
cdef float [:,:,::1] cView = obj.getData3D()[0][0]
cView[:] = data
cdef CFloat32Data3D * getObject(i) except NULL:
cdef CFloat32Data3D * pDataObject = man3d.get(i)
if pDataObject == NULL:
raise Exception("Data object not found")
if not pDataObject.isInitialized():
raise Exception("Data object not initialized properly.")
return pDataObject
@cython.boundscheck(False)
@cython.wraparound(False)
def get(i):
cdef CFloat32Data3DMemory * pDataObject = dynamic_cast_mem(getObject(i))
outArr = np.empty((pDataObject.getDepth(),pDataObject.getHeight(), pDataObject.getWidth()),dtype=np.float32,order='C')
cdef float [:,:,::1] mView = outArr
cdef float [:,:,::1] cView = pDataObject.getData3D()[0][0]
mView[:] = cView
return outArr
def get_shared(i):
cdef CFloat32Data3DMemory * pDataObject = dynamic_cast_mem(getObject(i))
outArr = np.empty((pDataObject.getDepth(),pDataObject.getHeight(), pDataObject.getWidth()),dtype=np.float32,order='C')
cdef np.npy_intp shape[3]
shape[0] = pDataObject.getDepth()
shape[1] = pDataObject.getHeight()
shape[2] = pDataObject.getWidth()
return np.PyArray_SimpleNewFromData(3,shape,np.NPY_FLOAT32,pDataObject.getData3D()[0][0])
def get_single(i):
raise Exception("Not yet implemented")
def store(i,data):
cdef CFloat32Data3D * pDataObject = getObject(i)
fillDataObject(dynamic_cast_mem(pDataObject), data)
def dimensions(i):
cdef CFloat32Data3D * pDataObject = getObject(i)
return (pDataObject.getDepth(),pDataObject.getHeight(),pDataObject.getWidth())
def delete(ids):
try:
for i in ids:
man3d.remove(i)
except TypeError:
man3d.remove(ids)
def clear():
man3d.clear()
def info():
six.print_(wrap_from_bytes(man3d.info()))