From ebc0292cbd3ab2e890c9b6ea92396e6de00f736e Mon Sep 17 00:00:00 2001
From: Andrei Savici <saviciat@ornl.gov>
Date: Thu, 18 Sep 2014 18:39:26 -0400
Subject: [PATCH] Deal with all dimensions. Refs #9202

Also use proton charge
---
 .../inc/MantidMDAlgorithms/SXDMDNorm.h        |   8 --
 .../Framework/MDAlgorithms/src/SXDMDNorm.cpp  | 100 ++++++++----------
 2 files changed, 45 insertions(+), 63 deletions(-)

diff --git a/Code/Mantid/Framework/MDAlgorithms/inc/MantidMDAlgorithms/SXDMDNorm.h b/Code/Mantid/Framework/MDAlgorithms/inc/MantidMDAlgorithms/SXDMDNorm.h
index 6978b7d6474..c6f199868a9 100644
--- a/Code/Mantid/Framework/MDAlgorithms/inc/MantidMDAlgorithms/SXDMDNorm.h
+++ b/Code/Mantid/Framework/MDAlgorithms/inc/MantidMDAlgorithms/SXDMDNorm.h
@@ -60,14 +60,6 @@ namespace MDAlgorithms
     size_t hIndex,kIndex,lIndex;
     ///name of other dimensions
     std::vector<std::string> otherDims;
-    ///limits for other dimensions
-    std::vector<coord_t> otherDimsMin,otherDimsMax;
-    ///flag id other dimensions are integrated
-    std::vector<bool> otherDimsIntegrated;
-    ///index of other dimensions in the output workspaces
-    std::vector<size_t> otherDimsIndex;
-    ///map dimensions
-    std::unordered_map<size_t,size_t> dim;
     ///(2*PiRUBW)^-1
     Mantid::Kernel::DblMatrix transf;
     /// limits for momentum
diff --git a/Code/Mantid/Framework/MDAlgorithms/src/SXDMDNorm.cpp b/Code/Mantid/Framework/MDAlgorithms/src/SXDMDNorm.cpp
index 07cf688ed30..b8d85d4fb86 100644
--- a/Code/Mantid/Framework/MDAlgorithms/src/SXDMDNorm.cpp
+++ b/Code/Mantid/Framework/MDAlgorithms/src/SXDMDNorm.cpp
@@ -6,27 +6,7 @@ TODO: Enter a full wiki-markup description of your algorithm here. You can then
 #include "MantidMDEvents/MDEventWorkspace.h"
 #include "MantidMDEvents/MDHistoWorkspace.h"
 #include "MantidAPI/WorkspaceValidators.h"
-#include "MantidAPI/ImplicitFunctionFactory.h"
-#include "MantidGeometry/MDGeometry/MDBoxImplicitFunction.h"
-#include "MantidGeometry/MDGeometry/MDHistoDimension.h"
-#include "MantidKernel/CPUTimer.h"
-#include "MantidKernel/Strings.h"
-#include "MantidKernel/System.h"
-#include "MantidKernel/Utils.h"
-#include "MantidMDEvents/CoordTransformAffineParser.h"
-#include "MantidMDEvents/CoordTransformAligned.h"
-#include "MantidMDEvents/MDBoxBase.h"
-#include "MantidMDEvents/MDBox.h"
-#include "MantidMDEvents/MDEventFactory.h"
-#include "MantidMDEvents/MDEventWorkspace.h"
-#include "MantidMDEvents/MDHistoWorkspace.h"
 #include "MantidMDAlgorithms/BinMD.h"
-#include <boost/algorithm/string.hpp>
-#include <Poco/DOM/Document.h>
-#include <Poco/DOM/DOMParser.h>
-#include <Poco/DOM/Element.h>
-#include "MantidKernel/EnabledWhenProperty.h"
-#include "MantidMDEvents/CoordTransformAffine.h"
 #include "MantidKernel/TimeSeriesProperty.h"
 #include<algorithm>
 #include "MantidDataObjects/EventWorkspace.h"
@@ -169,8 +149,6 @@ namespace MDAlgorithms
           float dimMin=static_cast<float>(m_inputWS->getDimension(i)->getMinimum());
           float dimMax=static_cast<float>(m_inputWS->getDimension(i)->getMaximum());
 
-          otherDimsMin.push_back(dimMin);
-          otherDimsMax.push_back(dimMax);
           Kernel::TimeSeriesProperty<double> *run_property = dynamic_cast<Kernel::TimeSeriesProperty<double> *>(m_inputWS->getExperimentInfo(0)->run().getProperty(m_inputWS->getDimension(i)->getName()));
           double value=run_property->firstValue();
           otherValues.push_back(value);
@@ -204,7 +182,7 @@ namespace MDAlgorithms
       //get indices of the original dimensions in the output workspace, and if not found, the corresponding dimension is integrated
       Mantid::Kernel::Matrix<coord_t> mat=m_normWS->getTransformFromOriginal(0)->makeAffineMatrix();
 
-      for (size_t row=0; row<mat.numRows()-1; row++)
+      for (size_t row=0; row<mat.numRows()-1; row++)//affine matrix, ignore last row
       {
           if(mat[row][0]==1.)
           {
@@ -212,7 +190,6 @@ namespace MDAlgorithms
               hIndex=row;
               if(hMin<m_normWS->getDimension(row)->getMinimum()) hMin=m_normWS->getDimension(row)->getMinimum();
               if(hMax>m_normWS->getDimension(row)->getMaximum()) hMax=m_normWS->getDimension(row)->getMaximum();
-              dim[row]=0;
               if((hMin>m_normWS->getDimension(row)->getMaximum())||(hMax<m_normWS->getDimension(row)->getMinimum()))
               {
                   skipProcessing=true;
@@ -224,7 +201,6 @@ namespace MDAlgorithms
               kIndex=row;
               if(kMin<m_normWS->getDimension(row)->getMinimum()) kMin=m_normWS->getDimension(row)->getMinimum();
               if(kMax>m_normWS->getDimension(row)->getMaximum()) kMax=m_normWS->getDimension(row)->getMaximum();
-              dim[row]=1;
               if((kMin>m_normWS->getDimension(row)->getMaximum())||(kMax<m_normWS->getDimension(row)->getMinimum()))
               {
                   skipProcessing=true;
@@ -236,14 +212,22 @@ namespace MDAlgorithms
               lIndex=row;
               if(lMin<m_normWS->getDimension(row)->getMinimum()) lMin=m_normWS->getDimension(row)->getMinimum();
               if(lMax>m_normWS->getDimension(row)->getMaximum()) lMax=m_normWS->getDimension(row)->getMaximum();
-              dim[row]=2;
               if((lMin>m_normWS->getDimension(row)->getMaximum())||(lMax<m_normWS->getDimension(row)->getMinimum()))
               {
                   skipProcessing=true;
               }
 
-              //TODO: do the same thing for otherdimensions
-
+              for(size_t col=3;col<mat.numCols()-1;col++) //affine matrix, ignore last column
+              {
+                  if(mat[row][col]==1.)
+                  {
+                      double val=otherValues.at(col-3);
+                      if((val>m_normWS->getDimension(row)->getMaximum())||(val<m_normWS->getDimension(row)->getMinimum()))
+                      {
+                          skipProcessing=true;
+                      }
+                  }
+              }
           }
       }
 
@@ -260,6 +244,7 @@ namespace MDAlgorithms
       }
       else
       {
+          double PC=m_normWS->getExperimentInfo(0)->run().getProtonCharge();
           Kernel::PropertyWithValue< std::vector<double> > *prop=dynamic_cast< Mantid::Kernel::PropertyWithValue<std::vector<double> >*>(m_normWS->getExperimentInfo(0)->getLog("RUBW_MATRIX"));
           Mantid::Kernel::DblMatrix RUBW((*prop)()); //includes the 2*pi factor but not goniometer for now :)
           transf=m_normWS->getExperimentInfo(0)->run().getGoniometerMatrix()*RUBW;
@@ -270,20 +255,21 @@ namespace MDAlgorithms
           Mantid::API::Progress *prog=new Mantid::API::Progress(this,0.3,1,static_cast<int64_t>(detIDS.size()));
           const detid2index_map d2m=fluxW->getDetectorIDToWorkspaceIndexMap();
           const detid2index_map d2mSA=sA->getDetectorIDToWorkspaceIndexMap();
-          //TODO make parallel
-          PARALLEL_FOR3(m_normWS,fluxW,sA)
+
+          PARALLEL_FOR1(m_normWS)
           for(size_t i=0;i<detIDS.size();i++)
           {
               PARALLEL_START_INTERUPT_REGION
               Mantid::Geometry::IDetector_const_sptr detector=m_normWS->getExperimentInfo(0)->getInstrument()->getDetector(detIDS[i]);
               if(!detector->isMonitor()&&!detector->isMasked())
               {
+                  //get intersections
                   std::vector<Mantid::Kernel::VMD> intersections=calculateIntersections(detector);
                   if(!intersections.empty())
                   {
                       //calculate indices
                       //add to the correct signal at that particular index
-                      //NOTE: if parallel it has to be atomic
+                      //NOTE: if parallel it has to be atomic/critical
 
                       //get event vector
                       size_t sp=d2m.find(detIDS[i])->second;
@@ -292,35 +278,38 @@ namespace MDAlgorithms
                       std::vector<Mantid::DataObjects::WeightedEventNoTime>::iterator start=el.begin();
                       while((*start).tof()<(*intersections.begin())[3]) ++start;
 
-                      double solid=sA->readY(d2mSA.find(detIDS[i])->second)[0];
+                      double solid=sA->readY(d2mSA.find(detIDS[i])->second)[0]*PC;
                       std::vector<Mantid::Kernel::VMD>::iterator it;
                       for (it=intersections.begin()+1;it!=intersections.end();it++)
                       {
-                          Mantid::Kernel::VMD deltav=(*it)-(*(it-1));
-                          Mantid::Kernel::VMD avev=((*it)+(*(it-1)))*0.5;
-                          double eps=1e-7;
+                          Mantid::Kernel::VMD deltav=(*it)-(*(it-1));//difference between consecutive intersections
+                          Mantid::Kernel::VMD avev=((*it)+(*(it-1)))*0.5;//average between two intersection (to get position)
+                          double eps=1e-7;//do not integrate if momemntum difference is smaller than eps, assume contribution is 0
                           if (deltav[3]>eps)
                           {
-                            size_t hind,kind,lind;
-                            hind=static_cast<size_t>((avev[0]-m_normWS->getDimension(0)->getMinimum())/m_normWS->getDimension(0)->getBinWidth());
-                            kind=static_cast<size_t>((avev[1]-m_normWS->getDimension(1)->getMinimum())/m_normWS->getDimension(1)->getBinWidth());
-                            lind=static_cast<size_t>((avev[2]-m_normWS->getDimension(2)->getMinimum())/m_normWS->getDimension(2)->getBinWidth());
-                            if((hind<m_normWS->getDimension(hIndex)->getNBins())&&(kind<m_normWS->getDimension(kIndex)->getNBins()) &&(lind<m_normWS->getDimension(lIndex)->getNBins()))
-                            {
-                                double signal=0.;
-                                while((*start).tof()<(*it)[3])
-                                {
-                                    signal+=(*start).weight();
-                                    ++start;
-                                }
-                                signal*=solid;
-                                //TODO: replace with some locks instead
-                                PARALLEL_CRITICAL(updateMD)
-                                {
-                                    signal+=m_normWS->getSignalAt(m_normWS->getLinearIndex(hind,kind,lind));
-                                    m_normWS->setSignalAt(m_normWS->getLinearIndex(hind,kind,lind),signal);
-                                }
-                            }
+                              std::vector<coord_t> pos=avev.toVector<coord_t>();
+                              pos.insert(pos.end()-1,otherValues.begin(),otherValues.end());
+                              VMD posNew=mat*pos;
+                              size_t linIndex=m_normWS->getLinearIndexAtCoord(posNew.toVector<coord_t>().data());
+
+                              if(linIndex!=size_t(-1))
+                              {
+                                  double signal=0.;
+                                  while((*start).tof()<(*it)[3])
+                                  {
+                                      signal+=(*start).weight();
+                                      if (start==el.end())
+                                          break;
+                                      ++start;
+                                  }
+                                  signal*=solid;
+
+                                  PARALLEL_CRITICAL(updateMD)
+                                  {
+                                    signal+=m_normWS->getSignalAt(linIndex);
+                                    m_normWS->setSignalAt(linIndex,signal);
+                                  }
+                              }
                           }
                       }
 
@@ -330,6 +319,7 @@ namespace MDAlgorithms
               PARALLEL_END_INTERUPT_REGION
           }
           PARALLEL_CHECK_INTERUPT_REGION
+
       }
 
       this->setProperty("OutputNormalizationWorkspace",m_normWS);
-- 
GitLab