Skip to content
Snippets Groups Projects
Geant4AssemblyVolume.cpp 6.59 KiB
Newer Older
//==========================================================================
//  AIDA Detector description implementation 
//--------------------------------------------------------------------------
// Copyright (C) Organisation europeenne pour la Recherche nucleaire (CERN)
// All rights reserved.
//
// For the licensing terms see $DD4hepINSTALL/LICENSE.
// For the list of contributors see $DD4hepINSTALL/doc/CREDITS.
//
// Author     : M.Frank
//
//==========================================================================

// Geant4 include files
#include "DD4hep/DetectorTools.h"
#include "G4LogicalVolume.hh"
#include "G4VPhysicalVolume.hh"
#include "G4ReflectionFactory.hh"

// C/C++ include files
#include <sstream>
#include <string>

// Framework include files
#include "DDG4/Geant4GeometryInfo.h"
#include "DDG4/Geant4AssemblyVolume.h"

using namespace dd4hep::sim;

Markus FRANK's avatar
Markus FRANK committed
/// Default constructor with initialization
Geant4AssemblyVolume::Geant4AssemblyVolume() {
  m_assembly = new G4AssemblyVolume();
}

      
/// Default destructor
Geant4AssemblyVolume::~Geant4AssemblyVolume()   {
  // Do not delete the G4 assembly. This is done by Geant4!
  m_assembly = nullptr;
}

long Geant4AssemblyVolume::placeVolume(const TGeoNode* n,
				       G4LogicalVolume* pPlacedVolume,
				       G4Transform3D& transformation)
{
Markus FRANK's avatar
Markus FRANK committed
  size_t id = m_assembly->TotalTriplets();
Markus FRANK's avatar
Markus FRANK committed
  m_assembly->AddPlacedVolume(pPlacedVolume, transformation);
  return (long)id;
}

long Geant4AssemblyVolume::placeAssembly(const TGeoNode* n,
					 Geant4AssemblyVolume* pPlacedVolume,
					 G4Transform3D& transformation)
{
Markus FRANK's avatar
Markus FRANK committed
  size_t id = m_assembly->TotalTriplets();
Markus FRANK's avatar
Markus FRANK committed
  m_assembly->AddPlacedAssembly(pPlacedVolume->m_assembly, transformation);
void Geant4AssemblyVolume::imprint(Geant4GeometryInfo&   info,
                                   const TGeoNode*       parent,
                                   Chain                 chain,
                                   Geant4AssemblyVolume* pParentAssembly,
                                   G4LogicalVolume*      pMotherLV,
                                   G4Transform3D&        transformation,
                                   G4int                 copyNumBase,
                                   G4bool                surfCheck)
{
Markus FRANK's avatar
Markus FRANK committed
  struct _Wrap : public G4AssemblyVolume  {
    void imprintsCountPlus()  {   this->G4AssemblyVolume::ImprintsCountPlus(); }
  } *parent_wrapper = nullptr;
  static int level=0;
  TGeoVolume* vol = parent->GetVolume();
  unsigned int numberOfDaughters = (copyNumBase == 0) ? pMotherLV->GetNoDaughters() : copyNumBase;

Markus FRANK's avatar
Markus FRANK committed
  parent_wrapper = (_Wrap*)pParentAssembly->m_assembly;
  ++level;

  // We start from the first available index
  //
  numberOfDaughters++;
Markus FRANK's avatar
Markus FRANK committed
  parent_wrapper->imprintsCountPlus();

  //cout << " Assembly:" << detail::tools::placementPath(chain) << endl;

Markus FRANK's avatar
Markus FRANK committed
  std::vector<G4AssemblyTriplet>::iterator iter = pParentAssembly->m_assembly->GetTripletsIterator();
  for( unsigned int i = 0, n = parent_wrapper->TotalTriplets(); i < n; i++, iter++ )  {
Markus FRANK's avatar
Markus FRANK committed
    const auto& triplet = *iter;
    const TGeoNode* node = pParentAssembly->m_entries[i];

    new_chain.emplace_back(node);
    //cout << " Assembly: Entry: " << detail::tools::placementPath(new_chain) << endl;

Markus FRANK's avatar
Markus FRANK committed
    G4Transform3D Ta( *(triplet.GetRotation()), triplet.GetTranslation() );
    if ( triplet.IsReflection() )  {
      Ta = Ta * G4ReflectZ3D();
    }
    G4Transform3D Tfinal = transformation * Ta;
Markus FRANK's avatar
Markus FRANK committed
    if ( triplet.GetVolume() )    {
      // Generate the unique name for the next PV instance
      // The name has format:
      //
      // av_WWW_impr_XXX_YYY_ZZZ
      // where the fields mean:
      // WWW - assembly volume instance number
      // XXX - assembly volume imprint number
      // YYY - the name of a log. volume we want to make a placement of
      // ZZZ - the log. volume index inside the assembly volume
      //
      std::stringstream pvName;
#if 0
      pvName << "av_"
Markus FRANK's avatar
Markus FRANK committed
             << m_assembly->GetAssemblyID()
Markus FRANK's avatar
Markus FRANK committed
             << triplet.GetVolume()->GetName().c_str()
Markus FRANK's avatar
Markus FRANK committed
             << m_assembly->GetAssemblyID()
             << std::ends;
      // Generate a new physical volume instance inside a mother
      // (as we allow 3D transformation use G4ReflectionFactory to
      //  take into account eventual reflection)
      //
#if 0
      printout(INFO,"Geant4Converter","++ Place %svolume %s in assembly.",
Markus FRANK's avatar
Markus FRANK committed
	       triplet.IsReflection() ? "REFLECTED " : "",
	       detail::tools::placementPath(new_chain).c_str());
#endif
      G4PhysicalVolumesPair pvPlaced
        = G4ReflectionFactory::Instance()->Place( Tfinal,
                                                  pvName.str().c_str(),
Markus FRANK's avatar
Markus FRANK committed
                                                  triplet.GetVolume(),
                                                  pMotherLV,
                                                  false,
                                                  numberOfDaughters + i,
                                                  surfCheck );

      // Register the physical volume created by us so we can delete it later
      //
      //fPVStore.emplace_back( pvPlaced.first );
      info.g4VolumeImprints[vol].emplace_back(new_chain,pvPlaced.first);
#if 0
      cout << " Assembly:Parent:" << parent->GetName() << " " << node->GetName()
           << " " <<  (void*)node << " G4:" << pvName.str() << " Daughter:"
           << detail::tools::placementPath(new_chain) << endl;
      cout << endl;
#endif

      if ( pvPlaced.second )  {
        G4Exception("Geant4AssemblyVolume::imprint(..)", "GeomVol0003", FatalException,
                    "Fancy construct popping new mother from the stack!");
        //fPVStore.emplace_back( pvPlaced.second );
      }
    }
Markus FRANK's avatar
Markus FRANK committed
    else if ( triplet.GetAssembly() )  {
      // Place volumes in this assembly with composed transformation
Markus FRANK's avatar
Markus FRANK committed
      imprint(info, parent, new_chain, (Geant4AssemblyVolume*)triplet.GetAssembly(),
              pMotherLV, Tfinal, i*100+copyNumBase, surfCheck );
    }
    else   {
      --level;
      G4Exception("Geant4AssemblyVolume::imprint(..)", "GeomVol0003", FatalException,
                  "Triplet has no volume and no assembly");
    }
  }
  //cout << "Imprinted assembly level:" << level << " in mother:" << pMotherLV->GetName() << endl;
  --level;
}