#include #include #include #include #include #include #include using namespace std; #include #include #include #include "HDGEOMETRY/DMagneticFieldMapSpoiled.h" #include "HDGEOMETRY/DMagneticFieldMapParameterized.h" //---------------------------------------------------------- // This file contains routines used to implement dynamic // geometry linking in hdgeant. The routines generated // by the hdds-geant utility (part of the HDDS) package // are what are linked here. Here is how it works: // // The routines that were called from the hdgeant core // have been replaced with a corresponding wrapper routine // that is defined here. The wrapper simply calls the // actual function using a function pointer. For example: // // void md5geom_wrapper_(char *md5){ // // (*md5geom_ptr)(md5); // } // // where the global variable md5geom_ptr is of type // pointer to a function with format: // "void md5geom_(char *md5)" // // Whereas previously one might simply call: // // md5geom_(md5); // // now one calls: // // md5geom_wrapper(md5); // // // The function pointers are all initialized to point to // the statically linked routines so that they are what // is used if the user does not specify the XML geometry // be used. The pointers are initialized to point to the // static routines with statements like this: // // typeof(md5geom_) *md5geom_ptr = md5geom_; // // where the use of typeof() makes it so we don't have // to enter the signature of the function twice. // // If the user does specify that the XML source be used // to dynamically regenerate the geometry, then the // pointers are updated to point to routines found in // the shared object. The shared object is generated at // run time using the hdds-geant utility to generate a // FORTRAN source file which is then compiled into a // shared object and opened using the dl library. // // This should work for most all modifications to the // geometry. The one problem would be if a major change // was made that required another routine be generated // by hdds-geant. For example, a getsection_() routine // is added. In that case, one would need to add the // declaration of a wrapper routine below and a function // pointer for it. One would also need to add a call // to GetRoutine() and then make sure all places in the // code using the routine called the wrapper function. //---------------------------------------------------------- extern "C" { void init_runtime_xml_(void); // Declare statically linked routines void hddsgeant3_(void); void md5geom_(char *md5); float guplsh_(int *medi0, int *medi1); void gufld_(float *r, float *B); void getoptical_(int *imat, float *E, float *refl, float *abs1, float *rind, float *plsh, float *eff); // Initialize routine pointers to use statically linked routines __typeof__(hddsgeant3_) *hddsgeant3_ptr = hddsgeant3_; __typeof__(md5geom_) *md5geom_ptr = md5geom_; __typeof__(guplsh_) *guplsh_ptr = guplsh_; __typeof__(gufld_) *gufld_ptr = gufld_; __typeof__(getoptical_) *getoptical_ptr = getoptical_; // Trivial wrapper routines use pointer to dispatch call void hddsgeant3_wrapper_(void){ (*hddsgeant3_ptr)(); } void md5geom_wrapper_(char *md5){ (*md5geom_ptr)(md5); } float guplsh_wrapper_(int *medi0, int *medi1){ return (*guplsh_ptr)(medi0, medi1); } void gufld_wrapper_(float *r, float *B){ (*gufld_ptr)(r, B); } void getoptical_wrapper_(int *imat, float *E, float *refl, float *abs1, float *rind, float *plsh, float *eff){ (*getoptical_ptr)(imat, E, refl, abs1, rind, plsh, eff);} // Below are several more routines which need to be implemented // using the same three steps as above. However, all of them // have the same format of returning an int and taking no // arguments. We compact things a bit using the following // macro which would expand to something like this: // // int getcolumn_(void); // int (*getcolumn_ptr)(void) = getcolumn_; // int getcolumn_wrapper_(void) { return ( *getcolumn_ptr)(); } // #define MakeDispatcherINT(N) \ int N(void); \ int (* N ## ptr)(void) = N; \ int N ## wrapper_(void) { return ( *N ## ptr)(); } MakeDispatcherINT(getcolumn_); MakeDispatcherINT(getlayer_); MakeDispatcherINT(getmap_); MakeDispatcherINT(getmodule_); MakeDispatcherINT(getpackage_); MakeDispatcherINT(getplane_); MakeDispatcherINT(getring_); MakeDispatcherINT(getrow_); MakeDispatcherINT(getsector_); } void GetRoutine(void **ptr, const char *rname); static void *dlgeom_handle=NULL; string HDDS_XML = "$HDDS_HOME/main_HDDS.xml"; //------------------ // init_runtime_xml_ //------------------ void init_runtime_xml_(void) { int retcode; cout< tmp.F"; cout << cmd << endl; retcode = system(cmd.c_str()); // Compile FORTRAN into shared object cout<