diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/CMakeLists.txt b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/CMakeLists.txt
new file mode 100644
index 0000000000000000000000000000000000000000..3510a90caa617b8f7f3f7141b1f0ad20d81085dc
--- /dev/null
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/CMakeLists.txt
@@ -0,0 +1,59 @@
+################################################################################
+# Package: PileupReweighting
+################################################################################
+
+# Declare the package name:
+atlas_subdir( PileupReweighting )
+
+# Declare the package's dependencies:
+atlas_depends_on_subdirs( PUBLIC
+                          Control/AthToolSupport/AsgTools
+                          Event/xAOD/xAODEventInfo
+                          GaudiKernel
+                          PhysicsAnalysis/AnalysisCommon/PATInterfaces
+                          PRIVATE
+                          Control/AthenaBaseComps
+                          Tools/PathResolver )
+
+# External dependencies:
+find_package( ROOT COMPONENTS MathCore RIO Cint Core Tree Hist pthread MathMore Minuit Minuit2 Matrix Physics HistPainter Rint )
+
+# tag ROOTBasicLibs was not recognized in automatic conversion in cmt2cmake
+
+# tag ROOTCintexLibs was not recognized in automatic conversion in cmt2cmake
+
+# Component(s) in the package:
+atlas_add_root_dictionary( PileupReweightingLib
+                           PileupReweightingLibDictSource
+                           ROOT_HEADERS PileupReweighting/TPileupReweighting.h Root/LinkDef.h
+                           EXTERNAL_PACKAGES ROOT )
+
+atlas_add_library( PileupReweightingLib
+                   Root/*.cxx
+                   ${PileupReweightingLibDictSource}
+                   PUBLIC_HEADERS PileupReweighting
+                   PRIVATE_INCLUDE_DIRS ${ROOT_INCLUDE_DIRS}
+                   LINK_LIBRARIES AsgTools xAODEventInfo GaudiKernel PATInterfaces
+                   PRIVATE_LINK_LIBRARIES ${ROOT_LIBRARIES} AthenaBaseComps PathResolver )
+
+atlas_add_component( PileupReweighting
+                     src/components/*.cxx
+                     INCLUDE_DIRS ${ROOT_INCLUDE_DIRS}
+                     LINK_LIBRARIES ${ROOT_LIBRARIES} AsgTools xAODEventInfo GaudiKernel PATInterfaces AthenaBaseComps PathResolver PileupReweightingLib )
+
+atlas_add_dictionary( PileupReweightingDict
+                      PileupReweighting/PileupReweightingDict.h
+                      PileupReweighting/selection.xml
+                      INCLUDE_DIRS ${ROOT_INCLUDE_DIRS}
+                      LINK_LIBRARIES ${ROOT_LIBRARIES} AsgTools xAODEventInfo GaudiKernel PATInterfaces AthenaBaseComps PathResolver PileupReweightingLib )
+
+atlas_add_executable( testPRW
+                      src/testPRW.C
+                      INCLUDE_DIRS ${ROOT_INCLUDE_DIRS}
+                      LINK_LIBRARIES ${ROOT_LIBRARIES} AsgTools xAODEventInfo GaudiKernel PATInterfaces AthenaBaseComps PathResolver PileupReweightingLib )
+
+# Install files from the package:
+atlas_install_python_modules( python/*.py )
+atlas_install_joboptions( share/*.py )
+atlas_install_data( share/*.root )
+
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/IPileupReweightingTool.h b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/IPileupReweightingTool.h
index b99122c814dcebdbe2a1181f7acd06b0a7c45fc2..151760209caf6e1346b0e559d46d6744dc3c0916 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/IPileupReweightingTool.h
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/IPileupReweightingTool.h
@@ -12,28 +12,86 @@
 #include "TString.h"
 #include "xAODEventInfo/EventInfo.h"
 
-namespace Root {
-class TPileupReweighting;
-}
+#include "PATInterfaces/ISystematicsTool.h"
+
+class TH1;
 
 namespace CP {
+class TPileupReweighting;
 
-   class IPileupReweightingTool : public virtual asg::IAsgTool {
+   class IPileupReweightingTool : public virtual asg::IAsgTool, virtual public CP::ISystematicsTool {
 
       ASG_TOOL_INTERFACE( CP::IPileupReweightingTool )
 
       public:
-         /// Decorates with: PileupWeight , RandomRunNumber, RandomLumiBlockNumber
-         virtual void apply ( const xAOD::EventInfo* eventInfo ) = 0;
+         /// Return combined pileup weight
+         virtual float getCombinedWeight( const xAOD::EventInfo& eventInfo ) = 0;
+
+         /// Same as above, but for a 'custom weight' variable
+         virtual float getCombinedWeight( const xAOD::EventInfo& eventInfo,Double_t x, Double_t y=0. ) = 0;
+
+         /// When using UnrepresentedDataAction=2, you may want to apply this additional weight to ensure sum of weights are preserved
+         virtual float getUnrepresentedDataWeight( const xAOD::EventInfo& eventInfo ) = 0;
+
+         /// Get the dataWeight used to 'unprescale' data collected from a given trigger combination. mu_dependency is recommended to be true
+         virtual float getDataWeight( const xAOD::EventInfo& eventInfo, const TString& trigger, bool mu_dependent=true ) = 0;
+
+         /// Get a random run number for this MC event, mu_dependency is recommended ... jetetmiss seem to like it muchly
+         virtual int getRandomRunNumber( const xAOD::EventInfo& eventInfo , bool mu_dependent=true) = 0;
+
+         /// Get the mu of a lumiblock ... needed to 'correct' the number in datasets
+         virtual float getLumiBlockMu( const xAOD::EventInfo& eventInfo ) = 0;
+         /// Get the integrated lumi of a lumiblock (in pb-1)
+         virtual double getLumiBlockIntegratedLumi( const xAOD::EventInfo& eventInfo ) = 0;
+
+         /// Decorates with: 
+         /// MC: PileupWeight (CombinedWeight[*UnrepresentedDataWeight if action=2]), RandomRunNumber, RandomLumiBlockNumber, PRWHash
+         /// Data: corrected_averageInteractionsPerCrossing
+         /// mu_dependent says if the mu_dependency should be used for random run numbers or the data weights. You will get random run numbers of 0 for events with zero pileup weight
+         virtual StatusCode apply ( const xAOD::EventInfo& eventInfo, bool mu_dependent=true ) = 0;
+
+
+         /// return the prw hash used for fast updates of weights at the post-processing level ... see the share/makeWeightTree.C script for usage
+         virtual ULong64_t getPRWHash( const xAOD::EventInfo& eventInfo ) = 0;
+
+         // methods that go straight to the underlying pileup tool 
+
+         /// Get a random lumiblock number for the given run number 
+         virtual UInt_t GetRandomLumiBlockNumber(UInt_t runNumber) = 0;
+
+         /// possible alternative to using the EventBookkeepers info ... assuming you made your PRW Config file!!
+         virtual Double_t GetSumOfEventWeights(Int_t channel) = 0;
+         virtual Double_t GetNumberOfEvents(Int_t channel) = 0;
+
+         /// Get the integrated luminosity (in pb-1) between start and end run (inclusive)
+         virtual Double_t GetIntegratedLumi(UInt_t start, UInt_t end) = 0;
+         /// Total lumi (in pb-1) for a given trigger combination .. leave blank for the unprescaled
+         virtual Double_t GetIntegratedLumi(const TString& trigger="") = 0;
+         /** similar to above, but for only the given mcRunNumber/periodNumber */
+         virtual Double_t GetIntegratedLumi(Int_t periodNumber, UInt_t start, UInt_t end) = 0;
+         /** return fraction of lumi assigned to periodNumber (or mcRunNumber) that is between start and end data run numbers*/
+         virtual Double_t GetIntegratedLumiFraction(Int_t periodNumber, UInt_t start, UInt_t end) = 0;
+         /** return fraction of lumi assigned to periodNumber (or mcRunNumber) with given mu, that is between start and end data run numbers*/
+         virtual Double_t GetIntegratedLumiFraction(Int_t periodNumber, Double_t mu, UInt_t start, UInt_t end) = 0;
+
+
+         /// use these methods when generating config files 
+         virtual Int_t AddPeriod(Int_t periodNumber, UInt_t start, UInt_t end) = 0;
+         virtual Int_t SetBinning(Int_t nbinsx, Double_t* xbins, Int_t nbinsy=0, Double_t* ybins=0) = 0;
+         virtual Int_t SetUniformBinning(Int_t nbinsx, Double_t xlow, Double_t xup, Int_t nbinsy=0, Double_t ylow=0, Double_t yup=0) = 0;
+         virtual Int_t SetBinning(TH1* hist) = 0;
+
+         /// Call this method once per event when in config file generating mode and you want standard mu reweighting
+         virtual int fill( const xAOD::EventInfo& eventInfo ) = 0;
+
+         /// Use this method if you want to do a generic reweighting instead
+         virtual int fill( const xAOD::EventInfo& eventInfo, Double_t x, Double_t y=0.) = 0;
+
 
-         /// Get the dataWeight used to 'unprescale' data collected from a given trigger combination
-         virtual double dataWeight ( const xAOD::EventInfo* eventInfo, TString trigger, bool mu_dependent=true ) = 0;
+         /// Get pointer to the underlying tool - expert use only:  Will require #include "PileupReweighting/TPileupReweighting.h"
+         virtual CP::TPileupReweighting* expert() = 0;
 
-         /// Get pointer to the underlying tool - expert use only
-         virtual Root::TPileupReweighting* get() = 0;
 
-         /// execute method - retrieves the eventInfo from the storegate and decorates it for you 
-         virtual StatusCode execute() = 0;
    };
 
 }
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/PileupReweightingTool.h b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/PileupReweightingTool.h
index 597d125cdec68ba4e3e87dd7850d29a71fa74881..7c54e8ebe20b05cd026d7397105054f0354d792f 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/PileupReweightingTool.h
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/PileupReweightingTool.h
@@ -14,6 +14,9 @@
 #include "PileupReweighting/IPileupReweightingTool.h"
 #include "PileupReweighting/TPileupReweighting.h"
 
+#include "AsgTools/ToolHandle.h"
+#include "PATInterfaces/IWeightTool.h"
+
 namespace CP {
 
    /// Implementation of the PileupReweighting tool
@@ -21,13 +24,36 @@ namespace CP {
    /// @author Will Buttinger
    ///
 
-   class PileupReweightingTool : public virtual IPileupReweightingTool,
-                             public asg::AsgTool, public Root::TPileupReweighting {
+   class PileupReweightingTool : 
+                             public asg::AsgTool, public virtual CP::TPileupReweighting,public virtual IPileupReweightingTool {
 
       /// Create a proper constructor for Athena
       ASG_TOOL_CLASS( PileupReweightingTool, CP::IPileupReweightingTool )
 
    public:
+
+      /// Get a random lumiblock number for the given run number 
+      virtual UInt_t GetRandomLumiBlockNumber(UInt_t runNumber) { return m_activeTool->GetRandomLumiBlockNumber(runNumber); }
+      /// Get the integrated luminosity (in pb-1) between start and end run (inclusive)
+      virtual Double_t GetIntegratedLumi(UInt_t start, UInt_t end)  { return m_activeTool->GetIntegratedLumi(start, end); }
+      /// Total lumi (in pb-1) for a given trigger combination .. leave blank for the unprescaled
+      virtual Double_t GetIntegratedLumi(const TString& trigger="")  { return m_activeTool->GetIntegratedLumi(trigger); }
+      /** similar to above, but for only the given mcRunNumber/periodNumber */
+      virtual Double_t GetIntegratedLumi(Int_t periodNumber, UInt_t start, UInt_t end)  { return m_activeTool->GetIntegratedLumi(periodNumber, start, end); }
+      /** return fraction of lumi assigned to periodNumber (or mcRunNumber) that is between start and end data run numbers*/
+      virtual Double_t GetIntegratedLumiFraction(Int_t periodNumber, UInt_t start, UInt_t end)  { return m_activeTool->GetIntegratedLumiFraction(periodNumber, start, end); }
+      /** return fraction of lumi assigned to periodNumber (or mcRunNumber) with given mu, that is between start and end data run numbers*/
+      virtual Double_t GetIntegratedLumiFraction(Int_t periodNumber, Double_t mu, UInt_t start, UInt_t end) { return m_activeTool->GetIntegratedLumiFraction(periodNumber, mu, start, end); }
+      /// use these methods when generating config files 
+      virtual Int_t AddPeriod(Int_t periodNumber, UInt_t start, UInt_t end)  { return m_activeTool->AddPeriod(periodNumber, start, end); }
+      virtual Int_t SetBinning(Int_t nbinsx, Double_t* xbins, Int_t nbinsy=0, Double_t* ybins=0)  { return m_activeTool->SetBinning(nbinsx, xbins, nbinsy, ybins); }
+      virtual Int_t SetUniformBinning(Int_t nbinsx, Double_t xlow, Double_t xup, Int_t nbinsy=0, Double_t ylow=0, Double_t yup=0)  { return m_activeTool->SetUniformBinning(nbinsx, xlow, xup, nbinsy, ylow, yup); }
+      virtual Int_t SetBinning(TH1* hist)  { return m_activeTool->SetBinning(hist); }
+      /// helpful alternative to using the EventBookkeepers info ... assuming you made your PRW Config file!!
+      virtual Double_t GetSumOfEventWeights(Int_t channel)  { return m_activeTool->GetSumOfEventWeights(channel); }
+      virtual Double_t GetNumberOfEvents(Int_t channel)  { return m_activeTool->GetNumberOfEvents(channel) ; }
+
+
       /// Constructor for standalone usage
       PileupReweightingTool( const std::string& name );
 
@@ -37,43 +63,77 @@ namespace CP {
       /// Finalize - can call the WriteToFile for us
       virtual StatusCode finalize();
 
-      /// Uses Input and Output properties to get and decorate an eventinfo object
-      virtual StatusCode execute();
-
       /// Decorates with: 
-      /// MC: PileupWeight , RandomRunNumber, RandomLumiBlockNumber 
-      /// Data: data weights from trigger expressions you've specified 
-      virtual void apply ( const xAOD::EventInfo* eventInfo );
-
-      /// Get the dataWeight used to 'unprescale' data collected from a given trigger combination
-      virtual double dataWeight ( const xAOD::EventInfo* eventInfo, TString trigger, bool mu_dependent=true );
-
-
-      /// Get pointer to the underlying tool - expert use only
-      virtual Root::TPileupReweighting* get() { return this; }
-
+      /// MC: PileupWeight (CombinedWeight[*UnrepresentedDataWeight if action=2]), RandomRunNumber, RandomLumiBlockNumber 
+      /// Data: corrected_averageInteractionsPerCrossing
+      /// mu_dependent says if the mu_dependency should be used for random run numbers or the data weights. You will get random run numbers of 0 for events with zero pileup weight
+      virtual StatusCode apply ( const xAOD::EventInfo& eventInfo, bool mu_dependent=true );
+
+      /// Return combined pileup weight
+      virtual float getCombinedWeight( const xAOD::EventInfo& eventInfo );
+
+      /// Same as above, but for a 'custom weight' variable
+      virtual float getCombinedWeight( const xAOD::EventInfo& eventInfo,Double_t x, Double_t y=0. );
+
+      /// return the prw hash used for fast updates of weights at the post-processing level ... see the share/makeWeightTree.C script for usage
+      virtual ULong64_t getPRWHash( const xAOD::EventInfo& eventInfo );
+
+      /// Get the mu of a lumiblock ... needed to 'correct' the number in datasets
+      virtual float getLumiBlockMu( const xAOD::EventInfo& eventInfo );
+      /// Get the integrated lumi of a lumiblock (in pb-1)
+      virtual double getLumiBlockIntegratedLumi(const xAOD::EventInfo& eventInfo);
+
+      /// When using UnrepresentedDataAction=2, you may want to apply this additional weight to ensure sum of weights are preserved
+      virtual float getUnrepresentedDataWeight( const xAOD::EventInfo& eventInfo );
+   
+      /// Get the dataWeight used to 'unprescale' data collected from a given trigger combination. mu_dependency is recommended to be true
+      virtual float getDataWeight( const xAOD::EventInfo& eventInfo, const TString& trigger, bool mu_dependent=true );
+   
+      /// Get a random run number for this MC event, using mu-dependent randomization by default ... jetetmiss seem to like it muchly
+      virtual int getRandomRunNumber( const xAOD::EventInfo& eventInfo , bool mu_dependent=true);
+   
+      /// Call this method once per event when in config file generating mode and you want standard mu reweighting
+      virtual int fill( const xAOD::EventInfo& eventInfo );
+   
+      /// Use this method if you want to do a generic reweighting instead
+      virtual int fill( const xAOD::EventInfo& eventInfo, Double_t x, Double_t y=0.);
+
+      /// Get pointer to the underlying tool - expert use only. Will require #include "PileupReweighting/TPileupReweighting.h"
+      virtual CP::TPileupReweighting* expert() { return m_activeTool; }
+
+
+
+      /// The ISystematicsTool methods 
+      bool isAffectedBySystematic( const CP::SystematicVariation& systematic ) const;
+      CP::SystematicSet affectingSystematics() const;
+      CP::SystematicSet recommendedSystematics() const;
+      CP::SystematicCode applySystematicVariation( const CP::SystematicSet& systConfig );
+
+#ifndef XAOD_STANDALONE
+      void updateHandler(Property& /*p*/);
+#endif
 
    private:
+      std::string m_configStream;
       bool m_inConfigMode;
-      //Root::TPileupReweighting* m_tool;
+      CP::TPileupReweighting *m_upTool, *m_downTool; //systematic variation instances for the reweighting
+
+      double m_upVariation; double m_downVariation;
+      CP::SystematicVariation m_systUp, m_systDown;
+      CP::TPileupReweighting* m_activeTool; //defaults to this
 
       std::vector<std::string> m_prwFiles;
       std::vector<std::string> m_lumicalcFiles;
-      std::vector<std::string> m_lumicalcTriggers;
-
-      std::vector<std::string> m_dataWeightTriggers;
-      bool m_useMuDependent;
 
       std::string m_prefix;
-      std::string m_inputKey, m_outputKey;
 
-      int m_unrepdataAction;
       int m_defaultChannel;
-      double m_dataScaleFactor;
       std::string m_usePeriodConfig;
+      std::map<int, bool> m_doneConfigs;
 
+      bool m_noWeightsMode;
 
-      void checkInit(const xAOD::EventInfo* eventInfo);
+      ToolHandle<IWeightTool> m_weightTool;
 
       // MN:  this prevents ROOT dict generator from complaining about lack of ClassDef()
       // Note: inheriting from TObject and not having ClassDef makes this class unsuitable for I/O
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/TPileupReweighting.h b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/TPileupReweighting.h
index a224bac9292e7ed146499471f5b5cfc2ef817510..161bb9c054228134579040052e7f949fdc315545 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/TPileupReweighting.h
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/TPileupReweighting.h
@@ -7,8 +7,6 @@
 #ifndef __TPILEUPREWEIGHTING__
 #define __TPILEUPREWEIGHTING__
 
-#define PILEUPREWEIGHTINGVERSION 3 //useful maybe for other packages to detect with. Each release I'll try to remember to increment this
-
 /**
    @class TPileupReweighting
    @brief Tool to get the calculated MC pileup weight. Also does custom weights and other useful stuff.
@@ -46,7 +44,8 @@ class TTree;
 class TFile;
 
 
-namespace Root {
+
+namespace CP {
   class TPileupReweighting : public TNamed {
 
   public: 
@@ -58,51 +57,61 @@ namespace Root {
 
 
   public:
-      /** Initialize this class once before the event loop starts 
-          If distribution information is provided, it is assumed to be 
-          for the standard pileup reweighting */
-      Int_t initialize(const TString dataRootFileName="",
-                    const TString dataRootHistName="",
-                    const TString mcRootFileName="",
-                    const TString mcRootHistName="", int runNumber=0, int channel=0);
 
-      Int_t Initialize();
+      /** use a hardcoded period configuration */
+      Int_t UsePeriodConfig(const TString configName);
+      /** Add a histogram binning config. To modify the pileup histo binning, use "pileup" as name */
+      Int_t SetBinning(Int_t nbinsx, Double_t* xbins, Int_t nbinsy=0, Double_t* ybins=0);
+      Int_t SetUniformBinning(Int_t nbinsx, Double_t xlow, Double_t xup, Int_t nbinsy=0, Double_t ylow=0, Double_t yup=0);
+      Int_t SetBinning(TH1* hist);
 
-      //-------------------------------------------------------------
-      //Bonus method for retrieving the int lumi in each data period
-      //the MuonEfficiencyCorrections package wants this
-      //-------------------------------------------------------------
-      std::vector<Double_t> getIntegratedLumiVector();
+      /** Set which channel should be used as a default when specific mc channel distributions cannot be found */
+      /** default channels can now be assigned for each mc run number .. mcRunNumber=-1 is the global default */
+      void SetDefaultChannel(Int_t channel, Int_t mcRunNumber=-1);
+      Int_t GetDefaultChannel(Int_t mcRunNumber=-1);
 
-      /** total luminosity loaded and accepted by the tool (in inverse pb) */
-      Double_t GetIntegratedLumi(TString trigger="");
+      /** total luminosity loaded and accepted by the tool (in inverse pb) */ 
+      Double_t GetIntegratedLumi(const TString& trigger="");
       /** totalMC maps should also hold the total number of entries for each channel */
       inline Double_t GetNumberOfEvents(Int_t channel) {
-         std::map<Int_t, Double_t>& tMap = globalNumberOfEntries["pileup"];
-         if(tMap.find(channel)==tMap.end()) {
+         Period* global = m_periods[-1];
+         if(!global) return 0;
+         if(global->numberOfEntries.find(channel)==global->numberOfEntries.end()) {
             Error("GetNumberOfEvents", "Unknown channel: %d",channel);
             return 0;
          }
-         return tMap[channel];
+         return global->numberOfEntries[channel];
       }
       inline Double_t GetSumOfEventWeights(Int_t channel) {
-         std::map<Int_t, Double_t>& tMap = globalTotals["pileup"];
-         if(tMap.find(channel)==tMap.end()) {
+         Period* global = m_periods[-1];
+         if(!global) return 0;
+         if(global->sumOfWeights.find(channel)==global->sumOfWeights.end()) {
             Error("GetSumOfEventWeights", "Unknown channel: %d",channel);
             return 0;
          }
-         return tMap[channel];
+         return global->sumOfWeights[channel];
       }
+
+      /** Combine two period numbers. Histos are merged and the first number will be redirected to the second (the second is created if it doesn't exist) */
+      void RemapPeriod(Int_t periodNumber1, Int_t periodNumber2);
+
+      
       /** return fraction of lumi assigned to periodNumber (or mcRunNumber) that is between start and end data run numbers*/
       Double_t GetIntegratedLumiFraction(Int_t periodNumber, UInt_t start, UInt_t end);
       /** return fraction of lumi assigned to periodNumber (or mcRunNumber) with given mu, that is between start and end data run numbers*/
       Double_t GetIntegratedLumiFraction(Int_t periodNumber, Double_t mu, UInt_t start, UInt_t end);
 
       /** get the amount of integrated lumi between the two given runs */
-      Double_t GetIntegratedLumi(UInt_t start, UInt_t end);
+      inline Double_t GetIntegratedLumi(UInt_t start, UInt_t end) {    return GetIntegratedLumi(-1,start,end); }
       /** similar to above, but for only the given mcRunNumber/periodNumber */
       Double_t GetIntegratedLumi(Int_t periodNumber, UInt_t start, UInt_t end);
 
+      /** get integrated lumi for specific run and lumiblock number .. comes from the 'unprescaled lumi', and is in pb*/
+      Double_t GetLumiBlockIntegratedLumi(Int_t runNumber, UInt_t lb);
+
+      /** get the lumiblock mu, useful for 'updating' the mu coming from data to account for new lumitags */
+      Float_t GetLumiBlockMu(Int_t runNumber, UInt_t lb);
+
       //-----------------------------------------------------
       //General Tool Configuring methods
       //-----------------------------------------------------
@@ -110,80 +119,66 @@ namespace Root {
       inline void DisableWarnings(Bool_t in) { m_SetWarnings = !in;}
       /** Indicate if additional debugging information should be output */
       inline void EnableDebugging(Bool_t in) { m_debugging = in;}
-      /** Set which channel should be used as a default when specific mc channel distributions cannot be found */
-      inline void SetDefaultChannel(Int_t channel) { m_defaultChannel=channel;}
-      /** default channels can now be assigned for each mc run number */
-      inline void SetDefaultChannelByMCRunNumber(Int_t channel, Int_t mcRunNumber) { m_defaultChannelsByMCRunNumber[mcRunNumber]=channel; }
-      Int_t GetDefaultChannel(Int_t mcRunNumber=0);
+
       /** Set how to handle configurations where some of your data has no corresponding mc to describe it 
           0=Default (Throw exception), 1=Subtract lumi from normalizations (i.e. discard data), 2=keep lumi and continue*/
       inline void SetUnrepresentedDataAction(Int_t action, Double_t tolerance=0.05) {
-         if(action<0 || action>2) {
+         if(action<0 || action>3) {
             Fatal("SetUnrepresentedDataAction","Set action=%d - I'm afraid I can't let you do that, Dave",action);
             throw std::runtime_error("Throwing 2001");
          } 
          m_unrepresentedDataAction=action;
          m_unrepDataTolerance=tolerance; //applicable for action=2 mode. Default is 5%
       }
+      /** return the unrepresented data fraction in a given channel .. when using action=2, you will want to scale up all MC events by 1/(1-unrepFraction) to account for missed data */
+      Double_t GetUnrepresentedDataFraction(Int_t periodNumber,Int_t channel);
+      inline Float_t GetUnrepresentedDataWeight(Int_t periodNumber,Int_t channel) { 
+         if(m_unrepresentedDataAction!=2) {
+            Warning("GetUnrepresentedDataWeight","You should not be applying this weight unless the UnrepresentedDataAction=2");
+         }
+         return 1./(1. - GetUnrepresentedDataFraction(periodNumber,channel));
+      }
+      //-----------------------------------------------------
+      //Methods to veto data in Action=1 mode
+      //-----------------------------------------------------
+      Bool_t IsUnrepresentedData(Int_t runNumber, Float_t x, Float_t y=0.);
+
       /** Should the tool ignore period assignments encoded in config file */
       inline void IgnoreConfigFilePeriods(Bool_t in) { m_ignoreFilePeriods=in; }
 
       /** Assign an mc RunNumber to a data period */
-      Int_t AddPeriod(Int_t runNumber, UInt_t start, UInt_t end);
-      /** Get the PeriodNumber of the given runNumber,start and end. If not found, returns -1 */
-      Int_t GetPeriodNumber(Int_t runNumber, UInt_t start, UInt_t end);
+      Int_t AddPeriod(Int_t periodNumber, UInt_t start, UInt_t end);
       /** Get the first period number with the data run number contained - assume all periods are disconnected for this to be useful */
       Int_t GetFirstFoundPeriodNumber(UInt_t runNumber);
-      /** Alternative method where you give it the mcRunNumber and the 'data' runNumber */
-      Int_t GetPeriodNumber(Int_t mcRunNumber, UInt_t runNumber);
-      /** use a hardcoded period configuration */
-      Int_t UsePeriodConfig(const TString configName);
-      /** Add a histogram binning config. To modify the pileup histo binning, use "pileup" as name */
-      Int_t AddBinning(const TString weightName,Int_t nbinsx, Double_t* xbins);
-      Int_t AddBinning(const TString weightName,Int_t nbinsx, Double_t xlow, Double_t xup);
-      Int_t AddBinning(const TString weightName,Int_t nbinsx, Double_t xlow, Double_t xup, Int_t nbinsy, Double_t ylow, Double_t yup);
-      Int_t AddBinning(const TString weightName,TH1* hist);
 
-      /** Combine two mc RunNumbers. Histos are merged and the second number will be redirected to the first */
-      void MergeMCRunNumbers(Int_t runNumber1, Int_t runNumber2, Int_t newRunNumber=0);
+
+
       /** Scale the LumiMetaData mu values by this factor */
-      void SetDataScaleFactors(Float_t x,Float_t y=1., Float_t z=1.) { m_dataScaleFactorX=x;m_dataScaleFactorY=y;m_dataScaleFactorZ=z; }
-      void SetMCScaleFactors(Float_t x,Float_t y=1.,Float_t z=1.) { m_mcScaleFactorX=x;m_mcScaleFactorY=y;m_mcScaleFactorZ=z;}
+      inline void SetDataScaleFactors(Float_t x,Float_t y=1.) { m_dataScaleFactorX=x;m_dataScaleFactorY=y; }
+      inline void SetMCScaleFactors(Float_t x,Float_t y=1.) { m_mcScaleFactorX=x;m_mcScaleFactorY=y;}
+
+
 
       //-----------------------------------------------------
-      //Methods to load config files or histograms
+      //Methods to load config files
       //-----------------------------------------------------
       Int_t AddConfigFile(const TString fileName);
       Int_t AddLumiCalcFile(const TString fileName, const TString trigger="None");
       Int_t AddMetaDataFile(const TString fileName,const TString channelBranchName="mc_channel_number");
 
-      Int_t AddDistribution(TH1* hist, const TString weightName, Int_t runNumber, Int_t channelNumber);
-      Int_t AddDistribution(TH1* hist, Int_t runNumber, Int_t channelNumber);
-      Int_t AddDistributionFromFile(const TString fileName, const TString histName, const TString weightName, Int_t runNumber=0, Int_t channelNumber=0);
-      Int_t AddDistributionFromFile(const TString fileName, const TString histName, Int_t runNumber=0, Int_t channelNumber=0);
+      /** Initialize this class once before the event loop starts 
+          If distribution information is provided, it is assumed to be 
+          for the standard pileup reweighting */
+      Int_t Initialize();
+
 
       //-----------------------------------------------------
       //Methods to get the various weights
       //-----------------------------------------------------
-      Float_t GetCombinedWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber,Float_t x,Float_t y=0., Float_t z=0.);
-      Float_t GetPeriodWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber);
-      Float_t GetPrimaryWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber,Float_t x);
-      Float_t GetSecondaryWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber,Float_t x, Float_t y);
-
-
-      //inline methods for above methods with the default "pileup" weight
-      inline Float_t GetCombinedWeight(Int_t periodNumber, Int_t channelNumber,Float_t x, Float_t y=0., Float_t z=0.) {
-         return GetCombinedWeight("pileup",periodNumber,channelNumber,x,y,z);
-      }
-      inline Float_t GetPeriodWeight(Int_t periodNumber, Int_t channelNumber) {
-         return GetPeriodWeight("pileup",periodNumber,channelNumber);
-      }
-      inline Float_t GetPrimaryWeight(Int_t periodNumber, Int_t channelNumber,Float_t x) {
-         return GetPrimaryWeight("pileup",periodNumber,channelNumber,x);
-      }
-      inline Float_t GetSecondaryWeight(Int_t periodNumber, Int_t channelNumber,Float_t x, Float_t y) {
-         return GetSecondaryWeight("pileup",periodNumber,channelNumber,x,y);
-      }
+      Float_t GetCombinedWeight(Int_t periodNumber, Int_t channelNumber,Float_t x,Float_t y=0.);
+      Float_t GetPeriodWeight(Int_t periodNumber, Int_t channelNumber);
+      Float_t GetPrimaryWeight(Int_t periodNumber, Int_t channelNumber,Float_t x);
+      Float_t GetSecondaryWeight(Int_t periodNumber, Int_t channelNumber,Float_t x,Float_t y);
 
       //-----------------------------------------------------
       //Methods to work with the metadata
@@ -207,17 +202,16 @@ namespace Root {
       //-----------------------------------------------------
       //Methods to generate config files
       //-----------------------------------------------------
-      Int_t Fill(const TString weightName,Int_t runNumber,Int_t channelNumber,Float_t w,Float_t x, Float_t y=0., Float_t z=0.);
-      Int_t Fill(Int_t runNumber,Int_t channelNumber,Float_t w,Float_t x, Float_t y=0., Float_t z=0.);
+      Int_t Fill(Int_t runNumber,Int_t channelNumber,Float_t w,Float_t x, Float_t y=0.);
       Int_t WriteToFile(const TString filename=""); //if no name given, will use tool name
       Int_t WriteToFile(TFile* outFile);
 
-      //-----------------------------------------------------
-      //Methods to veto data in Action=1 mode
-      //-----------------------------------------------------
-      Bool_t IsUnrepresentedData(const TString weightName, Int_t runNumber, Float_t x, Float_t y=0., Float_t z=0.);
-      Bool_t IsUnrepresentedData(Int_t runNumber, Float_t x, Float_t y=0., Float_t z=0.);
 
+      //
+      //Methods used to do the prwTree - for fast recalculation of prw weights when processing ntuples
+      //
+      ULong64_t GetPRWHash(Int_t periodNumber, Int_t channelNumber, Float_t x, Float_t y=0.);
+      Bool_t MakeWeightTree(TString channelNumbers, TString outFile, TString hashBranch="PRWHash", TString weightBranch="PileupWeight");
 
       //-----------------------------------------------------
       //RandomDataPeriod functionality stuff
@@ -226,9 +220,8 @@ namespace Root {
       void SetRandomSeed(int seed) {m_random3->SetSeed(seed);}
       int GetRandomSeed() {return m_random3->GetSeed();}
       /** Gets a random data run number according to the integrated lumi distribution associated to this mcRunNumber */
+      /** allows use of custom reweighting config files  */
       UInt_t GetRandomRunNumber(Int_t mcRunNumber);
-      /** Similar to above method, but allows use of custom reweighting config files .. using weightName='pileup' has same behaviour as above */
-      UInt_t GetRandomRunNumber(Int_t mcRunNumber, TString weightName);
       /** Get random run number according to integrated lumi distribution for the given mu value (uses the binning of the mu histogram) */
       UInt_t GetRandomRunNumber(Int_t mcRunNumber, Double_t mu);
       /** Get random period number from the sub-periods assigned to this run number */
@@ -241,75 +234,56 @@ namespace Root {
       //Methods for PROOF cluster merging of generated histos
       //-----------------------------------------------------
       Int_t Merge(TCollection *coll);
-      std::map<TString, std::map<Int_t,std::map<Int_t, TH1*> > >& GetInputHistograms() { return m_inputHistograms;}
+      //*std::map<Int_t,std::map<Int_t, TH1*> > & GetInputHistograms() { return m_inputHistograms;}
 
 
       //-----------------------------------------------------
       //Methods to inspect the input and weighting histograms
       //-----------------------------------------------------
-      TH1* GetInputHistogram(const TString weightName, Int_t channelNumber, Int_t runNumber) {
-         if(m_inputHistograms.find(weightName)==m_inputHistograms.end()||
-            m_inputHistograms[weightName].find(channelNumber)==m_inputHistograms[weightName].end()||
-            m_inputHistograms[weightName][channelNumber].find(runNumber)==m_inputHistograms[weightName][channelNumber].end()) {
-            return 0;
-         }
-         return m_inputHistograms[weightName][channelNumber][runNumber];
-      }
-
-      TH1D* GetPrimaryDistribution(const TString weightName, Int_t channelNumber, Int_t periodNumber) {
-         if(primaryDistributions.find(weightName)==primaryDistributions.end()||
-            primaryDistributions[weightName].find(channelNumber)==primaryDistributions[weightName].end()||
-            primaryDistributions[weightName][channelNumber].find(periodNumber)==primaryDistributions[weightName][channelNumber].end()) {
+      TH1* GetInputHistogram(Int_t channelNumber, Int_t periodNumber) {
+         if(m_periods.find(periodNumber)==m_periods.end()||
+            m_periods[periodNumber]->inputHists.find(channelNumber)==m_periods[periodNumber]->inputHists.end()) {
             return 0;
          }
-         return primaryDistributions[weightName][channelNumber][periodNumber];
+         return m_periods[periodNumber]->inputHists[channelNumber];
       }
 
-      TH2D* GetSecondaryDistribution(const TString weightName, Int_t channelNumber, Int_t periodNumber) {
-         if(secondaryDistributions.find(weightName)==secondaryDistributions.end()||
-            secondaryDistributions[weightName].find(channelNumber)==secondaryDistributions[weightName].end()||
-            secondaryDistributions[weightName][channelNumber].find(periodNumber)==secondaryDistributions[weightName][channelNumber].end()) {
+      TH1D* GetPrimaryDistribution(Int_t channelNumber, Int_t periodNumber) {
+         if(m_periods.find(periodNumber)==m_periods.end()||
+            m_periods[periodNumber]->primaryHists.find(channelNumber)==m_periods[periodNumber]->primaryHists.end()) {
             return 0;
          }
-         return secondaryDistributions[weightName][channelNumber][periodNumber];
+         return m_periods[periodNumber]->primaryHists[channelNumber];
       }
 
       TH1D* GetPrimaryTriggerDistribution(const TString trigger, Int_t periodNumber) {
-         if(m_triggerPrimaryDistributions.find(trigger)==m_triggerPrimaryDistributions.end()||
-            m_triggerPrimaryDistributions[trigger].find(periodNumber)==m_triggerPrimaryDistributions[trigger].end()) {
+         if(m_periods.find(periodNumber)==m_periods.end()||
+            m_periods[periodNumber]->triggerHists.find(trigger)==m_periods[periodNumber]->triggerHists.end()) {
             return 0;
          }
-         return m_triggerPrimaryDistributions[trigger][periodNumber];
+         return m_periods[periodNumber]->triggerHists[trigger];
       }
 
       /** Method for weighting data to account for prescales and mu bias. Use by giving the tool multiple lumicalc files, one for each trigger */
       Double_t GetDataWeight(Int_t runNumber, TString trigger, Double_t x);
-      Double_t GetDataWeight(Int_t runNumber, TString trigger); //version without mu dependence
+      Double_t GetDataWeight(Int_t runNumber, TString trigger);//version without mu dependence
 
 
       // other methods 
       Bool_t IsInitialized() { return m_isInitialized; }
 
-      //-----------------------------------------------------
-      //Depricated methods
-      //-----------------------------------------------------
-      /** Add a data distribution histogram or ttree (do this after mc) */
-      void AddDataDistribution(const TString& dataRootFileName,const TString& dataRootHistName, int runNumber=0);
-      /** Add a mc distribution histogram or ttree */
-      void AddMCDistribution(const TString& mcRootFileName,const TString& mcRootHistName, int runNumber=0, int channel=0);
-      /** Add a custom data distribution histogram or ttree (do this after mc) */
-      void AddCustomDataDistribution(const TString& customRootFileName,const TString& customRootHistName, TString customName="", int runNumber=0);
-      /** Add a custom mc distribution histogram or ttree */
-      void AddCustomMCDistribution(const TString& customRootFileName,const TString& customRootHistName, TString customName="", int runNumber=0, int channel=0);
-      Float_t getPileupWeight( float mu, int runNumber=0, int channel=0 );
-
-  private:
-
-      TH1* CloneEmptyHistogram(const TString weightName, Int_t runNumber, Int_t channelNumber);
+      Int_t AddDistribution(TH1* hist, Int_t runNumber, Int_t channelNumber);
+
+  protected:
+      Int_t GetNearestGoodBin(Int_t thisMCRunNumber, Int_t bin);
+      Int_t IsBadBin(Int_t thisMCRunNumber, Int_t bin);
+
+
+      TH1* CloneEmptyHistogram(Int_t runNumber, Int_t channelNumber);
       /** Normalize histograms */
       void normalizeHistogram(TH1* histo);
       void AddDistributionTree(TTree *tree, TFile *file);
-      Int_t FactorizeDistribution(TH1* hist, const TString weightName, Int_t channelNumber, Int_t periodNumber,bool includeInMCRun,bool includeInGlobal);
+      //*Int_t FactorizeDistribution(TH1* hist, const TString weightName, Int_t channelNumber, Int_t periodNumber,bool includeInMCRun,bool includeInGlobal);
 
       void CalculatePrescaledLuminosityHistograms(const TString trigger);
 
@@ -317,84 +291,34 @@ namespace Root {
       Bool_t m_SetWarnings;
       Bool_t m_debugging;
       Bool_t m_countingMode;
-      Int_t m_defaultChannel;
       Int_t m_unrepresentedDataAction;
-      Bool_t m_isInitialized;
-      Bool_t m_lumiVectorIsLoaded;
-      Float_t m_dataScaleFactorX;Float_t m_dataScaleFactorY;Float_t m_dataScaleFactorZ;
-      Float_t m_mcScaleFactorX;Float_t m_mcScaleFactorY;Float_t m_mcScaleFactorZ;
+      Bool_t m_isInitialized;Bool_t m_lumiVectorIsLoaded;
+      Float_t m_dataScaleFactorX;Float_t m_dataScaleFactorY;
+      Float_t m_mcScaleFactorX;Float_t m_mcScaleFactorY;
       Int_t m_nextPeriodNumber;
       Bool_t m_ignoreFilePeriods;
       TTree *m_metadatatree;
       Double_t m_unrepDataTolerance;
       Bool_t m_doGlobalDataWeight; //used in GetDataWeight to flag mu-independent version
+      Int_t m_lumicalcRunNumberOffset; //used for 'faking' a lumicalc file for run2
 
-      //-----------------------------------------------------
-      //Bonus method members
-      //-----------------------------------------------------
-      std::vector<Double_t> m_integratedLumiVector;
+      /** map storing the lumicalc file locations - used when building DataPileupWeights */
+      std::map<TString,TString> m_lumicalcFiles;
 
       //-----------------------------------------------------
       //Shared private data members
       //-----------------------------------------------------
-      /** Map from mc RunNumber to (periodNumber,[start,end]) period definitions. One mc runNumber can have many disconnected periods */
-      std::map<Int_t,std::map<Int_t, std::pair<UInt_t,UInt_t> > > m_periods;
-      /** Quick map back from periodNumber to mcRunNumber */
-      std::map<Int_t,Int_t> m_periodToMCRun;
-      /** Map of empty histograms to use for custom weights. Also holds the standard pileup histogram in "pileup" */
-      std::map<TString, TH1*> m_emptyHistograms;
-      /** Remappings of mc run numbers. Used through the MergeMCRunNumbers method */
-      std::map<Int_t,Int_t> m_mcRemappings;
-      std::map<Int_t,std::map<Int_t,Int_t> > m_mcMerges;
 
-      //-----------------------------------------------------
-      //The standard pileup private members
-      //-----------------------------------------------------
-      /** [weightName,channelNumber] -> total (N,L,D) */
-      std::map<TString, std::map<Int_t, Double_t> > globalTotals;
-      /** [weightName,channelNumber,periodNumber] -> total (N_A,L_A,D_A) */
-      std::map<TString, std::map<Int_t, std::map<Int_t, Double_t> > > periodTotals;
-      /** [weightName,channelNumber,periodNumber] -> 1D distribution */
-      std::map<TString, std::map<Int_t, std::map<Int_t, TH1D*> > > primaryDistributions;
-      /** [weightName,channelNumber,periodNumber] -> 2D distribution */
-      std::map<TString, std::map<Int_t, std::map<Int_t, TH2D*> > > secondaryDistributions;
-
-      /** [weightName,channelNumber] -> NumberOfEntries from histograms */
-      std::map<TString, std::map<Int_t, Double_t> > globalNumberOfEntries;
-
-      //data is stored in channel=-1
-      std::map<TString, std::map<Int_t,std::map<Int_t, TH1*> > > m_inputHistograms;
-
-      //[trigger,runNumber] -> lumicalc derived histograms used to weight data to account for prescaling, based on mu
-      std::map<TString, std::map<Int_t, TH1*> > m_triggerInputHistograms;
-      //[trigger,periodNumber] -> assembled into the individual periods
-      std::map<TString, std::map<Int_t, TH1D*> > m_triggerPrimaryDistributions;
 
-      /** channel metadata map */
-      std::map<TString, std::map<Int_t, Double_t> > m_metadata; 
 
-      //-----------------------------------------------------
-      //RandomDataPeriod functionality stuff
-      //numbers generated seperately for each mc period
-      //-----------------------------------------------------
-      TRandom3 *m_random3;
-      /** Integrated lumi in each run, split in to the different mc periods: [weightName,mcRunNumber,runNumber]->lumi */
-      std::map<TString, std::map<Int_t, std::map<UInt_t, Double_t> > > dataPeriodRunTotals;
-      /** Integrated lumi in each run for each bin of the different mc periods: [weightName, binNumber, mcRunNumber, runNumber]->lumi*/
-#ifndef __MAKECINT__ //this map is too deep for rootcint to handle
-      std::map<TString, std::map<UInt_t, std::map<Int_t, std::map<UInt_t, Double_t> > > > dataPeriodBinnedRunTotals;
-#endif
-      /** [weightName,datarunnum,binnum] -> badbin flag */
-      std::map<TString, std::map<Int_t, std::map<Int_t, Bool_t> > > m_badbins;
+      /** the empty histogram used for this weight... effectively holds the configuration of the binning */
+      TH1* m_emptyHistogram; 
 
-      /** mapping for default channels by mc run number - introduced for mc12a/b combined functionality */
-      std::map<Int_t,Int_t> m_defaultChannelsByMCRunNumber;
 
-      /** lumi in each lumiblock in each run loaded. used for GetRandomLumiBlockNumber */
-      std::map<UInt_t, std::map<UInt_t, Double_t> > m_lumiByRunAndLbn;
 
-      /** map storing the lumicalc file locations - used when building DataPileupWeights */
-      std::map<TString,TString> m_lumicalcFiles;
+      /** channel metadata map */
+      std::map<TString, std::map<Int_t, Double_t> > m_metadata; 
+
 
       struct CompositeTrigger {
          int op;
@@ -422,7 +346,62 @@ namespace Root {
 
       CompositeTrigger* makeTrigger(TString s);
 
-      ClassDef(TPileupReweighting,1)
+public:
+      inline void PrintPeriods() { for(auto p : m_periods) {std::cout << p.first << " -> "; p.second->print("");} }
+      struct Period {
+         Period(Int_t _id, UInt_t _start, UInt_t _end, Int_t _defaultChannel) : id(_id),start(_start),end(_end),defaultChannel(_defaultChannel) {};
+         Int_t id;
+         UInt_t start;
+         UInt_t end;
+         Int_t defaultChannel;
+         std::map<Int_t, Int_t> inputBinRedirect;
+         std::map<Int_t, Double_t> unrepData; //indexed by channel number
+         std::vector<Period*> subPeriods; 
+         std::vector<UInt_t> runNumbers; //populated with runNumbers that actually had some data available
+         std::map<Int_t, TH1*> inputHists;
+         std::map<Int_t, Double_t> sumOfWeights; 
+         std::map<Int_t, Int_t> numberOfEntries;
+         std::map<Int_t, TH1D*> primaryHists; //normalized histograms, indexed by channelNumber. -1 holds the data
+         std::map<Int_t, TH2D*> secondaryHists; //semi-normalized histograms, only used in 2D reweighting
+         std::map<TString, TH1D*> triggerHists; //unnormalized ... i.e. integral should be equal to the lumi! ... should really only be filled in the channel=-1 case (i.e. data)
+         bool contains(unsigned int runNumber) {
+            if(runNumber >= start && runNumber <= end) return true;
+            for(auto p : subPeriods) if(p->contains(runNumber)) return true;
+            return false;
+         };
+         void SetDefaultChannel(Int_t channel) {
+            defaultChannel=channel;
+            for(auto p : subPeriods) p->SetDefaultChannel(channel);
+         };
+         void print(const char* prefix) { 
+            std::cout << prefix << id << "[" << start << "," << end << "] : ";
+            for(auto hist : inputHists) std::cout << hist.first << " , ";
+            std::cout << std::endl;
+            for(auto p : subPeriods) p->print(TString::Format(" %s",prefix).Data()); };
+      };
+      struct Run {
+         std::map<TString,TH1*> inputHists; //key is the 'trigger' 
+         std::map<Int_t,Double_t> badBins; 
+         Double_t lumi; //total data in run
+         std::map<UInt_t, std::pair<Double_t,Double_t> > lumiByLbn; //key=lbn, value = <lumi,mu>
+         TH1D* muDist; //mu distribution for this run
+      };
+protected:
+      std::map<Int_t, Period*> m_periods; //periods mapped by id. -1 = the "global" period (0->9999999). Uses a pointer so can easily implement remap as two entries pointing at same period
+      std::map<UInt_t, Run> m_runs; //runs mapped by runNumber
+
+      std::map<Int_t, Double_t> unrepDataByChannel; // -1 holds the total unrepData!
+
+      std::string m_prwFilesPathPrefix;
+
+      //-----------------------------------------------------
+      //RandomDataPeriod functionality stuff
+      //numbers generated seperately for each mc period
+      //-----------------------------------------------------
+      TRandom3 *m_random3;
+
+public:
+      ClassDef(TPileupReweighting,0)
 
 
   }; // End: class definition
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/selection.xml b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/selection.xml
index 5e88bb132270854112fd876882173d61a925bca1..824ad0976961128cf856251006633c4ffd056535 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/selection.xml
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/PileupReweighting/selection.xml
@@ -1,4 +1,5 @@
  <lcgdict>      
+      <namespace name="CP"/>
         <class name="CP::PileupReweightingTool" />
         <class name="CP::IPileupReweightingTool" />
 </lcgdict> 
\ No newline at end of file
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/LinkDef.h b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/LinkDef.h
index 1b26fea9a28e053b8a04711384525341cc46059d..765abae5de05df8103fa05d4ba738ef53fcf05ba 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/LinkDef.h
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/LinkDef.h
@@ -2,11 +2,15 @@
   Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration
 */
 
+#ifndef __PRWLINKDEF__
+#define __PRWLINKDEF__
+
+
 #include <map>
 #include <string>
 #include "TString.h"
-#include <PileupReweighting/TPileupReweighting.h>
-#include "PathResolver/PathResolver.h"
+#include "PileupReweighting/TPileupReweighting.h"
+
 
 #ifdef __CINT__
 
@@ -14,62 +18,13 @@
 #pragma link off all classes;
 #pragma link off all functions;
 
-#pragma link C++ namespace Root ;
 
-#pragma link C++ class Root::TPileupReweighting+ ; 
 
-#pragma link C++ class map<int,map<int,pair<unsigned int,unsigned int > > >+ ;
-#pragma link C++ class map<int,pair<unsigned int,unsigned int > >+ ;
-#pragma link C++ class pair<unsigned int,unsigned int>+ ;
-#pragma link C++ class map<TString,TH1*>+ ;
-#pragma link C++ class map<int,map<int,int> >+ ;
-#pragma link C++ class map<TString,map<int, double> >+ ;
-#pragma link C++ class map<int,double>+ ;
-#pragma link C++ class map<TString,map<int,map<int,double> > >+;
-#pragma link C++ class map<int,map<int,double> >+;
-#pragma link C++ class map<TString,map<int, map<int,TH1D*> > >+;
-#pragma link C++ class map<int,map<int,TH1D*> >+;
-#pragma link C++ class map<int,TH1D*>+;
-#pragma link C++ class map<TString,map<int, map<int,TH2D*> > >+;
-#pragma link C++ class map<int,map<int,TH2D*> >+;
-#pragma link C++ class map<int,TH2D*>+;
-#pragma link C++ class map<TString,map<int, map<int,TH1*> > >+;
-#pragma link C++ class map<int,map<int,TH1*> >+;
-#pragma link C++ class map<int,TH1*>+;
-//#pragma link C++ class map<TString,map<unsigned int,map<int,map<unsigned int,double> > > >+; 
-//#pragma link C++ class map<unsigned int,map<int,map<unsigned int,double> > >+; 
-#pragma link C++ class map<TString,map<int,map<unsigned int,double> > >+; 
-#pragma link C++ class map<int,map<unsigned int,double> >+; 
-#pragma link C++ class map<unsigned int,double>+; 
-#pragma link C++ class map<TString,map<int,map<int,bool> > >+;
-#pragma link C++ class map<int,map<int,bool> >+;
-#pragma link C++ class map<int,bool>+;
-#pragma link C++ class map<TString,TString>+;
+#pragma link C++ namespace CP;
 
-#pragma link C++ class pair<int,map<int,pair<unsigned int,unsigned int > > >+ ;
-#pragma link C++ class pair<int,pair<unsigned int,unsigned int > >+ ;
-#pragma link C++ class pair<TString,TH1*>+ ;
-#pragma link C++ class pair<int,map<int,int> >+ ;
-#pragma link C++ class pair<TString,map<int, double> >+ ;
-#pragma link C++ class pair<TString,map<int,map<int,double> > >+;
-#pragma link C++ class pair<int,map<int,double> >+;
-#pragma link C++ class pair<TString,map<int, map<int,TH1D*> > >+;
-#pragma link C++ class pair<int,map<int,TH1D*> >+;
-#pragma link C++ class pair<int,TH1D*>+;
-#pragma link C++ class pair<TString,map<int, map<int,TH2D*> > >+;
-#pragma link C++ class pair<int,map<int,TH2D*> >+;
-#pragma link C++ class pair<int,TH2D*>+;
-#pragma link C++ class pair<TString,map<int, map<int,TH1*> > >+;
-#pragma link C++ class pair<int,map<int,TH1*> >+;
-#pragma link C++ class pair<int,TH1*>+;
-//#pragma link C++ class pair<TString,map<unsigned int,map<int,map<unsigned int,double> > > >+; 
-//#pragma link C++ class pair<unsigned int,map<int,map<unsigned int,double> > >+; 
-#pragma link C++ class pair<TString,map<int,map<unsigned int,double> > >+; 
-#pragma link C++ class pair<int,map<unsigned int,double> >+; 
-#pragma link C++ class pair<unsigned int,double>+; 
-#pragma link C++ class pair<TString,map<int,map<int,bool> > >+;
-#pragma link C++ class pair<int,map<int,bool> >+;
-#pragma link C++ class pair<int,bool>+;
+#pragma link C++ class CP::TPileupReweighting+ ; 
 
 
+#endif
+
 #endif
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/PileupReweightingTool.cxx b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/PileupReweightingTool.cxx
index dae691f71a9660904e9060391880890f22475b04..0accfd4d1f41e9860b5ea0135d2c78ede3133e7a 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/PileupReweightingTool.cxx
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/PileupReweightingTool.cxx
@@ -9,35 +9,144 @@
 
 #include "PathResolver/PathResolver.h"
 
-#include "xAODEventInfo/EventInfoAuxContainer.h"
+#include "PATInterfaces/SystematicRegistry.h"
+
+#ifndef XAOD_STANDALONE
+#include "GaudiKernel/ITHistSvc.h"
+#endif
+
+#ifdef XAOD_STANDALONE
+#include "ReweightUtils/McEventWeight.h"
+#endif
+
+#include "TH1.h"
+#include "TTree.h"
+
 
 namespace CP {
 
-PileupReweightingTool::PileupReweightingTool( const std::string& name ) : asg::AsgTool( name ), Root::TPileupReweighting(name.c_str()), m_inConfigMode(false) {
+PileupReweightingTool::PileupReweightingTool( const std::string& name ) :CP::TPileupReweighting(name.c_str()), asg::AsgTool( name ), 
+									 m_inConfigMode(false), 
+									 m_upTool(0), m_downTool(0), m_systUp("PRW_DATASF", 1 ), m_systDown("PRW_DATASF", -1), 
+									 m_activeTool(this), 
+									 m_noWeightsMode(false), 
+									 m_weightTool("McEventWeight/myWeightTool") {
+
+#ifndef XAOD_STANDALONE
+   declareProperty("ConfigOutputStream", m_configStream="", "When creating PRW config files, this is the THistSvc stream it goes into. If blank, it wont write this way");
+#endif
 
    declareProperty("ConfigFiles", m_prwFiles, "List of prw config files"); //array of files
-   declareProperty("LumiCalcFiles", m_lumicalcFiles, "List of lumicalc files"); //array of files
-   declareProperty("LumiCalcTriggers",m_lumicalcTriggers, "List of triggers associated to each lumicalc file. 'None' is a keyword. If none specified, will assume all are 'None'"); //alias names - if not same length of LumiCalcFiles, will just use "None" (the default)
-   declareProperty("UnrepresentedDataAction",m_unrepdataAction=2);
-   declareProperty("DataScaleFactor",m_dataScaleFactor=1.);
+   declareProperty("ConfigFilesPathPrefix", m_prwFilesPathPrefix="", "Path of additional folder structure in prw config files"); //string prefix
+   declareProperty("LumiCalcFiles", m_lumicalcFiles, "List of lumicalc files, in the format '<filename>:<trigger>' .. if no trigger given, 'None' is assumed"); //array of files
+   declareProperty("Prefix",m_prefix="","Prefix to attach to all decorations ... only used in the 'apply' method");
+   declareProperty("UnrepresentedDataAction",m_unrepresentedDataAction=3,"1 = remove unrepresented data, 2 = leave it there, 3 = reassign it to nearest represented bin");
+   declareProperty("DataScaleFactor",m_dataScaleFactorX=1.);
    declareProperty("DefaultChannel",m_defaultChannel=0); //when channel info not present in config file, use this channel instead
-   declareProperty("UsePeriodConfig",m_usePeriodConfig="","Use this period configuration when in config generating mode. Leave blank to try auto-detect");
-   declareProperty("DataWeightTriggers",m_dataWeightTriggers,"List of triggers used to calculate dataweights and attach to eventinfo during apply method, if running on data");
-   declareProperty("MuDependentDataWeights",m_useMuDependent=true,"Mu dependent data weights means calculating dataweights from luminosity in the bins of mu - more accurate than doing a global weight with all the luminosity");
-   declareProperty("Prefix",m_prefix="","Prefix to attach to all decorations");
-   declareProperty("Input",m_inputKey="","Specify a specific EventInfo object");
-   declareProperty("Output",m_outputKey="","Specify an output EventInfo object. If differs from input, will create a clone of EventInfo and decorate that");
+   declareProperty("UsePeriodConfig",m_usePeriodConfig="auto","Use this period configuration when in config generating mode. Set to 'auto' to auto-detect");
+
+   declareProperty("DataScaleFactorUP",m_upVariation=0.,"Set to a value representing the 'up' fluctuation - will report a PRW_DATASF uncertainty to Systematic Registry");
+   declareProperty("DataScaleFactorDOWN",m_downVariation=0.,"Set to a value representing the 'down' fluctuation - will report a PRW_DATASF uncertainty to Systematic Registry");
+
+   declareProperty("LumiCalcRunNumberOffset",m_lumicalcRunNumberOffset=0,"Use to 'fake' a Run2 lumicalc file. Suggest using a value of 22000 to do this from an 8TeV lumicalc file");
+
+#ifdef XAOD_STANDALONE
+   declareProperty( "WeightTool", m_weightTool = new McEventWeight("myWeightTool"),"The tool to compute the weight in the sumOfWeights");
+#else
+   declareProperty( "WeightTool", m_weightTool,"The tool to compute the weight in the sumOfWeights");
+#endif
+
+#ifndef XAOD_STANDALONE
+   //attached update handler to the outputlevel property, so we can pass changes on to the underlying tool 
+   auto props = getProperties();
+ 	for( Property* prop : props ) {
+      if( prop->name() != "OutputLevel" ) {
+         continue;
+      }
+      prop->declareUpdateHandler( &PileupReweightingTool::updateHandler, this );
+      break;
+   }
+#endif
+
+}
+
+#ifndef XAOD_STANDALONE
+//rootcore can't do this yet!
+void PileupReweightingTool::updateHandler(Property& p) {
+   //call the original update handler 
+   this->msg_update_handler(p);
+   EnableDebugging(this->msgLvl(MSG::DEBUG));
+}
+#endif
+
+bool PileupReweightingTool::isAffectedBySystematic( const CP::SystematicVariation& systematic ) const {
+   CP::SystematicSet sys = affectingSystematics(); return sys.find( systematic ) != sys.end();
+}
+
+CP::SystematicSet PileupReweightingTool::affectingSystematics() const {
+   CP::SystematicSet result;
+   if(m_upVariation) result.insert( m_systUp  );
+   if(m_downVariation) result.insert( m_systDown );
+   return result;
+}
+
+CP::SystematicSet PileupReweightingTool::recommendedSystematics() const {
+   return affectingSystematics();
+}
+
+CP::SystematicCode PileupReweightingTool::applySystematicVariation( const CP::SystematicSet& systConfig ) { 
+   if(systConfig.find( m_systUp ) != systConfig.end() && systConfig.find( m_systDown ) != systConfig.end()) {
+      ATH_MSG_ERROR("Errr... what the!? You're trying to do both PRW_DATASF directions at the same time!!!??"); 
+      return SystematicCode::Unsupported; 
+   }
+   if(systConfig.find( m_systUp ) != systConfig.end()) {
+      if(!m_upTool) { ATH_MSG_ERROR("Requested up variation of PRW_DATASF, but not configured to do this :-("); return SystematicCode::Unsupported; }
+       m_activeTool = m_upTool;
+   }
+   else if(systConfig.find( m_systDown ) != systConfig.end() ) {
+      if(!m_downTool) { ATH_MSG_ERROR("Requested down variation of PRW_DATASF, but not configured to do this :-("); return SystematicCode::Unsupported; }
+      m_activeTool = m_downTool;
+   }
+   else m_activeTool = this;
+   return SystematicCode::Ok;
 }
 
+
+float PileupReweightingTool::getLumiBlockMu( const xAOD::EventInfo& eventInfo ) { return CP::TPileupReweighting::GetLumiBlockMu(eventInfo.runNumber(),eventInfo.lumiBlock()); }
+
+double PileupReweightingTool::getLumiBlockIntegratedLumi(const xAOD::EventInfo& eventInfo) { return CP::TPileupReweighting::GetLumiBlockIntegratedLumi(eventInfo.runNumber(),eventInfo.lumiBlock()); }
+
 StatusCode PileupReweightingTool::initialize() {
 
+   //set debugging if debugging is on:
+   EnableDebugging(this->msgLvl(MSG::DEBUG));
+
+   //see if we need variations 
+   if(m_upVariation) {
+      m_upTool = new TPileupReweighting((name()+"_upVariation").c_str());
+      m_upTool->SetUnrepresentedDataAction(m_unrepresentedDataAction);
+      m_upTool->SetDataScaleFactors(m_upVariation);
+   }
+   if(m_downVariation) {
+      m_downTool = new TPileupReweighting((name()+"_downVariation").c_str());
+      m_downTool->SetUnrepresentedDataAction(m_unrepresentedDataAction);
+      m_downTool->SetDataScaleFactors(m_downVariation);
+   }
+
+   SetDefaultChannel(m_defaultChannel); if(m_upTool) m_upTool->SetDefaultChannel(m_defaultChannel); if(m_downTool) m_downTool->SetDefaultChannel(m_defaultChannel);
+
    //should we set the period config (file maker mode)
    if(m_prwFiles.size()+m_lumicalcFiles.size()==0) {
       m_inConfigMode=true;
       ATH_MSG_INFO("In Config file making mode."); 
-      if(m_usePeriodConfig!="") {
+      if(m_usePeriodConfig!="auto" && m_usePeriodConfig!="") {
          /*m_tool->*/UsePeriodConfig(m_usePeriodConfig);
-      } 
+      }
+#ifndef XAOD_STANDALONE
+      //retrieve histsvc now, otherwise it wont get correctly configured by the time it's used in the finalize ... so it seems
+      ServiceHandle<ITHistSvc> histSvc("THistSvc",name());
+      CHECK( histSvc.retrieve() );
+#endif
    } else {
       //have we any prw to load
       for(unsigned int j=0;j<m_prwFiles.size();j++) {
@@ -45,117 +154,243 @@ StatusCode PileupReweightingTool::initialize() {
             std::string file = PathResolverFindCalibFile(m_prwFiles[j]);
             if(file=="") { ATH_MSG_ERROR("Unable to find the PRW Config file: " << m_prwFiles[j]); return StatusCode::FAILURE; }
             /*m_tool->*/AddConfigFile(file.c_str());
+            if(m_upTool) m_upTool->AddConfigFile(file.c_str()); if(m_downTool) m_downTool->AddConfigFile(file.c_str());
       }
 
-      //have we any lumicalc files to load
-      bool useNoneLabel = (m_lumicalcFiles.size() != m_lumicalcTriggers.size());
+      //have we any lumicalc files to load? .. if we do and had no prwFiles then the user must specify the period configuration
+      if(m_lumicalcFiles.size()>0 && m_prwFiles.size()==0) {
+         if(m_usePeriodConfig!="auto" && m_usePeriodConfig!="") {
+            ATH_MSG_INFO("Setting up without a PRW config file, but with period config " << m_usePeriodConfig << ". You will only be able to use random run number and data weight functionality... no reweighting!");
+            if(UsePeriodConfig(m_usePeriodConfig)!=0) {
+               ATH_MSG_ERROR("Unrecognised PeriodConfig: " << m_usePeriodConfig); return StatusCode::FAILURE;
+            }
+            m_noWeightsMode=true; //will stop the prw weight being decorated in apply method
+            if(m_upTool) m_upTool->UsePeriodConfig(m_usePeriodConfig); if(m_downTool) m_downTool->UsePeriodConfig(m_usePeriodConfig);
+         } else {
+            ATH_MSG_INFO("No config files provided, but " << m_lumicalcFiles.size() << " lumicalc file provided. Please specify a UsePeriodConfig if you want to use the tool without a config file (e.g. do 'MC15') ");
+            return StatusCode::FAILURE;
+         }
+      }
+   
       for(unsigned int j=0;j<m_lumicalcFiles.size();j++) {
-            ATH_MSG_INFO("Locating File: " << m_lumicalcFiles[j]);
-            std::string file = PathResolverFindCalibFile(m_lumicalcFiles[j]);
-            if(file=="") { ATH_MSG_ERROR("Unable to find the Lumicalc file: " << m_lumicalcFiles[j]); return StatusCode::FAILURE; }
-            /*m_tool->*/SetDataScaleFactors(m_dataScaleFactor);
-            /*m_tool->*/AddLumiCalcFile(file.c_str(),(useNoneLabel)? "None" : m_lumicalcTriggers[j]);
+            //see if there's a trigger at the end of the filename .. format is "file:trigger" 
+            TString myFile = m_lumicalcFiles[j];
+            TString myTrigger = (myFile.Contains(':')) ? TString(myFile(myFile.Last(':')+1,myFile.Length()-myFile.Last(':'))) : TString("None");
+            myFile = (myFile.Contains(':')) ? TString(myFile(0,myFile.Last(':'))) : myFile;
+            ATH_MSG_INFO("Locating File: " << myFile);
+            std::string file = PathResolverFindCalibFile(myFile.Data());
+            if(file=="") { ATH_MSG_ERROR("Unable to find the Lumicalc file: " << myFile); return StatusCode::FAILURE; }
+            /*m_tool->*/AddLumiCalcFile(file.c_str(),myTrigger);
+            if(m_upTool) m_upTool->AddLumiCalcFile(file.c_str(),myTrigger); if(m_downTool) m_downTool->AddLumiCalcFile(file.c_str(),myTrigger);
       }
-      /*m_tool->*/SetUnrepresentedDataAction(m_unrepdataAction);
-      /*m_tool->*/SetDefaultChannel(m_defaultChannel);
    }
 
+   //register ourselves with the systematic registry! 
+   CP::SystematicRegistry& registry = CP::SystematicRegistry::getInstance();
+   if( registry.registerSystematics( *this ) != CP::SystematicCode::Ok ) return StatusCode::FAILURE;
+
+
    //delay initializing underlying tool until first usage, just in case user wants to do any advanced initialization options
 
+   ATH_MSG_DEBUG("Retrieving weight tool...");
+   StatusCode sc = m_weightTool.retrieve();
+   if (sc != StatusCode::SUCCESS) {
+     ATH_MSG_ERROR("    " << m_weightTool->name() << " could not be retrieved.");
+     return sc;
+   }
+   else {
+     ATH_MSG_DEBUG("    " << m_weightTool->name() << " retrieved.");
+     m_weightTool->print();
+   }
+
    return StatusCode::SUCCESS;
 }
 
 StatusCode PileupReweightingTool::finalize() {
    if(m_inConfigMode) {
+#ifndef XAOD_STANDALONE
+   if(m_configStream=="") {
       //write the prw config files
       std::string nameWithoutParent = this->name().substr(this->name().find(".")+1);
       TString fileName = TString::Format("%s.prw.root", nameWithoutParent.c_str());
       /*m_tool->*/WriteToFile(fileName);
-   }
-   return StatusCode::SUCCESS;
-}
+   } else {
+      //write to the histsvc stream instead ...
+      
+      ServiceHandle<ITHistSvc> histSvc("THistSvc",name());
+      CHECK( histSvc.retrieve() );
+
+      TTree *outTreeMC=0;
+      TTree *outTreeData=0;
+      Int_t channel = 0;UInt_t runNumber = 0; 
+      std::vector<UInt_t>* pStarts = 0;std::vector<UInt_t>* pEnds = 0;
+      Char_t histName[150];
+
+
+      //loop over periods ... periods only get entry in table if they have an input histogram 
+      for(auto period : m_periods) {
+         if(!period.second) continue; //should never happen, but just in case!
+         if(period.first != period.second->id) continue; //skips redirects 
+         runNumber = period.first;
+         pStarts = new std::vector<UInt_t>;pEnds = new std::vector<UInt_t>;
+         if(period.second->subPeriods.size()==0) { 
+            pStarts->push_back(period.second->start); pEnds->push_back(period.second->end); 
+         }
+         else {
+            for(auto subp : period.second->subPeriods) {
+               pStarts->push_back(subp->start); pEnds->push_back(subp->end);
+            }
+         }
+         for(auto inHist : period.second->inputHists) {
+            channel = inHist.first;
+            TH1* hist = inHist.second;
+            strcpy(histName,hist->GetName());
+            CHECK( histSvc->regHist(TString::Format("/%s/PileupReweighting/%s",m_configStream.c_str(),hist->GetName()).Data(),hist) );
+            if(!outTreeMC) {
+               outTreeMC = new TTree("MCPileupReweighting","MCPileupReweighting");
+               outTreeMC->Branch("Channel",&channel);
+               outTreeMC->Branch("RunNumber",&runNumber);
+               outTreeMC->Branch("PeriodStarts",&pStarts);
+               outTreeMC->Branch("PeriodEnds",&pEnds);
+               outTreeMC->Branch("HistName",&histName,"HistName/C");
+               CHECK( histSvc->regTree(TString::Format("/%s/PileupReweighting/%s",m_configStream.c_str(),outTreeMC->GetName()).Data(),outTreeMC) );
+            }
+            outTreeMC->Fill();
+         }
+         delete pStarts; delete pEnds;
+      }
 
-void PileupReweightingTool::checkInit(const xAOD::EventInfo* eventInfo) {
-   if(!/*m_tool->*/IsInitialized()) {
-      if(m_usePeriodConfig=="") { //try autodetect based on runnum of first event 
-         switch(eventInfo->runNumber()) {
-            case 180164: case 183003: case 186169: case 189751:  m_usePeriodConfig="MC12c";/*m_tool->*/UsePeriodConfig("MC12c");break;
-            case 195847: case 195848: m_usePeriodConfig="MC12ab";/*m_tool->*/UsePeriodConfig("MC12ab");break;
-            case 212272: m_usePeriodConfig="MC14";/*m_tool->*/UsePeriodConfig("MC14");break;
+      //loop over data 
+      for(auto& run : m_runs) {
+         runNumber = run.first;
+         if(run.second.inputHists.find("None")==run.second.inputHists.end()) continue;
+   
+         TH1* hist = run.second.inputHists["None"];
+         strcpy(histName,hist->GetName());
+         CHECK( histSvc->regHist(TString::Format("/%s/PileupReweighting/%s",m_configStream.c_str(),hist->GetName()).Data(),hist) );
+         if(!outTreeData) {
+            outTreeData = new TTree("DataPileupReweighting","DataPileupReweighting");
+            outTreeData->Branch("RunNumber",&runNumber);
+            outTreeData->Branch("HistName",&histName,"HistName/C");
+            CHECK( histSvc->regTree(TString::Format("/%s/PileupReweighting/%s",m_configStream.c_str(),outTreeData->GetName()).Data(),outTreeData) );
          }
+         outTreeData->Fill();
       }
-      if(/*m_tool->*/Initialize()!=0) throw std::runtime_error("PileupReweighting: Failure in init");
-   }
-   return;
-}
 
-void PileupReweightingTool::apply(const xAOD::EventInfo* eventInfo) {
-   checkInit(eventInfo);
+      Info("WriteToFile", "Successfully generated config file to stream: %s",m_configStream.c_str());
+      Info("WriteToFile", "Happy Reweighting :-)");
 
-   if(!eventInfo->eventType(xAOD::EventInfo::IS_SIMULATION)) {
-      //allowed to do data weights 
-      for(auto& trigger : m_dataWeightTriggers) {
-         eventInfo->auxdecor<double>(m_prefix+trigger) = dataWeight(eventInfo, trigger, m_useMuDependent);
-      }
-      return;
    }
 
-   if(m_inConfigMode)  {
-      Fill(eventInfo->runNumber(), eventInfo->mcChannelNumber(), eventInfo->mcEventWeight(), eventInfo->averageInteractionsPerCrossing());
-      return;
+#else
+      //write the prw config files
+      std::string nameWithoutParent = this->name().substr(this->name().find(".")+1);
+      TString fileName = TString::Format("%s.prw.root", nameWithoutParent.c_str());
+      /*m_tool->*/WriteToFile(fileName);
+#endif
    }
 
-   //decorate with standard PileupWeight 
-   eventInfo->auxdecor<double>(m_prefix+"PileupWeight") = /*m_tool->*/GetCombinedWeight(   eventInfo->runNumber(), eventInfo->mcChannelNumber(), eventInfo->averageInteractionsPerCrossing() );
-   //decorate with random run number etc
-   //using mu-dependent randomization by default ... jetetmiss seem to like it muchly
-   /*m_tool->*/SetRandomSeed(314159+eventInfo->mcChannelNumber()*2718+eventInfo->eventNumber()); //to make randomization repeatable
-   eventInfo->auxdecor<unsigned int>(m_prefix+"RandomRunNumber") = /*m_tool->*/GetRandomRunNumber(   eventInfo->runNumber(), eventInfo->averageInteractionsPerCrossing()  );
-   eventInfo->auxdecor<unsigned int>(m_prefix+"RandomLumiBlockNumber") = (eventInfo->auxdecor<unsigned int>(m_prefix+"RandomRunNumber")==0) ? 0 : /*m_tool->*/GetRandomLumiBlockNumber(  eventInfo->auxdecor<unsigned int>(m_prefix+"RandomRunNumber")  );
+   if(m_upTool) {delete m_upTool;m_upTool=0;} if(m_downTool)  {delete m_downTool;m_downTool=0;} 
 
-   return; //FIXME return statuscode instead maybe...
+   return StatusCode::SUCCESS;
 }
 
-double PileupReweightingTool::dataWeight(const xAOD::EventInfo* eventInfo, TString trigger, bool mu_dependent) {
-   checkInit(eventInfo);
+ULong64_t PileupReweightingTool::getPRWHash( const xAOD::EventInfo& eventInfo ) {
+   return m_activeTool->GetPRWHash(eventInfo.runNumber(), eventInfo.mcChannelNumber(), eventInfo.averageInteractionsPerCrossing() );
+}
 
-   if(eventInfo->eventType(xAOD::EventInfo::IS_SIMULATION)) {
-      ATH_MSG_WARNING("Requesting Data Weight is not intended for simulated events. Returning 1.");
-      return 1; //no data weights for simulated events
+/// Return combined pileup weight
+float PileupReweightingTool::getCombinedWeight( const xAOD::EventInfo& eventInfo ) {
+   return m_activeTool->GetCombinedWeight(   eventInfo.runNumber(), eventInfo.mcChannelNumber(), eventInfo.averageInteractionsPerCrossing() );
+}
+
+float PileupReweightingTool::getCombinedWeight( const xAOD::EventInfo& eventInfo, Double_t x, Double_t y ) {
+   return m_activeTool->GetCombinedWeight(   eventInfo.runNumber(), eventInfo.mcChannelNumber(), x, y );
+}
+
+/// When using UnrepresentedDataAction=2, you may want to apply this additional weight to ensure sum of weights are preserved
+float PileupReweightingTool::getUnrepresentedDataWeight( const xAOD::EventInfo& eventInfo ) {
+    return m_activeTool->GetUnrepresentedDataWeight( eventInfo.runNumber(), eventInfo.mcChannelNumber() );
+}
+
+
+/// Get a random run number for this MC event, mu_dependency is not recommended for this
+int PileupReweightingTool::getRandomRunNumber( const xAOD::EventInfo& eventInfo , bool mu_dependent) {
+   m_activeTool->SetRandomSeed(314159+eventInfo.mcChannelNumber()*2718+eventInfo.eventNumber()); //to make randomization repeatable
+   if(mu_dependent) return m_activeTool->GetRandomRunNumber(   eventInfo.runNumber(), eventInfo.averageInteractionsPerCrossing()  );
+   else return m_activeTool->GetRandomRunNumber( eventInfo.runNumber() );
+}
+
+int PileupReweightingTool::fill( const xAOD::EventInfo& eventInfo ) {
+   return fill(eventInfo,eventInfo.averageInteractionsPerCrossing());
+}
+
+int PileupReweightingTool::fill( const xAOD::EventInfo& eventInfo, Double_t x, Double_t y) {
+   
+   //auto-detect the period config if necessary 
+   if(m_usePeriodConfig=="auto" && !m_doneConfigs[eventInfo.runNumber()]) { //try autodetect based on runnum of first event ... 
+      //if data, we only need to ensure a binning is done ... for now, we assume the MC15 binning
+      if(!eventInfo.eventType(xAOD::EventInfo::IS_SIMULATION)) {
+         if(!m_emptyHistogram) { SetUniformBinning(100,0,100.); } //use a default binning
+      } else {
+         switch(eventInfo.runNumber()) {
+            //case 180164: case 183003: case 186169: case 189751:  m_usePeriodConfig="MC12c";/*m_tool->*/UsePeriodConfig("MC12c");break;
+            //case 195847: case 195848: m_usePeriodConfig="MC12ab";/*m_tool->*/UsePeriodConfig("MC12ab");break;
+            case 212272: UsePeriodConfig("MC14_8TeV");break;
+            case 222222: UsePeriodConfig("MC14_13TeV");break;
+            case 222510: case 222525: UsePeriodConfig("MC15"); break;
+         }
+      }
+      m_doneConfigs[eventInfo.runNumber()] = true;
    }
 
-   if(!mu_dependent) return /*m_tool->*/GetDataWeight(eventInfo->runNumber(), trigger);
-   return /*m_tool->*/GetDataWeight( eventInfo->runNumber(), trigger, eventInfo->averageInteractionsPerCrossing() );
+   return TPileupReweighting::Fill(eventInfo.runNumber(), eventInfo.eventType(xAOD::EventInfo::IS_SIMULATION) ? eventInfo.mcChannelNumber() : -1 /*data*/, eventInfo.eventType(xAOD::EventInfo::IS_SIMULATION) ? m_weightTool->getWeight() : 1., x, y);
 
 }
 
-StatusCode PileupReweightingTool::execute() {
-   const xAOD::EventInfo* evtInfo =0;
+StatusCode PileupReweightingTool::apply(const xAOD::EventInfo& eventInfo, bool mu_dependent) {
 
-   if(m_inputKey.length()>0) ATH_CHECK(evtStore()->retrieve(evtInfo,m_inputKey));
-   else {
-      #ifdef XAOD_STANDALONE
-         ATH_CHECK( evtStore()->retrieve(evtInfo,"") ); //apparently TEvent can't do keyless retrieves!!??
-      #else
-         ATH_CHECK(evtStore()->retrieve(evtInfo));
-      #endif
+   if(m_inConfigMode)  {
+      fill( eventInfo );
+      return StatusCode::SUCCESS;
    }
 
-   //do we need to make a copy??
-   if(m_inputKey!=m_outputKey && m_outputKey!="") {
-      xAOD::EventInfo* evtInfoCopy = new xAOD::EventInfo( *evtInfo );
-      xAOD::EventInfoAuxContainer* aux = new xAOD::EventInfoAuxContainer;
-      evtInfoCopy->setStore(aux);
-      ATH_CHECK( evtStore()->record( evtInfoCopy , m_outputKey ) );
-      ATH_CHECK( evtStore()->record( aux , m_outputKey+"Aux." ) );
-      evtInfo = evtInfoCopy;
+   if(!eventInfo.eventType(xAOD::EventInfo::IS_SIMULATION)) {
+      eventInfo.auxdecor<float>(m_prefix+"corrected_averageInteractionsPerCrossing") = getLumiBlockMu(eventInfo);
+      return StatusCode::SUCCESS;
    }
 
-   apply(evtInfo);
+   //just copy the value over for MC
+   eventInfo.auxdecor<float>(m_prefix+"corrected_averageInteractionsPerCrossing") = eventInfo.averageInteractionsPerCrossing();
+
+   //decorate with standard PileupWeight 
+   if(!m_noWeightsMode) {
+      double weight =  /*m_tool->*/getCombinedWeight( eventInfo );
+      if(m_unrepresentedDataAction==2) weight *= getUnrepresentedDataWeight( eventInfo );
+      eventInfo.auxdecor<double>(m_prefix+"PileupWeight") = weight;
+   }
+   //decorate with random run number etc
+   eventInfo.auxdecor<unsigned int>(m_prefix+"RandomRunNumber") = /*m_tool->*/getRandomRunNumber( eventInfo, mu_dependent );
+   eventInfo.auxdecor<unsigned int>(m_prefix+"RandomLumiBlockNumber") = (eventInfo.auxdecor<unsigned int>(m_prefix+"RandomRunNumber")==0) ? 0 : /*m_tool->*/GetRandomLumiBlockNumber(  eventInfo.auxdecor<unsigned int>(m_prefix+"RandomRunNumber")  );
+
+   eventInfo.auxdecor<ULong64_t>(m_prefix+"PRWHash") = getPRWHash( eventInfo );
+
+   ATH_MSG_VERBOSE("PileupWeight = " << eventInfo.auxdecor<double>(m_prefix+"PileupWeight") << " RandomRunNumber = " << eventInfo.auxdecor<unsigned int>(m_prefix+"RandomRunNumber") << " RandomLumiBlockNumber = " << eventInfo.auxdecor<unsigned int>(m_prefix+"RandomLumiBlockNumber"));
 
    return StatusCode::SUCCESS;
+}
+
+float PileupReweightingTool::getDataWeight(const xAOD::EventInfo& eventInfo, const TString& trigger, bool mu_dependent) {
+
+   if(eventInfo.eventType(xAOD::EventInfo::IS_SIMULATION)) {
+      ATH_MSG_WARNING("Requesting Data Weight is not intended for simulated events. Returning 1.");
+      return 1; //no data weights for simulated events
+   }
+
+   if(!mu_dependent) return /*m_tool->*/m_activeTool->GetDataWeight(eventInfo.runNumber(), trigger);
+   return /*m_tool->*/m_activeTool->GetDataWeight( eventInfo.runNumber(), trigger, getLumiBlockMu(eventInfo) /*use the 'correct' mu instead of the one from the file!!*/ );
 
 }
 
 
-} // namespace CP
\ No newline at end of file
+} // namespace CP
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/TPileupReweighting.cxx b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/TPileupReweighting.cxx
index cc6d00cc5f57f2da91b2d0b56c01e33d68a85e45..784d075a8b7cfc86b7eaade2225d3b249e244fc8 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/TPileupReweighting.cxx
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/Root/TPileupReweighting.cxx
@@ -36,189 +36,177 @@ Description: Tool to get the calculated MC pileup weight.
 #include <TRandom3.h>
 
 
-ClassImp(Root::TPileupReweighting)
+ClassImp(CP::TPileupReweighting)
 
 
 //=============================================================================
 // Constructor
 //=============================================================================
-Root::TPileupReweighting::TPileupReweighting(const char* name) :
+CP::TPileupReweighting::TPileupReweighting(const char* name) :
   TNamed(name,"notitle"), 
    m_SetWarnings(true),m_debugging(false),
-   m_countingMode(true),m_defaultChannel(0),m_unrepresentedDataAction(0),m_isInitialized(false),m_lumiVectorIsLoaded(false),
-   m_dataScaleFactorX(1.),m_dataScaleFactorY(1.),m_dataScaleFactorZ(1.),
-   m_mcScaleFactorX(1.),m_mcScaleFactorY(1.),m_mcScaleFactorZ(1.),
-   m_nextPeriodNumber(1),m_ignoreFilePeriods(false),m_metadatatree(0),m_unrepDataTolerance(0.05),m_doGlobalDataWeight(false)
+   m_countingMode(true),m_unrepresentedDataAction(0),m_isInitialized(false),m_lumiVectorIsLoaded(false),
+   m_dataScaleFactorX(1.),m_dataScaleFactorY(1.),
+   m_mcScaleFactorX(1.),m_mcScaleFactorY(1.),
+   m_nextPeriodNumber(1),m_ignoreFilePeriods(false),m_metadatatree(0),m_unrepDataTolerance(0.05),m_doGlobalDataWeight(false),m_lumicalcRunNumberOffset(0), m_emptyHistogram(0), m_random3(0)
 {
    m_random3 = new TRandom3(0);
    m_random3->SetSeed(1);
-   //load the default pileup histogram binning
-   //this is now done in the periodConfig (2012-05-09). We start with no default so that the default is picked up from the prw file
-   //m_emptyHistograms["pileup"] = new TH1D("pileup_default","pileup_default",100,0,50);
-   //m_emptyHistograms["pileup"]->SetDirectory(0);
-   //set the lumi vector to size=11
-   m_integratedLumiVector.resize(11,0.);
+
+   //create the global period 
+   m_periods[-1] = new Period(-1,0,9999999,0/* the global default channel*/);
 
    //MC12ab merger by default
-   MergeMCRunNumbers(195847,195848);
+   //RemapPeriod(195848,195847);
+}
+
+
+
+//=============================================================================
+// Destructor
+//=============================================================================
+CP::TPileupReweighting::~TPileupReweighting() {
+   delete m_random3; 
+}
+
+void CP::TPileupReweighting::RemapPeriod(Int_t periodNumber1, Int_t periodNumber2) {
+   //check if periodNumber2 exists
+   if(m_periods.find(periodNumber2)==m_periods.end()) {
+      m_periods[periodNumber2] = new Period(periodNumber2,0,0,m_periods[-1]->defaultChannel);
+   }
+   m_periods[periodNumber1] = m_periods[periodNumber2];
 }
 
-Double_t Root::TPileupReweighting::GetIntegratedLumiFraction(Int_t mcRunNumber, UInt_t start, UInt_t end) {
-   if(!m_isInitialized) {
-      Error("GetIntegratedLumiFraction", "Please initialize the tool before retrieving lumi fractions");
-      throw std::runtime_error("Throwing 1");
+void CP::TPileupReweighting::SetDefaultChannel(Int_t channel, Int_t periodNumber) {
+   //check if periodNumber2 exists
+   if(m_periods.find(periodNumber)==m_periods.end()) {
+      m_periods[periodNumber] = new Period(periodNumber,0,0,channel);
+   } else {
+      m_periods[periodNumber]->SetDefaultChannel(channel); 
    }
+}
+
+
+Int_t CP::TPileupReweighting::SetBinning(Int_t nbinsx, Double_t* xbins, Int_t nbinsy, Double_t* ybins) {
+   if(m_emptyHistogram) { delete m_emptyHistogram; }
+   if(nbinsy>0) m_emptyHistogram = new TH2D("default","default",nbinsx,xbins,nbinsy,ybins);
+   else m_emptyHistogram = new TH1D("default","default",nbinsx,xbins);
+   m_emptyHistogram->SetDirectory(0);
+   return 0;
+}
+Int_t CP::TPileupReweighting::SetUniformBinning(Int_t nbinsx, Double_t xlow, Double_t xup, Int_t nbinsy, Double_t ylow, Double_t yup) {
+   std::vector<double> xbins(nbinsx+1);
+   std::vector<double> ybins(nbinsy+1);
+   for(int i=0;i<nbinsx+1;i++) xbins[i] = xlow + i*(xup-xlow)/nbinsx;
+   for(int i=0;i<nbinsy+1;i++) ybins[i] = ylow + i*(yup-ylow)/nbinsy;
+   return SetBinning(nbinsx,&xbins[0],nbinsy,&ybins[0]);
+}
+Int_t CP::TPileupReweighting::SetBinning(TH1* hist) {
+   if(!hist) return -1;
+   if(m_emptyHistogram) { delete m_emptyHistogram; }
+   m_emptyHistogram = dynamic_cast<TH1*>(hist->Clone("default"));
+   m_emptyHistogram->SetDirectory(0);
+   return 0;
+}
+
+Int_t CP::TPileupReweighting::GetDefaultChannel(Int_t mcRunNumber) {
+   return m_periods[mcRunNumber]->defaultChannel;
+}
+
+
+Double_t CP::TPileupReweighting::GetIntegratedLumi(const TString& trigger) { 
+   if(!m_isInitialized) { Info("GetIntegratedLumi","Initializing the subtool.."); Initialize(); }
    if(!m_lumiVectorIsLoaded) {
-      Error("GetIntegratedLumiFraction","No Lumicalc file loaded, so no lumi fraction possible, returning 0");
+      Error("GetIntegratedLumi","No Lumicalc file loaded, so cannot get integrated lumi, returning 0");
       return 0;
    }
-   //check if the runNumber has been reassigned
-   if(m_mcRemappings.find(mcRunNumber)!=m_mcRemappings.end()) mcRunNumber=m_mcRemappings[mcRunNumber];
-   const std::map<Int_t, Double_t>::const_iterator it = periodTotals["pileup"][-1].find(mcRunNumber);
-   if(it==periodTotals["pileup"][-1].end() || it->second == 0) { 
-      if(m_SetWarnings) Warning("GetIntegratedLumiFraction","%d has no associated lumi. Returning 0",mcRunNumber);
-      return 0; /*throw 33;*/
+   if(trigger=="" || trigger=="None") return GetSumOfEventWeights(-1)/1E6; 
+   //check the triggerPrimaryDistributions in the global period, if not present, build it
+   Period* global = m_periods[-1];
+   if(global->triggerHists.find(trigger)==global->triggerHists.end()) {
+      //try to do a subtrigger calculation we need to reopen all the lumicalc files and do the fiddily prescaled luminosity calculation
+      //will throw errors if can't
+      CalculatePrescaledLuminosityHistograms(trigger);
    }
-   //loop over assigned runs, if in range then include in numerator 
-   double total = 0;
-   std::map<UInt_t, Double_t>::const_iterator itend = dataPeriodRunTotals["pileup"][mcRunNumber].end();
-   for(std::map<UInt_t, Double_t>::const_iterator it2 = dataPeriodRunTotals["pileup"][mcRunNumber].begin(); it2 != itend; ++it2) {
-      if(it2->first >= start && it2->first <= end) total += it2->second;
+   return global->triggerHists[trigger]->Integral(0,global->triggerHists[trigger]->GetNbinsX()+1)/1E6;
+}
+
+
+
+Double_t CP::TPileupReweighting::GetIntegratedLumi(Int_t periodNumber, UInt_t start, UInt_t end) {
+   if(!m_isInitialized) { Info("GetIntegratedLumi","Initializing the subtool.."); Initialize(); }
+   //look through dataPeriodRunTotals["pileup"][periodNumber] for runs inside the given period
+   double total = 0; 
+   for(auto run : m_periods[periodNumber]->runNumbers) {
+      if(run >= start && run <= end) total += m_runs[run].inputHists["None"]->GetSumOfWeights();
    }
+   return total*1E-6;
+}
+
+Double_t CP::TPileupReweighting::GetLumiBlockIntegratedLumi(Int_t runNumber, UInt_t lb) {
+   if(!m_isInitialized) { Info("GetIntegratedLumi","Initializing the subtool.."); Initialize(); }
+   if(m_runs.find(runNumber)==m_runs.end()) return 0.;
+   auto& run = m_runs[runNumber];
 
-   return total / it->second;
+   if(run.lumiByLbn.find(lb)==run.lumiByLbn.end()) return 0.;
+   return run.lumiByLbn[lb].first*1E-6;
 }
 
-Double_t Root::TPileupReweighting::GetIntegratedLumiFraction(Int_t mcRunNumber, Double_t mu, UInt_t start, UInt_t end) {
-   if(!m_isInitialized) {
-      Error("GetIntegratedLumiFraction", "Please initialize the tool before retrieving lumi fractions");
-      throw std::runtime_error("Throwing 1");
-   }
+Float_t CP::TPileupReweighting::GetLumiBlockMu(Int_t runNumber, UInt_t lb) {
+   if(!m_isInitialized) { Info("GetLumiBlockMu","Initializing the subtool.."); Initialize(); }
+   if(m_runs.find(runNumber)==m_runs.end()) return 0.;
+   auto& run = m_runs[runNumber];
+
+   if(run.lumiByLbn.find(lb)==run.lumiByLbn.end()) return 0.;
+   return run.lumiByLbn[lb].second;
+}
+
+Double_t CP::TPileupReweighting::GetIntegratedLumiFraction(Int_t periodNumber, UInt_t start, UInt_t end) {
+   if(!m_isInitialized) { Info("GetIntegratedLumiFraction","Initializing the subtool.."); Initialize(); }
+
    if(!m_lumiVectorIsLoaded) {
       Error("GetIntegratedLumiFraction","No Lumicalc file loaded, so no lumi fraction possible, returning 0");
       return 0;
    }
-   //check if the runNumber has been reassigned
-   if(m_mcRemappings.find(mcRunNumber)!=m_mcRemappings.end()) mcRunNumber=m_mcRemappings[mcRunNumber];
-
-   //determine the mubin
-   TH1* hist = m_emptyHistograms["pileup"];
-   if(!hist) { Error("GetIntegratedLumiFraction", "Cannot do this without a lumicalc file!"); return 0; }
-   Int_t muBin = hist->FindFixBin(mu);
-
-   //get the total lumi of this mcRunNumber with this mu - (was stored in the run=0 slot of the map)
-   //check that this mcRunNumber actually has some lumi with this mu
-   std::map<UInt_t, std::map<Int_t, std::map<UInt_t, Double_t> > >::iterator myMap1 = dataPeriodBinnedRunTotals["pileup"].find(muBin);
-   if(myMap1 == dataPeriodBinnedRunTotals["pileup"].end()) {
-      //no lumi with this mu
-      return 0;
+   //total lumi in period is given by sumOfWeights[-1]
+   double total = m_periods[periodNumber]->sumOfWeights[-1];
+   //go through associated runs, adding up lumi from all within start and end (inclusive)
+   double numer(0);
+   for(auto run : m_periods[periodNumber]->runNumbers) {
+      if(run >= start && run <= end) numer += m_runs[run].inputHists["None"]->GetSumOfWeights();
    }
-   std::map<Int_t, std::map<UInt_t, Double_t> >::iterator myMap2 = myMap1->second.find(mcRunNumber);
-   if(myMap2 == myMap1->second.end()) {
-      //no lumi with this mu in this mcRunNumber
+   return numer / total;
+}
+
+Double_t CP::TPileupReweighting::GetIntegratedLumiFraction(Int_t periodNumber, Double_t mu, UInt_t start, UInt_t end) {
+   if(!m_isInitialized) { Info("GetIntegratedLumiFraction","Initializing the subtool.."); Initialize(); }
+   if(!m_lumiVectorIsLoaded) {
+      Error("GetIntegratedLumiFraction","No Lumicalc file loaded, so no lumi fraction possible, returning 0");
       return 0;
    }
+   //determine the mubin
+   if(!m_emptyHistogram) { Error("GetIntegratedLumiFraction", "Cannot do this without a lumicalc file!"); return 0; }
+   Int_t muBin = m_emptyHistogram->FindFixBin(mu);
+   //the total mu in this bin in this period is given by the triggerHist = "None" hist 
+   double total = m_periods[periodNumber]->triggerHists["None"]->GetBinContent(muBin);
    //loop over assigned runs, if in range then include in numerator 
-   double total = 0;
-   std::map<UInt_t, Double_t>::const_iterator itend = myMap2->second.end();
-   for(std::map<UInt_t, Double_t>::const_iterator it2 = myMap2->second.begin(); it2 != itend; ++it2) {
-      if(it2->first >= start && it2->first <= end) total += it2->second;
+   double numer = 0;
+   for(auto run : m_periods[periodNumber]->runNumbers) {
+      if(run >= start && run <= end) numer += m_runs[run].muDist->GetBinContent(muBin);
    }
 
-   return total / myMap2->second[0];
-}
-
-Double_t Root::TPileupReweighting::GetIntegratedLumi(UInt_t start, UInt_t end) {
-   //look through dataPeriodRunTotals["pileup"][-1] for runs inside the given period
-   double total = 0; 
-   std::map<UInt_t, Double_t>::const_iterator itend = dataPeriodRunTotals["pileup"][-1].end();
-   for(std::map<UInt_t, Double_t>::const_iterator it2 = dataPeriodRunTotals["pileup"][-1].begin(); it2 != itend; ++it2) {
-      if(it2->first >= start && it2->first <= end) total += it2->second;
-   }
-   return total/1E6;
+   return numer / total;
 }
 
-Double_t Root::TPileupReweighting::GetIntegratedLumi(Int_t periodNumber, UInt_t start, UInt_t end) {
-   //look through dataPeriodRunTotals["pileup"][periodNumber] for runs inside the given period
-   double total = 0; 
-   std::map<UInt_t, Double_t>::const_iterator itend = dataPeriodRunTotals["pileup"][periodNumber].end();
-   for(std::map<UInt_t, Double_t>::const_iterator it2 = dataPeriodRunTotals["pileup"][periodNumber].begin(); it2 != itend; ++it2) {
-      if(it2->first >= start && it2->first <= end) total += it2->second;
-   }
-   return total/1E6;
-}
 
-//=============================================================================
-// Destructor
-//=============================================================================
-Root::TPileupReweighting::~TPileupReweighting() {
-   delete m_random3; 
-   //delete all histograms
-   for(std::map<TString, std::map<Int_t,std::map<Int_t, TH1*> > >::iterator it0=m_inputHistograms.begin();it0!=m_inputHistograms.end();++it0) {
-      for(std::map<Int_t,std::map<Int_t, TH1*> >::iterator it1=it0->second.begin();it1!=it0->second.end();++it1) {
-         for(std::map<Int_t, TH1*>::iterator it2=it1->second.begin();it2!=it1->second.end();++it2) {
-            if(it2->second) delete it2->second;
-         }
-         it1->second.clear();
-      }
-      it0->second.clear();
-   }
-   m_inputHistograms.clear();
-   if(m_debugging) Info("destructor","Deleting primary distributions");
-   for(std::map<TString, std::map<Int_t,std::map<Int_t, TH1D*> > >::iterator it0=primaryDistributions.begin();it0!=primaryDistributions.end();++it0) {
-      for(std::map<Int_t,std::map<Int_t, TH1D*> >::iterator it1=it0->second.begin();it1!=it0->second.end();++it1) {
-         for(std::map<Int_t, TH1D*>::iterator it2=it1->second.begin();it2!=it1->second.end();++it2) {
-            if(it2->second) {
-               if(m_debugging) { Info("destructor","weight=%s chan=%d run=%d",it0->first.Data(),it1->first,it2->first);
-                  Info("destructor","Deleting %s",it2->second->GetName());
-               }
-               delete it2->second;
-            }
-         }
-         it1->second.clear();
-      }
-      it0->second.clear();
-   }
-   primaryDistributions.clear();
-   if(m_debugging) Info("destructor","Deleting secondary distributions");
-   for(std::map<TString, std::map<Int_t,std::map<Int_t, TH2D*> > >::iterator it0=secondaryDistributions.begin();it0!=secondaryDistributions.end();++it0) {
-      for(std::map<Int_t,std::map<Int_t, TH2D*> >::iterator it1=it0->second.begin();it1!=it0->second.end();++it1) {
-         for(std::map<Int_t, TH2D*>::iterator it2=it1->second.begin();it2!=it1->second.end();++it2) {
-            if(it2->second) {
-               if(m_debugging) { Info("destructor","weight=%s chan=%d run=%d",it0->first.Data(),it1->first,it2->first);
-                  Info("destructor","Deleting %s",it2->second->GetName());
-               }
-               delete it2->second;
-            }
-         }
-         it1->second.clear();
-      }
-      it0->second.clear();
-   }
-   secondaryDistributions.clear();
-   if(m_debugging) Info("destructor","Deleting trigger primary distributions");
-   for(std::map<TString, std::map<Int_t, TH1D*> >::iterator it0=m_triggerPrimaryDistributions.begin();it0!=m_triggerPrimaryDistributions.end();++it0) {
-      for(std::map<Int_t, TH1D*>::iterator it=it0->second.begin();it!=it0->second.end();++it) {
-         if(it->second) {
-            if(m_debugging) { Info("destructor","trigger=%s run=%d",it0->first.Data(),it->first);
-               Info("destructor","Deleting %s",it->second->GetName());
-            }
-            delete it->second;
-         }
-      }
-      it0->second.clear();
-   }
-   m_triggerPrimaryDistributions.clear();
-   
-}
 
-Int_t Root::TPileupReweighting::UsePeriodConfig(const TString configName) {
+Int_t CP::TPileupReweighting::UsePeriodConfig(const TString configName) {
    if(configName=="MC11a") {
       AddPeriod(180164, 177986,180481); //associates mc runnumber 180164 with data period 177986 to 180481 (period B-D)
       AddPeriod(183003, 180614,184169); //period E-H
       AddPeriod(185649, 185353,186934); //period I-K1. For I-K you would change the last number to 187815
       AddPeriod(185761, 186935,191933); //everything else. Thanks Ellie!
-      AddBinning("pileup",100,0,50);
+      SetUniformBinning(100,0,50);
       Info("UsePeriodConfig","Using MC11a Period configuration");
       return 0;
    } else if(configName=="MC11b" || configName=="MC11c") {
@@ -226,192 +214,150 @@ Int_t Root::TPileupReweighting::UsePeriodConfig(const TString configName) {
       AddPeriod(183003, 180614, 184169);
       AddPeriod(186169, 185353, 187815);
       AddPeriod(189751, 188902, 191933);
-      AddBinning("pileup",100,0,50);
+      SetUniformBinning(100,0,50);
       Info("UsePeriodConfig","Using MC11b/c Period configuration");
       return 0;
    } else if(configName=="MC12a") {
       AddPeriod(195847,200804,216432);
       //mc12a binning is in integer values of mu
-      AddBinning("pileup",50,-0.5,49.5);
+
+      if(m_emptyHistogram && (m_emptyHistogram->GetNbinsX()!=50 || fabs(m_emptyHistogram->GetXaxis()->GetBinLowEdge(1)+0.5)>0.01|| fabs(m_emptyHistogram->GetXaxis()->GetBinUpEdge(50)-49.5)>0.01) ) {
+         Error("UsePeriodConfig","Cannot use MC12a, an incompatible config has already been set up");
+         throw std::runtime_error("Throwing 13: Cannot use MC14, an incompatible config has already been set up");
+         return -1;
+      }
+
+      SetUniformBinning(50,-0.5,49.5);
       Info("UsePeriodConfig","Using MC12a Period configuration");
       return 0;
    } else if(configName=="MC12b") {
       AddPeriod(195848,200804,216432);
       //mc12b binning is in integer values of mu
-      AddBinning("pileup",50,-0.5,49.5);
+
+      if(m_emptyHistogram && (m_emptyHistogram->GetNbinsX()!=50 || fabs(m_emptyHistogram->GetXaxis()->GetBinLowEdge(1)+0.5)>0.01|| fabs(m_emptyHistogram->GetXaxis()->GetBinUpEdge(50)-49.5)>0.01) ) {
+         Error("UsePeriodConfig","Cannot use MC12b, an incompatible config has already been set up");
+         throw std::runtime_error("Throwing 13: Cannot use MC14, an incompatible config has already been set up");
+         return -1;
+      }
+
+      SetUniformBinning(50,-0.5,49.5);
       Info("UsePeriodConfig","Using MC12b Period configuration");
       return 0;
    } else if(configName=="MC12ab") {
       AddPeriod(195847,200804,216432);
-      //disable warnings temporarily for next AddPeriod, as we know we are causing a conflict
-      bool tmp = m_SetWarnings; 
-      m_SetWarnings=false;
       AddPeriod(195848,200804,216432);
-      m_SetWarnings = tmp;
+
+      if(m_emptyHistogram && (m_emptyHistogram->GetNbinsX()!=50 || fabs(m_emptyHistogram->GetXaxis()->GetBinLowEdge(1)+0.5)>0.01|| fabs(m_emptyHistogram->GetXaxis()->GetBinUpEdge(50)-49.5)>0.01) ) {
+         Error("UsePeriodConfig","Cannot use MC12ab, an incompatible config has already been set up");
+         throw std::runtime_error("Throwing 13: Cannot use MC14, an incompatible config has already been set up");
+         return -1;
+      }
+
       //mc12a/b binning is in integer values of mu
-      AddBinning("pileup",50,-0.5,49.5);
+      SetUniformBinning(50,-0.5,49.5);
       Info("UsePeriodConfig","Using MC12ab Period configuration");
       return 0;
-   } else if(configName=="MC14") {
+   } else if(configName=="MC14_8TeV") {
       AddPeriod(212272,200804,216432);
-      AddBinning("pileup",50,-0.5,49.5);
-      Info("UsePeriodConfig","Using MC14 Period configuration");
+      if(m_emptyHistogram && (m_emptyHistogram->GetNbinsX()!=50 || fabs(m_emptyHistogram->GetXaxis()->GetBinLowEdge(1)+0.5)>0.01 || fabs(m_emptyHistogram->GetXaxis()->GetBinUpEdge(50)-49.5)>0.01) ) {
+         Error("UsePeriodConfig","Cannot use MC14_8TeV, an incompatible config has already been set up");
+         throw std::runtime_error("Throwing 13: Cannot use MC14_8TeV, an incompatible config has already been set up");
+         return -1;
+      }
+      SetUniformBinning(50,-0.5,49.5);
+      Info("UsePeriodConfig","Using MC14_8TeV Period configuration");
+      return 0;
+   } else if(configName=="MC14_13TeV") {
+      AddPeriod(222222,222222,999999);
+      if(m_emptyHistogram && (m_emptyHistogram->GetNbinsX()!=100 || fabs(m_emptyHistogram->GetXaxis()->GetBinLowEdge(1)+0.5)>0.01|| fabs(m_emptyHistogram->GetXaxis()->GetBinUpEdge(100)-99.5)>0.01) ) {
+         Error("UsePeriodConfig","Cannot use MC14_13TeV, an incompatible config has already been set up");
+         throw std::runtime_error("Throwing 13: Cannot use MC14_13TeV, an incompatible config has already been set up");
+         return -1;
+      }
+      SetUniformBinning(100,-0.5,99.5);
+      Info("UsePeriodConfig","Using MC14_13TeV Period configuration");
+      return 0;
+   } else if(configName=="MC15") {
+      AddPeriod(222510,222222,999999);
+      AddPeriod(222525,222222,999999);
+      if(m_emptyHistogram && (m_emptyHistogram->GetNbinsX()!=100 || fabs(m_emptyHistogram->GetXaxis()->GetBinLowEdge(1))>0.01 || fabs(m_emptyHistogram->GetXaxis()->GetBinUpEdge(100)-100.)>0.01) ) {
+         Error("UsePeriodConfig","Cannot use MC15, an incompatible config has already been set up");
+         throw std::runtime_error("Throwing 13: Cannot use MC15, an incompatible config has already been set up");
+         return -1;
+      }
+      SetUniformBinning(100,0,100); //Thanks Eric </sarcasm>
+      Info("UsePeriodConfig","Using MC15 Period configuration");
       return 0;
    }
    Error("UsePeriodConfig","Unrecognized period config");
    return -1;
 }
 
-std::vector<Double_t> Root::TPileupReweighting::getIntegratedLumiVector() { 
-   if(!m_lumiVectorIsLoaded) {
-      Error("getIntegratedLumiVector","The vector has not been filled. You must AddDataDistribution, using the iLumiCalc file, to fill it");
-      throw std::runtime_error("Throwing 99");
-   }
-   if(!m_isInitialized) {
-      Error("getIntegratedLumiVector","Please initialize the tool before retrieving the lumi vector");
-      throw std::runtime_error("Throwing 1");
-   }
-   return m_integratedLumiVector; 
-}
 
 ///returns a PeriodID. These count up from 1
-Int_t Root::TPileupReweighting::AddPeriod(Int_t mcRunNumber, UInt_t start, UInt_t end) {
-   if(m_isInitialized) {
-      Error("AddPeriod","You cannot AddPeriod after initializing the tool. Reorder your code!");
-      throw std::runtime_error("Throwing 1: You cannot AddPeriod after initializing the tool. Reorder your code!");
-   }
-   //check if this period is already defined
-   for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=m_periods[mcRunNumber].begin();periods!=m_periods[mcRunNumber].end();++periods) {
-      std::pair<UInt_t,UInt_t>& period = periods->second;
-      if(period.first == start && period.second==end) {
-         return 0;
-      } else if(start>=period.first && end<=period.second) {
-         //fully contained in existing period, just print a warning
-         Warning("AddPeriod","Period (%d,%d,%d) fully contained within existing period (%d,%d,%d). Doing nothing.",mcRunNumber,start,end,mcRunNumber,period.first,period.second);
-         return 0;
-      } else if(start==period.first && end==999999) {
-         //special case where we are 'extending' the period to infinity. This is a kludge to make old config files hadded to new ones work
-         Warning("AddPeriod","Extending period [%d,%d] to infinity. I hope this only happened because you are using an old prw config file (made pre 00-02-10)",period.first,period.second);
-         period.second = 999999;
-         return 0;
-      } else if((start <= period.second && start >= period.first) || (end <= period.second && end >= period.first) ) {
-         //overlaps with existing period. reject this Add
-         Error("AddPeriod","(%d,%d,%d) overlaps with existing period setup",mcRunNumber,start,end);
-         throw std::runtime_error("Throwing 2: (%d,%d,%d) overlaps with existing period setup");
-      }
-   } 
-   //check if the period being added conflicts with a period in another mcRunNumber... we just print a warning at this stage... not sure if it is a problem...
-   for(std::map<Int_t, std::map<Int_t, std::pair<UInt_t,UInt_t> > >::iterator periodRuns=m_periods.begin();periodRuns!=m_periods.end();++periodRuns) {
-      if(periodRuns->first==mcRunNumber) continue; //already checked this one
-      for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=periodRuns->second.begin();periods!=periodRuns->second.end();++periods) {
-         std::pair<UInt_t,UInt_t>& period = periods->second;
-         //if the period start and end match perfectly and this is a registered mc merger, then no warning
-         if(start==period.first && end==period.second && m_mcMerges.find(periodRuns->first)!=m_mcMerges.end()&& m_mcMerges[periodRuns->first].find(periodRuns->first)!=m_mcMerges[periodRuns->first].end() && m_mcMerges[periodRuns->first][periodRuns->first]==mcRunNumber) continue;
-         if(start==period.first && end==period.second && m_mcMerges.find(mcRunNumber)!=m_mcMerges.end()&& m_mcMerges[mcRunNumber].find(mcRunNumber)!=m_mcMerges[mcRunNumber].end() && m_mcMerges[mcRunNumber][mcRunNumber]==periodRuns->first) continue;
-         if((start <= period.second && start >= period.first) || (end <= period.second && end >= period.first) ) {
-            //overlaps with existing period. show a warning
-            if(m_SetWarnings) Warning("AddPeriod","(%d,%d,%d) conflicts with existing period setup (%d,%d,%d)",mcRunNumber,start,end,periodRuns->first,period.first,period.second);
-         }
-      }
-   } 
-   std::pair<UInt_t,UInt_t> thePair; thePair.first=start;thePair.second=end;
-   m_periods[mcRunNumber][m_nextPeriodNumber] = thePair;
-   //also put this in a handy backmapping for getting this period's periodWeight and mc distributions (uses same ones) 
-   m_periodToMCRun[m_nextPeriodNumber]=mcRunNumber;
-   m_nextPeriodNumber++;
-   return m_nextPeriodNumber-1;
-}
+Int_t CP::TPileupReweighting::AddPeriod(Int_t periodNumber, UInt_t start, UInt_t end) {
+   //you can add more periods only in counting mode, but they must not be with subperiods!
 
-Int_t Root::TPileupReweighting::GetPeriodNumber(Int_t mcRunNumber, UInt_t start, UInt_t end) {
-   //try to locate this period number in the map 
-   for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=m_periods[mcRunNumber].begin();periods!=m_periods[mcRunNumber].end();++periods) {
-      std::pair<UInt_t,UInt_t> period = periods->second;
-      if(period.first == start && period.second==end) {
-         return periods->first;
-      } 
-   } 
-   return -1;
-}
+   if(m_isInitialized && !m_countingMode) {
+      Error("AddPeriod","You cannot AddPeriod after initializing the tool, except when in config file generating mode. Reorder your code!");
+      throw std::runtime_error("Throwing 1: You cannot AddPeriod after initializing the tool, except when in config file generating mode. Reorder your code!");
+   }
 
-Int_t Root::TPileupReweighting::GetFirstFoundPeriodNumber(UInt_t runNumber) {
-   //try to locate this period number in the map 
-   for(std::map<Int_t,std::map<Int_t, std::pair<UInt_t,UInt_t> > >::iterator it = m_periods.begin();it!=m_periods.end();++it) {
-      for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=it->second.begin();periods!=it->second.end();++periods) {
-         std::pair<UInt_t,UInt_t> period = periods->second;
-         if(period.first <= runNumber && period.second>=runNumber) {
-            return periods->first;
-         } 
-      } 
+   if(m_periods.find(periodNumber)==m_periods.end()) {
+      m_periods[periodNumber] = new Period(periodNumber,start,end,m_periods[-1]->defaultChannel);
+      return periodNumber;
    }
-   return -1;
-}
 
-Int_t Root::TPileupReweighting::GetPeriodNumber(Int_t mcRunNumber, UInt_t runNumber) {
-   //try to locate this period number in the map 
-   for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=m_periods[mcRunNumber].begin();periods!=m_periods[mcRunNumber].end();++periods) {
-      std::pair<UInt_t,UInt_t> period = periods->second;
-      if(period.first <= runNumber && period.second>=runNumber) {
-         return periods->first;
-      } 
-   } 
-   return -1;
-}
+   //if period exists but both start and end are zero, then just assign that
+   Period* p = m_periods[periodNumber];
 
-Int_t Root::TPileupReweighting::AddBinning(const TString weightName,Int_t nbinsx, Double_t* xbins) {
-   if(m_emptyHistograms.find(weightName)!=m_emptyHistograms.end()) {
-        delete m_emptyHistograms[weightName];
-   }
-   m_emptyHistograms[weightName] = new TH1D(weightName+"_default","default",nbinsx,xbins);
-   m_emptyHistograms[weightName]->SetDirectory(0);
-   return 0;
-}
-Int_t Root::TPileupReweighting::AddBinning(const TString weightName,Int_t nbinsx, Double_t xlow, Double_t xup) {
-   if(m_emptyHistograms.find(weightName)!=m_emptyHistograms.end()) {
-        delete m_emptyHistograms[weightName];
+   if(p->subPeriods.size() == 0 && p->start==0 && p->end==0) { p->start=start; p->end=end; return periodNumber; }
+
+   //check if period already contained (i.e. same period or one of the existing subperiods) ... if it is, do nothing 
+   if(p->start==start && p->end==end) return periodNumber;
+   for(auto pp : p->subPeriods) {
+      if(pp->start==start && pp->end==end) return pp->id;
    }
-   m_emptyHistograms[weightName] = new TH1D(weightName+"_default","default",nbinsx,xlow,xup);
-   m_emptyHistograms[weightName]->SetDirectory(0);
-   return 0;
-}
-Int_t Root::TPileupReweighting::AddBinning(const TString weightName,Int_t nbinsx, Double_t xlow, Double_t xup, Int_t nbinsy, Double_t ylow, Double_t yup) {
-   if(m_emptyHistograms.find(weightName)!=m_emptyHistograms.end()) {
-        delete m_emptyHistograms[weightName];
+
+   //if get here, and in counting mode, this isn't allowed! .. i.e. we can't have subperiods
+   if(m_isInitialized && m_countingMode) {
+      Error("AddPeriod","You cannot have subperiods when in Config File Generating mode");
+      throw std::runtime_error("Throwing 44: In Config File Generating mode, but detected subperiods in the period definition. This is not allowed.");
    }
-   m_emptyHistograms[weightName] = new TH2D(weightName+"_default","default",nbinsx,xlow,xup,nbinsy,ylow,yup);
-   m_emptyHistograms[weightName]->SetDirectory(0);
-   return 0;
-}
 
-Int_t Root::TPileupReweighting::AddBinning(const TString weightName,TH1* hist) {
-   if(!hist) return -1;
-   if(m_emptyHistograms.find(weightName)!=m_emptyHistograms.end()) {
-        delete m_emptyHistograms[weightName];
+   //if period is already defined, we create a new period with start and end, and assign our period to this one 
+   //also create a new period for the existing period configuration of this period 
+
+   if(p->subPeriods.size()==0) {
+      while(m_periods.find(m_nextPeriodNumber) != m_periods.end()) m_nextPeriodNumber++;
+      m_periods[m_nextPeriodNumber] = new Period(m_nextPeriodNumber,p->start,p->end,p->defaultChannel);
+      p->subPeriods.push_back(m_periods[m_nextPeriodNumber]);
+      p->start = 0; p->end=0;
    }
-   m_emptyHistograms[weightName] = dynamic_cast<TH1*>(hist->Clone(weightName+"_default"));
-   m_emptyHistograms[weightName]->SetDirectory(0);
-   return 0;
-}
 
-Int_t Root::TPileupReweighting::GetDefaultChannel(Int_t mcRunNumber) {
-   if(m_defaultChannelsByMCRunNumber.find(mcRunNumber)!=m_defaultChannelsByMCRunNumber.end()) return m_defaultChannelsByMCRunNumber[mcRunNumber];
-   else return m_defaultChannel; 
+   while(m_periods.find(m_nextPeriodNumber) != m_periods.end()) m_nextPeriodNumber++;
+   m_periods[m_nextPeriodNumber] = new Period(m_nextPeriodNumber,start,end, p->defaultChannel);
+   p->subPeriods.push_back(m_periods[m_nextPeriodNumber]);
+
+   return m_nextPeriodNumber;
 }
 
 
-//combine run number's histograms, periods, everything, then remove trace of second run number
-void Root::TPileupReweighting::MergeMCRunNumbers(Int_t mcRunNumber1, Int_t mcRunNumber2, Int_t newMCRunNumber) {
-   if(m_isInitialized) {
-      Error("MergeMCRunNumbers","You cannot MergeMCRunNumbers after initializing the tool. Reorder your code!");
-      throw std::runtime_error("Throwing 5: You cannot MergeMCRunNumbers after initializing the tool. Reorder your code!");
+Int_t CP::TPileupReweighting::GetFirstFoundPeriodNumber(UInt_t runNumber) {
+   //go through periods, first period that includes this runNumber wins!
+   //otherwise it returns the 'global period number': -1
+   for(auto period : m_periods) {
+      if(period.second->id==-1) continue;
+      if(period.second->contains(runNumber)) return period.second->id;
    }
-   if(newMCRunNumber==0) newMCRunNumber=mcRunNumber1;
-   m_mcMerges[newMCRunNumber][mcRunNumber1]=mcRunNumber2;
+   return -1;
 
 }
 
-TH1* Root::TPileupReweighting::CloneEmptyHistogram(const TString weightName, Int_t runNumber, Int_t channelNumber) {
 
-   TString s(weightName);
+TH1* CP::TPileupReweighting::CloneEmptyHistogram(Int_t runNumber, Int_t channelNumber) {
+
+   TString s("pileup");
    if(channelNumber>=0) {
       s += "_chan"; s += channelNumber;
    } else {
@@ -419,12 +365,12 @@ TH1* Root::TPileupReweighting::CloneEmptyHistogram(const TString weightName, Int
    }
    s+="_run"; s+= runNumber;
    //get the empty histogram
-   if(m_emptyHistograms.find(weightName)==m_emptyHistograms.end()) {
-      Error("CloneEmptyHistogram","There is no binning info for weight %s - use AddBinning or load a prw config file (This usually means you need to call AddConfigFile BEFORE AddLumiCalcFile)",weightName.Data());
-      throw std::runtime_error("Throwing 47: There is no binning info - use AddBinning or load a prw config file (This usually means you need to call AddConfigFile BEFORE AddLumiCalcFile)");
+   if(!m_emptyHistogram) {
+      Error("CloneEmptyHistogram","There is no binning info - use SetBinning/SetUniformBinning or load a prw config file (This usually means you need to call AddConfigFile BEFORE AddLumiCalcFile)");
+      throw std::runtime_error("Throwing 47: There is no binning info - use SetBinning/SetUniformBinning or load a prw config file (This usually means you need to call AddConfigFile BEFORE AddLumiCalcFile)");
    }
 
-   TH1* out = dynamic_cast<TH1*>(m_emptyHistograms[weightName]->Clone(s));
+   TH1* out = dynamic_cast<TH1*>(m_emptyHistogram->Clone(s));
    out->SetTitle(s);
    out->SetDirectory(0); //take ownership
    out->SetEntries(0);
@@ -432,7 +378,7 @@ TH1* Root::TPileupReweighting::CloneEmptyHistogram(const TString weightName, Int
 
 }
 
-Int_t Root::TPileupReweighting::GenerateMetaDataFile(const TString fileName,const TString channelBranchName) {
+Int_t CP::TPileupReweighting::GenerateMetaDataFile(const TString fileName,const TString channelBranchName) {
 
    TTree inTree("in","in");
    inTree.ReadFile(fileName);
@@ -476,7 +422,7 @@ Int_t Root::TPileupReweighting::GenerateMetaDataFile(const TString fileName,cons
       return 0;
 }
 
-Int_t Root::TPileupReweighting::AddMetaDataFile(const TString fileName,const TString channelBranchName) {
+Int_t CP::TPileupReweighting::AddMetaDataFile(const TString fileName,const TString channelBranchName) {
    TDirectory* origDir = gDirectory;
    TTree* tmp = 0;
    TFile* rootFile = 0;
@@ -533,7 +479,7 @@ Int_t Root::TPileupReweighting::AddMetaDataFile(const TString fileName,const TSt
       }
    }
 
-   if(tmp) delete tmp;
+   delete tmp;
    if(rootFile) { rootFile->Close();delete rootFile;}
 
    // Return to the original ROOT directory
@@ -542,7 +488,7 @@ Int_t Root::TPileupReweighting::AddMetaDataFile(const TString fileName,const TSt
    return 0;
 }
 
-TTree* Root::TPileupReweighting::GetMetaDataTree() {
+TTree* CP::TPileupReweighting::GetMetaDataTree() {
    if(m_metadatatree) delete m_metadatatree;
    m_metadatatree = new TTree(TString(this->GetName())+"MetaData",TString(this->GetName())+"MetaData");
    m_metadatatree->SetDirectory(0);
@@ -567,7 +513,8 @@ TTree* Root::TPileupReweighting::GetMetaDataTree() {
    data["SumOfEventWeights"]=0.;
    m_metadatatree->Branch("SumOfEventWeights",&(data["SumOfEventWeights"]));
    //and add to channels list any channels that have events 
-   for(std::map<Int_t,Double_t>::iterator it=globalNumberOfEntries["pileup"].begin();it!=globalNumberOfEntries["pileup"].end();++it) {
+   auto& myMap = m_periods[-1]->sumOfWeights;
+   for(std::map<Int_t,Double_t>::iterator it=myMap.begin();it!=myMap.end();++it) {
       if(it->first>=0 && it->second>0.) channels[it->first]=true;
    }
 
@@ -577,27 +524,29 @@ TTree* Root::TPileupReweighting::GetMetaDataTree() {
       for(std::map<TString,Double_t>::iterator it2=data.begin();it2!=data.end();++it2) {
          if(it2->first=="NumberOfEvents") {
             //check for this in globalNumberOfEntries 
-            if(globalNumberOfEntries["pileup"].find(channel)==globalNumberOfEntries["pileup"].end()) {
+            if(myMap.find(channel)==myMap.end()) {
                data[it2->first]=0.;
                Warning("GetMetaDataTree","Channel %d does not have MetaData %s",it->first,(it2->first).Data());
             } else {
-               data[it2->first]=globalNumberOfEntries["pileup"][channel];
+               data[it2->first]=m_periods[-1]->numberOfEntries[channel];
             }
          } else if(it2->first=="SumOfEventWeights") {
             //check for this in globalTotals 
-            if(globalTotals["pileup"].find(channel)==globalTotals["pileup"].end()) {
+            auto& myMap2 = m_periods[-1]->sumOfWeights;
+            if(myMap2.find(channel)==myMap2.end()) {
                data[it2->first]=0.;
                Warning("GetMetaDataTree","Channel %d does not have MetaData %s",it->first,(it2->first).Data());
             } else {
-               data[it2->first]=globalTotals["pileup"][channel];
+               data[it2->first]=m_periods[-1]->sumOfWeights[channel];
             }
          } else {
-            if(m_metadata[it2->first].find(channel)==m_metadata[it2->first].end()) {
+            auto& myMap2 = m_metadata[it2->first];
+            if(myMap2.find(channel)==myMap2.end()) {
                //this channel doesn't have this property
                data[it2->first]=0.;
                Warning("GetMetaDataTree","Channel %d does not have MetaData %s",it->first,(it2->first).Data());
             } else {
-               data[it2->first]=m_metadata[it2->first][channel];
+               data[it2->first]=myMap2[channel];
             }
          }
       }
@@ -610,11 +559,14 @@ TTree* Root::TPileupReweighting::GetMetaDataTree() {
    return m_metadatatree;
 }
 
-Int_t Root::TPileupReweighting::AddDistribution(TH1* hist, Int_t runNumber, Int_t channelNumber) {
-   return AddDistribution(hist,"pileup",runNumber,channelNumber);
-}
 
-void Root::TPileupReweighting::AddDistributionTree(TTree *tree, TFile *file) {
+
+
+
+
+
+
+void CP::TPileupReweighting::AddDistributionTree(TTree *tree, TFile *file) {
 
    bool isMC=true;
    //should have the standard structure 
@@ -660,9 +612,6 @@ void Root::TPileupReweighting::AddDistributionTree(TTree *tree, TFile *file) {
       tree->GetEntry(i);
       TString sHistName(histName);
       TString weightName(customName);
-      if(hasDefaultsBranch) {
-         if(isDefaultForRunNumber) m_defaultChannelsByMCRunNumber[runNbr]=channel;
-      }
       if(loadedHistos.find(sHistName)==loadedHistos.end()) {
          loadedHistos[sHistName]=true;
          if(!m_ignoreFilePeriods && isMC) {
@@ -673,145 +622,97 @@ void Root::TPileupReweighting::AddDistributionTree(TTree *tree, TFile *file) {
          }
          //load the histogram
          TH1 *histo = (TH1*)file->Get( sHistName );
+         if(!histo) histo = (TH1*)file->Get( TString::Format("%sPileupReweighting/%s",m_prwFilesPathPrefix.c_str(),sHistName.Data()) ); //try from subdir
          if(!histo) {
             Error("AddDistributionTree","Unable to find the histogram %s in the File %s",sHistName.Data(),file->GetName());
             throw std::runtime_error("Throwing 21");
          }
          //add it to the histograms 
-         AddDistribution(histo, weightName, runNbr, channel);
+         AddDistribution(histo, runNbr, channel);
+         //we also add it to the global count, if this isn't data 
+         if(channel>=0) AddDistribution(histo,-1,channel);
+      }
+      if(hasDefaultsBranch) {
+         if(isDefaultForRunNumber && m_periods.find(runNbr)!=m_periods.end()) m_periods[runNbr]->defaultChannel=channel;
       }
    }
 
 }
 
 //data is signalled by a negative channel number
-Int_t Root::TPileupReweighting::AddDistribution(TH1* hist,const TString weightName,  Int_t runNumber, Int_t channelNumber) {
+Int_t CP::TPileupReweighting::AddDistribution(TH1* hist,Int_t runNumber, Int_t channelNumber) {
 
    if(m_isInitialized) {
       Error("AddDistribution","You cannot AddDistribution after initializing the tool. Reorder your code!");
       throw std::runtime_error("Throwing 5: You cannot AddLumiCalcFile after initializing the tool. Reorder your code!");
    }
 
-   
-   //std::map<Int_t,TH1*>& inputHistograms = (channelNumber>=0) ? m_inputMCHistograms[weightName][channelNumber] : m_inputDataHistograms[weightName];
-   std::map<Int_t,TH1*>& inputHistograms = m_inputHistograms[weightName][channelNumber];
+   if(channelNumber>=0 && !m_periods[runNumber]) {
+      Error("AddDistribution","Unrecognised periodNumber: %d .. please use AddPeriod to define a period",runNumber);
+      throw std::runtime_error("Throwing 6: Unrecognised periodNumber. Please use AddPeriod to define a period");
+   }
 
-   //check if we need to create an empty histogram 
-   if(inputHistograms.find(runNumber) == inputHistograms.end()) {
+   TH1*& inputHist = (channelNumber<0) ? m_runs[runNumber].inputHists["None"] : m_periods[runNumber]->inputHists[channelNumber];
+
+   if(!inputHist) {
       //if no emptyHistogram specified, we will use this histogram as the structure for the empty;
-      if(m_emptyHistograms.find(weightName)==m_emptyHistograms.end()) {
-         TString s = weightName; s+= "_default";
+      if(!m_emptyHistogram) {
          //check if the input histogram is a TH1D or not. We like TH1D, not yucky TH1F
          if(strcmp(hist->IsA()->GetName(),"TH1D")==0 || strcmp(hist->IsA()->GetName(),"TH2D")==0) {
-            m_emptyHistograms[weightName] = dynamic_cast<TH1*>(hist->Clone(s));
+            m_emptyHistogram = dynamic_cast<TH1*>(hist->Clone("default"));
          } else {
             //check dimensionality
             if(hist->GetDimension()==1) {
                std::vector<Double_t> binsX;
                for(int i=0;i<=hist->GetNbinsX();i++) binsX.push_back(hist->GetXaxis()->GetBinLowEdge(i+1));
                TH1D tmpHist("tmpHist","tmpHist",binsX.size()-1,&binsX[0]);
-               m_emptyHistograms[weightName] = dynamic_cast<TH1*>(tmpHist.Clone(s));
+               m_emptyHistogram = dynamic_cast<TH1*>(tmpHist.Clone("default"));
             } else if(hist->GetDimension()==2) {
                std::vector<Double_t> binsX;std::vector<Double_t> binsY;
                for(int i=0;i<=hist->GetNbinsX();i++) binsX.push_back(hist->GetXaxis()->GetBinLowEdge(i+1));
                for(int i=0;i<=hist->GetNbinsY();i++) binsY.push_back(hist->GetYaxis()->GetBinLowEdge(i+1));
                TH2D tmpHist("tmpHist","tmpHist",binsX.size()-1,&binsX[0],binsY.size()-1,&binsY[0]);
-               m_emptyHistograms[weightName] = dynamic_cast<TH1*>(tmpHist.Clone(s));
-            } else if(hist->GetDimension()==3) {
-               std::vector<Double_t> binsX;std::vector<Double_t> binsY;std::vector<Double_t> binsZ;
-               for(int i=0;i<=hist->GetNbinsX();i++) binsX.push_back(hist->GetXaxis()->GetBinLowEdge(i+1));
-               for(int i=0;i<=hist->GetNbinsY();i++) binsY.push_back(hist->GetYaxis()->GetBinLowEdge(i+1));
-               for(int i=0;i<=hist->GetNbinsZ();i++) binsZ.push_back(hist->GetZaxis()->GetBinLowEdge(i+1));
-               TH3D tmpHist("tmpHist","tmpHist",binsX.size()-1,&binsX[0],binsY.size()-1,&binsY[0],binsZ.size()-1,&binsZ[0]);
-               m_emptyHistograms[weightName] = dynamic_cast<TH1*>(tmpHist.Clone(s));
+               m_emptyHistogram = dynamic_cast<TH1*>(tmpHist.Clone("default"));
             } else {
                Error("AddDistribution","Unknown input histogram dimensionality: %d",hist->GetDimension());
                throw std::runtime_error("Throwing 98");
             }
          }
-         m_emptyHistograms[weightName]->Reset();//clear the histogram
-         m_emptyHistograms[weightName]->SetEntries(0);
-         m_emptyHistograms[weightName]->SetDirectory(0);
+         m_emptyHistogram->Reset();m_emptyHistogram->SetEntries(0);m_emptyHistogram->SetDirectory(0);//clear the histogram
       }
-      inputHistograms[runNumber] = CloneEmptyHistogram(weightName,runNumber,channelNumber);
+      inputHist = CloneEmptyHistogram(runNumber,channelNumber);
+
    }
+
+
    //iterator over bins of histogram, filling the TH1 stored in the data map
-   Double_t numEntries = inputHistograms[runNumber]->GetEntries();
-   Int_t bin,binx,biny,binz;
-   for(binz=1; binz<=hist->GetNbinsZ(); binz++) {
-      for(biny=1; biny<=hist->GetNbinsY(); biny++) {
-         for(binx=1; binx<=hist->GetNbinsX(); binx++) {
-            bin = hist->GetBin(binx,biny,binz);
-            Double_t value = hist->GetBinContent(bin);
-            Double_t x = hist->GetXaxis()->GetBinCenter(binx);
-            Double_t y = hist->GetYaxis()->GetBinCenter(biny);
-            Double_t z = hist->GetZaxis()->GetBinCenter(binz);
-            //shift x,y,z by the MCScaleFactors as appropriate 
-            if(channelNumber>=0) {x *= m_mcScaleFactorX; y *= m_mcScaleFactorY; z *= m_mcScaleFactorZ;}
-            else { x *= m_dataScaleFactorX; y *= m_dataScaleFactorY; z *= m_dataScaleFactorZ; }
-#if ROOT_VERSION_CODE >= ROOT_VERSION(5,28,0)
-            Int_t inBin = inputHistograms[runNumber]->FindFixBin(x,y,z);
-#else
-             Int_t inBin = inputHistograms[runNumber]->GetXaxis()->FindFixBin(x);
-#endif
-            Double_t inValue = inputHistograms[runNumber]->GetBinContent(inBin);
-            inputHistograms[runNumber]->SetBinContent(inBin,inValue+value);
-         }
+   Double_t numEntries = inputHist->GetEntries();
+   Int_t bin,binx,biny;
+   for(biny=1; biny<=hist->GetNbinsY(); biny++) {
+      for(binx=1; binx<=hist->GetNbinsX(); binx++) {
+         bin = hist->GetBin(binx,biny);
+         Double_t value = hist->GetBinContent(bin);
+         Double_t x = hist->GetXaxis()->GetBinCenter(binx);
+         Double_t y = hist->GetYaxis()->GetBinCenter(biny);
+         //shift x,y,z by the MCScaleFactors as appropriate 
+         if(channelNumber>=0) {x *= m_mcScaleFactorX; y *= m_mcScaleFactorY;}
+         else { x *= m_dataScaleFactorX; y *= m_dataScaleFactorY; }
+         Int_t inBin = inputHist->FindFixBin(x,y);
+         Double_t inValue = inputHist->GetBinContent(inBin);
+         inputHist->SetBinContent(inBin,inValue+value);
       }
    }
+
    //also keep track of the number of entries 
    //SetBinContent screws with the entry count, so had to record it before the loops above
-   inputHistograms[runNumber]->SetEntries(numEntries+hist->GetEntries());
+   inputHist->SetEntries(numEntries+hist->GetEntries());
    m_countingMode=false;
    return 0;
 
 } 
 
-Int_t Root::TPileupReweighting::AddDistributionFromFile(const TString fileName, const TString histName, const TString weightName, Int_t runNumber, Int_t channelNumber) {
-   if(m_isInitialized) {
-      Error("AddDistributionFromFile","You cannot AddDistributionFromFile after initializing the tool. Reorder your code!");
-      throw std::runtime_error("Throwing 5: You cannot AddLumiCalcFile after initializing the tool. Reorder your code!");
-   }
-
-    // Cache the current directory in the root file
-   TDirectory* origDir = gDirectory;
-   // Load the data ROOT file
-   TFile* rootFile = TFile::Open( fileName, "READ" );
-   if ( rootFile->IsZombie() ) {
-      Error("AddConfigFile","Could not open file: %s",fileName.Data());
-      std::string toThrow = "Throwing 6: Could not open file: "; toThrow += fileName;
-      throw std::runtime_error(toThrow);
-   } else {
-      //decide if this is a histogram or a ttree 
-      if(!rootFile->Get(histName)) {
-         Error("AddDistributionFromFile","Could not find object %s in file %s",histName.Data(),fileName.Data());
-         throw std::runtime_error("Throwing 7");
-      }
-      if(rootFile->Get(histName)->InheritsFrom("TH1")) {
-         TH1 *tmp = (TH1*)rootFile->Get( histName );
-         //tmp->SetDirectory(0);//take ownership 
-         AddDistribution(tmp, weightName, runNumber, channelNumber);
-      } else {
-         Error("AddDistributionFromFile","%s is not a Histogram",histName.Data());
-         throw std::runtime_error("Throwing 8");
-      }
-   }
-   m_countingMode=false;
-
-   delete rootFile;
-
-   // Return to the original ROOT directory
-   gDirectory = origDir;
-
-   return 0;
-
-}
 
-Int_t Root::TPileupReweighting::AddDistributionFromFile(const TString fileName, const TString histName, Int_t runNumber, Int_t channelNumber) {
-   return AddDistributionFromFile(fileName,histName,"pileup",runNumber,channelNumber);
-}
-
-Int_t Root::TPileupReweighting::AddLumiCalcFile(const TString fileName, const TString trigger) {
+Int_t CP::TPileupReweighting::AddLumiCalcFile(const TString fileName, const TString trigger) {
 
    if(m_isInitialized) {
       Error("AddLumiCalcFile","You cannot AddLumiCalcFile after initializing the tool. Reorder your code!");
@@ -831,44 +732,48 @@ Int_t Root::TPileupReweighting::AddLumiCalcFile(const TString fileName, const TS
          m_lumicalcFiles[trigger] = fileName;
          if(trigger=="None") {
             Info("AddLumiCalcFile","Adding LumiMetaData (scale factor=%f)...",m_dataScaleFactorX);
+            //structure expected is as given by iLumiCalc:
+            //   RunNbr, AvergeInteractionPerXing, IntLumi
+            UInt_t runNbr=0;Float_t intLumi=0;UInt_t lbn=0;TBranch *b_runNbr=0;TBranch *b_intLumi=0;TBranch *b_lbn=0;
+            Float_t mu=0.; TBranch *b_mu=0;
+            if(tmp->SetBranchAddress("RunNbr",&runNbr,&b_runNbr)!=0) {
+               Error("AddLumiCalcFile","Could not find RunNbr branch in Data TTree");throw std::runtime_error("Could not find RunNbr branch in Data TTree");
+            }
+            if(tmp->SetBranchAddress("AvergeInteractionPerXing",&mu,&b_mu)!=0) {
+               Error("AddLumiCalcFile","Could not find AvergeInteractionPerXing branch in Data TTree");throw std::runtime_error("Could not find AvergeInteractionPerXing branch in Data TTree");
+            }
+            if(tmp->SetBranchAddress("IntLumi",&intLumi,&b_intLumi)!=0) {
+               Error("AddLumiCalcFile","Could not find IntLumi branch in Data TTree");throw std::runtime_error("Could not find IntLumi branch in Data TTree");
+            }
+            if(tmp->SetBranchAddress("LBStart",&lbn,&b_lbn)!=0) {
+               Error("AddLumiCalcFile","Could not find LBStart branch in Data TTree");throw std::runtime_error("Could not find LBStart branch in Data TTree");
+            }
+            long nEntries = tmp->GetEntries();
+   
+            for(long i=0;i<nEntries;i++) {
+               b_runNbr->GetEntry(i);b_intLumi->GetEntry(i);b_mu->GetEntry(i);
+               runNbr += m_lumicalcRunNumberOffset;
+               //save the lumi by run and lbn 
+               b_lbn->GetEntry(i);
+   
+               Run& r = m_runs[runNbr];
+               //rescale the mu value 
+               mu *= m_dataScaleFactorX;
+               if(trigger=="None") {r.lumiByLbn[lbn].first += intLumi; r.lumiByLbn[lbn].second = mu;}
+               //fill into input data histograms 
+               //check if we need to create an empty histogram 
+               
+               if(r.inputHists.find(trigger) == r.inputHists.end()) {
+                  r.inputHists[trigger] = CloneEmptyHistogram(runNbr,-1);
+               }
+               r.inputHists[trigger]->Fill(mu,intLumi);
+            }
+            m_countingMode=false;
+            m_lumiVectorIsLoaded=true;
          } else {
             Info("AddLumiCalcFile","Adding LumiMetaData for DataWeight (trigger=%s) (scale factor=%f)...",trigger.Data(),m_dataScaleFactorX);
          }
-         //structure expected is as given by iLumiCalc:
-         //   RunNbr, AvergeInteractionPerXing, IntLumi
-         UInt_t runNbr=0;Float_t intLumi=0;UInt_t lbn=0;TBranch *b_runNbr=0;TBranch *b_intLumi=0;TBranch *b_lbn=0;
-         Float_t mu=0.; TBranch *b_mu=0;
-         if(tmp->SetBranchAddress("RunNbr",&runNbr,&b_runNbr)!=0) {
-            Error("AddLumiCalcFile","Could not find RunNbr branch in Data TTree");throw std::runtime_error("Could not find RunNbr branch in Data TTree");
-         }
-         if(tmp->SetBranchAddress("AvergeInteractionPerXing",&mu,&b_mu)!=0) {
-            Error("AddLumiCalcFile","Could not find AvergeInteractionPerXing branch in Data TTree");throw std::runtime_error("Could not find AvergeInteractionPerXing branch in Data TTree");
-         }
-         if(tmp->SetBranchAddress("IntLumi",&intLumi,&b_intLumi)!=0) {
-            Error("AddLumiCalcFile","Could not find IntLumi branch in Data TTree");throw std::runtime_error("Could not find IntLumi branch in Data TTree");
-         }
-         if(tmp->SetBranchAddress("LBStart",&lbn,&b_lbn)!=0) {
-            Error("AddLumiCalcFile","Could not find LBStart branch in Data TTree");throw std::runtime_error("Could not find LBStart branch in Data TTree");
-         }
-         long nEntries = tmp->GetEntries();
-         //std::map<Int_t, TH1*>& histograms = m_inputDataHistograms["pileup"];
-         std::map<Int_t, TH1*>& histograms = (trigger=="None") ? m_inputHistograms["pileup"][-1] : m_triggerInputHistograms[trigger];
-         for(long i=0;i<nEntries;i++) {
-            b_runNbr->GetEntry(i);b_intLumi->GetEntry(i);b_mu->GetEntry(i);
-            //save the lumi by run and lbn 
-            b_lbn->GetEntry(i);
-            m_lumiByRunAndLbn[runNbr][lbn] = intLumi;
-            //rescale the mu value 
-            mu *= m_dataScaleFactorX;
-            //fill into input data histograms 
-            //check if we need to create an empty histogram 
-            if(histograms.find(runNbr) == histograms.end()) {
-               histograms[runNbr] = CloneEmptyHistogram("pileup",runNbr,-1);
-            }
-            histograms[runNbr]->Fill(mu,intLumi);
-         }
-         m_countingMode=false;
-         m_lumiVectorIsLoaded=true;
+         
       }
    }
 
@@ -880,7 +785,7 @@ Int_t Root::TPileupReweighting::AddLumiCalcFile(const TString fileName, const TS
    return 0;
 }
 
-Int_t Root::TPileupReweighting::AddConfigFile(const TString fileName) {
+Int_t CP::TPileupReweighting::AddConfigFile(const TString fileName) {
 
    //open the file and look for config TTrees 
    //recognised TTrees are: ChannelMetaData, MCPileupReweighting, MCCustomReweighting, DataCustomReweighting
@@ -899,19 +804,37 @@ Int_t Root::TPileupReweighting::AddConfigFile(const TString fileName) {
       throw std::runtime_error(toThrow);
    } else {
       //try to get the the known TTrees 
-      TTree *tmp = (TTree*)rootFile->Get( "MCPileupReweighting" );
+      TTree *tmp = (TTree*)rootFile->Get( std::string(m_prwFilesPathPrefix+"MCPileupReweighting").c_str() );
+      if(tmp) {
+         Info("AddConfigFile","Adding MCPileupReweighting...");
+         AddDistributionTree(tmp,rootFile);
+         m_countingMode=false;
+      }
+      tmp = 0;tmp = (TTree*)rootFile->Get( std::string(m_prwFilesPathPrefix+"MCCustomReweighting").c_str() );
+      if(tmp) {
+         Info("AddConfigFile","Adding MCCustomReweighting...");
+         AddDistributionTree(tmp,rootFile);
+         m_countingMode=false;
+      }
+      tmp = 0;tmp = (TTree*)rootFile->Get( std::string(m_prwFilesPathPrefix+"DataCustomReweighting").c_str() );
+      if(tmp) {
+         Info("AddConfigFile","Adding DataCustomReweighting...");
+         AddDistributionTree(tmp,rootFile);
+         m_countingMode=false;
+      }
+      tmp=0; tmp = (TTree*)rootFile->Get( std::string(m_prwFilesPathPrefix+"PileupReweighting/MCPileupReweighting").c_str() ); //allow trees in the PileupReweighting dir instead
       if(tmp) {
          Info("AddConfigFile","Adding MCPileupReweighting...");
          AddDistributionTree(tmp,rootFile);
          m_countingMode=false;
       }
-      tmp = 0;tmp = (TTree*)rootFile->Get( "MCCustomReweighting" );
+      tmp = 0;tmp = (TTree*)rootFile->Get( std::string(m_prwFilesPathPrefix+"PileupReweighting/MCCustomReweighting").c_str() );
       if(tmp) {
          Info("AddConfigFile","Adding MCCustomReweighting...");
          AddDistributionTree(tmp,rootFile);
          m_countingMode=false;
       }
-      tmp = 0;tmp = (TTree*)rootFile->Get( "DataCustomReweighting" );
+      tmp = 0;tmp = (TTree*)rootFile->Get( std::string(m_prwFilesPathPrefix+"PileupReweighting/DataCustomReweighting").c_str() );
       if(tmp) {
          Info("AddConfigFile","Adding DataCustomReweighting...");
          AddDistributionTree(tmp,rootFile);
@@ -927,1101 +850,759 @@ Int_t Root::TPileupReweighting::AddConfigFile(const TString fileName) {
    return 0;
 }
 
-void Root::TPileupReweighting::AddDataDistribution(const TString& dataRootFileName,
-                    const TString& dataRootHistName,Int_t runNumber) {
-   Warning("AddDataDistribution","Method Depricated. Please use either AddConfigFile or AddDistributionFromFile");
-   if(dataRootHistName=="LumiMetaData") AddLumiCalcFile(dataRootFileName);
-   else AddDistributionFromFile(dataRootFileName,dataRootHistName,"pileup",runNumber,-1);
-}
-
-void Root::TPileupReweighting::AddMCDistribution(const TString& mcRootFileName,
-                    const TString& mcRootHistName, int runNumber, int channelNumber) {
-   Warning("AddMCDistribution","Method Depricated. Please use either AddConfigFile or AddDistributionFromFile");
-   if(mcRootHistName=="MCPileupReweighting") AddConfigFile(mcRootFileName);
-   else AddDistributionFromFile(mcRootFileName,mcRootHistName,"pileup",runNumber,channelNumber);
-}
-
-void Root::TPileupReweighting::AddCustomDataDistribution(const TString& customRootFileName,
-                    const TString& customRootHistName, TString customName, int runNumber) {
-   Warning("AddCustomDataDistribution","Method Depricated. Please use either AddConfigFile or AddDistributionFromFile");
-   if(customRootHistName=="DataCustomReweighting") AddConfigFile(customRootFileName);
-   else AddDistributionFromFile(customRootFileName,customRootHistName,customName,runNumber,-1);
-}
+Int_t CP::TPileupReweighting::Initialize() {
 
+   //Need to calculate a period weight for each period
+   //Need a reweighting histogram for each period 
+   //for merged mc run numbers, we must generate or modify the existing mc distributions 
 
-void Root::TPileupReweighting::AddCustomMCDistribution(const TString& customRootFileName,
-                    const TString& customRootHistName, TString customName, int runNumber, int channelNumber) {
-   Warning("AddCustomMCDistribution","Method Depricated. Please use either AddConfigFile or AddDistributionFromFile");
-   if(customRootHistName=="MCCustomReweighting") AddConfigFile(customRootFileName);
-   else AddDistributionFromFile(customRootFileName,customRootHistName,customName,runNumber,channelNumber);
-}
 
 
-//channelNumber=-1 is data
-Int_t Root::TPileupReweighting::FactorizeDistribution(TH1* hist, const TString weightName, 
-                           Int_t channelNumber, Int_t periodNumber,bool includeInMCRun,bool includeInGlobal)   {
+   //print out the period assignments if in debug mode 
+   if(m_debugging) {
+      PrintPeriods();
+   }
 
-   //determine which mc runNumber this periodNumber is assigned to 
-   Int_t mcRunNumber = m_periodToMCRun[periodNumber];
 
-   if(includeInGlobal) {
-      globalTotals[weightName][channelNumber] += hist->GetSumOfWeights();
-      globalNumberOfEntries[weightName][channelNumber] += hist->GetEntries();
-   }
-   if(includeInMCRun) {
-      periodTotals[weightName][channelNumber][mcRunNumber] += hist->GetSumOfWeights();
-      //initialize the corresponding data value if necessary
-      if(periodTotals[weightName][-1].find(mcRunNumber)==periodTotals[weightName][-1].end()) {
-         periodTotals[weightName][-1][mcRunNumber]=0;
+   if(m_countingMode) {
+      Info("Initialize","In Config File Generating mode. Remember to call WriteToFile!");
+      //need to check no periods have subperiods, this is not allowed in counting mode 
+      for(auto period : m_periods) {
+         if(period.second->subPeriods.size()!=0) {
+            Error("Initialize","You cannot have subperiods when in Config File Generating mode");
+            throw std::runtime_error("Throwing 44: In Config File Generating mode, but detected subperiods in the period definition. This is not allowed.");
+         }
       }
+     //histograms are instantiated in the event loop as the channels occur
+      //can delay period definition here
+      //but since we are here, check that the user has at least defined some periods .. i.e. 
+//       if(m_periods.size()==1) {
+//          Error("Initialize", "In Config File Generating mode, but no periods have been defined. This makes no sense!? Define some periods please (with AddPeriod method)!");
+//          throw std::runtime_error("Throwing 43: In Config File Generating mode, but no periods have been defined. This makes no sense!? Define some periods please (with AddPeriod method)!");
+//       }
+      m_isInitialized=true;
+      return 0;
    }
-   periodTotals[weightName][channelNumber][periodNumber] += hist->GetSumOfWeights();
-
-   //initialize the corresponding data value if necessary
-   if(periodTotals[weightName][-1].find(periodNumber)==periodTotals[weightName][-1].end()) {
-      periodTotals[weightName][-1][periodNumber]=0;
-   }
-
-   //add to the primary/secondary/tertiary histograms
-   if(hist->InheritsFrom("TH3")) {
-      //not supported yet 
-      Error("initialize","3d not supported yet");throw 40;
-   } else if(hist->InheritsFrom("TH2")) {
-      TH2* histo2d = dynamic_cast<TH2*>(hist);
-      //project out and add that to primary 
-      TH1D* proj = histo2d->ProjectionX();
-      if(!primaryDistributions[weightName][channelNumber][periodNumber]) {
-         //need to make a copy
-         TH1D* projCopy = dynamic_cast<TH1D*>(proj->Clone(proj->GetName()));
-         projCopy->SetDirectory(0);
-         primaryDistributions[weightName][channelNumber][periodNumber] = projCopy;
-         //initialize the corresponding data histogram if necessary
-         if(!primaryDistributions[weightName][-1][periodNumber]) {
-            TH1D* projEmpty = dynamic_cast<TH1D*>(proj->Clone(proj->GetName()));
-            projEmpty->Reset();
-            projEmpty->SetDirectory(0);
-            primaryDistributions[weightName][-1][periodNumber]=projEmpty;
+
+
+   //go through all periods, use inputHists to populate the primary and secondary dists, and fill sumOfWeights and numberOfEntries
+   for(auto period : m_periods) {
+      if(period.first != period.second->id) continue; //skips remappings 
+      for(auto& inputHist : period.second->inputHists) {
+         int channel = inputHist.first;
+         TH1* hist = inputHist.second;
+         period.second->sumOfWeights[channel] += hist->GetSumOfWeights();
+         period.second->numberOfEntries[channel] += hist->GetEntries();
+         if(hist->GetDimension()==1) {
+            period.second->primaryHists[channel] = dynamic_cast<TH1D*>(CloneEmptyHistogram(period.first,channel));
+            period.second->primaryHists[channel]->Add(hist);
          }
-      } else {
-         primaryDistributions[weightName][channelNumber][periodNumber]->Add(proj);
       }
-      if(!secondaryDistributions[weightName][channelNumber][periodNumber]) {
-         secondaryDistributions[weightName][channelNumber][periodNumber] = dynamic_cast<TH2D*>(CloneEmptyHistogram(weightName,periodNumber,channelNumber));
-         //initialize the corresponding data histogram if necessary
-         if(!secondaryDistributions[weightName][-1][periodNumber]) {
-            secondaryDistributions[weightName][-1][periodNumber]=dynamic_cast<TH2D*>(CloneEmptyHistogram(weightName,periodNumber,-1));
-         }
-      }
-      secondaryDistributions[weightName][channelNumber][periodNumber]->Add(hist);
-
-      if(includeInMCRun) {
-         if(!primaryDistributions[weightName][channelNumber][mcRunNumber]) {
-            //need to make a copy for the run number, to avoid double deleting
-            TH1D* projCopy = dynamic_cast<TH1D*>(proj->Clone(proj->GetName()));
-            projCopy->SetDirectory(0);
-            primaryDistributions[weightName][channelNumber][mcRunNumber] = projCopy;
-            //initialize the corresponding data histogram if necessary
-            if(!primaryDistributions[weightName][-1][mcRunNumber]) {
-               TH1D* projEmpty = dynamic_cast<TH1D*>(proj->Clone(proj->GetName()));
-               projEmpty->Reset();
-               projEmpty->SetDirectory(0);
-               primaryDistributions[weightName][-1][mcRunNumber]=projEmpty;
-            }
-         } else {
-            primaryDistributions[weightName][channelNumber][mcRunNumber]->Add(proj);
-         }
-         if(!secondaryDistributions[weightName][channelNumber][mcRunNumber]) {
-            secondaryDistributions[weightName][channelNumber][mcRunNumber] = dynamic_cast<TH2D*>(CloneEmptyHistogram(weightName,mcRunNumber,channelNumber));
-            //initialize the corresponding data histogram if necessary
-            if(!secondaryDistributions[weightName][-1][mcRunNumber]) {
-               secondaryDistributions[weightName][-1][mcRunNumber]=dynamic_cast<TH2D*>(CloneEmptyHistogram(weightName,mcRunNumber,-1));
-            }
-         }
-         secondaryDistributions[weightName][channelNumber][mcRunNumber]->Add(hist);
-      }
-      delete proj;
-
-   } else {
-      //regular 1D distribution 
-      if(!primaryDistributions[weightName][channelNumber][periodNumber]) {
-         primaryDistributions[weightName][channelNumber][periodNumber] = dynamic_cast<TH1D*>(CloneEmptyHistogram(weightName,periodNumber,channelNumber));
-         //initialize the corresponding data histogram if necessary
-         if(!primaryDistributions[weightName][-1][periodNumber]) {
-            primaryDistributions[weightName][-1][periodNumber]=dynamic_cast<TH1D*>(CloneEmptyHistogram(weightName,periodNumber,-1));
-         }
-      }
-      primaryDistributions[weightName][channelNumber][periodNumber]->Add(hist);
-      if(includeInMCRun) {
-         if(!primaryDistributions[weightName][channelNumber][mcRunNumber]) {
-            primaryDistributions[weightName][channelNumber][mcRunNumber] = dynamic_cast<TH1D*>(CloneEmptyHistogram(weightName,mcRunNumber,channelNumber));
-            //initialize the corresponding data histogram if necessary
-            if(!primaryDistributions[weightName][-1][mcRunNumber]) {
-               primaryDistributions[weightName][-1][mcRunNumber]=dynamic_cast<TH1D*>(CloneEmptyHistogram(weightName,mcRunNumber,-1));
-            }
-         }
-         primaryDistributions[weightName][channelNumber][mcRunNumber]->Add(hist);
-      }
-   }
-
-   return 0;
-}
-
-//=============================================================================
-// Initialization
-//=============================================================================
-Int_t Root::TPileupReweighting::initialize( const TString dataRootFileName,
-                                          const TString dataRootHistName,
-                                          const TString mcRootFileName,
-                                          const TString mcRootHistName, int runNumber, int channel )
-{
-   if(dataRootFileName != "") {
-      AddMCDistribution(mcRootFileName,mcRootHistName,runNumber,channel);
-      AddDataDistribution(dataRootFileName,dataRootHistName,runNumber);
-      if(m_periods.size()==0) {
-         Info("initialize","No Periods have been defined. I'm guessing you are doing old-style reweighting");
-         Info("initialize","Creating dummy period and ignoring unrepresented data issues");
-         AddPeriod(0,0,0); //assign mcRunNumber 0 to data runNumber 0-0.
-         SetUnrepresentedDataAction(2);
-      }
-   }
-   return Initialize();
-}
-
-Int_t Root::TPileupReweighting::Initialize() {
-
-   //Need to calculate a period weight for each period
-   //Need a reweighting histogram for each period 
-   //for merged mc run numbers, we must generate or modify the existing mc distributions 
-
-
-   //1. Deal with the registered mc mergers 
-   std::map<Int_t, bool> newMCRuns; //records if new mcRunNumbers have been created by merging
-   for(std::map<Int_t, std::map<Int_t, Int_t> >::iterator mergers = m_mcMerges.begin();mergers != m_mcMerges.end();++mergers) {
-      for(std::map<Int_t,Int_t>::iterator mergePair = (mergers->second).begin();mergePair!=(mergers->second).end();++mergePair) {
-         Int_t run1 = mergePair->first;
-         Int_t run2 = mergePair->second;
-         Int_t runOut = mergers->first;
-         if(run1==runOut) {
-            if(m_debugging) Info("Initialize","Merging %d into %d",run2,run1);
-            //merging run2 directly in to run1. 
-            //just combine run2 and run1 mc histos, and clear empty the run2 histograms 
-            for(std::map<TString, std::map<Int_t, std::map<Int_t,TH1*> > >::iterator mcHistos = m_inputHistograms.begin();mcHistos!=m_inputHistograms.end();++mcHistos) {
-               TString weightName = mcHistos->first;
-               for(std::map<Int_t, std::map<Int_t,TH1*> >::iterator channels = mcHistos->second.begin();channels!=mcHistos->second.end();++channels) {
-                  if((channels->first)<0) continue; //skip data channels
-                  if(channels->second.find(run2)!=channels->second.end()) {
-                     if(channels->second.find(run1)!=channels->second.end()) {
-                        channels->second[run1]->Add(channels->second[run2]);
-                     } else {
-                        channels->second[run1] = CloneEmptyHistogram(weightName,run1,channels->first);
-                        channels->second[run1]->Add(channels->second[run2]);
+      //also add all runs that belong to us ... only do the "None" trigger
+      for(auto& run : m_runs) {
+         if(!period.second->contains(run.first)) continue;
+         if(run.second.inputHists.find("None") == run.second.inputHists.end()) continue;
+         //add to list of runNumbers for this period 
+         period.second->runNumbers.push_back(run.first);
+         TH1* hist = run.second.inputHists["None"];
+
+         //sweep over bins, checking for unrepresented data and acting accordingly 
+         Int_t bin,binx,biny;
+         for(biny=1; biny<=hist->GetNbinsY(); biny++) {
+            for(binx=1; binx<=hist->GetNbinsX(); binx++) {
+               bin = hist->GetBin(binx,biny);
+               Double_t value = hist->GetBinContent(bin);
+               if(value==0.) continue;
+               //loop over channels, check for zero in this bin  ... only need to do if we haven't checked ourselves 
+               if(period.second->inputBinRedirect.find(bin) == period.second->inputBinRedirect.end()) {
+                  for(auto& inputHist : period.second->inputHists) {
+                     if(inputHist.first<0) continue; //skips data
+                     Double_t mcValue = (inputHist.second)->GetBinContent(bin);
+                     if(mcValue==0.) { 
+                        if(m_unrepresentedDataAction!=0 && m_debugging) Info("Initialize","Unrepresented data at coords [%f,%f] caused by periodNumber %d in channel %d",hist->GetXaxis()->GetBinCenter(binx),hist->GetYaxis()->GetBinCenter(biny),period.first,inputHist.first);
+                        //if we are doing unrepaction=2, just set to 'not the bin number'. if we are doing action=3, find the nearest good bin 
+                        if(m_unrepresentedDataAction==0) {period.second->inputBinRedirect[bin] = bin-1;Error("Initialize","Unrepresented data at coords [%f,%f] caused by periodNumber %d in channel %d",hist->GetXaxis()->GetBinCenter(binx),hist->GetYaxis()->GetBinCenter(biny),period.first,inputHist.first);}
+                        if(m_unrepresentedDataAction==1||m_unrepresentedDataAction==2) period.second->inputBinRedirect[bin] = bin-1;
+                        else if(m_unrepresentedDataAction==3) period.second->inputBinRedirect[bin] = GetNearestGoodBin(period.first, bin);
+                        break;
                      }
-                     channels->second[run2]->Reset();
                   }
+                  if(period.second->inputBinRedirect.find(bin) == period.second->inputBinRedirect.end()) period.second->inputBinRedirect[bin] = bin; //a good bin 
                }
-            }
-            //merge the periods that were assigned 
-            //combine the periods
-            if(m_periods.find(run2)!=m_periods.end()) {
-               for(std::map<Int_t,std::pair<UInt_t,UInt_t> >::iterator periods2 = m_periods[run2].begin();periods2!=m_periods[run2].end();++periods2) {
-                  AddPeriod(run1,periods2->second.first,periods2->second.second);
-               }
-               m_periods.erase(run2);
-            }
-            //and remap run2 in to run1 
-            m_mcRemappings[run2]=run1;
-         } else {
-            //merging run1 and run2 in to a new run (run3).
-            newMCRuns[runOut]=true;
-            for(std::map<TString, std::map<Int_t, std::map<Int_t,TH1*> > >::iterator mcHistos = m_inputHistograms.begin();mcHistos!=m_inputHistograms.end();++mcHistos) {
-               TString weightName = mcHistos->first;
-               for(std::map<Int_t, std::map<Int_t,TH1*> >::iterator channels = mcHistos->second.begin();channels!=mcHistos->second.end();++channels) {
-                  if((channels->first)<0) continue;
-                  if(channels->second.find(run2)!=channels->second.end()) {
-                     if(channels->second.find(runOut)!=channels->second.end()) {
-                        channels->second[runOut]->Add(channels->second[run2]);
-                     } else {
-                        channels->second[runOut] = CloneEmptyHistogram(weightName,runOut,channels->first);
-                        channels->second[runOut]->Add(channels->second[run2]);
+               //if is a bad bin, act accordingly 
+               if(period.second->inputBinRedirect[bin] != bin) {
+                  run.second.badBins[bin]=value;
+                  if(m_unrepresentedDataAction==1) {
+                     //remove this data. we have to remember to do this for the trigger data too if this is the pileup weight
+                     hist->SetBinContent(bin,0);
+                     //also modify the m_inputTriggerHistograms ... 
+                     for(auto& triggerDists : run.second.inputHists) triggerDists.second->SetBinContent(bin,0);
+                  } else if(m_unrepresentedDataAction==3) {
+                     //reassign the content to that bin 
+                     hist->SetBinContent(period.second->inputBinRedirect[bin],hist->GetBinContent(period.second->inputBinRedirect[bin])+hist->GetBinContent(bin));
+                     hist->SetBinContent(bin,0);
+                     //also modify the m_inputTriggerHistograms ... 
+                     for(auto& triggerDists : run.second.inputHists) {
+                        triggerDists.second->SetBinContent(period.second->inputBinRedirect[bin],triggerDists.second->GetBinContent(period.second->inputBinRedirect[bin])+triggerDists.second->GetBinContent(bin));
+                        triggerDists.second->SetBinContent(bin,0);
                      }
-                  }
-                  if(channels->second.find(run1)!=channels->second.end()) {
-                     if(channels->second.find(runOut)!=channels->second.end()) {
-                        channels->second[runOut]->Add(channels->second[run1]);
-                     } else {
-                        channels->second[runOut] = CloneEmptyHistogram(weightName,runOut,channels->first);
-                        channels->second[runOut]->Add(channels->second[run1]);
-                     }
-                  }
-               }
-            }
-            //merge the periods that were assigned, but do not clear the period assignment 
-            if(m_periods.find(run2)!=m_periods.end()) {
-               for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods2 = m_periods[run2].begin();periods2!=m_periods[run2].end();++periods2) {
-                  AddPeriod(runOut,periods2->second.first,periods2->second.second);
-               }
+                   }
+                }
             }
-            if(m_periods.find(run1)!=m_periods.end()) {
-               for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods2 = m_periods[run1].begin();periods2!=m_periods[run1].end();++periods2) {
-                  AddPeriod(runOut,periods2->second.first,periods2->second.second);
+         }
+         //create the period's 'data' hist if necessary 
+         if( hist->GetDimension()==1 ) {
+              if(!period.second->primaryHists[-1] )  {
+                  period.second->primaryHists[-1] = dynamic_cast<TH1D*>(CloneEmptyHistogram(period.first,-1));
+                  period.second->triggerHists["None"] = dynamic_cast<TH1D*>(CloneEmptyHistogram(period.first,-1)); //this will remain unnormalized, unlike the 'primaryHist'
                }
+         }
+         else if( hist->GetDimension()==2 ) {
+            if(!period.second->secondaryHists[-1] ) {
+               period.second->secondaryHists[-1] =  dynamic_cast<TH2D*>(CloneEmptyHistogram(period.first,-1));
+               period.second->primaryHists[-1] = period.second->secondaryHists[-1]->ProjectionX(); period.second->primaryHists[-1]->SetDirectory(0);
+               period.second->primaryHists[-1]->Reset();
+               period.second->triggerHists["None"] = static_cast<TH1D*>(period.second->primaryHists[-1]->Clone("triggerHist"));
+               period.second->triggerHists["None"]->SetDirectory(0);
             }
+            
          }
-      }
-   } 
 
-   //print out the period assignments if in debug mode 
-   if(m_debugging) {
-      for(std::map<Int_t,std::map<Int_t, std::pair<UInt_t,UInt_t> > >::iterator mcruns = m_periods.begin();mcruns!=m_periods.end();++mcruns) {
-         Info("Initialize","MCRunNumber %d has following periods:",mcruns->first);
-         for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods = mcruns->second.begin();periods!=mcruns->second.end();++periods) {
-            Info("Initialize","  PeriodNumber %d: %d - %d",periods->first,periods->second.first,periods->second.second);
-         }
       }
+
    }
 
-   //Factorize the MC distributions to build up the totals and primary/seconday/tertiary distributions
-   for(std::map<TString, std::map<Int_t, std::map<Int_t,TH1*> > >::iterator mcHistos = m_inputHistograms.begin();mcHistos!=m_inputHistograms.end();++mcHistos) {
-      TString weightName = mcHistos->first;
-      for(std::map<Int_t, std::map<Int_t,TH1*> >::iterator channels = mcHistos->second.begin();channels!=mcHistos->second.end();++channels) {
-         Int_t channel = channels->first;
-         if(channel<0) continue;
-         for(std::map<Int_t,TH1*>::iterator mcruns = channels->second.begin();mcruns!=channels->second.end();++mcruns) {
-            Int_t mcRunNum = mcruns->first;
-            //all the periods associated to the mcrun will get the same distribution 
-            //only include the first subperiod in the totals and mcrunnum's distributions
-            //and only include genuine mcrunnums (not ones generated in mergers) in global totals
-            bool isFirst = true;
-            for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator period = m_periods[mcRunNum].begin(); period!=m_periods[mcRunNum].end();++period) {
-               FactorizeDistribution(mcruns->second,weightName,channel,period->first,isFirst,isFirst&&(!newMCRuns[mcRunNum]));
-               isFirst=false;
+   Double_t unrepData(0);
+   //count up the unrepresented data, indexed by channel 
+
+   for(auto& run : m_runs) {
+      for(auto& b : run.second.badBins) {
+         if(!b.second) continue;
+         int bin = b.first;
+         unrepData += b.second;
+         //find out which channel in which period caused this bad bin!
+         std::map<Int_t, bool> doneChannels;
+         for(auto& period : m_periods) {
+            if(period.first != period.second->id) continue;
+            if(!period.second->contains(run.first)) continue;
+            for(auto& inHist : period.second->inputHists) {
+               if(inHist.first<0) continue; //skips any data hists (shouldn't happen really)
+               if((inHist.second)->GetBinContent(bin)==0) { period.second->unrepData[inHist.first] += b.second; } //store unrep data by period too, because will use in GetUnrepresentedFraction 
+               if(doneChannels[inHist.first]) continue; //dont doublecount the data if the channel was to blame across multiple periods
+               if((inHist.second)->GetBinContent(bin)==0) {unrepDataByChannel[inHist.first] += b.second;doneChannels[inHist.first]=true;}
             }
          }
       }
    }
 
-   //build any trigger "data weight" histograms. Have one for each period, for each trigger
-   for(std::map<Int_t, std::map<Int_t,std::pair<UInt_t,UInt_t> > >::iterator runPeriods = m_periods.begin();runPeriods != m_periods.end();++runPeriods) {
-      for(std::map<Int_t,std::pair<UInt_t,UInt_t> >::iterator period = runPeriods->second.begin();period!=runPeriods->second.end();++period) {
-         Int_t periodNumber = period->first;
-         UInt_t pStart = period->second.first; UInt_t pEnd = period->second.second;
-         if(m_inputHistograms.find("pileup")==m_inputHistograms.end()) continue; //need the "None" unprescaled mu distribution
-         std::map<Int_t,TH1*> unprescaledDataHists = m_inputHistograms["pileup"][-1];
-         m_triggerPrimaryDistributions["None"][periodNumber] = dynamic_cast<TH1D*>(CloneEmptyHistogram("pileup",periodNumber,-1));
-         //loop over runs and see if we have corresponding in the unprescaled data map
-         for(UInt_t r=pStart;r<=pEnd;r++) {
-            if(unprescaledDataHists.find(r)!=unprescaledDataHists.end()) {
-               if(unprescaledDataHists[r]) m_triggerPrimaryDistributions["None"][periodNumber]->Add(unprescaledDataHists[r]);
-            }
-            //also now look for trigger input histograms
-            for(std::map<TString, std::map<Int_t, TH1*> >::iterator triggerDists = m_triggerInputHistograms.begin();triggerDists!=m_triggerInputHistograms.end();++triggerDists) {
-               if(m_triggerPrimaryDistributions[triggerDists->first][periodNumber]==0) {
-                  m_triggerPrimaryDistributions[triggerDists->first][periodNumber] = dynamic_cast<TH1D*>(CloneEmptyHistogram("pileup",periodNumber,-1));
-                  Info("Initialize","Created Data Weight Histogram for [%s,%d]",triggerDists->first.Data(),periodNumber);
-               }
-               if(triggerDists->second.find(r)!=triggerDists->second.end()) {
-                  if(triggerDists->second[r]) {
-                     m_triggerPrimaryDistributions[triggerDists->first][periodNumber]->Add(triggerDists->second[r]);
-                     delete triggerDists->second[r]; triggerDists->second[r]=0;
-                  }
-               }
-            }
+   //now that all periods and runs have been checked, with redirects set up where necessary, we actually accumulate the data across the runs 
+   //we should also check if there are any data runs that are not covered by the period assignments 
+
+   double ignoredData(0);
+   for(auto& run : m_runs) {
+      bool covered(false);
+      TH1* hist = run.second.inputHists["None"];
+      for(auto& period : m_periods) {
+         if(!period.second->contains(run.first)) continue;
+         if(period.first!=-1) covered=true; //don't count the global period
+         if(hist->GetDimension()==1) {
+            period.second->primaryHists[-1]->Add(hist);
+            period.second->triggerHists["None"]->Add(hist);
+         } else if(hist->GetDimension()==2) {
+            period.second->secondaryHists[-1]->Add(hist);
+            TH1* proj = ((TH2*)hist)->ProjectionX();
+            period.second->primaryHists[-1]->Add(proj);
+            period.second->triggerHists["None"]->Add(proj);
+            delete proj;
          }
+         period.second->sumOfWeights[-1] += hist->GetSumOfWeights();
+         period.second->numberOfEntries[-1] += hist->GetEntries();
       }
+      if(!covered && m_periods.size()>1) {
+         Warning("Initialize","loaded data in run %d that is not covered by period assignments",run.first);
+         ignoredData += hist->GetSumOfWeights();
+      } 
    }
+   double totalData =  (m_unrepresentedDataAction==1) ? (m_periods[-1]->sumOfWeights[-1]+unrepData) : m_periods[-1]->sumOfWeights[-1];
 
-   std::map<TString,Double_t> unrepData;
-   std::map<TString,std::map<Int_t,Bool_t> > doneRun; //keeps record of which data runs have been included in the global totals
-
-   for(std::map<Int_t, std::map<Int_t,std::pair<UInt_t,UInt_t> > >::iterator runPeriods = m_periods.begin();runPeriods != m_periods.end();++runPeriods) {
-      Int_t thisMCRunNumber = runPeriods->first;
-      for(std::map<Int_t,std::pair<UInt_t,UInt_t> >::iterator period = runPeriods->second.begin();period!=runPeriods->second.end();++period) {
-         Int_t periodNumber = period->first;
-         UInt_t pStart = period->second.first; UInt_t pEnd = period->second.second;
-         //for(std::map<TString,std::map<Int_t,TH1*> >::iterator dataHists = m_inputDataHistograms.begin();dataHists!=m_inputDataHistograms.end();++dataHists) {
-         for(std::map<TString,std::map<Int_t,std::map<Int_t,TH1*> > >::iterator dataHists = m_inputHistograms.begin();dataHists!=m_inputHistograms.end();++dataHists) {
-            TString weightName = dataHists->first;
-            //get list of loaded runs associated to this period 
-            std::vector<UInt_t> associatedRuns;
-            //std::map<Int_t,TH1*> hists = m_inputDataHistograms[weightName];
-            std::map<Int_t,TH1*> hists = (dataHists->second)[-1];
-            //look for loaded runs
-            for(UInt_t run=pStart;run<=pEnd;run++) {
-               if(hists.find(run)!=hists.end() && hists[run]) {
-                  associatedRuns.push_back(run);
-               }
-            }
-            //loop over data runs in this period and build up the data distributions
-            for(std::vector<UInt_t>::iterator run = associatedRuns.begin();run!=associatedRuns.end();++run) {
-               TH1* thisHist = hists[*run];
-               //look for unrepresented data in any of the bins of any of the corresponding channels
-               for(std::map<Int_t,std::map<Int_t,TH1*> >::iterator channels=m_inputHistograms[weightName].begin();channels!=m_inputHistograms[weightName].end();++channels) {
-                  if((channels->first)<0) continue; //skip data channels
-                  if((channels->second)[thisMCRunNumber]) {
-                     Int_t bin,binx,biny,binz;
-                     for(binz=1; binz<=thisHist->GetNbinsZ(); binz++) {
-                        for(biny=1; biny<=thisHist->GetNbinsY(); biny++) {
-                           for(binx=1; binx<=thisHist->GetNbinsX(); binx++) {
-                              bin = thisHist->GetBin(binx,biny,binz);
-                              Double_t value = thisHist->GetBinContent(bin);
-                              if(value==0.) continue;
-                              Double_t mcValue = (channels->second)[thisMCRunNumber]->GetBinContent(bin);
-                              if(mcValue==0. && !m_badbins[weightName][*run][bin]) { //we don't want to double-count as we loop over channels 
-                                 //unrepresented data bin 
-                                 m_badbins[weightName][*run][bin]=true;
-                                 unrepData[weightName] += value;
-                                 if(m_unrepresentedDataAction==0 || m_debugging) {
-                                    if(m_debugging) Error("Initialize","Unrepresented data at coords [%f,%f,%f] caused by mcrun %d in channel %d",thisHist->GetXaxis()->GetBinCenter(binx),thisHist->GetYaxis()->GetBinCenter(biny),thisHist->GetZaxis()->GetBinCenter(binz),thisMCRunNumber,channels->first);
-                                 }
-                                 if(m_unrepresentedDataAction==1) thisHist->SetBinContent(bin,0);
-                              }
-                           }
-                        }
-                     }
-                  }
-               }
-                  dataPeriodRunTotals[weightName][periodNumber][*run] += thisHist->GetSumOfWeights(); //for random runnumber
-                  dataPeriodRunTotals[weightName][thisMCRunNumber][*run] += thisHist->GetSumOfWeights(); //for random runnumber
-                  //also record the binned totals - wont this use up loads of memory!!?? Do only for "pileup" weight then
-                  if(weightName=="pileup") {
-                     for(int binNum=0;binNum<=thisHist->GetNbinsX();++binNum) {
-                        if(thisHist->GetBinContent(binNum)==0) continue;
-                        dataPeriodBinnedRunTotals["pileup"][binNum][periodNumber][*run]+=thisHist->GetBinContent(binNum);
-                        dataPeriodBinnedRunTotals["pileup"][binNum][thisMCRunNumber][*run]+=thisHist->GetBinContent(binNum);
-                        //keep track of totals in the run=0 slot - used by GetRandomRunNumber
-                        dataPeriodBinnedRunTotals["pileup"][binNum][periodNumber][0]+=thisHist->GetBinContent(binNum);
-                        dataPeriodBinnedRunTotals["pileup"][binNum][thisMCRunNumber][0]+=thisHist->GetBinContent(binNum);
-                     }
-                  }
-                  //don't double-count the runs. we track the run totals in dataPeriodRunTotals["pileup"][-1]
-                  if(weightName=="pileup" && dataPeriodRunTotals["pileup"][-1].find(*run)==dataPeriodRunTotals["pileup"][-1].end()) {
-                     dataPeriodRunTotals["pileup"][-1][*run]=thisHist->GetSumOfWeights();
-                     int lumiVectorPosition = -1;
-                     int runNbr = *run;
-                     if(runNbr>=177986&&runNbr<=178109) lumiVectorPosition=0;//period B
-                     else if(runNbr>=179710&&runNbr<=180481) lumiVectorPosition=1;//period D
-                     else if(runNbr>=180614&&runNbr<=180776) lumiVectorPosition=2;//period E
-                     else if(runNbr>=182013&&runNbr<=182519) lumiVectorPosition=3;//period F
-                     else if(runNbr>=182726&&runNbr<=183462) lumiVectorPosition=4;//period G
-                     else if(runNbr>=183544&&runNbr<=184169) lumiVectorPosition=5;//period H
-                     else if(runNbr>=185353&&runNbr<=186493) lumiVectorPosition=6;//period I
-                     else if(runNbr>=186516&&runNbr<=186755) lumiVectorPosition=7;//period J
-                     else if(runNbr>=186873&&runNbr<=187815) lumiVectorPosition=8;//period K
-                     else if(runNbr>=188902&&runNbr<=190343) lumiVectorPosition=9; //period L
-                     else if(runNbr>=190503&&runNbr<=191933) lumiVectorPosition=10; //period M
-                     if(lumiVectorPosition>=0) { m_integratedLumiVector[lumiVectorPosition]+=(thisHist->GetSumOfWeights())/1E6; }
-                  }
-                  //we will manage the global total addition here... to prevent double counting when config file loaded overlapping periods
-                  //NOTE: May want to check at AddConfigFile stage if config file contains conflicting period config.... need to decide if it is ok or not!!??
-                  if(!doneRun[weightName][*run]) {
-                     globalTotals[weightName][-1] += thisHist->GetSumOfWeights();
-                     globalNumberOfEntries[weightName][-1] += thisHist->GetEntries();
-                     doneRun[weightName][*run] = true;
-                  }
-                  //factorize out the distribution and add it to existing distributions
-                  //include this distribution in the mcRunNumber distribution and in the global total (only if mcRunNumber is a genuine run num. Don't want to double-count)
-                  FactorizeDistribution(thisHist,weightName,-1,periodNumber,true,false/* !newMCRuns[thisMCRunNumber]*/);
-            } //end loop over data in period
-         } //loop over weights
-      } //loop over periods in mcrun
-   } //loop over mcruns
+   if(ignoredData>0.) Warning("Initialize", "Period Assignments missed %f%% data",100.*ignoredData/totalData);
 
-   //we should also check if there are any data runs that are not covered by the period assignments 
-   for(std::map<TString, std::map<Int_t,std::map<Int_t, TH1*> > >::iterator weights=m_inputHistograms.begin();weights!=m_inputHistograms.end();++weights) {
-      Double_t ignoredData = 0;
-      if(weights->second.find(-1)==weights->second.end()) continue; //no data
-      for(std::map<Int_t,TH1*>::iterator runs = weights->second[-1].begin();runs!=weights->second[-1].end();++runs) {
-         //look for period assignment for run 
-         Int_t periodNum=-1;
-         for(std::map<Int_t, std::map<Int_t, std::pair<UInt_t,UInt_t> > >::iterator mcruns = m_periods.begin();mcruns!=m_periods.end();++mcruns) {
-            for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=mcruns->second.begin();periods!=mcruns->second.end();++periods) {
-               if((UInt_t)runs->first >= periods->second.first && (UInt_t)runs->first <= periods->second.second) {
-                  periodNum = periods->first;break;
-               }
-            }
-            if(periodNum!=-1) break;
-         }
-         if(periodNum==-1) {
-            //this run isn't assigned to a period. Give warning 
-            Warning("Initialize","%s weight contains loaded data in run %d that is not covered by period assignments",weights->first.Data(),runs->first);
-            ignoredData += (runs->second)->GetSumOfWeights();
+
+
+   if(unrepData) {
+      double frac = unrepData/totalData;
+      if( frac  > m_unrepDataTolerance) {
+               Error("Initialize", "%f%% unrepresented data, which suggests something is wrong with your prw config. Try EnableDebugging(true) to investigate",100.* (unrepData/m_periods[-1]->sumOfWeights[-1]));
+      }
+      if(m_unrepresentedDataAction==1) {
+         Warning("Initialize","has %f%% unrepresented data. This was removed (UnrepresentedDataAction=1)",100.*frac);
+      }  else if(m_unrepresentedDataAction==2) {
+         Warning("Initialize","has %f%% unrepresented data. This was kept in (UnrepresentedDataAction=2)",100.*frac);
+      } else if(m_unrepresentedDataAction==3) {
+         Warning("Initialize","has %f%% unrepresented data. This was reassigned (UnrepresentedDataAction=3)",100.*frac);
+      } else if(m_unrepresentedDataAction==0) {
+         Error("Initialize","has %f%% unrepresented data:",100.*frac);
+         //print the report of which channels caused it 
+         for(auto& it : unrepDataByChannel) {
+            Error("Initialize","   Channel %d caused %f%% of the unrepresented data (nEntries=%d)",it.first,100.*it.second/unrepData,m_periods[-1]->numberOfEntries[it.first]);
          }
+         Error("Initialize","Exiting. You must decide how to proceed...");
+         Error("Initialize","1) use AddPeriod or RemapPeriod to redefine the mc periods to include this data. You should not need to regenerate your mc config file");
+         Error("Initialize","2) use SetUnrepresentedDataAction(1) to remove this data from the weight calculations. You should also veto such data events (using IsUnrepresentedData(..,..) method)");
+         Error("Initialize","3) use SetUnrepresentedDataAction(2) to leave this data in the calculation. I hope you know what you're doing!!");
+         Error("Initialize","4) use SetUnrepresentedDataAction(3) to reassign the data to the nearest representable bin");
+         throw std::runtime_error("Throwing exception 22: Unrepresented data exists. Make a choice how to handle this. See err log for details");
       }
-      if(ignoredData>0.) Warning("Initialize", "Period Assignments missed %f data in %s weight",ignoredData,weights->first.Data());
-   }
-
-   bool displayOptions = false;
-   bool toleranceViolated = false;
-   for(std::map<TString,Double_t>::iterator it=unrepData.begin();it!=unrepData.end();++it) {
-      if(it->second>0.) {
-            //check if unrepresented data violates tolerance criteria, regardless of action mode 
-           if((it->second/globalTotals[it->first][-1]) > m_unrepDataTolerance) {
-               Error("Initialize", "%s weight has %f%% unrepresented data, which suggests something is wrong with your prw config. Try EnableDebugging(true) to investigate",it->first.Data(),100.*it->second/globalTotals[it->first][-1]);
-               toleranceViolated=true;
-           }
-           if(m_unrepresentedDataAction==0 || m_debugging) {
-               Error("Initialize","%s weight has %f%% unrepresented data",it->first.Data(),100.*it->second/globalTotals[it->first][-1]);
-               //sweep back over the bad bins and add up the unrepresented data grouped by channel
-               std::map<Int_t,Double_t> unrepChannelTotals;
-               for(std::map<Int_t, std::map<Int_t, Bool_t> >::iterator badRuns=m_badbins[it->first].begin();badRuns!=m_badbins[it->first].end();++badRuns) {
-                  //find out which mcrunNum this run belongs to 
-                  Int_t mcrunNum=-1;
-                  for(std::map<Int_t, std::map<Int_t, std::pair<UInt_t,UInt_t> > >::iterator mcruns = m_periods.begin();mcruns!=m_periods.end();++mcruns) {
-                     for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::iterator periods=mcruns->second.begin();periods!=mcruns->second.end();++periods) {
-                        if((UInt_t)badRuns->first >= periods->second.first && (UInt_t)badRuns->first <= periods->second.second) {
-                           mcrunNum = mcruns->first;break;
-                        }
-                     }
-                     if(mcrunNum!=-1) break;
-                  }
-                  //now sweep over mc channels and check this periodNum for the bad bins 
-                  for(std::map<Int_t, std::map<Int_t, TH1*> >::iterator channels=m_inputHistograms[it->first].begin();channels!=m_inputHistograms[it->first].end();++channels) {
-                     if(channels->first < 0) continue;
-                     for(std::map<Int_t, Bool_t>::iterator bins=badRuns->second.begin();bins!=badRuns->second.end();++bins) {
-                        if(bins->second && (channels->second)[mcrunNum]->GetBinContent(bins->first)==0.) {
-                           //found one of the bad bins. 
-                           unrepChannelTotals[channels->first] += m_inputHistograms[it->first][-1][badRuns->first]->GetBinContent(bins->first);
-                        }
-                     }
-                  }
-               }
-               //display the results of the scan 
-               for(std::map<Int_t,Double_t>::iterator badChans = unrepChannelTotals.begin();badChans!=unrepChannelTotals.end();++badChans) {
-                  if(m_unrepresentedDataAction!=0 && (badChans->second*100./it->second)<0.000001) continue; //don't display the tiny channel errors if just debugging
-                  //quickly count up the number of entries in this channel
-                  long nEntries(0);
-                  for(std::map<Int_t,TH1*>::iterator cIt = m_inputHistograms[it->first][badChans->first].begin();cIt!=m_inputHistograms[it->first][badChans->first].end();++cIt) {
-                     nEntries += cIt->second->GetEntries();
-                  }
-                  Error("Initialize","   Channel %d has %f%% of the unrepresented (nEntries=%ld)",badChans->first,badChans->second*100./it->second,nEntries);
-               }
-               if(m_unrepresentedDataAction==0) displayOptions=true;
-           } else if(m_unrepresentedDataAction==1) {
-               Warning("Initialize","%s weight has %f%% unrepresented data. This was removed (UnrepresentedDataAction=1)",it->first.Data(),100.*it->second/(globalTotals[it->first][-1]+it->second));
-           } else if(m_unrepresentedDataAction==2) {
-               Warning("Initialize","%s weight has %f%% unrepresented data. This was kept in (UnrepresentedDataAction=2)",it->first.Data(),100.*it->second/(globalTotals[it->first][-1]));
-           }
+      //if get here, through the exception if frac is bad
+      if( frac  > m_unrepDataTolerance) {
+         throw std::runtime_error("Throwing exception 222: Some channel had too much unrepresented data. You should fix your prw file");
       }
    }
 
-   if(toleranceViolated) {
-      Error("Initialize","Some channel had over %f%% of unrepresented data. You should fix your prw file",m_unrepDataTolerance*100.);
-      throw std::runtime_error("Throwing exception 222: Some channel had too much unrepresented data. You should fix your prw file");
-   }
 
-   if(displayOptions) {
-      Error("Initialize","Exiting. You must decide how to proceed...");
-      Error("Initialize","1) use AddPeriod or MergeMCRunNumbers to redefine the mc periods to include this data. You should not need to regenerate your mc config file");
-      Error("Initialize","2) use SetUnrepresentedDataAction(1) to remove this data from the weight calculations. You should also veto such data events (using IsUnrepresentedData(..,..) method)");
-      Error("Initialize","3) use SetUnrepresentedDataAction(2) to leave this data in the calculation. I hope you know what you're doing!!");
-      throw std::runtime_error("Throwing exception 22: Unrepresented data exists. Make a choice how to handle this. See err log for details");
-   }
-
-   if(m_debugging) Info("Initialize","Normalizing 1D histograms...");
+   if(m_debugging) Info("Initialize","Normalizing histograms and cleaning up...");
    //now that all the distributions are built. Normalize them all 
-   for(std::map<TString, std::map<Int_t, std::map<Int_t, TH1D*> > >::iterator weights=primaryDistributions.begin();weights!=primaryDistributions.end();++weights) {
-      for(std::map<Int_t, std::map<Int_t, TH1D*> >::iterator channels=weights->second.begin();channels!=weights->second.end();++channels) {
-         for(std::map<Int_t,TH1D*>::iterator mcrunsAndPeriods=channels->second.begin();mcrunsAndPeriods!=channels->second.end();++mcrunsAndPeriods) {
-            normalizeHistogram(mcrunsAndPeriods->second);
-         }
+   for(auto period : m_periods) {
+      if(period.first != period.second->id) continue;
+      for(auto pHist : period.second->primaryHists) {
+         normalizeHistogram(pHist.second);
       }
-   }
-   if(m_debugging) Info("Initialize","Normalizing 2D histograms...");
-   for(std::map<TString, std::map<Int_t, std::map<Int_t, TH2D*> > >::iterator weights=secondaryDistributions.begin();weights!=secondaryDistributions.end();++weights) {
-      for(std::map<Int_t, std::map<Int_t, TH2D*> >::iterator channels=weights->second.begin();channels!=weights->second.end();++channels) {
-         for(std::map<Int_t,TH2D*>::iterator mcrunsAndPeriods=channels->second.begin();mcrunsAndPeriods!=channels->second.end();++mcrunsAndPeriods) {
-            normalizeHistogram(mcrunsAndPeriods->second);
-         }
+      for(auto pHist : period.second->secondaryHists) {
+         normalizeHistogram(pHist.second);
       }
-   }
-
-   if(m_debugging) Info("Initialize","Deleting input histograms...");
-   //to save space, delete all the input histograms 
-   for(std::map<TString, std::map<Int_t,std::map<Int_t, TH1*> > >::iterator it0=m_inputHistograms.begin();it0!=m_inputHistograms.end();++it0) {
-      for(std::map<Int_t,std::map<Int_t, TH1*> >::iterator it1=it0->second.begin();it1!=it0->second.end();++it1) {
-         for(std::map<Int_t, TH1*>::iterator it2=it1->second.begin();it2!=it1->second.end();++it2) {
-            delete it2->second;
-         }
-         it1->second.clear();
+      for(auto pHist : period.second->inputHists) {
+         delete pHist.second;
       }
-      it0->second.clear();
+      period.second->inputHists.clear();
    }
-   m_inputHistograms.clear();
+
+   //we keep the inputHists for data, because can use that to do random run numbers based on mu
+
    if(m_debugging) Info("Initialize","...Done");
    //if no input histograms were added, we are in counting mode
-   if(m_countingMode) {
-      Info("Initialize","In Config File Generating mode. Remember to call WriteToFile!");
-     //histograms are instantiated in the event loop as the channels occur
-      //but since we are here, check that the user has at least defined some periods
-      if(m_periods.size()==0) {
-         Error("Initialize", "In Config File Generating mode, but no periods have been defined. This makes no sense!? Define some periods please (with AddPeriod method)!");
-         throw std::runtime_error("Throwing 43: In Config File Generating mode, but no periods have been defined. This makes no sense!? Define some periods please (with AddPeriod method)!");
-      }
-   }
+   
 
    m_isInitialized=true;
    
   return 0;
 }
 
-Bool_t Root::TPileupReweighting::IsUnrepresentedData(const TString weightName, Int_t runNumber, Float_t x, Float_t y, Float_t z) {
-   if(m_emptyHistograms.find(weightName)==m_emptyHistograms.end()) {
-      Error("IsUnrepresentedData", "Unknown weight %s", weightName.Data());
-      throw std::runtime_error("Throwing 23: Unknown weight");
-   }
-   TH1* hist = m_emptyHistograms[weightName];
-   Int_t bin = hist->GetXaxis()->FindFixBin(x);
-   if(hist->GetDimension()==2) {
-      Int_t biny = hist->GetYaxis()->FindFixBin(y);
-      Int_t nx = hist->GetXaxis()->GetNbins()+2;
-      bin += nx*biny;
-   }
-   if(hist->GetDimension()==3) {
-      Int_t biny = hist->GetYaxis()->FindFixBin(y);
-      Int_t nx = hist->GetXaxis()->GetNbins()+2;
-      Int_t ny = hist->GetYaxis()->GetNbins()+2;
-      Int_t binz = hist->GetZaxis()->FindFixBin(z);
-      bin += nx*(biny+ny*binz);
-   }
-   return m_badbins[weightName][runNumber][bin];
-}
+//loop over the mc histograms, find bin closest to given bin that is "ok" (has all nonzero)
+//0 = ok bin, 1 = bad bin, 2 = out of range bin
+Int_t CP::TPileupReweighting::IsBadBin(Int_t periodNumber, Int_t bin) {
+
+   if(bin<0) return 2; //out of range 
+
+   Period* p = m_periods[periodNumber];
+   for(auto& inHist : p->inputHists) {
+      int channelNumber = inHist.first;
+      if(channelNumber<0) continue; //skip data hists 
+      if(bin > (inHist.second->GetNbinsX()+2)*(inHist.second->GetNbinsY()+2)*(inHist.second->GetNbinsZ()+2)) return 2; //definitely out of range by this point!
+      if(inHist.second->GetBinContent(bin)==0) return 1;
+   }
+   return 0;
 
-Bool_t Root::TPileupReweighting::IsUnrepresentedData(Int_t runNumber, Float_t x, Float_t y, Float_t z) {
-   return IsUnrepresentedData("pileup",runNumber,x,y,z);
 }
 
-Int_t Root::TPileupReweighting::WriteToFile(const TString fname) {
-   
-   if(!m_countingMode) {Warning("WriteToFile","Not in counting mode, so no file will be written");return 0;}
+//loop over the mc histograms, find bin closest to given bin that is "ok" (has all nonzero)
+Int_t CP::TPileupReweighting::GetNearestGoodBin(Int_t thisMCRunNumber, Int_t bin) {
 
+   int binDistance = 1;
+   do {
+      int res1 = IsBadBin(thisMCRunNumber,bin+binDistance);
+      if(!res1) return bin+binDistance;
+      int res2 = IsBadBin(thisMCRunNumber,bin-binDistance);
+      if(!res2) return bin-binDistance;
+      if(res1==2 && res2==2) { //run out of bins 
+         Error("GetNearestGoodBin", "None of the bins are good!!??");
+         return -1;
+      }
+      binDistance++;
+   } while(true); //scary!
 
-   //build a TTree with the correct mc structure, and dump the mc histogram info in to it
-   //also build aggregate histograms across all channels - shouldn't be used as input histograms unless for a single channel
-   TString filename = fname;
-   filename += (filename=="") ? TString(this->GetName()) + ".prw.root" : "";
-   
-   //loop over the weights. "pileup" gets it own MCPileupReweighting ttree. 
-   //data goes in to DataCustomReweighting. other goes in to MCCustomReweighting 
+   return -1;
 
-   TFile* outFile = TFile::Open(filename,"RECREATE");
-   Int_t r = WriteToFile(outFile);
-   outFile->Close();
-   return r;
 }
 
-Int_t Root::TPileupReweighting::WriteToFile(TFile* outFile) {
-   TDirectory* origDir = gDirectory;
-
-   outFile->cd();
+Double_t CP::TPileupReweighting::GetUnrepresentedDataFraction(Int_t periodNumber,Int_t channel) {
 
-   TTree *outTreeMC=0;
-   TTree *outTreeCustomMC=0;
-   TTree *outTreeData=0;
-   TTree *outTreeCustomData=0;
-   TTree *outTree = 0;//points to tree currently being written to
-   Int_t channel = 0;UInt_t runNumber = 0; 
-   std::vector<UInt_t>* pStarts = 0;std::vector<UInt_t>* pEnds = 0;
-   Char_t histName[150];Char_t customName[150];
+  //return unrepDataByChannel[channel]/GetSumOfEventWeights(-1); //the denominator is equivalent to asking for the total data!
 
-   bool lastChannelWasData = true;
 
-   for(std::map<TString, std::map<Int_t, std::map<Int_t, TH1*> > >::iterator weights=m_inputHistograms.begin();weights!=m_inputHistograms.end();++weights) {
-      TString weightName = weights->first;
-      strcpy(customName,weightName);
-      if(outTree) {
-         outTree=0;
-      }
-      for(std::map<Int_t, std::map<Int_t, TH1*> >::iterator channels=weights->second.begin();channels!=weights->second.end();++channels) {
-         channel = channels->first;
-         if(lastChannelWasData&&channel>=0) {
-            if(outTree) {outTree=0;}
-            lastChannelWasData=false;
-         }
-         if(channel<0) lastChannelWasData=true;
-         if(weightName=="pileup") {
-            if(channel>=0 && !outTree) {
-               if(!outTreeMC) {
-                  outTreeMC = new TTree("MCPileupReweighting","MCPileupReweighting");
-                  outTreeMC->Branch("Channel",&channel);
-                  outTreeMC->Branch("RunNumber",&runNumber);
-                  outTreeMC->Branch("PeriodStarts",&pStarts);
-                  outTreeMC->Branch("PeriodEnds",&pEnds);
-                  outTreeMC->Branch("HistName",&histName,"HistName/C");
-               }
-               outTree = outTreeMC;
-            } else if(channel<0 && !outTree) {
-               if(!outTreeData) {
-                  outTreeData = new TTree("DataPileupReweighting","DataPileupReweighting");
-                  outTreeData->Branch("RunNumber",&runNumber);
-                  outTreeData->Branch("HistName",&histName,"HistName/C");
-               }
-               outTree = outTreeData;
-            }
-         } else {
-            if(channel>=0 && !outTree) {
-               if(!outTreeCustomMC) {
-                  outTreeCustomMC = new TTree("MCCustomReweighting","MCCustomReweighting");
-                  outTreeCustomMC->Branch("CustomName",&customName,"CustomName/C");
-                  outTreeCustomMC->Branch("Channel",&channel);
-                  outTreeCustomMC->Branch("RunNumber",&runNumber);
-                  outTreeCustomMC->Branch("PeriodStarts",&pStarts);
-                  outTreeCustomMC->Branch("PeriodEnds",&pEnds);
-                  outTreeCustomMC->Branch("HistName",&histName,"HistName/C");
-               }
-               outTree = outTreeCustomMC;
-            } else if(channel<0 && !outTree) {
-               if(!outTreeCustomData) {
-                  outTreeCustomData = new TTree("DataCustomReweighting","DataCustomReweighting");
-                  outTreeCustomData->Branch("CustomName",&customName,"CustomName/C");
-                  outTreeCustomData->Branch("RunNumber",&runNumber);
-                  outTreeCustomData->Branch("HistName",&histName,"HistName/C");
-               }
-               outTree = outTreeCustomData;
-            }
-         }
-         for(std::map<Int_t, TH1*>::iterator runs=channels->second.begin();runs!=channels->second.end();++runs) {
-            runNumber = runs->first;
-            if(!lastChannelWasData) { //load periods only for mc 
-               pStarts = new std::vector<UInt_t>;pEnds = new std::vector<UInt_t>;
-               for(std::map<Int_t,std::pair<UInt_t,UInt_t> >::iterator periods=m_periods[runNumber].begin();periods!=m_periods[runNumber].end();++periods) {
-                  pStarts->push_back(periods->second.first);pEnds->push_back(periods->second.second);
-               }
-            }
-            strcpy(histName,runs->second->GetName());
-            outTree->Fill();
-            runs->second->Write();
-            if(!lastChannelWasData) { //delete periods only for mc 
-               delete pStarts;delete pEnds;
-            }
-         }
-      }
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetUnrepresentedDataFraction","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
-   //write the non-zero ttrees 
-   if(outTreeMC) {outTreeMC->Write();delete outTreeMC;outTreeMC=0;}
-   if(outTreeData) {outTreeData->Write();delete outTreeData;outTreeData=0;}
-   if(outTreeCustomMC) {outTreeCustomMC->Write();delete outTreeCustomMC;outTreeCustomMC=0;}
-   if(outTreeCustomData) {outTreeCustomData->Write();delete outTreeCustomData;outTreeCustomData=0;}
 
+   //look for channel through sumOfWeights - because if the channel has no unrepData, it will have no entry in the unrepData map
+   if(p->sumOfWeights.find(channel) == p->sumOfWeights.end()) channel = m_periods[periodNumber]->defaultChannel;
 
-   Info("WriteToFile", "Successfully generated config file: %s",outFile->GetName());
-   Info("WriteToFile", "Happy Reweighting :-)");
 
-   gDirectory = origDir;
+   if(p->sumOfWeights.find(channel) == p->sumOfWeights.end()) {
+      Error("GetUnrepresentedDataFraction","Unrecognised channel: %d",channel);
+      throw std::runtime_error("GetUnrepresentedDataFraction: Unrecognised channel");
+   }
 
-   return 0;
-}
+   return p->unrepData[channel]/p->sumOfWeights[-1];
 
-Double_t Root::TPileupReweighting::GetDataWeight(Int_t runNumber, TString trigger) {
-   //special mu-independent version of GetDataWeight. Will just use the global luminosity
-   m_doGlobalDataWeight=true;
-   double out = GetDataWeight(runNumber, trigger, 0);
-   m_doGlobalDataWeight=false;
-   return out;
 }
 
-Double_t Root::TPileupReweighting::GetDataWeight(Int_t runNumber, TString trigger, Double_t x) {
-   if(!m_isInitialized) { Error("GetDataWeight","Tool is not initialized"); return 0;}
-   if(m_countingMode) return 0.;
-   if(m_countingMode) {
-      Error("GetDataWeight","You cannot get a weight when in config file generating mode. Please use FillWeight method");
-      throw std::runtime_error("GetDataWeight: You cannot get a weight when in config file generating mode. Please use FillWeight method");
-   }
-   //determine which mc run number this is
-   Int_t periodNumber = GetFirstFoundPeriodNumber(runNumber);
-   if(periodNumber==-1) {
-      Error("GetDataWeight","Could not find corresponding period for runNumber %d",runNumber);
-      throw std::runtime_error("GetDataWeight: Could not find period for runNumber");
-   }
-   if(m_triggerPrimaryDistributions.find(trigger)==m_triggerPrimaryDistributions.end()) {
-      //try to do a subtrigger calculation we need to reopen all the lumicalc files and do the fiddily prescaled luminosity calculation
-      //will throw errors if can't
-      CalculatePrescaledLuminosityHistograms(trigger);
+
+
+UInt_t CP::TPileupReweighting::GetRandomRunNumber(Int_t periodNumber) {
+   if(!m_isInitialized) { Info("GetRandomRunNumber","Initializing the subtool.."); Initialize(); }
+   if(m_countingMode) { return 0; } //do nothing when in counting mode
+
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetRandomRunNumber","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
-   TH1D* denomHist = m_triggerPrimaryDistributions[trigger][periodNumber];
-   if(denomHist==0) {
-      Error("GetDataWeight","Could not find trigger %s",trigger.Data());
-      throw std::runtime_error("GetDataWeight: Could not find trigger");
+
+   double lumi = p->sumOfWeights[-1] * m_random3->Rndm();
+
+   //go through the runs assigned to this period ... 
+   double lumiSum=0;
+   for(auto runNum : p->runNumbers) {
+      lumiSum += m_runs[runNum].inputHists["None"]->GetSumOfWeights();
+      if(lumiSum >= lumi) return runNum;
    }
-   TH1D* numerHist = m_triggerPrimaryDistributions["None"][periodNumber];
-   if(numerHist==0) {
-      Error("GetDataWeight","Could not find unprescaled trigger");
-      throw std::runtime_error("GetDataWeight: Could not find unprescaled trigger. Please AddLumiCalc with a 'None' trigger");
+   Error("GetRandomRunNumber","overran integrated luminosity for periodNumber=%d",periodNumber);
+   throw std::runtime_error("Throwing 46.1: overran integrated luminosity for GetRandomRunNumber");
+   return 0;
+}
+
+UInt_t CP::TPileupReweighting::GetRandomRunNumber(Int_t periodNumber, Double_t x) {
+   if(!m_isInitialized) { Info("GetRandomRunNumber","Initializing the subtool.."); Initialize(); }
+   if(m_countingMode) { return 0; } //do nothing when in counting mode
+
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetRandomRunNumber","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
 
-   if(m_doGlobalDataWeight) return numerHist->Integral(0,numerHist->GetNbinsX()+1)/denomHist->Integral(0,denomHist->GetNbinsX()+1);
+   int bin = m_emptyHistogram->FindFixBin(x);
 
-   Int_t bin=numerHist->FindFixBin(x);
-   return numerHist->GetBinContent(bin)/denomHist->GetBinContent(bin);
+   double lumi = p->primaryHists[-1]->GetBinContent(bin) * p->sumOfWeights[-1] * m_random3->Rndm();
 
-}
+   if(!lumi) { return 0; } //no lumi with that mu
 
-/** alternative random runnumber method for people who want to use custom reweighting config files to determine the random numbers */
-//Could tidy up code by removing the version below, and defaulting weightName to "pileup"
-UInt_t Root::TPileupReweighting::GetRandomRunNumber(Int_t mcRunNumber, TString weightName) {
-   if(!m_isInitialized) { Error("GetRandomRunNumber","Tool is not initialized"); return 0;}
-   if(m_countingMode) { return 0; } //do nothing when in counting mode
-   //check if the runNumber has been reassigned
-   if(m_mcRemappings.find(mcRunNumber)!=m_mcRemappings.end()) mcRunNumber=m_mcRemappings[mcRunNumber];
-   const std::map<Int_t, Double_t>::const_iterator it = periodTotals[weightName][-1].find(mcRunNumber);
-   if(it==periodTotals[weightName][-1].end()|| it->second == 0) { 
-      if(m_SetWarnings) Warning("GetRandomRunNumber","%d has no associated custom data. Returning 0",mcRunNumber);
-      return 0; /*throw 33;*/
-   }
-   double lumi = it->second * m_random3->Rndm();
-    double lumisum = 0.;
-    std::map<UInt_t, Double_t>::const_iterator itend = dataPeriodRunTotals[weightName][mcRunNumber].end();
-    for(std::map<UInt_t, Double_t>::const_iterator it2 = dataPeriodRunTotals[weightName][mcRunNumber].begin(); it2 != itend; ++it2)
-      {
-	lumisum += it2->second;
-	if (lumisum >= lumi)
-	  return it2->first;
-      }
-    Error("GetRandomRunNumber","overran integrated luminosity for mcRunNumber=%d in custom weight=%s",mcRunNumber,weightName.Data());
-     throw std::runtime_error("Throwing 46.1: overran integrated luminosity for mcRunNumber in custom weight");
-    return 0;
-}
 
-/** generate random run number from lumicalc mu distributions */
-UInt_t Root::TPileupReweighting::GetRandomRunNumber(Int_t mcRunNumber) {
-   if(!m_isInitialized) { Error("GetRandomRunNumber","Tool is not initialized"); return 0;}
-   if(m_countingMode) { return 0; } //do nothing when in counting mode
-   //check if the runNumber has been reassigned
-   if(m_mcRemappings.find(mcRunNumber)!=m_mcRemappings.end()) mcRunNumber=m_mcRemappings[mcRunNumber];
-   const std::map<Int_t, Double_t>::const_iterator it = periodTotals["pileup"][-1].find(mcRunNumber);
-   if(it==periodTotals["pileup"][-1].end()|| it->second == 0) { 
-      if(m_SetWarnings) Warning("GetRandomRunNumber","%d has no associated lumi. Returning 0",mcRunNumber);
-      return 0; /*throw 33;*/
-   }
-   double lumi = it->second * m_random3->Rndm();
-    double lumisum = 0.;
-    std::map<UInt_t, Double_t>::const_iterator itend = dataPeriodRunTotals["pileup"][mcRunNumber].end();
-    for(std::map<UInt_t, Double_t>::const_iterator it2 = dataPeriodRunTotals["pileup"][mcRunNumber].begin(); it2 != itend; ++it2)
-      {
-	lumisum += it2->second;
-	if (lumisum >= lumi)
-	  return it2->first;
-      }
-    Error("GetRandomRunNumber","overran integrated luminosity for mcRunNumber=%d",mcRunNumber);
-     throw std::runtime_error("Throwing 46: overran integrated luminosity for mcRunNumber");
-    return 0;
+   //go through the runs assigned to this period ... 
+   double lumiSum=0;
+   for(auto runNum : p->runNumbers) {
+      lumiSum += m_runs[runNum].inputHists["None"]->GetBinContent(bin);
+      if(lumiSum >= lumi) return runNum;
+   }
+   Error("GetRandomRunNumber","overran integrated luminosity for periodNumber=%d",periodNumber);
+   throw std::runtime_error("Throwing 46.1: overran integrated luminosity for GetRandomRunNumber");
+   return 0;
 }
 
-///return random run numbers but only for runs where the mu is present. If the mu has no associated data, returns 0
-UInt_t Root::TPileupReweighting::GetRandomRunNumber(Int_t mcRunNumber, Double_t mu) {
-   if(!m_isInitialized) { Error("GetRandomRunNumber","Tool is not initialized"); return 0;}
+UInt_t CP::TPileupReweighting::GetRandomLumiBlockNumber(UInt_t runNumber) {
+   if(!m_isInitialized) {Info("GetRandomLumiBlockNumber","Initializing the subtool.."); Initialize(); }
    if(m_countingMode) { return 0; } //do nothing when in counting mode
 
-   //determine the mubin
-   TH1* hist = m_emptyHistograms["pileup"];
-   if(!hist) { Error("GetRandomRunNumber", "Cannot do this without a lumicalc file!"); return 0; }
-   Int_t muBin = hist->FindFixBin(mu);
-
-   //check if the runNumber has been reassigned
-   if(m_mcRemappings.find(mcRunNumber)!=m_mcRemappings.end()) mcRunNumber=m_mcRemappings[mcRunNumber];
+   double lumi = GetIntegratedLumi(runNumber,runNumber) * m_random3->Rndm() * 1E6 /* dont forget the lumi was divided by a million to get to pb */; 
+   double lumisum = 0.;
 
-   //check that this mcRunNumber actually has some lumi with this mu
-   std::map<UInt_t, std::map<Int_t, std::map<UInt_t, Double_t> > >::iterator myMap1 = dataPeriodBinnedRunTotals["pileup"].find(muBin);
-   if(myMap1 == dataPeriodBinnedRunTotals["pileup"].end()) {
-      //no lumi with this mu
-      return 0;
-   }
-   std::map<Int_t, std::map<UInt_t, Double_t> >::iterator myMap2 = myMap1->second.find(mcRunNumber);
-   if(myMap2 == myMap1->second.end()) {
-      //no lumi with this mu in this mcRunNumber
-      return 0;
+   for(auto& lbn : m_runs[runNumber].lumiByLbn) {
+      if(m_unrepresentedDataAction==1 && IsUnrepresentedData(runNumber,lbn.second.second)) continue; //skip over bad lumi 
+      lumisum += lbn.second.first;
+      if(lumisum >= lumi) return lbn.first;
    }
-
-   //the total lumi is stored in the run=0 slot of this map
-   double lumi = myMap2->second[0] * m_random3->Rndm();
-    double lumisum = 0.;
-    std::map<UInt_t, Double_t>::const_iterator itend = myMap2->second.end();
-    for(std::map<UInt_t, Double_t>::const_iterator it2 = myMap2->second.begin(); it2 != itend; ++it2)
-      {
-         //skip case of it2->first=0 (which is the total lumi)
-         if(it2->first==0) continue;
-	lumisum += it2->second;
-	if (lumisum >= lumi)
-	  return it2->first;
-      }
-    Error("GetRandomRunNumber","overran integrated luminosity for mcRunNumber=%d, mubin=%d",mcRunNumber,muBin);
-     throw std::runtime_error("Throwing 46: overran integrated luminosity for mcRunNumber");
-    return 0;
+   Error("GetRandomLumiBlockNumber","overran integrated luminosity for RunNumber=%d",runNumber);
+   throw std::runtime_error("Throwing 46: overran integrated luminosity for runNumber");
+   return 0;
 }
 
-Int_t Root::TPileupReweighting::GetRandomPeriodNumber(Int_t mcRunNumber) {
-   if(!m_isInitialized) { Error("GetRandomPeriodNumber","Tool is not initialized"); return 0;}
+//only considers periods assigned directly!
+Int_t CP::TPileupReweighting::GetRandomPeriodNumber(Int_t periodNumber) {
+   if(!m_isInitialized) { Info("GetRandomPeriodNumber","Initializing the subtool.."); Initialize(); }
    if(m_countingMode) { return 0; } //do nothing when in counting mode
-   //check if the runNumber has been reassigned
-   if(m_mcRemappings.find(mcRunNumber)!=m_mcRemappings.end()) mcRunNumber=m_mcRemappings[mcRunNumber];
-   
-   if(m_periods.find(mcRunNumber)==m_periods.end()) {
-      Error("GetRandomPeriodNumber","Unknown MC RunNumber: %d",mcRunNumber);
-      throw std::runtime_error("Throwing 46: Unknown MC RunNumber");
-   }
 
-   //if this runnumber has only one period, return itself
-   if(m_periods[mcRunNumber].size()==1) {
-      return mcRunNumber;
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetRandomPeriodNumber","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
 
-   const std::map<Int_t, Double_t>::const_iterator it = periodTotals["pileup"][-1].find(mcRunNumber);
-   if(it==periodTotals["pileup"][-1].end()) { 
-      if(m_SetWarnings) Warning("GetRandomPeriodNumber","%d has no associated lumi. Returning 0",mcRunNumber);
-      return 0; /*throw 33;*/
+   //if no subperiods, we just return ourselves 
+   if(p->subPeriods.size()==0) return p->id;
+
+   //need the total lumi of this period
+   double lumi = p->sumOfWeights[-1] * m_random3->Rndm();
+
+   //loop over subPeriods .. 
+   double lumiSum=0;
+   for(auto subp : p->subPeriods) {
+      lumiSum += subp->sumOfWeights[-1];
+      if(lumiSum >= lumi) return subp->id;
    }
-   Double_t lumi = it->second * m_random3->Rndm();
-   Double_t lumisum = 0.;
-   for(std::map<Int_t, std::pair<UInt_t,UInt_t> >::const_iterator it2 = m_periods[mcRunNumber].begin(); it2 != m_periods[mcRunNumber].end(); ++it2) {
-      lumisum += periodTotals["pileup"][-1][it2->first];
-      if (lumisum >= lumi)
-	  return it2->first;
-      }
-    Error("GetRandomPeriodNumber","overran integrated luminosity for mcRunNumber=%d",mcRunNumber);
-     throw std::runtime_error("Throwing 46: overran integrated luminosity for mcRunNumber");
-    return 0;
+
+   Error("GetRandomPeriodNumber","overran integrated luminosity for periodNumber=%d",periodNumber);
+   throw std::runtime_error("Throwing 46.1: overran integrated luminosity for GetRandomPeriodNumber");
+   return 0;
+
 }
 
-UInt_t Root::TPileupReweighting::GetRandomLumiBlockNumber(UInt_t runNumber) {
-   if(!m_isInitialized) { Error("GetRandomRunNumber","Tool is not initialized"); return 0;}
-   if(m_countingMode) { return 0; } //do nothing when in counting mode
+Bool_t CP::TPileupReweighting::IsUnrepresentedData(Int_t runNumber, Float_t x, Float_t y) {
+   int bin = m_emptyHistogram->FindFixBin(x,y);
+   return ( m_runs[runNumber].badBins[bin] );
+}
 
-   double lumi = GetIntegratedLumi(runNumber,runNumber) * m_random3->Rndm();
-   double lumisum = 0.;
-   std::map<UInt_t, std::map<UInt_t, Double_t> >::iterator it = m_lumiByRunAndLbn.find(runNumber);
-   if(it==m_lumiByRunAndLbn.end()) {
-      Error("GetRandomLumiBlockNumber","Unknown run number: %d",runNumber);
-      throw std::runtime_error("Throwing 46.1: Unknown Run number");
+//this method builds a file get can be friended to a TTree with a prw-hash branch
+Bool_t CP::TPileupReweighting::MakeWeightTree(TString channelNumbers, TString outFile, TString hashBranch, TString weightBranch) {
+   if(!m_isInitialized) { Info("MakeWeightTree","Initializing the subtool.."); Initialize(); }
+   TH1* hist = m_emptyHistogram;
+   if(!hist) {
+      Error("MakeWeightTree","Tool not configured properly ... please report this!");
+      throw std::runtime_error("Throwing 47: Tool not configured properly ... please report this!");
    }
-   std::map<UInt_t, Double_t>::const_iterator itend = it->second.end();
-   for(std::map<UInt_t, Double_t>::const_iterator it2 = it->second.begin(); it2 != itend; ++it2) {
-      lumisum += it2->second;
-      if (lumisum >= lumi) return it2->first;
+
+   //loop over given channels, and loop over all periods (except the global period) and get the pileup weight in each case for each bin 
+   TFile f1(outFile,"RECREATE");
+
+   TTree* outTree = new TTree("prwTree","prwTree");
+   ULong64_t prwHash(0);Float_t weight(0.);
+   outTree->Branch(hashBranch,&prwHash);
+   outTree->Branch(weightBranch,&weight);
+
+   TObjArray *tx = channelNumbers.Tokenize(",");
+   for (Int_t i = 0; i < tx->GetEntries(); i++) {
+      int channelNumber = ((TObjString *)(tx->At(i)))->String().Atoi();
+      if(channelNumber>999999 || channelNumber<0) { 
+         Error("MakeWeightTree","ChannelNumber can not be bigger than 999999 or less than 0 ... got %d",channelNumber);
+         f1.Close();
+         return 0;
+      }
+      for(auto& period : m_periods) {
+         if(period.first==-1) continue;
+         int periodNumber = period.first;
+         for(int i=1;i<=hist->GetNbinsX();i++) {
+            double x = hist->GetXaxis()->GetBinCenter(i);
+            for(int j=1;j<=hist->GetNbinsY();j++) {
+               double y = hist->GetYaxis()->GetBinCenter(j);
+               weight = GetCombinedWeight(periodNumber,channelNumber,x,y);
+               prwHash = GetPRWHash(periodNumber,channelNumber,x,y);
+               outTree->Fill();
+            }
+         }
+
+      }
    }
-   Error("GetRandomLumiBlockNumber","overran integrated luminosity for RunNumber=%d",runNumber);
-   throw std::runtime_error("Throwing 46: overran integrated luminosity for runNumber");
-   return 0;
+
+   outTree->BuildIndex(hashBranch.Data());
+   outTree->Write();
+   f1.Close();
+
+   Info("MakeWeightTree","Successfully wrote prwTree to %s",outFile.Data());
+
+   return true;
 }
 
-Float_t Root::TPileupReweighting::getPileupWeight(Float_t mu, Int_t runNumber, Int_t channel ) {
-   if(!m_isInitialized) { Error("getPileupWeight","Tool is not initialized"); return 0;}
-   if(m_SetWarnings) Warning("getPileupWeight","Depricated. Please use GetCombinedWeight(runNumber,channelNumber,mu)");
-   return GetCombinedWeight(runNumber,channel,mu);
+ULong64_t CP::TPileupReweighting::GetPRWHash(Int_t periodNumber, Int_t channelNumber, Float_t x, Float_t y) {
+   if(!m_isInitialized) { Info("GetPRWHash","Initializing the subtool.."); Initialize(); }
+   TH1* hist = m_emptyHistogram;
+   if(!hist) {
+      Error("GetPRWHash","Tool not configured properly ... please report this!");
+      throw std::runtime_error("Throwing 47: Tool not configured properly ... please report this!");
+   }
+
+   ULong64_t out = hist->FindFixBin(x,y);
+   out += (unsigned long)(channelNumber)*10000000000;
+   out += (unsigned long)(periodNumber)*10000;
+
+   return out;
+
 }
 
-Float_t Root::TPileupReweighting::GetCombinedWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber, Float_t x, Float_t y, Float_t /*z*/) {
-   if(!m_isInitialized) { Error("GetCombinedWeight","Tool is not initialized"); return 0;}
+Float_t CP::TPileupReweighting::GetCombinedWeight(Int_t periodNumber, Int_t channelNumber, Float_t x, Float_t y) {
+   if(!m_isInitialized) { Info("GetCombinedWeight","Initializing the subtool.."); Initialize(); }
    if(m_countingMode) return 0.;
    if(m_countingMode) {
       Error("GetCombinedWeight","You cannot get a weight when in config file generating mode. Please use FillWeight method");
       throw std::runtime_error("Throwing 2: You cannot get a weight when in config file generating mode. Please use FillWeight method");
    }
    //decide how many dimensions this weight has - use the emptyHistogram to tell... 
-   TH1* hist = m_emptyHistograms[weightName];
+   TH1* hist = m_emptyHistogram;
    if(!hist) {
-      Error("GetCombinedWeight","Weight not recognised: %s",weightName.Data());
+      Error("GetCombinedWeight","Tool not configured properly ... please report this!");
+      throw std::runtime_error("Throwing 47: Tool not configured properly ... please report this!");
    }
 
-   //get the PeriodWeight for the associated runNumber - NOTE: Delayed implementing ... see Recipe H... check dependence... 09/11/2012 - looks like H was wrong to apply IntegratedLumiFraction and old CombinedWeight... since old CombinedWeight used the periodNumber's periodWeight, which appears to include the fraction already
-   Int_t mcRunNumber = periodNumber;
-   std::map<Int_t,Int_t>::iterator it = m_periodToMCRun.find(periodNumber);
-   if(it!=m_periodToMCRun.end()) mcRunNumber = it->second;
-   Float_t out = GetPeriodWeight(weightName,mcRunNumber/*periodNumber*/,channelNumber)*GetPrimaryWeight(weightName,periodNumber,channelNumber,x);
-   if(hist->GetDimension()>1) out *= GetSecondaryWeight(weightName,periodNumber,channelNumber,x,y);
-   //if(hist->GetDimension()>2) out *= GetTertiaryWeight(weightName,periodNumber,channelNumber,x,y,z);
-
+   Float_t out = GetPeriodWeight(periodNumber,channelNumber)*GetPrimaryWeight(periodNumber,channelNumber,x);
+   if(hist->GetDimension()>1) out *= GetSecondaryWeight(periodNumber,channelNumber,x,y);
+   
    return out;
 }
 
-Float_t Root::TPileupReweighting::GetPeriodWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber) {
-   if(!m_isInitialized) { Error("GetPeriodWeight","Tool is not initialized"); return 0;}
+Float_t CP::TPileupReweighting::GetPeriodWeight(Int_t periodNumber, Int_t channelNumber) {
+   //= L_A/L / N_A/N  
+   if(!m_isInitialized) { Info("GetPeriodWeight","Initializing the subtool.."); Initialize(); }
    if(m_countingMode) return 0.;
-   //check if the runNumber has been reassigned
-   //only do this for mc, never for data
-   Int_t inPeriodNumber = periodNumber; //hold for getting the default channel if necessary
-   if(channelNumber>=0) {
-      if(m_mcRemappings.find(periodNumber)!=m_mcRemappings.end()) periodNumber=m_mcRemappings[periodNumber];
-   }
 
-   if(periodTotals[weightName][-1].find(periodNumber)==periodTotals[weightName][-1].end()) {
-      Error("GetPeriodWeight","Unrecognised periodNumber/mcRunNumber: %d",periodNumber);
-      throw std::runtime_error("Throwing 1: Unrecognised periodNumber/mcRunNumber");
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetPeriodWeight","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
+   if(p->sumOfWeights.find(channelNumber) == p->sumOfWeights.end()) channelNumber = p->defaultChannel;
 
-   Double_t data = periodTotals[weightName][-1][periodNumber];
-   Double_t totalData = globalTotals[weightName][-1];
+   double n_a = p->sumOfWeights[channelNumber];
+   double n = m_periods[-1]->sumOfWeights[channelNumber];
 
-   if(totalData==0.) {
-      Error("GetPeriodWeight","No data for weight %s",weightName.Data());
-      throw std::runtime_error("Throwing 2: No data for weight");
-   }
+   double l_a = p->sumOfWeights[-1];
+   double l = m_periods[-1]->sumOfWeights[-1];
 
-   if(periodTotals[weightName].find(channelNumber)==periodTotals[weightName].end()) {
-      //use the default
-      if(periodTotals[weightName].find(GetDefaultChannel(inPeriodNumber))==periodTotals[weightName].end()) {
-         Error("GetPeriodWeight","Unrecognised channel %d and no default found: %d",channelNumber,GetDefaultChannel(inPeriodNumber));
-         throw std::runtime_error("Throwing 3: Unrecognised channel");
-      } 
-      channelNumber=GetDefaultChannel(inPeriodNumber);
-   }
-   if(periodTotals[weightName][channelNumber].find(periodNumber)==periodTotals[weightName][channelNumber].end()) {
-      Error("GetPeriodWeight","Unrecognised periodNumber/mcRunNumber %d in channelNumber %d",periodNumber,channelNumber);
-      throw std::runtime_error("Throwing 1: Unrecognised periodNumber/mcRunNumber");
-   }
+   return (l_a/l) / (n_a/n);
+}
 
-   Double_t mc = periodTotals[weightName][channelNumber][periodNumber];
-   Double_t totalMC = globalTotals[weightName][channelNumber];
+Float_t CP::TPileupReweighting::GetPrimaryWeight(Int_t periodNumber, Int_t channelNumber,Float_t x) {
+   //= L_i/L_A / N_i/N_A .. primaryHists have already been normalized
+   if(!m_isInitialized) { Info("GetPrimaryWeight","Initializing the subtool.."); Initialize(); }
+   if(m_countingMode) return 0.;
 
-   if(totalMC==0.) {
-      if(m_SetWarnings) Warning("GetPeriodWeight","No mc for weight '%s' in channelNumber %d",weightName.Data(),channelNumber);
-      return 0.;
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetPrimaryWeight","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
-   if(mc==0.) {
-      if(m_SetWarnings) Warning("GetPeriodWeight","No mc for weight '%s' in channelNumber %d and periodNumber/runNumber %d",weightName.Data(),channelNumber,periodNumber);
-      return 0.;
+   int oChanNumber = channelNumber;
+   if(p->sumOfWeights.find(channelNumber) == p->sumOfWeights.end()) channelNumber = p->defaultChannel;
+
+   if(!p->primaryHists[channelNumber]) {
+      Error("GetPrimaryWeight","Unrecognised channelNumber: %d",oChanNumber);
+      throw std::runtime_error("Throwing 2: Unrecognised channelNumber");
    }
 
-   return (data/totalData) / (mc/totalMC) ;
+   int bin = p->primaryHists[channelNumber]->FindFixBin(x);
 
-}
+   double n = p->primaryHists[channelNumber]->GetBinContent(bin);
 
-Float_t Root::TPileupReweighting::GetPrimaryWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber, Float_t x) {
-   if(!m_isInitialized) { Error("GetPrimaryWeight","Tool is not initialized"); return 0;}
-   if(m_countingMode) return 0.;
-   //check if the runNumber has been reassigned
-   //only do this for mc, never for data
-   Int_t inPeriodNumber = periodNumber;
-   if(channelNumber>=0) {
-      if(m_mcRemappings.find(periodNumber)!=m_mcRemappings.end()) periodNumber=m_mcRemappings[periodNumber];
+   if(!p->primaryHists[-1]) {
+      Error("GetPrimaryWeight","No data loaded for period %d. Did you forget to load a lumicalc file or data config file?",periodNumber);
+      throw std::runtime_error("Throwing 3: No data loaded. Did you forget to load a lumicalc file or data config file?");
    }
 
-   //determine bin number 
-   TH1D* data = primaryDistributions[weightName][-1][periodNumber];
-   if(!data) {
-      Error("GetPrimaryWeight","Unrecognised periodNumber/mcRunNumber: %d",periodNumber);
-      throw std::runtime_error("Throwing 4: Unrecognised periodNumber/mcRunNumber");
-   }
-   if(primaryDistributions[weightName].find(channelNumber)==primaryDistributions[weightName].end()) {
-      //try switching to default channel 
-      if(primaryDistributions[weightName].find(GetDefaultChannel(inPeriodNumber))==primaryDistributions[weightName].end()) {
-         Error("GetPrimaryWeight","Unrecognised channelNumber %d, and no distribution found for defaultChannel %d",channelNumber,GetDefaultChannel(inPeriodNumber));
-         throw std::runtime_error("Throwing 5: Unrecognised channelNumber");
-      }
-      channelNumber=GetDefaultChannel(inPeriodNumber);
-   }
-   TH1D* mc = primaryDistributions[weightName][channelNumber][periodNumber];
-   if(!mc) {
-      Error("GetPrimaryWeight","Unable to find distribution for [%s,%d,%d]",weightName.Data(),channelNumber,periodNumber);
-      throw std::runtime_error("Throwing 6: Unable to find distribution");
-   }
+   double l = p->primaryHists[-1]->GetBinContent(bin);
 
-   //MC10b correction
-   if ( fabs(x - 100.0) < 0.00001 && mc->GetXaxis()->GetXmax() < 99.0 ) x = 0.0;
+   return l/n;
+}
 
-   Int_t bin = data->GetXaxis()->FindFixBin(x);
+Float_t CP::TPileupReweighting::GetSecondaryWeight(Int_t periodNumber, Int_t channelNumber,Float_t x,Float_t y) {
+   //= L_j/L_i / N_j/N_i .. secondary hists have already been normalized
+   if(!m_isInitialized) { Info("GetSecondaryWeight","Initializing the subtool.."); Initialize(); }
+   if(m_countingMode) return 0.;
 
-   if(mc->GetBinContent(bin)==0.) {
-      if(m_SetWarnings) Warning("GetPrimaryWeight","No mc for weight '%s' in channelNumber=%d, periodNumber/runNumber=%d,x=%f",weightName.Data(),channelNumber,periodNumber,x);
-      return 0.;
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetSecondaryWeight","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
+   if(p->sumOfWeights.find(channelNumber) == p->sumOfWeights.end()) channelNumber = p->defaultChannel;
+   int bin = p->secondaryHists[channelNumber]->FindFixBin(x,y);
+   double n = p->secondaryHists[channelNumber]->GetBinContent(bin);
+   double l = p->secondaryHists[-1]->GetBinContent(bin);
+
+   return l/n;
+}
 
-   return (data->GetBinContent(bin)/mc->GetBinContent(bin));
+Double_t CP::TPileupReweighting::GetDataWeight(Int_t runNumber, TString trigger) {
+   //special mu-independent version of GetDataWeight. Will just use the global luminosity
+   m_doGlobalDataWeight=true;
+   double out = GetDataWeight(runNumber, trigger, 0);
+   m_doGlobalDataWeight=false;
+   return out;
 }
 
-Float_t Root::TPileupReweighting::GetSecondaryWeight(const TString weightName, Int_t periodNumber, Int_t channelNumber,Float_t x, Float_t y) {
-   if(!m_isInitialized) { Error("GetSecondaryWeight","Tool is not initialized"); return 0;}
+Double_t CP::TPileupReweighting::GetDataWeight(Int_t runNumber, TString trigger, Double_t x) {
+
+   if(!m_isInitialized) { Info("GetDataWeight","Initializing the subtool.."); Initialize(); }
    if(m_countingMode) return 0.;
-   //check if the runNumber has been reassigned
-   //only do this for mc, never for data
-   Int_t inPeriodNumber = periodNumber;
-   if(channelNumber>=0) {
-      if(m_mcRemappings.find(periodNumber)!=m_mcRemappings.end()) periodNumber=m_mcRemappings[periodNumber];
+   if(m_countingMode) {
+      Error("GetDataWeight","You cannot get a weight when in config file generating mode. Please use FillWeight method");
+      throw std::runtime_error("GetDataWeight: You cannot get a weight when in config file generating mode. Please use FillWeight method");
    }
+   //determine which period this run number is in
+   Int_t periodNumber = GetFirstFoundPeriodNumber(runNumber);
 
-   //determine bin number 
-   TH2D* data = secondaryDistributions[weightName][-1][periodNumber];
-   if(!data) {
-      Error("GetSecondaryWeight","Unrecognised periodNumber/mcRunNumber: %d",periodNumber);
-      throw std::runtime_error("Throwing 4: Unrecognised periodNumber/mcRunNumber");
+   Period* p = m_periods[periodNumber];
+   if(!p) {
+      Error("GetDataWeight","Unrecognised periodNumber: %d",periodNumber);
+      throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
    }
-   if(secondaryDistributions[weightName].find(channelNumber)==secondaryDistributions[weightName].end()) {
-      //try switching to default channel 
-      if(secondaryDistributions[weightName].find(GetDefaultChannel(inPeriodNumber))==secondaryDistributions[weightName].end()) {
-         Error("GetSecondaryWeight","Unrecognised channelNumber %d, and no distribution found for defaultChannel %d",channelNumber,GetDefaultChannel(inPeriodNumber));
-         throw std::runtime_error("Throwing 5: Unrecognised channelNumber");
-      }
-      channelNumber=GetDefaultChannel(inPeriodNumber);
+
+   //see if we already made this trigger hist 
+   
+
+   if(p->triggerHists.find(trigger)==p->triggerHists.end()) {
+      //try to do a subtrigger calculation we need to reopen all the lumicalc files and do the fiddily prescaled luminosity calculation
+      //will throw errors if can't
+      CalculatePrescaledLuminosityHistograms(trigger);
    }
-   TH2D* mc = secondaryDistributions[weightName][channelNumber][periodNumber];
-   if(!mc) {
-      Error("GetSecondaryWeight","Unable to find distribution for [%s,%d,%d]",weightName.Data(),channelNumber,periodNumber);
-      throw std::runtime_error("Throwing 6: Unable to find distribution");
+   TH1D* denomHist = p->triggerHists[trigger];
+   if(denomHist==0) {
+      Error("GetDataWeight","Could not find trigger %s in period %d",trigger.Data(),p->id);
+      throw std::runtime_error("GetDataWeight: Could not find trigger");
    }
-   Int_t nx = data->GetXaxis()->GetNbins()+2;
-   Int_t binx = data->GetXaxis()->FindFixBin(x);
-   Int_t biny = data->GetYaxis()->FindFixBin(y);
-   Int_t bin = binx + nx*biny;
-
-   if(mc->GetBinContent(bin)==0.) {
-      if(m_SetWarnings) Warning("GetSecondaryWeight","No mc for weight '%s' in channelNumber %d and periodNumber/runNumber %d",weightName.Data(),channelNumber,periodNumber);
-      return 0.;
+   TH1D* numerHist = p->triggerHists["None"];
+   if(numerHist==0) {
+      Error("GetDataWeight","Could not find unprescaled trigger in period %d",p->id);
+      throw std::runtime_error("GetDataWeight: Could not find unprescaled trigger. Please AddLumiCalc with a 'None' trigger");
    }
+   
+   if(m_doGlobalDataWeight) return numerHist->Integral(0,numerHist->GetNbinsX()+1)/denomHist->Integral(0,denomHist->GetNbinsX()+1);
 
-   return (data->GetBinContent(bin)/mc->GetBinContent(bin));
-
-}
+   Int_t bin=numerHist->FindFixBin(x);
+   return numerHist->GetBinContent(bin)/denomHist->GetBinContent(bin);
 
-Int_t Root::TPileupReweighting::Fill(Int_t runNumber,Int_t channelNumber,Float_t w, Float_t x, Float_t y, Float_t z) {
-   return Fill("pileup",runNumber,channelNumber,w,x,y,z);
 }
 
 //fills the appropriate inputHistograms
-Int_t Root::TPileupReweighting::Fill(const TString weightName,Int_t runNumber,Int_t channelNumber,Float_t w,Float_t x, Float_t y, Float_t z) {
-   if(!m_isInitialized) { Error("Fill","Tool is not initialized!"); return 0;}
+Int_t CP::TPileupReweighting::Fill(Int_t runNumber,Int_t channelNumber,Float_t w,Float_t x, Float_t y) {
+   if(!m_isInitialized) { Info("Fill","Initializing the subtool.."); Initialize(); }
    //should only be given genuine mcRunNumbers if mc (channel>=0). We don't fill periodNumber distributions 
-   if(channelNumber>=0) {
-      //if(m_mcRemappings.find(runNumber)!=m_mcRemappings.end()) runNumber=m_mcRemappings[runNumber]; //- no remappings in fill mode!
 
-      //also check that channel histograms exist for all defined periods. If not, define them 
-      if(m_inputHistograms[weightName].find(channelNumber)==m_inputHistograms[weightName].end()) {
-         //loop over mcruns, get empty histogram for each 
-         for(std::map<Int_t,std::map<Int_t, std::pair<UInt_t,UInt_t> > >::iterator it=m_periods.begin();it!=m_periods.end();++it) {
-            m_inputHistograms[weightName][channelNumber][it->first] = CloneEmptyHistogram(weightName,it->first,channelNumber);
-         }
-         //must create histograms for the merged run numbers as well, as config files are made without merging the numbers
-         //only do it if the remapping has some actual period assignments though!
-         for(std::map<Int_t,Int_t>::iterator it=m_mcRemappings.begin();it!=m_mcRemappings.end();++it) {
-            if(m_periods.find(it->second) == m_periods.end()) continue; ///added this line in july 2014... need to check doesnt break MC12ab
-            m_inputHistograms[weightName][channelNumber][it->first] = CloneEmptyHistogram(weightName,it->first,channelNumber);
-         }
+   TH1* hist = 0;
+
+   //get the period if mc, get the data run if data 
+   if(channelNumber>=0) {
+      if(m_periods.find(runNumber)==m_periods.end()) {
+         Error("Fill","Unrecognised runNumber: %d.  Check your period configuration (AddPeriod or UsePeriodConfig)",runNumber);
+         throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
       }
-   } else {
-      //create histograms for data too, only for this particular run
-      if(m_inputHistograms[weightName][channelNumber].find(runNumber)==m_inputHistograms[weightName][channelNumber].end()) {
-         m_inputHistograms[weightName][channelNumber][runNumber] = CloneEmptyHistogram(weightName,runNumber,channelNumber);
+      Period* p = m_periods[runNumber];
+      if(!p) {
+         Error("Fill","Unrecognised runNumber: %d.  Check your period configuration (AddPeriod or UsePeriodConfig) ... but should never have got here so please report this!",runNumber);
+         throw std::runtime_error("Throwing 1: Unrecognised periodNumber");
       }
+      if(p->inputHists.find(channelNumber)==p->inputHists.end()) {
+         //need to create my period histogram 
+         p->inputHists[channelNumber] = CloneEmptyHistogram(runNumber,channelNumber);
+      }
+      hist = p->inputHists[channelNumber];
+   } else {
+      Run& r = m_runs[runNumber];
+      if(r.inputHists["None"]==0) r.inputHists["None"]=CloneEmptyHistogram(runNumber,channelNumber);
+      hist = r.inputHists["None"];
    }
 
-   TH1* hist = m_inputHistograms[weightName][channelNumber][runNumber];
-
    if(!hist) {
-      Error("Fill","Unknown [weight,run,channel] = [%s,%d,%d]",weightName.Data(),runNumber,channelNumber);
-      throw std::runtime_error("Throwing 45: Fill:: Unknown [weight,run,channel] ");
+      Error("Fill","Unknown [run,channel] = [%d,%d]",runNumber,channelNumber);
+      throw std::runtime_error("Throwing 45: Fill:: Unknown [run,channel] ");
    }
 
    if(hist->GetDimension()==1) {
       return hist->Fill(x,w);
    } else if(hist->GetDimension()==2) {
-      return (dynamic_cast<TH2*>(hist))->Fill(x,y,w);
-   } else if(hist->GetDimension()==3) {
-      return (dynamic_cast<TH3*>(hist))->Fill(x,y,z,w);
-   }
+      return (static_cast<TH2*>(hist))->Fill(x,y,w);
+   } 
    return -1;
 }
 
+Int_t CP::TPileupReweighting::WriteToFile(const TString fname) {
+   
+   if(!m_countingMode) {Warning("WriteToFile","Not in counting mode, so no file will be written");return 0;}
+
+
+   //build a TTree with the correct mc structure, and dump the mc histogram info in to it
+   //also build aggregate histograms across all channels - shouldn't be used as input histograms unless for a single channel
+   TString filename = fname;
+   filename += (filename=="") ? TString(this->GetName()) + ".prw.root" : "";
+   
+   //loop over the weights. "pileup" gets it own MCPileupReweighting ttree. ... since we got rid of weightNames in this version, everything goes in MCPileupReweighting now!
+   //data goes in to DataCustomReweighting. other goes in to MCCustomReweighting 
+   TFile* outFile = TFile::Open(filename,"RECREATE");
+   Int_t r = WriteToFile(outFile);
+   outFile->Close();
+   return r;
+}
+
+
+Int_t CP::TPileupReweighting::WriteToFile(TFile* outFile) {
+   if(!m_countingMode) {Warning("WriteToFile","Not in counting mode, so no file will be written");return 0;}
+
+   TDirectory* origDir = gDirectory;
+
+   outFile->cd();
+
+   TTree *outTreeMC=0;
+   TTree *outTreeData=0;
+   Int_t channel = 0;UInt_t runNumber = 0; 
+   std::vector<UInt_t>* pStarts = 0;std::vector<UInt_t>* pEnds = 0;
+   Char_t histName[150];
+
+
+   //loop over periods ... periods only get entry in table if they have an input histogram 
+   for(auto period : m_periods) {
+      if(period.first != period.second->id) continue; //skips redirects 
+      runNumber = period.first;
+      pStarts = new std::vector<UInt_t>;pEnds = new std::vector<UInt_t>;
+      if(period.second->subPeriods.size()==0) { 
+         pStarts->push_back(period.second->start); pEnds->push_back(period.second->end); 
+      }
+      else {
+         for(auto subp : period.second->subPeriods) {
+            pStarts->push_back(subp->start); pEnds->push_back(subp->end);
+         }
+      }
+      for(auto inHist : period.second->inputHists) {
+         channel = inHist.first;
+         TH1* hist = inHist.second;
+         strcpy(histName,hist->GetName());
+         hist->Write();
+
+         if(!outTreeMC) {
+            outTreeMC = new TTree("MCPileupReweighting","MCPileupReweighting");
+            outTreeMC->Branch("Channel",&channel);
+            outTreeMC->Branch("RunNumber",&runNumber);
+            outTreeMC->Branch("PeriodStarts",&pStarts);
+            outTreeMC->Branch("PeriodEnds",&pEnds);
+            outTreeMC->Branch("HistName",&histName,"HistName/C");
+         }
+         outTreeMC->Fill();
+      }
+      delete pStarts; delete pEnds;
+   }
+
+   //loop over data 
+   for(auto& run : m_runs) {
+      runNumber = run.first;
+      if(run.second.inputHists.find("None")==run.second.inputHists.end()) continue;
+
+      TH1* hist = run.second.inputHists["None"];
+      strcpy(histName,hist->GetName());
+      hist->Write();
+      if(!outTreeData) {
+         outTreeData = new TTree("DataPileupReweighting","DataPileupReweighting");
+         outTreeData->Branch("RunNumber",&runNumber);
+         outTreeData->Branch("HistName",&histName,"HistName/C");
+      }
+      outTreeData->Fill();
+   }
+
+
+   //write the non-zero ttrees 
+   if(outTreeMC) {outTreeMC->Write();delete outTreeMC;outTreeMC=0;}
+   if(outTreeData) {outTreeData->Write();delete outTreeData;outTreeData=0;}
+
+
+   Info("WriteToFile", "Successfully generated config file: %s",outFile->GetName());
+   Info("WriteToFile", "Happy Reweighting :-)");
+
+   gDirectory = origDir;
+
+   return 0;
+}
+
+
 //=============================================================================
 // Method to calculate the ratio histogram
 //=============================================================================
-void Root::TPileupReweighting::normalizeHistogram(TH1* hist){
+void CP::TPileupReweighting::normalizeHistogram(TH1* hist){
    // normalize the data histogram based on which sort of histogram it is
    if(hist){ 
       if(hist->InheritsFrom("TH3")) {
@@ -2064,7 +1645,7 @@ void Root::TPileupReweighting::normalizeHistogram(TH1* hist){
 
 
 //This allows PROOF to merge the generated histograms before the WriteToFile or WriteCustomToFile calls
-Int_t Root::TPileupReweighting::Merge(TCollection *coll) {
+Int_t CP::TPileupReweighting::Merge(TCollection *coll) {
    if(!coll) return 0;
    if(coll->IsEmpty()) return 0;
 
@@ -2074,27 +1655,32 @@ Int_t Root::TPileupReweighting::Merge(TCollection *coll) {
    while( ( obj = next() ) ) {
 
       // Check that the element is of the right type:
-      Root::TPileupReweighting* vobj = dynamic_cast< Root::TPileupReweighting* >( obj );
+      CP::TPileupReweighting* vobj = dynamic_cast< CP::TPileupReweighting* >( obj );
       if( ! vobj ) {
          Error( "Merge", "Unknown object type encountered: %s",obj->ClassName() );
          return 0;
       }
 
-      //merge the inputHistograms 
-      for(std::map<TString, std::map<Int_t,std::map<Int_t, TH1*> > >::iterator weights=vobj->GetInputHistograms().begin();weights!=vobj->GetInputHistograms().end();++weights) {
-         TString weightName = weights->first;
-         for(std::map<Int_t,std::map<Int_t, TH1*> >::iterator channels=weights->second.begin();channels!=weights->second.end();++channels) {
-            Int_t channel = channels->first;
-            for(std::map<Int_t, TH1*>::iterator runs=channels->second.begin();runs!=channels->second.end();++runs) {
-               Int_t run=runs->first;
-               if(runs->second) {
-                  if(m_inputHistograms[weightName][channel][run]) {
-                     m_inputHistograms[weightName][channel][run]->Add(runs->second);
-                  } else {
-                     m_inputHistograms[weightName][channel][run] = dynamic_cast<TH1*>(runs->second->Clone(runs->second->GetName()));
-                     m_inputHistograms[weightName][channel][run]->SetDirectory(0);
-                  }
-               }
+      //merge the inputHistograms ... all the periods should be identical 
+      for(auto period : vobj->m_periods) {
+         if(period.first != period.second->id) continue;
+         for(auto& iHist : period.second->inputHists) {
+            if(GetInputHistogram(period.first,iHist.first)==0) {
+               m_periods[period.first]->inputHists[iHist.first] = dynamic_cast<TH1*>(iHist.second->Clone(iHist.second->GetName()));
+               m_periods[period.first]->inputHists[iHist.first]->SetDirectory(0);
+            } else {
+               GetInputHistogram(period.first,iHist.first)->Add(iHist.second);
+            }
+         }
+      }
+      //also must remember to merge the runs too (where the data is held)
+      for(auto& run : vobj->m_runs) {
+         for(auto& iHist : run.second.inputHists) {
+            if(m_runs[run.first].inputHists[iHist.first]==0) {
+               m_runs[run.first].inputHists[iHist.first] = dynamic_cast<TH1*>(iHist.second->Clone(iHist.second->GetName()));
+               m_runs[run.first].inputHists[iHist.first]->SetDirectory(0);
+            } else {
+               m_runs[run.first].inputHists[iHist.first]->Add(iHist.second);
             }
          }
       }
@@ -2104,7 +1690,7 @@ Int_t Root::TPileupReweighting::Merge(TCollection *coll) {
 }
 
 
-void Root::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TString trigger) {
+void CP::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TString trigger) {
    //check if this is a composite trigger (user has provided OR of triggers that are passed before prescale)
    
    TString triggerCopy = trigger; triggerCopy.ReplaceAll(" ",""); triggerCopy.ReplaceAll("&&","&");triggerCopy.ReplaceAll("||","|");
@@ -2116,25 +1702,6 @@ void Root::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TStr
 
    std::vector<TString> subTriggers;
 
-/*
-   int fStart=0;int nextSep=trigger.Index(',');
-   while(nextSep>=0) {
-      TString subTrigger = trigger(fStart,nextSep-fStart);
-      if(m_triggerPrimaryDistributions.find(subTrigger)==m_triggerPrimaryDistributions.end()) {
-         Error("GetDataWeight","Could not find subTrigger %s in composite trigger set: %s",subTrigger.Data(),trigger.Data());
-         throw std::runtime_error("Could not find sub-trigger");
-      }
-      subTriggers.push_back(subTrigger);
-      fStart = nextSep+1;
-      nextSep = trigger.Index(',',fStart);
-   }
-   subTriggers.push_back(trigger(fStart,trigger.Length()-fStart)); //don't forget the last one
-   if(subTriggers.size()==0) {
-      Error("GetDataWeight","Could not find trigger %s",trigger.Data());
-      throw std::runtime_error("Could not find trigger");
-   }
-*/
-
    t->getTriggers(subTriggers); //fills the vector
 
    //1. Open all the lumicalc files
@@ -2180,6 +1747,7 @@ void Root::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TStr
             long nEntries = tmp->GetEntries();
             for(long i=0;i<nEntries;i++) {
                b_runNbr->GetEntry(i);b_L1Presc->GetEntry(i);b_L2Presc->GetEntry(i);b_L3Presc->GetEntry(i);b_lbn->GetEntry(i);
+               runNbr += m_lumicalcRunNumberOffset;
                //save the prescale by run and lbn 
                //if(runNbr==215643) Info("...","prescale in [%d,%d] = %f %f %f", runNbr,lbn,ps1,ps2,ps3);
                if(ps1>0&&ps2>0&&ps3>0) prescaleByRunAndLbn[*it][runNbr][lbn] = ps1*ps2*ps3;
@@ -2220,6 +1788,7 @@ void Root::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TStr
             b_runNbr->GetEntry(i);b_intLumi->GetEntry(i);b_mu->GetEntry(i);
             b_lbn->GetEntry(i);
 
+            runNbr += m_lumicalcRunNumberOffset;
             //for each subtrigger, lookup prescale, and calculate pFactor 
 /*
             double pFactor(1);
@@ -2235,12 +1804,23 @@ void Root::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TStr
 
             //Info("...","prescale in [%d,%d] = %f", runNbr,lbn,1./pFactor);
 
-            Int_t periodNumber = GetFirstFoundPeriodNumber(runNbr);
-            if(m_triggerPrimaryDistributions[trigger].find(periodNumber) == m_triggerPrimaryDistributions[trigger].end()) {
-               m_triggerPrimaryDistributions[trigger][periodNumber] = dynamic_cast<TH1D*>(CloneEmptyHistogram("pileup",periodNumber,-1));
-               Info("CalculatePrescaledLuminosityHistograms","Created Data Weight Histogram for [%s,%d]",trigger.Data(),periodNumber);
+            int bin = m_emptyHistogram->FindFixBin(mu*m_dataScaleFactorX);
+
+            //add into all periods that contain this runNbr 
+            for(auto p : m_periods) {
+               if(p.first != p.second->id) continue; //skips remappings 
+               if(!p.second->contains(runNbr)) continue;
+               std::map<TString,TH1D*>& triggerHists = p.second->triggerHists;
+               if(triggerHists.find(trigger) == triggerHists.end()) {
+                  triggerHists[trigger] = dynamic_cast<TH1D*>(CloneEmptyHistogram(p.first,-1));
+                  if(m_debugging) Info("CalculatePrescaledLuminosityHistograms","Created Data Weight Histogram for [%s,%d]",trigger.Data(),p.first);
+               }
+                //check if we were about to fill a bad bin ... if we are, we either skipp the fill (unrep action=1) or redirect (unrep action=3)
+               if( (m_unrepresentedDataAction==1) && p.second->inputBinRedirect[bin]!=bin) { } //do nothing
+               else if( (m_unrepresentedDataAction==3) ) triggerHists[trigger]->Fill(triggerHists[trigger]->GetBinCenter(p.second->inputBinRedirect[bin]), intLumi*pFactor);
+               else triggerHists[trigger]->Fill(mu*m_dataScaleFactorX,intLumi*pFactor);
+               
             }
-            m_triggerPrimaryDistributions[trigger][periodNumber]->Fill(mu*m_dataScaleFactorX,intLumi*pFactor);
          }
       }
    }
@@ -2256,8 +1836,8 @@ void Root::TPileupReweighting::CalculatePrescaledLuminosityHistograms(const TStr
 
 }
 
-Root::TPileupReweighting::CompositeTrigger* Root::TPileupReweighting::makeTrigger(TString s) {
-   Info("makeTrigger","Doing %s",s.Data());
+CP::TPileupReweighting::CompositeTrigger* CP::TPileupReweighting::makeTrigger(TString s) {
+   if(m_debugging) Info("makeTrigger","Doing %s",s.Data());
    //find the first operand
    TString oper1; CompositeTrigger* cOper1 = 0;
    if(s.BeginsWith("(")) {
@@ -2283,8 +1863,9 @@ Root::TPileupReweighting::CompositeTrigger* Root::TPileupReweighting::makeTrigge
       i++;
    }
 
+
    if(op==0) {
-      if(m_triggerPrimaryDistributions.find(s)==m_triggerPrimaryDistributions.end()) {
+      if(m_lumicalcFiles.find(s)==m_lumicalcFiles.end()) {
          Error("GetDataWeight","Could not find subTrigger %s",s.Data());
          return 0;
       }
@@ -2314,12 +1895,12 @@ Root::TPileupReweighting::CompositeTrigger* Root::TPileupReweighting::makeTrigge
    if(j==s.Length()+1 && op==2) { //no more OR found, set oper2 to remainder 
       if(cOper1) out->trig1=cOper1;  
       else {oper1=s(0,i); out->trig1=makeTrigger(oper1); }
-      TString oper2 = s(i+1,s.Length()); Info("makeTrigger","Found & %s %s",oper1.Data(),oper2.Data());
+      TString oper2 = s(i+1,s.Length()); if(m_debugging) Info("makeTrigger","Found & %s %s",oper1.Data(),oper2.Data());
       out->trig2 = makeTrigger(oper2); if(out->trig2==0) { delete out; return 0; } 
       return out;
    } else if(op==1) {   //found an OR, set oper1 to everything up to this 
       
-      oper1 = s(0,j); TString oper2 = s(j+1,s.Length()); Info("makeTrigger","Found & then | %s %s",oper1.Data(),oper2.Data());
+      oper1 = s(0,j); TString oper2 = s(j+1,s.Length()); if(m_debugging) Info("makeTrigger","Found & then | %s %s",oper1.Data(),oper2.Data());
       if(cOper1) delete cOper1;
       out->op=op; //updates to an OR
       out->trig1=makeTrigger(oper1); if(out->trig1==0) { delete out; return 0; }
@@ -2331,28 +1912,5 @@ Root::TPileupReweighting::CompositeTrigger* Root::TPileupReweighting::makeTrigge
    return 0;
 }
 
-Double_t Root::TPileupReweighting::GetIntegratedLumi(TString trigger) { 
-   if(!m_isInitialized) {
-      Error("GetIntegratedLumi", "Please initialize the tool before retrieving the total lumi");
-      throw std::runtime_error("Throwing 1");
-   }
-   if(!m_lumiVectorIsLoaded) {
-      Error("GetIntegratedLumi","No Lumicalc file loaded, so cannot get integrated lumi, returning 0");
-      return 0;
-   }
-   if(trigger=="") return globalTotals["pileup"][-1]/1E6; 
-   else { //check the triggerPrimaryDistributions, if not present, build it
-      if(m_triggerPrimaryDistributions.find(trigger)==m_triggerPrimaryDistributions.end()) {
-         //try to do a subtrigger calculation we need to reopen all the lumicalc files and do the fiddily prescaled luminosity calculation
-         //will throw errors if can't
-         CalculatePrescaledLuminosityHistograms(trigger);
-      }
-      //loop over all periods and sum up lumi 
-      Double_t out =0;
-      for(std::map<Int_t, TH1D*>::iterator it=m_triggerPrimaryDistributions[trigger].begin(); it!= m_triggerPrimaryDistributions[trigger].end(); ++it) {
-         out += it->second->Integral(0,it->second->GetNbinsX()+1);
-      }
-      return out;
-   }
-}
+
 
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/Makefile.RootCore b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/Makefile.RootCore
index 420f80bf502a8c365ddb8d8939e0ead9e206cc31..c88e9b711f76a356eb80ceaa823501ced71bc506 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/Makefile.RootCore
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/Makefile.RootCore
@@ -2,7 +2,8 @@ PACKAGE          = PileupReweighting
 PACKAGE_CXXFLAGS = 
 PACKAGE_LDFLAGS  = 
 PACKAGE_PRELOAD  = Hist
-PACKAGE_DEP = AsgTools xAODEventInfo PathResolver xAODCore
+PACKAGE_DEP = AsgTools ReweightUtils xAODEventInfo PathResolver xAODCore PATInterfaces
 PACKAGE_REFLEX = 1
+PACKAGE_PEDANTIC = 1
 
 include $(ROOTCOREDIR)/Makefile-common
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/requirements b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/requirements
index 074c81cfd491076054170ecc793966bdaadc0676..ee281aa266a6b82b48ab4479d99450ac9fcdc83e 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/requirements
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/cmt/requirements
@@ -9,7 +9,7 @@ use GaudiInterface              GaudiInterface-*        External
 use AsgTools        AsgTools-*        Control/AthToolSupport
 use xAODEventInfo xAODEventInfo-*   Event/xAOD
 
-
+use PATInterfaces  PATInterfaces-*  PhysicsAnalysis/AnalysisCommon
 
 private 
 use PathResolver PathResolver-*     Tools
@@ -23,19 +23,31 @@ end_private
 apply_tag       ROOTBasicLibs
 apply_tag       ROOTMathLibs
 
+apply_tag       ROOTCintexLibs
+
 # Specify the required ROOT components for cmake (transparent to CMT)
 apply_pattern cmake_add_command command="find_package(ROOT COMPONENTS MathCore RIO)"
 
 ## declare the directories CMT should know about
 branches python share doc src Root
 
-apply_pattern have_root_headers     root_headers="TPileupReweighting.h ../Root/LinkDef.h"    headers_lib="PileupReweightingLib" extra_includes=" -I$(PathResolver_include_dirs) -I$(AsgTools_include_dirs) -DXAOD_STANDALONE"
-apply_pattern dual_use_library files=" *.cxx ../Root/*.cxx"
+
+library PileupReweightingLib "../Root/*.cxx"
+apply_pattern named_installed_library library=PileupReweightingLib
+
+macro extra_in "" AthAnalysisBase "-DXAOD_ANALYSIS"
+apply_pattern have_root_headers     root_headers="TPileupReweighting.h ../Root/LinkDef.h"    headers_lib="PileupReweightingLib" extra_includes="$(extra_in)  "
+
+library PileupReweighting *.cxx -s=components *.cxx
+macro_append PileupReweighting_dependencies " PileupReweightingLib"
+apply_pattern component_library
+
+#apply_pattern dual_use_library files=" *.cxx ../Root/*.cxx"
 
 
 private
 use     AtlasReflex     AtlasReflex-*   External -no_auto_imports
-apply_pattern lcgdict dict=PileupReweightingDict selectionfile=selection.xml headerfiles="../PileupReweighting/PileupReweightingDict.h"
+apply_pattern lcgdict dict=PileupReweighting selectionfile=selection.xml headerfiles="../PileupReweighting/PileupReweightingDict.h"
 end_private
 
 
@@ -56,4 +68,6 @@ apply_pattern declare_calib files="../share/*.root"
 
 
 
-
+application testPRW testPRW.C
+#ensure the main library is compiled first
+macro_append testPRW_dependencies " PileupReweighting "
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/exampleJobOptions.py b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/exampleJobOptions.py
index f2c769d5ab0b7474c3f457584fc9ba91a51e543b..87f1b3e8be937b29d5879cf440328baa44ce1fe6 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/exampleJobOptions.py
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/exampleJobOptions.py
@@ -1,12 +1,27 @@
 
 #by default (i.e. with no config files) the tool will be created in config file making mode
 
+from glob import glob
+
+theApp.EvtMax=10                                        #says how many events to run over. Set to -1 for all events
+
+import AthenaPoolCnvSvc.ReadAthenaPool                   #sets up reading of POOL files (e.g. xAODs)
+svcMgr.EventSelector.InputCollections=glob(vars().get("FILES","/afs/cern.ch/atlas/project/PAT/xAODs/r5591/mc14_8TeV.117050.PowhegPythia_P2011C_ttbar.recon.AOD.e1727_s1933_s1911_r5591/AOD.01494882._111853.pool.root.1") )   #insert your list of input files here
+
+algseq = CfgMgr.AthSequencer("AthAlgSeq")
+
+
+#in your c++ code, create a ToolHandle<IPileupReweightingTool>
+#the ToolHandle constructor should be given "CP::PileupReweightingTool/myTool" as its string argument
 ToolSvc += CfgMgr.CP__PileupReweightingTool("myTool", 
-            ConfigFiles=["PileupReweighting/mc12ab_defaults.prw.root"],
+            ConfigFiles=["my.prw.root"],
             DataScaleFactor=1./1.09,
-            UnrepresentedDataAction=2,
-            LumiCalcFiles=["GRL_v61.None.lumicalc.root"])
+            UnrepresentedDataAction=2, DataScaleFactorUP=1./1.11, DataScaleFactorDOWN=1./1.07,
+            LumiCalcFiles=["/usera/will/testareas/ZZ8TeV/ZZAnalysis/share/GRL_v61.None.lumicalc.root"],OutputLevel=VERBOSE)
+
+algseq += CfgMgr.CP__PileupReweightingProvider(Tool=ToolSvc.myTool,OutputLevel=VERBOSE)                                 #adds an instance of your alg to it
+
+
+include("AthAnalysisBaseComps/SuppressLogging.py")       #Optional include to suppress as much athena output as possible
 
-#in your code, create a ToolHandle<IPileupReweightingTool>
-#the ToolHandle constructor should be given "CP::PileupReweightingTool/myTool" as its string argument
 
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/generatePRW_jobOptions.py b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/generatePRW_jobOptions.py
new file mode 100644
index 0000000000000000000000000000000000000000..c6e2898b4dd9e67738ec16ab773026fd012f745b
--- /dev/null
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/generatePRW_jobOptions.py
@@ -0,0 +1,23 @@
+
+theApp.EvtMax=-1                                         #says how many events to run over. Set to -1 for all events
+
+import AthenaPoolCnvSvc.ReadAthenaPool                   #sets up reading of POOL files (e.g. xAODs)
+from glob import glob
+svcMgr.EventSelector.InputCollections=glob( vars().get("FILES","/afs/cern.ch/atlas/project/PAT/xAODs/r5591/mc14_8TeV.117050.PowhegPythia_P2011C_ttbar.recon.AOD.e1727_s1933_s1911_r5591/AOD.01494882._111853.pool.root.1").strip() )
+
+if len(svcMgr.EventSelector.InputCollections)==0:
+   print("WARNING  >>>>>>>>>>  generatePRW_jobOptions.py:   NO INPUT FILES PROVIDED/FOUND FROM: %s ... this will produce a failure unless you are sending this job to the grid <<<<<<<<<<<<" % vars().get("FILES","/afs/cern.ch/atlas/project/PAT/xAODs/r5591/mc14_8TeV.117050.PowhegPythia_P2011C_ttbar.recon.AOD.e1727_s1933_s1911_r5591/AOD.01494882._111853.pool.root.1").strip())
+   
+
+algseq = CfgMgr.AthSequencer("AthAlgSeq")                #gets the main AthSequencer
+algseq += CfgMgr.CP__PileupReweightingProvider(ConfigOutputStream="METADATA")                           #adds an instance of your alg to it
+
+include("AthAnalysisBaseComps/SuppressLogging.py")       #Optional include to suppress as much athena output as possible
+
+svcMgr += CfgMgr.THistSvc()
+svcMgr.THistSvc.Output += ["METADATA DATAFILE='my.prw.root' OPT='RECREATE'"]
+
+#use on the grid like this:
+#pathena PileupReweighting/generatePRW_jobOptions.py --inDS="etc/,etc/,etc/" --outDS="user.whatever.myprw/" --extOutFile="auto.prw.root"
+
+
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/makeWeightTree.C b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/makeWeightTree.C
new file mode 100644
index 0000000000000000000000000000000000000000..d5afc46e43a1ad1e349e053c10fe020e13f90254
--- /dev/null
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/share/makeWeightTree.C
@@ -0,0 +1,37 @@
+
+///***************************************************************************************
+///PRW Weight Tree Making and Usage:
+///---------------------------------------------------------------------------------------
+///This is a small script you can run to generate a 'prw weight tree'
+///Modify the properties below to set up an instance of a prw tool as you would if you
+///processing all your MC in a usual event loop.
+///Then in the 'MakeWeightTree' line, specify all the channel numbers you want to generate
+///pileup weights for. Specify them as a comma delimited list in a string.
+///
+///Once you run this script (root makeWeightTree.C) you will get a prwTree.root file
+///To use it, you need to have made a branch in your main ntuple called 'PRWHash'
+///which contains the result of the GetPRWHash() method of the tool.
+///Then you just friend this tree to your main tree/chain, like this:
+///TChain c("MainTreeChain"); //add files to this
+///TChain* prwChain = new TChain("prwTree"); prwChain->Add("prwTree.root"); prwChain->BuildIndex("PRWHash");
+///c.AddFriend(prwChain);
+///Your main chain will then appear to have a 'PileupWeight' branch in it!!
+
+
+
+{
+   CP::PileupReweightingTool tool("tool");
+
+   std::vector<std::string> configFiles = {"/usera/will/testareas/ZdZd13TeV/ZdZdAnalysis/share/302073.prw.root","/usera/will/testareas/ZdZd13TeV/ZdZdAnalysis/share/361106_361108.r6630.prw.root"};
+   std::vector<std::string> lumiFiles = {"/usera/will/testareas/ZdZd13TeV/ZdZdAnalysis/share/data15_13TeV.periodA.None.lumicalc.gerl.root"};
+
+   tool.setProperty("ConfigFiles",configFiles);
+   tool.setProperty("LumicalcFiles",lumiFiles);
+   tool.setProperty("DefaultChannel",361106);
+
+
+   tool.initialize();
+   tool.expert()->MakeWeightTree("361106,361107,361108,123456","prwTree.root","PRWHash" /*change to name of branch in main tree*/ ,"PileupWeight" /*optional change to name of branch you want to be the weight*/);
+
+
+}
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/PileupReweightingProvider.h b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/PileupReweightingProvider.h
index 1196c13683f2a17294d6d04e28006396def18212..e13756df0caffff92d420ef68dca2b4818e88710 100644
--- a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/PileupReweightingProvider.h
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/PileupReweightingProvider.h
@@ -9,23 +9,83 @@
 #include "AthenaBaseComps/AthAlgorithm.h"
 #include "GaudiKernel/ToolHandle.h"
 
+#include "xAODEventInfo/EventInfoAuxContainer.h"
+
+#include "PATInterfaces/SystematicRegistry.h"
+
 namespace CP {
 
 class PileupReweightingProvider : public AthAlgorithm {
 
      public:
-      PileupReweightingProvider( const std::string& name, ISvcLocator* svcloc) : AthAlgorithm(name,svcloc),m_tool("PileupReweightingTool") {
+      PileupReweightingProvider( const std::string& name, ISvcLocator* svcloc) : AthAlgorithm(name,svcloc),m_tool("CP::PileupReweightingTool/auto") {
          declareProperty("Tool",m_tool,"The configured PileupReweightingTool");
+         declareProperty("Input",m_inputKey="","Specify a specific EventInfo object");
+         declareProperty("Output",m_outputKey="","Specify an output EventInfo object. If differs from input, will create a clone of EventInfo and decorate that");
+         declareProperty("ConfigOutputStream",m_configStream="","Specify the stream to output config file to");
       }
 
       ~PileupReweightingProvider() { }
 
-      virtual StatusCode initialize() { CHECK( m_tool.retrieve() ); return StatusCode::SUCCESS; }
-      virtual StatusCode execute() { CHECK( m_tool->execute() ); return StatusCode::SUCCESS; }
+      virtual StatusCode initialize() { 
+         CHECK( m_tool.retrieve() ); 
+
+         IProperty* myTool = dynamic_cast<IProperty*>(&*m_tool);
+         CHECK( myTool->setProperty("ConfigOutputStream",m_configStream) );
+
+         //get the list of systematics 
+         CP::SystematicRegistry& registry = CP::SystematicRegistry::getInstance();
+         m_allSysts = registry.recommendedSystematics();
+
+         if(m_configStream!="") ATH_MSG_INFO("Now running config file making .... please be patient! ... ");
+
+         return StatusCode::SUCCESS; 
+      }
+      virtual StatusCode execute() { 
+            const xAOD::EventInfo* evtInfo =0;
+
+            if(m_inputKey.length()>0) ATH_CHECK(evtStore()->retrieve(evtInfo,m_inputKey));
+            else {
+               #ifdef XAOD_STANDALONE
+                  ATH_CHECK( evtStore()->retrieve(evtInfo,"") ); //apparently TEvent can't do keyless retrieves!!??
+               #else
+                  ATH_CHECK(evtStore()->retrieve(evtInfo));
+               #endif
+            }
+
+            //do we need to make a copy??
+            if(m_inputKey!=m_outputKey && m_outputKey!="") {
+               xAOD::EventInfo* evtInfoCopy = new xAOD::EventInfo( *evtInfo );
+               xAOD::EventInfoAuxContainer* aux = new xAOD::EventInfoAuxContainer;
+               evtInfoCopy->setStore(aux);
+               ATH_CHECK( evtStore()->record( evtInfoCopy , m_outputKey ) );
+               ATH_CHECK( evtStore()->record( aux , m_outputKey+"Aux." ) );
+               evtInfo = evtInfoCopy;
+            }
+
+            CHECK( m_tool->apply(*evtInfo) );
+
+
+            //here's an example of systematic variations 
+            for(auto& syst : m_allSysts) {
+               ATH_MSG_VERBOSE("Doing systematic : " << syst.name());
+               if(! m_tool->isAffectedBySystematic( syst )) continue;
+               CP::SystematicSet tmp; tmp.insert( syst );
+               if( m_tool->applySystematicVariation( tmp ) != CP::SystematicCode::Ok ) continue;
+               CHECK( m_tool->apply(*evtInfo) );
+            }
+            //make sure we leave the tool in the nominal mode 
+            if( m_tool->applySystematicVariation( CP::SystematicSet() ) != CP::SystematicCode::Ok ) return StatusCode::FAILURE; 
+
+            return StatusCode::SUCCESS; 
+      }
 
    private:
       ToolHandle<IPileupReweightingTool> m_tool;
 
+      std::string m_inputKey,m_outputKey,m_configStream;
+
+      CP::SystematicSet m_allSysts;
 
 }; 
 
diff --git a/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/testPRW.C b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/testPRW.C
new file mode 100644
index 0000000000000000000000000000000000000000..81f897fb2d63dc85b3a91b1f849bb725adf7acd6
--- /dev/null
+++ b/PhysicsAnalysis/AnalysisCommon/PileupReweighting/src/testPRW.C
@@ -0,0 +1,184 @@
+/*
+  Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration
+*/
+
+
+#include "PileupReweighting/TPileupReweighting.h"
+#include "TTree.h"
+
+void testValue(double v1, double v2) {
+   if( (!v1 && v2) || (!v2 && v1) ) throw std::runtime_error("zero-non-zero mismatch");
+   if( fabs(v1-v2)/v1 > 0.001 ) throw std::runtime_error("mismatch > 0.1% in values");
+}
+
+
+int main() {
+   //create a 'dummy' lumicalc file, with only a few runs, with a few lumiblocks 
+   TFile l("dummy.None.lumicalc.root","RECREATE");
+   TTree *t = new TTree("LumiMetaData","LumiMetaData");
+   UInt_t runNbr=0;Float_t ps1=1;Float_t ps2=1; Float_t ps3=1;UInt_t lbn=0;Float_t intLumi=0;Float_t mu=0;
+   t->Branch("RunNbr",&runNbr);t->Branch("L1Presc",&ps1);t->Branch("L2Presc",&ps2);t->Branch("L3Presc",&ps3);t->Branch("LBStart",&lbn);t->Branch("IntLumi",&intLumi);t->Branch("AvergeInteractionPerXing",&mu);
+
+   runNbr=1;lbn=1;intLumi=5;mu=0.5;t->Fill();
+   runNbr=1;lbn=2;intLumi=10;mu=1.5;t->Fill();
+   runNbr=2;lbn=1;intLumi=3;mu=1.5;t->Fill();
+   runNbr=3;lbn=1;intLumi=7;mu=1.5;t->Fill();
+   runNbr=3;lbn=2;intLumi=2;mu=2.5;t->Fill(); //will be unrepresented data in the setup in channel 2000
+ 
+   t->Write();delete t;
+   l.Close();
+
+   //make some trigger lumicalcs... only the prescale numbers are important!
+   TFile l2("dummy.TriggerA.lumicalc.root","RECREATE");
+   t = new TTree("LumiMetaData","LumiMetaData");
+   t->Branch("RunNbr",&runNbr);t->Branch("L1Presc",&ps1);t->Branch("L2Presc",&ps2);t->Branch("L3Presc",&ps3);t->Branch("LBStart",&lbn);t->Branch("IntLumi",&intLumi);t->Branch("AvergeInteractionPerXing",&mu);
+
+   runNbr=1;lbn=1;ps1=2;mu=0.5;t->Fill();
+   runNbr=1;lbn=2;ps1=3;mu=1.5;t->Fill();
+   runNbr=2;lbn=1;ps1=3;mu=1.5;t->Fill();
+   runNbr=3;lbn=1;ps1=4;mu=1.5;t->Fill();
+   runNbr=3;lbn=2;ps1=2;mu=2.5;t->Fill(); 
+ 
+   t->Write();delete t;
+   l2.Close();
+
+   TFile l3("dummy.TriggerB.lumicalc.root","RECREATE");
+   t = new TTree("LumiMetaData","LumiMetaData");
+   t->Branch("RunNbr",&runNbr);t->Branch("L1Presc",&ps1);t->Branch("L2Presc",&ps2);t->Branch("L3Presc",&ps3);t->Branch("LBStart",&lbn);t->Branch("IntLumi",&intLumi);t->Branch("AvergeInteractionPerXing",&mu);
+
+   runNbr=1;lbn=1;ps1=4;mu=0.5;t->Fill();
+   runNbr=1;lbn=2;ps1=6;mu=1.5;t->Fill();
+   runNbr=2;lbn=1;ps1=6;mu=1.5;t->Fill();
+   runNbr=3;lbn=1;ps1=8;mu=1.5;t->Fill();
+   runNbr=3;lbn=2;ps1=4;mu=2.5;t->Fill(); 
+ 
+   t->Write();delete t;
+   l3.Close();
+
+   //now create a config file ... this test config file generating 
+
+   CP::TPileupReweighting g;
+   g.AddPeriod(100,1,2);
+   g.AddPeriod(101,3,3);
+   g.SetUniformBinning(10,0,10);
+   g.Initialize();
+
+   g.Fill(100,2002,1,0.5);
+   g.Fill(100,2002,1,1.5);
+   g.Fill(101,2002,1,1.5);
+   g.Fill(101,2002,1,2.5);
+   g.Fill(101,2002,1,3.5);
+
+   g.WriteToFile("dummy1.prw.root");
+
+   CP::TPileupReweighting* genConfig = new CP::TPileupReweighting("genConfig");
+   genConfig->AddPeriod(100,1,2);
+   genConfig->AddPeriod(101,3,3);
+   genConfig->SetUniformBinning(10,0,10);
+   genConfig->Initialize();
+
+   genConfig->Fill(100,2000/*channel number*/,1/*evt_weight*/,0.5);
+   genConfig->Fill(100,2000,1,1.5);
+   genConfig->Fill(101,2000,1,1.5);genConfig->Fill(101,2000,1,1.5);genConfig->Fill(101,2000,1,1.5); //three times stats in this bin 
+
+   genConfig->Fill(100,2001,0.5,1.5);
+   genConfig->Fill(101,2001,2,1.5);
+   genConfig->Fill(101,2001,1,2.5); 
+
+   genConfig->WriteToFile("dummy2.prw.root");
+
+   //using dummy1 alone should be ok, no unrepresented data... 
+   CP::TPileupReweighting prw1;
+   prw1.AddConfigFile("dummy1.prw.root");
+   prw1.AddLumiCalcFile("dummy.None.lumicalc.root");
+   prw1.AddLumiCalcFile("dummy.TriggerA.lumicalc.root","TriggerA");
+   prw1.AddLumiCalcFile("dummy.TriggerB.lumicalc.root","TriggerB");
+   //prw1.EnableDebugging(1);
+   prw1.Initialize();
+
+   std::cout << "prw1 Integrated lumi = " << prw1.GetIntegratedLumi() << " (expected=2.7e-5) " << std::endl;  testValue(prw1.GetIntegratedLumi(),2.7e-5);
+   std::cout << "prw1 periodWeights : " << prw1.GetPeriodWeight(100,2002) << " (expected=1.6666) " << prw1.GetPeriodWeight(101,2002) << " (expected=0.5555)" << std::endl; testValue(prw1.GetPeriodWeight(100,2002),1.6666);
+   std::cout << "prw1 primaryWeights : " << prw1.GetPrimaryWeight(100,2002,0.5) << " (expected=0.5555) " << prw1.GetPrimaryWeight(100,2002,1.5) << " (expected=1.4444) " << prw1.GetPrimaryWeight(101,2002,0.5) << " (expected=inf .. no MC) " <<  prw1.GetPrimaryWeight(101,2002,1.5) << " (expected=2.3333) " << prw1.GetPrimaryWeight(101,2002,2.5) << " (expected=0.6666) " << prw1.GetPrimaryWeight(101,2002,3.5) << " (expected=0.) " << std::endl;
+   testValue(prw1.GetPrimaryWeight(100,2002,0.5),0.5555);
+   testValue(prw1.GetPrimaryWeight(100,2002,1.5),1.4444);
+   testValue(prw1.GetPrimaryWeight(101,2002,1.5),2.3333);
+   testValue(prw1.GetPrimaryWeight(101,2002,2.5),0.6666);
+   testValue(prw1.GetPrimaryWeight(101,2002,3.5),0);
+
+   //test the random run number and lumiblock number generating 
+   int num1(0); int lnum1(0);
+   for(int i=0;i<1000000;i++) {
+      if(prw1.GetRandomRunNumber(100)==1) { //returns 1 or 2 in this test case .. should give 1 about 15/18ths of the time
+         num1++; 
+         if(prw1.GetRandomLumiBlockNumber(1)==1) lnum1++;
+      }
+   }
+   std::cout << "prw1 fraction of run1 = " << float(num1)/1000000 << " (expected=0.833333) " << " fraction of these with lbn1 = " << float(lnum1)/num1 << " (expected=0.333333) " << std::endl;
+   testValue(float(num1)/1000000,0.8333333);
+   testValue(float(lnum1)/num1,0.333333);
+
+   //test the DataWeights functionality .. first test the IntegratedLumi ...
+   std::cout << "prw1 TriggerA lumi = " << prw1.GetIntegratedLumi("TriggerA") << " (expected=9.58333e-6) " << std::endl; testValue(prw1.GetIntegratedLumi("TriggerA"),9.58333e-6);
+   std::cout << "prw1 TriggerB lumi = " << prw1.GetIntegratedLumi("TriggerB") << " (expected=4.79166e-6) " << std::endl; testValue(prw1.GetIntegratedLumi("TriggerB"),4.79166e-6);
+   std::cout << "prw1 TriggerA&&TriggerB lumi = " << prw1.GetIntegratedLumi("TriggerA&&TriggerB") << " (expected=1.81597e-6) " << std::endl;testValue(prw1.GetIntegratedLumi("TriggerA&&TriggerB"),1.81597e-6);
+
+   //now check the dataweight .. remember the runNumber is converted into a periodNumber (i.e. the weights aren't unique to a runNumber, they are unique to the periodNumber)
+   std::cout << "prw TriggerA DataWeights (mu independent): " << prw1.GetDataWeight(1,"TriggerA") << " (expected=2.63415) " << prw1.GetDataWeight(2,"TriggerA") << " (expected=2.63415) " << prw1.GetDataWeight(3,"TriggerA") << " (expected=3.2727)" << std::endl;
+   std::cout << "prw TriggerA DataWeights (mu dependent): " << prw1.GetDataWeight(1,"TriggerA",0.5) << " (expected=2) " << prw1.GetDataWeight(1,"TriggerA",1.5) << " (expected=3) " << prw1.GetDataWeight(2,"TriggerA",1.5) << " (expected=3.0000) " << prw1.GetDataWeight(3,"TriggerA",1.5) << " (expected=4) " << prw1.GetDataWeight(3,"TriggerA",2.5) << " (expected=2) " << prw1.GetDataWeight(3,"TriggerA",3.5) << " (expected=nan) " << std::endl;
+   testValue(prw1.GetDataWeight(1,"TriggerA"),2.63415);
+   testValue(prw1.GetDataWeight(2,"TriggerA"),2.63415);
+   testValue(prw1.GetDataWeight(3,"TriggerA"),3.2727);
+   testValue(prw1.GetDataWeight(1,"TriggerA",0.5),2.);
+   testValue(prw1.GetDataWeight(2,"TriggerA",1.5),3.);
+   testValue(prw1.GetDataWeight(3,"TriggerA",1.5),4);
+   testValue(prw1.GetDataWeight(3,"TriggerA",2.5),2);
+   //testValue(prw1.GetDataWeight(3,"TriggerA",3.5),nan);
+
+   try {
+      CP::TPileupReweighting t;
+      t.AddConfigFile("dummy2.prw.root");
+      t.AddLumiCalcFile("dummy.None.lumicalc.root");
+      t.Initialize();
+   } catch(...) {
+      std::cout << "exception thrown as expected" << std::endl;
+   }
+
+   CP::TPileupReweighting prw2;
+   prw2.AddConfigFile("dummy2.prw.root");
+   prw2.AddLumiCalcFile("dummy.None.lumicalc.root");
+   prw2.SetUnrepresentedDataAction(2,1.);
+   prw2.Initialize();
+
+   std::cout << "prw2 channel2000 combined weights: " << prw2.GetCombinedWeight(100,2000,0.5) << " (expected 0.9259) " << prw2.GetCombinedWeight(100,2000,1.5) << " (expected 2.40741) " << prw2.GetCombinedWeight(101,2000,1.5) << " (expected=0.432099) " << std::endl;
+   std::cout << "prw2 channel2000 'sum of weights': " << prw2.GetCombinedWeight(100,2000,0.5)+prw2.GetCombinedWeight(100,2000,1.5)+3*prw2.GetCombinedWeight(101,2000,1.5) << " from 5 events (sum of evtweights=5)" << std::endl;
+   std::cout << "prw2 channel2000 unrepresented weights: " << prw2.GetUnrepresentedDataWeight(100,2000) << " " << prw2.GetUnrepresentedDataWeight(101,2000) << std::endl;
+   std::cout << "prw2 channel2000 combined*unrepresented weight: "  << prw2.GetCombinedWeight(100,2000,0.5)*prw2.GetUnrepresentedDataWeight(100,2000)+prw2.GetCombinedWeight(100,2000,1.5)*prw2.GetUnrepresentedDataWeight(100,2000)+3*prw2.GetCombinedWeight(101,2000,1.5)*prw2.GetUnrepresentedDataWeight(101,2000) << " from 5 events (sum of evtweights=5)" << std::endl;
+
+   std::cout << "prw2 channel2001 combined weights: " << prw2.GetCombinedWeight(100,2001,1.5) << " (expected 3.37037) " << prw2.GetCombinedWeight(101,2001,1.5) << " (expected 0.45370) " << prw2.GetCombinedWeight(101,2001,2.5) << " (expected=0.259259) " << std::endl;
+   std::cout << "prw2 channel2001 'sum of weights': " << 0.5*prw2.GetCombinedWeight(100,2001,1.5)+2.*prw2.GetCombinedWeight(101,2001,1.5)+1.*prw2.GetCombinedWeight(101,2001,2.5) << " from 3 events (of various weights, sum of evtweights=3.5)" << std::endl;
+   std::cout << "prw2 channel2001 unrepresented weight: "  << prw2.GetUnrepresentedDataWeight(100,2001) << " " << prw2.GetUnrepresentedDataWeight(101,2001) << std::endl;
+   std::cout << "prw2 channel2001 combined*unrepresented weight: "  << 0.5*prw2.GetCombinedWeight(100,2001,1.5)*prw2.GetUnrepresentedDataWeight(100,2001)+2.*prw2.GetCombinedWeight(101,2001,1.5)*prw2.GetUnrepresentedDataWeight(101,2001)+1.*prw2.GetCombinedWeight(101,2001,2.5)*prw2.GetUnrepresentedDataWeight(101,2001) << " from 3 events (of various weights, sum of evtweights=3.5)" << std::endl;
+
+   //repeat with action=3 ... 
+   CP::TPileupReweighting prw3;
+   prw3.AddConfigFile("dummy2.prw.root");
+   prw3.AddLumiCalcFile("dummy.None.lumicalc.root");
+   prw3.SetUnrepresentedDataAction(3,1.);
+   prw3.Initialize();
+
+   std::cout << "prw3 channel2000 combined weights: " << prw3.GetCombinedWeight(100,2000,0.5) << " (expected 0.0) " << prw3.GetCombinedWeight(100,2000,1.5) << " (expected 3.33333) " << prw3.GetCombinedWeight(101,2000,1.5) << " (expected=0.555555) " << std::endl;
+   std::cout << "prw3 channel2000 'sum of weights': " << prw3.GetCombinedWeight(100,2000,0.5)+prw3.GetCombinedWeight(100,2000,1.5)+3*prw3.GetCombinedWeight(101,2000,1.5) << " from 5 events (sum of evtweights=5)" << std::endl;
+
+   std::cout << "prw3 channel2001 combined weights: " << prw3.GetCombinedWeight(100,2001,1.5) << " (expected 4.66666) " << prw3.GetCombinedWeight(101,2001,1.5) << " (expected 0.58333) " << prw3.GetCombinedWeight(101,2001,2.5) << " (expected=0.259259) " << std::endl;
+   std::cout << "prw3 channel2001 'sum of weights': " << 0.5*prw3.GetCombinedWeight(100,2001,1.5)+2.*prw3.GetCombinedWeight(101,2001,1.5)+1.*prw3.GetCombinedWeight(101,2001,2.5) << " from 3 events (of various weights, sum of evtweights=3.5)" << std::endl;
+
+
+   //CP::TPileupReweighting t;
+
+   //t.AddLumiCalcFile("/usera/will/testareas/ZZ8TeV/ZZAnalysis/share/GRL_v61.None.lumicalc.root");
+
+
+
+
+   return 0;
+}
\ No newline at end of file