Source code for xdesign.acquisition

#!/usr/bin/env python
# -*- coding: utf-8 -*-

# #########################################################################
# Copyright (c) 2016, UChicago Argonne, LLC. All rights reserved.         #
#                                                                         #
# Copyright 2016. UChicago Argonne, LLC. This software was produced       #
# under U.S. Government contract DE-AC02-06CH11357 for Argonne National   #
# Laboratory (ANL), which is operated by UChicago Argonne, LLC for the    #
# U.S. Department of Energy. The U.S. Government has rights to use,       #
# reproduce, and distribute this software.  NEITHER THE GOVERNMENT NOR    #
# UChicago Argonne, LLC MAKES ANY WARRANTY, EXPRESS OR IMPLIED, OR        #
# ASSUMES ANY LIABILITY FOR THE USE OF THIS SOFTWARE.  If software is     #
# modified to produce derivative works, such modified software should     #
# be clearly marked, so as not to confuse it with the version available   #
# from ANL.                                                               #
#                                                                         #
# Additionally, redistribution and use in source and binary forms, with   #
# or without modification, are permitted provided that the following      #
# conditions are met:                                                     #
#                                                                         #
#     * Redistributions of source code must retain the above copyright    #
#       notice, this list of conditions and the following disclaimer.     #
#                                                                         #
#     * Redistributions in binary form must reproduce the above copyright #
#       notice, this list of conditions and the following disclaimer in   #
#       the documentation and/or other materials provided with the        #
#       distribution.                                                     #
#                                                                         #
#     * Neither the name of UChicago Argonne, LLC, Argonne National       #
#       Laboratory, ANL, the U.S. Government, nor the names of its        #
#       contributors may be used to endorse or promote products derived   #
#       from this software without specific prior written permission.     #
#                                                                         #
# THIS SOFTWARE IS PROVIDED BY UChicago Argonne, LLC AND CONTRIBUTORS     #
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT       #
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS       #
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL UChicago     #
# Argonne, LLC OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,        #
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,    #
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;        #
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER        #
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT      #
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN       #
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE         #
# POSSIBILITY OF SUCH DAMAGE.                                             #
# #########################################################################

from __future__ import (absolute_import, division, print_function,
                        unicode_literals)

import numpy as np
from numbers import Number
from xdesign.geometry import *
from xdesign.geometry import beamintersect, beamcirc
import logging
import polytope as pt
from copy import copy

logger = logging.getLogger(__name__)


__author__ = "Doga Gursoy"
__copyright__ = "Copyright (c) 2016, UChicago Argonne, LLC."
__docformat__ = 'restructuredtext en'
__all__ = ['Beam',
           'Probe',
           'sinogram',
           'angleogram',
           'raster_scan',
           'angle_scan']


[docs]class Beam(Line): """Beam (thick line) in 2-D cartesian space. It is defined by two distinct points. Attributes ---------- p1 : Point p2 : Point size : scalar, optional Size of the beam. """ def __init__(self, p1, p2, size=0): if not isinstance(size, Number): raise TypeError("Size must be scalar.") super(Beam, self).__init__(p1, p2) self.size = float(size) def __str__(self): return "Beam(" + super(Beam, self).__str__() + ")" @property def half_space(self): """Returns the half space polytope respresentation of the infinite beam.""" # add half beam width along the normal direction to each of the points half = self.normal * self.size / 2 edges = [Line(self.p1 + half, self.p2 + half), Line(self.p1 - half, self.p2 - half)] A = np.ndarray((len(edges), self.dim)) B = np.ndarray(len(edges)) for i in range(0, 2): A[i, :], B[i] = edges[i].standard # test for positive or negative side of line if np.einsum('i, i', self.p1._x, A[i, :]) > B[i]: A[i, :] = -A[i, :] B[i] = -B[i] p = pt.Polytope(A, B) return p
[docs]class Probe(Beam): def __init__(self, p1, p2, size=0): super(Probe, self).__init__(p1, p2, size) self.history = []
[docs] def translate(self, dx): """Translates beam along its normal direction.""" vec = self.normal * dx self.p1 += vec self.p2 += vec
[docs] def measure(self, phantom, noise=False): """Return the probe measurement given phantom. When noise is > 0, poisson noise is added to the returned measurement.""" newdata = 0 for m in range(phantom.population): # print("%s Measure feature %i" % (str(self), m)) newdata += (beamintersect(self, phantom.feature[m].geometry) * phantom.feature[m].mass_atten) if noise > 0: newdata += newdata * noise * np.random.poisson(1) self.record() return newdata
[docs] def record(self): self.history.append(self.list)
[docs]def sinogram(sx, sy, phantom, noise=False): """Generates sinogram given a phantom. Parameters ---------- sx : int Number of rotation angles. sy : int Number of detection pixels (or sample translations). phantom : Phantom Returns ------- ndarray Sinogram. """ scan = raster_scan(sx, sy) sino = np.zeros((sx, sy)) for m in range(sx): for n in range(sy): sino[m, n] = next(scan).measure(phantom, noise) return sino
[docs]def angleogram(sx, sy, phantom, noise=False): """Generates angleogram given a phantom. Parameters ---------- sx : int Number of rotation angles. sy : int Number of detection pixels (or sample translations). phantom : Phantom Returns ------- ndarray Angleogram. """ scan = angle_scan(sx, sy) angl = np.zeros((sx, sy)) for m in range(sx): for n in range(sy): angl[m, n] = next(scan).measure(phantom, noise) return angl
[docs]def raster_scan(sx, sy): """Provides a beam list for raster-scanning. Parameters ---------- sx : int Number of rotation angles. sy : int Number of detection pixels (or sample translations). Yields ------ Probe """ # Step size of the probe. step = 1. / sy # Step size of the rotation angle. theta = np.pi / sx # Fixed probe location. p = Probe(Point([step / 2., -10]), Point([step / 2., 10]), step) for m in range(sx): for n in range(sy): yield copy(p) p.translate(step) p.translate(-1) p.rotate(theta, Point([0.5, 0.5]))
[docs]def angle_scan(sx, sy): """Provides a beam list for raster-scanning. Parameters ---------- sx : int Number of rotation angles. sy : int Number of detection pixels (or sample translations). Yields ------ Probe """ # Step size of the probe. step = 0.1 / sy # Fixed rotation points. p1 = Point([0, 0.5]) p2 = Point([0.5, 0.5]) # Step size of the rotation angle. beta = np.pi / (sx + 1) alpha = np.pi / sy # Fixed probe location. p = Probe(Point([step / 2., -10]), Point([step / 2., 10]), step) for m in range(sx): for n in range(sy): yield p p.rotate(-alpha, p1) p.rotate(np.pi, p1) p1.rotate(-beta, p2) p.rotate(-beta, p2)