GAUDI User Guide

Chapter 4
Getting started

4.1 Overview

In this chapter we walk through one of the example applications ( Histograms ) 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

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

.

Figure 4 Creating a job from the Histogram example application

 

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 2 on See The example main program.. 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 38 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 40 and 41 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 46.
Program execution
All of the code before line 54 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 (or possibly a database in future) for job configuration. The job options file of the Histograms example application is shown in Listing 3 on See The job options file for the Histograms example application..

The format of an options file is discussed fully in Chapter 11. Options may be set both for algorithms and services and the list of available options for standard components is given in Appendix Appendix B.

For the moment we look briefly at one of the options. The option TopAlg of the application manager 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. Thus, in the example in Listing 3 line 13, an instance of the class "HistoAlgorithm" will be created with name "HistoAlgorithm"

.

Listing 2 The example main program.

21: // Include files
22: #include "Gaudi/Kernel/SmartIF.h"
23: #include "Gaudi/Kernel/Bootstrap.h"
24: #include "Gaudi/Interfaces/IAppMgrUI.h"
25: #include "Gaudi/Interfaces/IProperty.h"
26: #include "Gaudi/JobOptionsSvc/Property.h"
27: //---------------------------------------------------------------------
28: // Package : Gaudi Examples
29: // Description: Main Program
30: //---------------------------------------------------------------------
31:
32: //--- Example main program
33: int main(int argc, char* argv[]) {
34:
35: StatusCode status = StatusCode::SUCCESS;
36:
37: // Create an instance of an application manager
38: IInterface* iface = Gaudi::createApplicationMgr();
39:
40: SmartIF<IProperty> propMgr ( IID_IProperty, iface );
41: SmartIF<IAppMgrUI> appMgr ( IID_IAppMgrUI, iface );
42:
43: // Set properties of algorithms and services
44: if ( propMgr == iface ) {
45: std::string opts = (argc>1) ? argv[1] : "jobOptions.txt";
46: status = propMgr->setProperty( StringProperty("JobOptionsPath", "jobOptions.txt") );
47: }
48: else {
49: exit(0);
50: }
51:
52: // Run the application manager and process events
53: if ( appMgr ) {
54: status = appMgr->run();
55: }
56: else {
57: return 0;
58: }
59:
60: // All done - exit
61: return 0;
62: }

 

Listing 3 The job options file for the Histograms example application.

1: //##############################################################
2: //
3: // Histograms job options file
4: //
5: //==============================================================
6: #include "$STDOPTS/Common.txt"
7:
8: //--------------------------------------------------------------
9: // Private Application Configuration options
10: //--------------------------------------------------------------
11:
12: ApplicationMgr.DLLs += {"RootHistCnv"};
13: ApplicationMgr.TopAlg = { "HistoAlgorithm" };
14:
15: // Specify ROOT or HBOOK persistency (HBOOK is default)
16: ApplicationMgr.HistogramPersistency = "ROOT";
17:
18: // Set output level threshold (2=DEBUG,3=INFO,4=WARNING,5=ERROR,6=FATAL)
19: MessageSvc.OutputLevel = 3;
20:
21: //--------------------------------------------------------------
22: // Event related parameters
23: //--------------------------------------------------------------
24: // Number of events to be processed (default is all events)
25: ApplicationMgr.EvtMax = 100;
26: // Print event number every event
27: EventSelector.PrintFreq = 1;
28:
29: //--------------------------------------------------------------
30: // Input file
31: //--------------------------------------------------------------
32: EventSelector.Input = {"FILE='$SITEROOT/lhcb/data/mc/sicb_bpipi_v233_100ev.dst1'"};
33: //EventSelector.Input = {"JOBID='16434'"};
34:
35: //--------------------------------------------------------------
36: // Histogram output file
37: //--------------------------------------------------------------
38:
39: // Specify Histogram Persistency Output File
40: HistogramPersistencySvc.OutputFile = "histo.rt";
41: //HistogramPersistencySvc.OutputFile = "histo.hbook";
42:
43: //--------------------------------------------------------------
44: // Algorithms Private Options
45: //--------------------------------------------------------------
46: HistoAlgorithm.HistogramFlag = true; // false = no histograms produced
47: HistoAlgorithm.OutputLevel = 3; // enable debug messages

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 HistoAlgorithm in the 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 HistoAlgorithm header file

The HistoAlgorithm class definition is shown in Listing 4.

Note the following:

 

Listing 4 The header file of the class: HistoAlgorithm.

1: #include "Gaudi/Algorithm/Algorithm.h" // Required for inheritance
2: class IHistogram1D; // Forward declaration
3: class IHistogram2D; // Forward declaration
4:
5: class HistoAlgorithm : public Algorithm {
6: public:
7: // Constructor of this form must be provided
8: HistoAlgorithm(const std::string& name, ISvcLocator* pSvcLocator);
9:
10: // Three mandatory member functions of any algorithm
11: StatusCode initialize();
12: StatusCode execute();
13: StatusCode finalize();
14:
15: private:
16: // These data members are used in the execution of this algorithm
17: // They are set in the initialisation phase by the job options service
18: bool m_produceHistogram;
19: // Two histograms ( used if m_produceHistogram = 1 (true) )
20: IHistogram1D* m_hTrackCount;
21: IHistogram2D* m_hPtvsP;
22: };

4.5.2 The HistoAlgorithm 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_produceHistogram flag to false and set the histogram pointers to zero. This is also the place to declare any member variables that you wish to be set by the job options service. This is done by making a call to the declareProperty() method.

 

1: HistoAlgorithm::HistoAlgorithm(const std::string& name,
2: ISvcLocator* pSvcLocator) :
3: Algorithm(name, pSvcLocator),
4: m_hTrackCount(0), m_hPtvsP(0) {
5: // Declare the algorithm's properties
6: declareProperty( "HistogramFlag", m_produceHistogram = false ); }
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 we see that depending on the value of the histogram flag the HistoAlgorithm class may also create two histograms at initialisation and declare them to the histogram data service (see Chapter 9 for details on the use of histograms in Gaudi):

Listing 5 Example of creation of histograms

1: if( m_produceHistogram ) {
2: log << MSG::INFO << "Histograms will be produced" << endreq;
3:
4: m_hTrackCount = histoSvc()->book( "/stat/simple1D/1", "TrackCount", 100,0,3000);
5: if( 0 == m_hTrackCount ) {
6: log << MSG::ERROR << "Cannot register histo TrackCount"<<endreq;
7: }
8: m_PtvsP = histoSvc()->book("stat/simple2D/3","Pt versus P", 100,0,200,100,0,5);
9: if( 0 == m_PtvsP ) {
10: log << MSG::ERROR << "Cannot register histo Pt versus P"<<endreq;
11: }
12: }
execution
The execute method is where most of the real action takes place. This method is called by the application manager once for every event. The HistoAlgorithm class accesses the event data store to retrieve a container of track objects (see Listing 6, line 2). It then fills one of the two previously created histograms with the number of particles (line 15).

In order to use the objects within the container an iterator is defined (line 19) and the second histogram is filled with the momentum of the tracks (line 24).

Listing 6 execute() method of the class HistoAlgorithm

1: // MCParticle
2: SmartDataPtr<MCParticleVector> particles( eventSvc(),
3: "/Event/MC/MCParticles" );
4:
5: if( !particles ) {
6: log << MSG::ERROR << "Unable to retrieve MCParticles" << endreq;
7: return 0;
8: }
9: log << MSG::INFO << "+++++ MCParticles retrieved +++++" << endreq;
10:
11: // Histograms
12: if( m_produceHistogram ) {
13:
14: // Fill the track count histogram
15: m_hTrackCount->fill(particles->size(), 1.);
16:
17: // Fill the Pt versus P scatterplot
18: // Iterate over all tracks in the track container
19: MCParticleVector::iterator iterP = 0;
20: for( iterP = particles->begin(); iterP != particles->end(); iterP++ ) {
21: // Get the energy of the track, convert it to GeV, and histogram it
22: double pt = (*iterP)->fourMomentum().perp();
23: double p = (*iterP)->fourMomentum().rho();
24: m_hPtvsP->fill( p / GeV, pt / GeV, 1. );
25: }
26: }

The details of the event data store and how data is accessed are discussed in Chapter 6.

Finalisation
At finalisation, the HistoAlgorithm class asks the histogram service to printout the histograms in different formats.
MsgStream
The HistoAlgorithm class makes use of the message service and the MsgStream utility class in order to print out some progress information, for example:
MsgStream          log( msgSvc(), name() );	log << MSG::INFO << "finalize" << endreq;

The first line creates a local MsgStream object, 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 11.

4.6 Job execution

From the main program and the requirements file we can make an executable. This executable together with the file of job options form a job which may be submitted for batch or run interactively. Figure 5 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 5 A sequence diagram showing a part of the execution of the 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 also an EmptyAlgorithm as well as the HistoAlgorithm of the example
  3. The top-level algorithms are initialised. They may request that their properties (if they have any) are set and may make use of the message service. If any algorithm fails to initialise the job is aborted.
  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. 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.
  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 Simple Physics Analysis Example

4.8 Other examples distributed with Gaudi

A number of examples is included in the current release of the framework. The intention is that each example exercises and shows to an end-user how to make use of some part of the functionality of the framework. The following table shows the list of available examples.

Table 4 List of available examples in current release

Example Name
Target Functionality

AlgSequencer

Illustrating the use of the sequencer algorithm provided in the GaudiAlg package

Associator

Illustrating the use of the AxPart2MCParticleAsct associator tool to navigate between reconstructed tracks and Monte Carlo particles

Common

Actually not a complete example: contains main program used by many examples (AlgSequencer, DDexample, DumpEvent, FieldGeom, FortranAlgorithm, Histograms, MCPrimaryVertex, ParticleProperties, SimpleAnalysis, ToolsAnalysis) and system specific Job Options include files common to all examples

DDexample

Illustrating the use of the detector description

DumpEvent

Navigation of the LHCb transient event data model

FieldGeom

Making available existing Sicb magnetic field and geometry data to Gaudi algorithms. Example of nested algorithms

FortranAlgorithm

Wrapping Fortran code in Gaudi

Histograms

Basic functionality of the framework to execute a simple algorithm, access event data and fill histograms.

MCPrimaryVertex

Retrieve data using SmartDataPtr

Ntuples

Two examples, reading and writing Ntuples

ParticleProperties

Access the Particle Properties service to retrieve Particle Properties

RandomNumber

Example of use of the Random Number service

Rio.Example1

Two examples, reading and writing persistent data with ROOT I/O

SimpleAnalysis

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

ToolsAnalysis

Example of use of framework tools in an analysis