TOC PREV NEXT INDEX

Gaudi logo


Chapter 4
Getting started
4.1 Overview

In this chapter we walk through one of the example applications (RandomNumber) which are distributed with the framework. We look briefly at the different files and go over the steps needed to compile and execute the code. We also outline where various subjects are covered in more detail in the remainder of the document. Finally we cover briefly the other example applications which are distributed and say a few words on what each one is intended to demonstrate.

4.2 Creating a job

Traditionally, a "job" is the running of a program on a specified set of input data to produce a set of output data, usually in batch.

For the example applications supplied this is essentially a two step process. First the executable must be produced, and secondly the necessary environment variables must be set and the required job options specified, as illustrated in Figure 4.1.
Figure 4.1 Creating a job from the AlgSequencer example application

The example applications consist of a number of "source code" files which together allow you to generate an executable program. These are:

· The main program.
· Header and implementation files for each of the concrete algorithm classes.
· A CMT requirements file.
· The set of Gaudi libraries.

In order for the job to run as desired you must provide the correct configuration information for the executable. This is done via entries in the job options file.

4.3 The main program

An example main program is shown in Listing 4.1.
Listing 4.1 The example main program.
1: // Include files
2: #include "GaudiKernel/SmartIF.h"
3: #include "GaudiKernel/Bootstrap.h"
4: #include "GaudiKernel/IAppMgrUI.h"
5: #include "GaudiKernel/IProperty.h"
6: #include "GaudiKernel/Property.h"
7:
8: //--- Example main program
9: int main(int argc, char* argv[]) {
10:
11: StatusCode status = StatusCode::SUCCESS;
12:
13: // Create an instance of an application manager
14: IInterface* iface = Gaudi::createApplicationMgr();
15:
16: SmartIF<IProperty> propMgr ( IID_IProperty, iface );
17: SmartIF<IAppMgrUI> appMgr ( IID_IAppMgrUI, iface );
18:
19: // Set properties of algorithms and services
20: if ( propMgr == iface ) {
21: std::string opts = (argc>1) ? argv[1] : "jobOptions.txt";
22: status=propMgr->setProperty(StringProperty("JobOptionsPath", opts));
23: }
24: else {
25: exit(0);
26: }
27:
28: // Run the application manager and process events
29: if ( appMgr ) {
30: status = appMgr->run();
31: }
32: else {
33: return 0;
34: }
35:
36: // All done - exit
37: return 0;
38: }
It is constructed as follows:

Include files
These are needed for the creation of the application manager and Smart interface pointers.
Application Manager instantiation
Line 14 instantiates an ApplicationMgr object. The application manager is essentially the job controller. It is responsible for creating and correctly initialising all of the services and algorithms required, for looping over the input data events and executing the algorithms specified in the job options file, and for terminating the job cleanly.
Retrieval of Interface pointers
The code on lines 16 and 17 retrieves the pointers to the IProperty and IAppMgrUI interfaces of the application manager.
Setting the application manager's properties
The only property which needs to be set explicitly in the main program is the name of the job options file which contains all of the other configuration information needed to run the job. This is done on line 22.
Program execution
All of the code before line 30 is essentially for setting up the job. Once this is done, a call to ApplicationMgr::run() is all that is needed to start the job proper! The steps that occur within this method are discussed briefly in section 4.6
4.4 Configuring the job

The application framework makes use of a job options file for job configuration. Part of the job options file of an example application is shown in Listing 4.2.
Listing 4.2 Part of the job options file for the RootIO example application.
1: // Include standard option files
2: #include "$STDOPTS/Common.txt"
3:
4: // Private Application Configuration options
5: ApplicationMgr.DLLs += { "GaudiDb", "GaudiRootDb" };
6: ApplicationMgr.ExtSvc += { "DbEventCnvSvc/RootEvtCnvSvc" };
7: ApplicationMgr.TopAlg = { "ReadAlg" };
8:
9: // Set output level threshold (2=DEBUG,3=INFO,4=WARNING,5=ERROR,6=FATAL)
10: MessageSvc.OutputLevel = 4;
11: EventSelector.OutputLevel = 2;
12:
13: // Input File
14: EventSelector.Input = {"DATAFILE='RootDst.root' TYP='ROOT' OPT='READ'"};
15: EventSelector.FirstEvent = 1;
16: ApplicationMgr.EvtMax = 5;
17:
18: // Persistency service setup:
19: EventPersistencySvc.CnvServices += { "RootEvtCnvSvc" };
20:
21: // Setup for ROOT I/O system
22: RootEvtCnvSvc.DbType = "ROOT";

The format of an options file is discussed fully in Chapter 12. Options may be set both for algorithms and services and the list of available options for standard components is given in Appendix B. Here we look briefly at a few of the more commonly used options.

4.4.1 Defining the algorithms to be executed

The option ApplicationMgr.TopAlg (line 7) is a list of algorithms that will be created and controlled directly by the application manager, the so-called top-level algorithms. The syntax is a list of the form:

ApplicationMgr.TopAlg = { "Type1/Name1", "Type2/Name2" };

The line above instructs the application manager to make two top level algorithms. One of type Type1 called "Name1" and one of type Type2 called "Name2".

In the case where the name of the algorithm is the same as the algorithm's type (i.e. class), only the class name is necessary. In the example, an instance of the class "ReadAlg" will be created with name "ReadAlg".

4.4.2 Defining the job input

Event data input is controlled by an EventSelector. The EventSelector uses a storage technology dependent data persistency service to load the data into the transient event data store, with the help of converters which are able to convert the data from the technology dependent persistent represenation, to the technology independent representation in the transient data store.

In order to set up this mechanism, one needs a number of job options:

- Line 14 defines the input data file, and the persistency technology (ROOT I/O in this example).
- Line 6 tells the application manager to create a new event conversion service, to be called RootEvtCnvSvc. Note that this is just a name for our convenience, the service is of type DbEventCnvSvc and does not (yet) know that it will deal with ROOT technology. The configuration of RootEvtCnvSvc to use the ROOT I/O technology is done in line 22.
- Line 19 tells the event persistency service (EventPersistencySvc created by the application manager by default) to use the RootEvtCnvSvc to do the conversion between persistent and transient data representations.
- Line 5 tells the application manager which additional libraries to load in order to find the required conversion service. In this example, the GaudiDb library contains the DbEventCnvSvc class, the GaudiRootDb library contains the ROOT specific database drivers.
- Finally, the options on lines 15 and 16 tell the EventSelector to start reading sequentially from the first event in the file, for five events.

In the special case where no event input is required (e.g. for event generation), one can replace the above options by the two options:

ApplicationMgr.EvtMax = 20; // events to be processed (default is 10)
ApplicationMgr.EvtSel = "NONE"; // do not use any event input

A discussion of event I/O can be found in Chapter 11. Converters and the conversion process are described in Chapter 14.

4.4.3 Defining job output

One can consider three types of job output: event data (including event collections and n-tuples), statistical data (histograms) and printout. Here we discuss only the simplest (printout); histograms are discussed in Chapter 10, event data in Chapter 11.

Printout in Gaudi is handled by the message service (described in Chapter 12), which allows to control the amount of printout according to severity level. The global threshold for printout is set be the option on line 10 - in this example only messages of severity level WARNING or above will be printed. This can be over-ridden for individual algorithms or services, as in line 11, where the threshold for EventSelector is set to DEBUG.

4.5 Algorithms

The subject of specialising the Algorithm base class to do something useful will be covered in detail in chapter 5. Here we will limit ourselves to looking at the class HelloWorld in the AlgSequencer example just to get an idea of the basics. For the full listing of this class, refer to the software distribution. Here we look only at those parts of the code which do something interesting and leave the technicalities for later.

4.5.1 The HelloWorld.h header file

The HelloWorld class definition is shown in Listing 4.3.
Listing 4.3 The header file of the class: HelloWorld.
1: // Include files
2: #include "GaudiKernel/Algorithm.h" // Required for inheritance
3: #include "GaudiKernel/Property.h"
4: #include "GaudiKernel/MsgStream.h"
5:
6: class HelloWorld : public Algorithm {
7: public:
8: /// Constructor of this form must be provided
9: HelloWorld(const std::string& name, ISvcLocator* pSvcLocator);
10:
11: /// Three mandatory member functions of any algorithm
12: StatusCode initialize();
13: StatusCode execute();
14: StatusCode finalize();
15: private:
16: bool m_initialized;
17: /// These data members are used in the execution of this algorithm
18: /// and are set in the initialisation phase by the job options service
19: IntegerProperty m_int;
20: DoubleProperty m_double;
21: StringProperty m_string;
22: };

Note the following:

· The class is derived from the Algorithm base class as must be all specialised algorithm classes. This implies that the Algorithm.h file must be included (line 6).
· All derived algorithm classes must provide a constructor with the parameters shown in line 9. The first parameter is the name of the algorithm and is used amongst other things to locate any options that may have been specified in the job options file.
· The HistoAlgorithm class has four (private) data members, defined in lines 16 to 21. These are a boolean flag, and pointers to Property objects (discussed later).
· The three methods on lines 12 to 14 must be implemented, since they are pure virtual in the base class.
4.5.2 The HelloWorld implementation file

The implementation file contains the actual code for the constructor and for the methods: initialize(), execute() and finalize(). It also contains two lines of code for the HistoAlgorithm factory, which we will discuss in section 5.3.1

The constructor
must call the base class constructor, passing on its two arguments. As usual, member variables should be initialised. Here we set the m_initialized flag to false. We also declare and initialise the member variables that we wish to be set by the job options service. This is done by calling the declareProperty() method.
1: HelloWorld::HelloWorld(const std::string& name, ISvcLocator* ploc)
2: : Algorithm(name, ploc) {
3: //-----------------------------------------------------------------------
4: m_initialized = false;
5: // Declare the algorithm's properties
6: declareProperty( "Int", m_int = 1);
7: declareProperty( "Double", m_double = 100.);
8: declareProperty( "String", m_string = std::string("gsgsgsgs"));
9: }

Initialisation

The application manager invokes the sysInitialize() method of the algorithm base class which, in turn, invokes the initialize() method of the base class, the setProperties() method, and finally the initialize() method of the concrete algorithm class. As a consequence all of an algorithm's properties will have been set before its initialize() method is invoked, and all of the standard services such as the message service are available. This is discussed in more detail in Chapter 5.

Looking at the code in the example (Listing 4.4) we see that we are now able to print out the values of the algorithm's properties, using the message service and the MsgStream utility class. A local MsgStream object is created (line 6), which uses the Algorithm's standard message service via the msgSvc() accessor, and the algorithm's name via the name() accessor. The use of these is discussed in more detail in Chapter 12.

Note that the job will stop if the initialize() method of any algorithm does not return StatusCode::SUCCESS. This is to avoid processing with a badly configured application.:
Listing 4.4 Example of initialize() method
1: StatusCode HelloWorld::initialize() {
2: //----------------------------------------------------------------------
3: // avoid calling initialize more than once
4: if( m_initialized ) return StatusCode::SUCCESS;
5:
6: MsgStream log(msgSvc(), name());
7: log << MSG::INFO << "initializing...." << endreq;
8: log << MSG::INFO << "Property Int = " << m_int << endreq;
9: log << MSG::INFO << "Property Double = " << m_double << endreq;
10: log << MSG::INFO << "Property String = " << m_string << endreq;
11:
12: m_initialized = true;
13: return StatusCode::SUCCESS;
14: }

execution
The execute() method is called by the application manager once for every event. This is where most of the real action should take place. The trivial HelloWorld class just prints out a message... Note that the method must return StatusCode::SUCCESS on successful completion. If a particular algorithm returns a FAILURE status code more than a (configurable) maximum number of times, the application manager will decide that this algorithm is badly configured and jump to the finalisation stage before all events have been processed.
1: StatusCode HelloWorld::execute() {
2: //----------------------------------------------------------------------
3: MsgStream log( msgSvc(), name() );
4: log << MSG::INFO << "executing...." << endreq;
5:
6: return StatusCode::SUCCESS;
7: }
.
Finalisation
The finalize() method is called at the end of the job. In the trivial example the m_initialized flag is reset and a message is printed.
1: StatusCode HelloWorld::finalize() {
2: //----------------------------------------------------------------------
3: MsgStream log(msgSvc(), name());
4: log << MSG::INFO << "finalizing...." << endreq;
5:
6: m_initialized = false;
7: return StatusCode::SUCCESS;
8: }
.
4.6 Job execution

From the main program and the CMT requirements file we can make an executable, as explained in section 3.5. This executable together with the file of job options form a job which may be submitted for batch or run interactively. Figure 4.2 shows a trace of an example program execution. The diagram is not intended to be complete, merely to illustrate a few of the points mentioned earlier in the chapter.
Figure 4.2 A sequence diagram showing a part of the execution of an example program.

1. The application manager instantiates the required services and initialises them. The message service is done first to allow the other services to use it, and the job options service is second so that the other services may be configured at run time.
2. The algorithms which have been declared to the application manager within the job options (via the TopAlg option) are created. We denote these algorithms "top-level" as they are the only ones controlled directly by the application manager. For illustration purposes we instantiate an EmptyAlgorithm and a HistoAlgorithm.
3. The top-level algorithms are initialised. Their properties (if they have any) are set and they may make use of the message service. If any algorithm fails to initialise, the job is stopped.
4. The application manager now starts to loop over events. After each event is read, it executes each of the top level algorithms in order. The order of execution of the algorithms is the order in which they appear in the TopAlg option. This will continue until the required number of events has been processed, unless one or more of the algorithms return a FAILURE status code more than the maximum number of times, in which case the application manager will jump to the finalisation stage before all events have been processed.
5. After the required data sample has been read the application manager finalises each top level algorithm.
6. Services are finalised.
7. All objects are deleted and resources freed. The program terminates.
4.7 Examples distributed with Gaudi

A number of examples is included in the current release of the framework, in the GaudiExamples package. The package has some sub-directories in addition to the standard ones shown in Figure 16.3. The options sub-directory contains files of standard job options common to many examples. These files are included in the job options of the specific examples when necessary. The specific job options files can be found in the home sub-directory.

The code of the examples is in sub-directories of the src directory, one sub-directory per example. The intention is that each example demonstrates how to make use of some part of the functionality of the framework. The list of available examples is shown in Table 4.1.
Table 4.1 List of examples available in Gaudi release v7
Example Name
Target Functionality
AlgSequencer
Illustrating the use of the sequencer algorithm provided in the GaudiAlg package
Common
Actually not a complete example: contains the main program used in the examples
GPython
Exercise the Python scripting packages
Histograms
Basic functionality of the framework to execute a simple algorithm, access event data and fill histograms.
Properties
Trivial algorithm showing how to set and retrieve Properties
RandomNumber
Example of use of the Random Number service
RootIO
Two examples, reading and writing persistent data with ROOT I/O

4.8 Additional LHCb specific examples

The examples described so far are rather simple and do not contain any specific knowledge about the LHCb event and detector data. A set of LHCb specific examples is provided in the Ex group of packages, as listed in Table 4.2
Table 4.2 List of LHCb specific examples available in the current release
Example Package Name
Target Functionality
Ex/AssociatorExample
Illustrating the use of the AxPart2MCParticleAsct associator tool to navigate between reconstructed tracks and Monte Carlo particles
Ex/DetDescExample
Illustrating the use of the detector description
ExDumpEventExample
Navigation of the LHCb transient event data model
Ex/EvtCollectionExample
Two examples, writing and reading event collections, starting from the vent data file produced by the RootIO example
Ex/FieldGeomExample
Making available existing Sicb magnetic field and geometry data to Gaudi algorithms. Example of nested algorithms
ExFortranAlgorithmExample
Wrapping Fortran code in Gaudi
Ex/GiGaExample
Uses GEANT4 to visualise the LHCb detector geometry and event data, as an example of using the GiGa service to interface Gaudi to GEANT4
Ex/HistogramsExample
Basic functionality of the framework to execute a simple algorithm, access event data and fill histograms.
Ex/MCPrimaryVertexExample
Retrieve data using SmartDataPtr
Ex/NtuplesExample
Two examples, reading and writing Ntuples
Ex/ParticlePropertiesExample
Access the Particle Properties service to retrieve Particle Properties
Ex/RootIOExample
Two examples, reading and writing persistent data with ROOT I/O
Ex/SimpleAnalysisExample
A realistic example of using the framework for a physics analysis, including access to Monte Carlo data, creation of reconstructed data and filling an n-tuple
Ex/ToolsAnalysisExample
Example of use of framework tools in an analysis
.

All examples share a single main program and some default job options, which can be found in the GaudiConf package.

4.8.1 Simple Physics Analysis Example

The algorithms in the examples of Table 4.2 use many of the Gaudi Services that someone would want to be able to utilize while doing physics analysis: histograms, ntuples, creating and retrieving private transient data, retrieving particle properties (like mass values), etc. Detailed examples on how to use the specific services are provided in the topical examples but in the SimpleAnalysisExample they are combined together. Tools to make physics analysis in a more elegant and complex way are under development and their concrete implementation will be part of the OO Physics Analysis Program. A trivial implementation of an algorithm similar to that of the SimpleAnalysisAlgorithm implemented using tools is provided in the ToolAnalysisExample.

The SimpleAnalysisAlgorithm is an example in which pi+ pi- invariant masses are made while requiring the component particles to satisfy some simple kinematic and quality cuts. Private containers of the particles satisfying successive cuts are created and filled (charged particles, detection in the silicon, best particle ID). Invariant masses are made and corresponding histograms are filled for all combinations of the final private containers, for combinations with Pt of both pions greater than a cut value and for combinations with impact parameter of both pions greater then a cut value. The Pt and impact parameter cut values are properties of the algorithm and as such can be specified in the jobOptions, where the number is taken in Gaudi Units. CLHEP vectors' classes are used to evaluate transverse momentum and invariant masses as well as to calculate the impact parameter. When nominal mass values are required they are retrieved via the ParticlePropertySvc. Since a primary vertex is required a "dummy" algorithm RecPrimaryVertex retrieves the Monte Carlo primary vertex and uses the quantites to fill a MyVertex object (/Event/MyAxVertices), which is then retrieved by the SimpleAnalysisAlgorithm. Since the MyVertex object is created and registered in the Transient Event store by the RecPrimaryVertex algorithm, the sequencing of RecPrimaryVertex and SimpleAnalysisAlgorithm in the jobOptions file is very important. A protection is put in place so that the SimpleAnalysisAlgorithm will return a failure code if not all of the necessary input data exist in the store.

When doing physics analysis on Monte Carlo data, it is necessary to compare the reconstructed decay with the Monte Carlo truth in order to calculate efficiencies. The MCDecayFinder algorithm is an example of how to find any one step decay. The parent of the decay and the list of its direct descendants are properties of the algorithm and can be specified in the jobOptions file. If no decay is specified in the jobOptions this example will look for a B0->pi+pi- decay. The Algorithm will retrieve the particle Geant3 ID from the ParticlePropertySvc (the identifying particleID in MCParticles) and search the MCParticles to find the requested parent and that is has the correct type and number of decay products. If a decay is found kinematic variables are stored in an ntuple that can be accessed by PAW. In addition the Algorithm uses the Message service with DEBUG or INFO levels to print a summary of its behaviour for each event as well as for the job.



Quadralay Corporation
http://www.webworks.com
Voice: (512) 719-3399
Fax: (512) 719-3606
sales@webworks.com
TOC PREV NEXT INDEX