source: trunk/MagicSoft/Mars/mhflux/MMcSpectrumWeight.cc@ 8886

Last change on this file since 8886 was 8883, checked in by tbretz, 17 years ago
*** empty log message ***
File size: 20.8 KB
Line 
1/* ======================================================================== *\
2!
3! *
4! * This file is part of MARS, the MAGIC Analysis and Reconstruction
5! * Software. It is distributed to you in the hope that it can be a useful
6! * and timesaving tool in analysing Data of imaging Cerenkov telescopes.
7! * It is distributed WITHOUT ANY WARRANTY.
8! *
9! * Permission to use, copy, modify and distribute this software and its
10! * documentation for any purpose is hereby granted without fee,
11! * provided that the above copyright notice appear in all copies and
12! * that both that copyright notice and this permission notice appear
13! * in supporting documentation. It is provided "as is" without express
14! * or implied warranty.
15! *
16!
17!
18! Author(s): Thomas Bretz 5/2005 <mailto:tbretz@astro.uni-wuerzburg.de>
19! Author(s): Marcos Lopez 10/2003 <mailto:marcos@gae.ucm.es>
20!
21! Copyright: MAGIC Software Development, 2000-2007
22!
23!
24\* ======================================================================== */
25
26//////////////////////////////////////////////////////////////////////////////
27//
28// MMcSpectrumWeight
29//
30// Change the spectrum of the MC showers simulated with Corsika (a power law)
31// to a new one, which can be either, again a power law but with a different
32// spectral index, or a generalizeed spectrum. The new spectrum can be
33// pass to this class in different ways:
34//
35// 1. If the new spectrum will be a power law, just introduce the slope
36// of this power law.
37// 2. If the new spectrum will have a general shape:
38// The new spectrum is passed as a char* (SetFormula())
39//
40// Method:
41// -------
42//
43// - Corsika spectrun: dN/dE = A * E^(a)
44// with a = fOldSlope, and A = N/integral{E*de} from ELowLim to EUppLim
45//
46// - New spectrum: dN/dE = B * g(E)
47// where B = N/integral{g*dE} from ELowLim to EUppLim, and N=NumEvents
48//
49// For converting the spectrum simulated with Corsika to the new one, we
50// apply a weight to each event, given by:
51//
52// W(E) = B/A * g(E)/E^(a)
53//
54// In the case the new spectrum is simply a power law: dN/dE = B * E^(b), we
55// have:
56//
57// W(E) = B/A * E^(b-a)
58//
59// (The factor B/A is used in order both the original and new spectrum have
60// the same area (i.e. in order they represent the same number of showers))
61//
62//
63// If using SetFormula you can specify formulas accepted by TF1, eg:
64// pow(X, -2.6)
65// (Rem: all capital (!) 'X' are replaced by the corresponding %s.fEnergy
66// automatically)
67//
68// For more details of the setup see MMcSpectrumWeight::ReadEnv
69//
70//
71// Input Containers:
72// MMcEvt
73// MMcCorsikaRunHeader
74// [MPointingPos]
75// [MHillas]
76//
77// Output Container:
78// MWeight [MParameterD]
79//
80//////////////////////////////////////////////////////////////////////////////
81#include "MMcSpectrumWeight.h"
82
83#include <TF1.h>
84#include <TH1.h>
85#include <TH2.h>
86#include <TH3.h>
87#include <TSpline.h>
88
89#include "MLog.h"
90#include "MLogManip.h"
91
92#include "MParList.h"
93#include "MParameters.h"
94
95#include "MHillas.h"
96#include "MPointingPos.h"
97
98#include "MMcEvt.hxx"
99#include "MMcCorsikaRunHeader.h"
100
101ClassImp(MMcSpectrumWeight);
102
103using namespace std;
104
105void MMcSpectrumWeight::Init(const char *name, const char *title)
106{
107 fName = name ? name : "MMcSpectrumWeight";
108 fTitle = title ? title : "Task to calculate weights to change the energy spectrum";
109
110 AddToBranchList("MMcEvt.fEnergy");
111
112 fNameWeight = "MWeight";
113 fNameMcEvt = "MMcEvt";
114
115 fNewSlope = -99;
116 fOldSlope = -99;
117
118 fEnergyMin = -1;
119 fEnergyMax = -2;
120
121 fNorm = 1;
122 fNormEnergy = -1;
123
124 fAllowChange = kFALSE;
125
126 fFunc = NULL;
127 fMcEvt = NULL;
128 fHillas = NULL;
129 fWeight = NULL;
130 fWeightsZd = NULL;
131 fWeightsSize = NULL;
132 fPointing = NULL;
133}
134
135// ---------------------------------------------------------------------------
136//
137// Default Constructor.
138//
139MMcSpectrumWeight::MMcSpectrumWeight(const char *name, const char *title)
140{
141 Init(name,title);
142}
143
144// ---------------------------------------------------------------------------
145//
146// Destructor. If necessary delete fFunc
147//
148MMcSpectrumWeight::~MMcSpectrumWeight()
149{
150 if (fFunc)
151 delete fFunc;
152// if (fWeightsSize)
153// delete fWeightsSize;
154}
155
156// ---------------------------------------------------------------------------
157//
158// Search for
159// - fNameMcEvt [MMcEvtBasic]
160//
161// Find/Create:
162// - fNameWeight [MWeight]
163//
164Int_t MMcSpectrumWeight::PreProcess(MParList *pList)
165{
166 fMcEvt = (MMcEvt*)pList->FindObject(fNameMcEvt, "MMcEvtBasic");
167 if (!fMcEvt)
168 {
169 *fLog << err << fNameMcEvt << " [MMcEvtBasic] not found... abort." << endl;
170 return kFALSE;
171 }
172
173 fWeight = (MParameterD*)pList->FindCreateObj("MParameterD", fNameWeight);
174 if (!fWeight)
175 return kFALSE;
176
177 if (fWeightsZd)
178 {
179 fPointing = (MPointingPos*)pList->FindObject("MPointingPos");
180 if (!fPointing)
181 {
182 *fLog << err << "MPointingPos not found... abort." << endl;
183 return kFALSE;
184 }
185 }
186
187 if (fWeightsSize)
188 {
189 fHillas = (MHillas*)pList->FindObject("MHillas");
190 if (!fHillas)
191 {
192 *fLog << err << "MHillas not found... abort." << endl;
193 return kFALSE;
194 }
195 }
196
197 return kTRUE;
198}
199
200// ---------------------------------------------------------------------------
201//
202// Replace {fNameMcEvt}.fEnergy by "(x)" and return the result.
203//
204TString MMcSpectrumWeight::ReplaceX(TString str) const
205{
206 return str.ReplaceAll(Form("%s.fEnergy", fNameMcEvt.Data()), "(x)");
207}
208
209// ---------------------------------------------------------------------------
210//
211// Return the function corresponding to the mc spectrum with
212// slope fOldSlope: pow({fNameMcEvt}.fEnergy, fOldSlope)
213//
214// The slope is returned as %.3f
215//
216TString MMcSpectrumWeight::GetFormulaSpecOld(const char *name) const
217{
218 return Form("pow(%s.fEnergy, %.3f)", name, fOldSlope);
219}
220
221// ---------------------------------------------------------------------------
222//
223// Return the function corresponding to the new spectrum with
224// slope fNewSlope: pow({fNameMcEvt}.fEnergy, fNewSlope)
225//
226// The slope is returned as %.3f
227//
228// If a different formula is set (SetFormula()) this formula is returned
229// unchanged.
230//
231TString MMcSpectrumWeight::GetFormulaSpecNew(const char *name) const
232{
233 TString str = fFormula.IsNull() ? Form("pow(%s.fEnergy, %.3f)", name, fNewSlope) : fFormula.Data();
234 if (!fFormula.IsNull())
235 str.ReplaceAll("X", Form("(%s.fEnergy)", name));
236
237 return str;
238}
239
240// ---------------------------------------------------------------------------
241//
242// Return the formula to calculate weights.
243// Is is compiled by
244// o1 = integral(fEnergyMin, fEnergyMax, GetFormulaSpecOldX());
245// n1 = integral(fEnergyMin, fEnergyMax, GetFormulaSpecNewX());
246// o2 = CalcSpecOld(fNormEnergy);
247// n2 = CalcSpecNew(fNormEnergy);
248//
249// result (fNormEnergy<0):
250// fNorm*o1/n1*GetFormulaNewSpec()/GetFormulaOldSpec()
251//
252// result (fNormEnergy>=0):
253// fNorm*o2/n2*GetFormulaNewSpec()/GetFormulaOldSpec()
254//
255// fNorm is 1 by default but can be overwritten using SetNorm()
256//
257// If the formulas GetFormulaSpecOldX() and GetFormulaSpecNewX()
258// are equal only fNorm is returned.
259//
260// The normalization constant is returned as %.16e
261//
262// Example: 0.3712780019*(pow(MMcEvt.fEnergy,-2.270))/(pow(MMcEvt.fEnergy,-2.600))
263//
264TString MMcSpectrumWeight::GetFormulaWeights(const char *name) const
265{
266 if (GetFormulaSpecOld()==GetFormulaSpecNew())
267 return Form("%.16e", fNorm);
268
269 const Double_t iold = fNormEnergy<0 ? GetSpecOldIntegral() : CalcSpecOld(fNormEnergy);
270 const Double_t inew = fNormEnergy<0 ? GetSpecNewIntegral() : CalcSpecNew(fNormEnergy);
271
272 const Double_t norm = fNorm*iold/inew;
273
274 return Form("%.16e*(%s)/(%s)", norm, GetFormulaSpecNew(name).Data(), GetFormulaSpecOld(name).Data());
275}
276
277// ---------------------------------------------------------------------------
278//
279// Returns the integral between fEnergyMin and fEnergyMax of
280// GetFormulaSpecNewX() describing the destination spectrum
281//
282Double_t MMcSpectrumWeight::GetSpecNewIntegral(Double_t emin, Double_t emax) const
283{
284 TF1 funcnew("Dummy", GetFormulaSpecNewX());
285 return funcnew.Integral(emin, emax);
286}
287
288// ---------------------------------------------------------------------------
289//
290// Returns the integral between fEnergyMin and fEnergyMax of
291// GetFormulaSpecOldX() describing the simulated spectrum
292//
293Double_t MMcSpectrumWeight::GetSpecOldIntegral(Double_t emin, Double_t emax) const
294{
295 TF1 funcold("Dummy", GetFormulaSpecOldX());
296 return funcold.Integral(emin, emax);
297}
298
299// ---------------------------------------------------------------------------
300//
301// Returns the value of GetFormulaSpecNewX() at the energy e describing
302// the destination spectrum
303//
304Double_t MMcSpectrumWeight::CalcSpecNew(Double_t e) const
305{
306 TF1 funcnew("Dummy", GetFormulaSpecNewX());
307 return funcnew.Eval(e);
308}
309
310// ---------------------------------------------------------------------------
311//
312// Returns the value of GetFormulaSpecOldX() at the energy e describing
313// the simulated spectrum
314//
315Double_t MMcSpectrumWeight::CalcSpecOld(Double_t e) const
316{
317 TF1 funcnew("Dummy", GetFormulaSpecOldX());
318 return funcnew.Eval(e);
319}
320
321void MMcSpectrumWeight::SetWeightsSize(TH1D *h)
322{
323 fWeightsSize=h;
324 /*
325 if (h==0)
326 {
327 fWeightsSize=0;
328 return;
329 }
330
331 if (fWeightsSize)
332 delete fWeightsSize;
333
334 const Double_t xmin = TMath::Log10(h->GetXaxis()->GetXmin());
335 const Double_t xmax = TMath::Log10(h->GetXaxis()->GetXmax());
336 const Double_t xnum = h->GetNbinsX()+1;
337
338 fWeightsSize = new TSpline3("WeightsSize", xmin, xmax,
339 h->GetArray()+1, xnum);*/
340}
341
342// ---------------------------------------------------------------------------
343//
344// Initialize fEnergyMin, fEnergymax and fOldSlope from MMcCorsikaRunHeader
345// by GetELowLim(), GetEUppLim() and GetSlopeSpec().
346//
347// If fEnergyMax>fEnergyMin (means: the values have already been
348// initialized) and !fAllowChange the consistency of the new values
349// with the present values is checked with a numerical precision of 1e-10.
350// If one doesn't match kFALSE is returned.
351//
352// If the mc slope is -1 kFALSE is returned.
353//
354// If the new slope for the spectrum is -1 it is set to the original MC
355// slope.
356//
357// fFunc is set to the formula returned by GetFormulaWeightsX()
358//
359Bool_t MMcSpectrumWeight::Set(const MMcCorsikaRunHeader &rh)
360{
361 if (fEnergyMax>fEnergyMin && !fAllowChange)
362 {
363 if (TMath::Abs(fEnergyMax-rh.GetEUppLim())>1e-10)
364 {
365 *fLog << err;
366 *fLog << "ERROR - The maximum simulated Monte Carlo energy is not allowed to change ";
367 *fLog << "(" << fEnergyMax << " --> " << rh.GetEUppLim() << ")... abort." << endl;
368 return kFALSE;
369 }
370
371 if (TMath::Abs(fOldSlope-rh.GetSlopeSpec())>1e-10)
372 {
373 *fLog << err;
374 *fLog << "ERROR - The slope of the Monte Carlo is not allowed to change ";
375 *fLog << "(" << fOldSlope << " --> " << rh.GetSlopeSpec() << ")... abort." << endl;
376 return kFALSE;
377 }
378
379 // No change happened
380 if (TMath::Abs(fEnergyMin-rh.GetELowLim())<=1e-10)
381 return kTRUE;
382
383 // The lower energy limit has changed
384 *fLog << warn;
385 *fLog << "The minimum simulated Monte Carlo energy has changed from ";
386 *fLog << fEnergyMin << "GeV to " << rh.GetELowLim() << "GeV." << endl;
387 fEnergyMin = rh.GetELowLim();
388 }
389
390 fOldSlope = rh.GetSlopeSpec();
391 fEnergyMin = rh.GetELowLim();
392 fEnergyMax = rh.GetEUppLim();
393
394 if (fNewSlope==-99 && fFormula.IsNull())
395 {
396 *fLog << inf << "A new slope for the power law has not yet been defined... using " << fOldSlope << "." << endl;
397 fNewSlope = fOldSlope;
398 }
399
400 if (fFunc)
401 delete fFunc;
402
403 if (GetFormulaSpecOld()==GetFormulaSpecNew())
404 *fLog << inf << "No spectral change requested..." << endl;
405 else
406 {
407 *fLog << inf << "Weighting from slope " << fOldSlope << " to ";
408 if (fFormula.IsNull())
409 *fLog << "slope " << fNewSlope << "." << endl;
410 else
411 *fLog << GetFormulaSpecNewX() << endl;
412 }
413
414 fFunc = new TF1("", GetFormulaWeightsX());
415 gROOT->GetListOfFunctions()->Remove(fFunc);
416 fFunc->SetName("SpectralWeighs");
417
418 return kTRUE;
419}
420
421// ---------------------------------------------------------------------------
422//
423// completes a simulated spectrum starting at an energy fEnergyMin down to
424// an energy emin.
425//
426// It is assumed that the contents of MMcSpectrumWeight for the new spectrum
427// correctly describe the spectrum within the histogram, and fEnergyMin
428// and fEnergyMax correctly describe the range.
429//
430// If scale is given the histogram statistics is further extended by the
431// new spectrum according to the scale factor (eg. 1.2: by 20%)
432//
433// In the 1D case it is assumed that the x-axis is a zenith angle binning.
434// In the 2D case the x-axis is assumed to be zenith angle, the y-axis
435// to be energy.
436//
437void MMcSpectrumWeight::CompleteEnergySpectrum(TH1 &h, Double_t emin, Double_t scale) const
438{
439 if (h.InheritsFrom(TH3::Class()))
440 {
441 return;
442 }
443
444 if (fEnergyMin < emin)
445 {
446 *fLog << err << "ERROR - MMcSpctrumWeight::CompleteEnergySpectrum: fEnergyMin (";
447 *fLog << fEnergyMin << ") smaller than emin (" << emin << ")." << endl;
448 return;
449 }
450
451 // Total number of events for the new spectrum in the same
452 // energy range as the current histogram is filled
453 const Double_t norm = GetSpecNewIntegral();
454
455 // Check if it is only a histogram in ZA
456 if (!h.InheritsFrom(TH2::Class()))
457 {
458 // Warning: Simply scaling the zenith angle distribution might
459 // increase fluctuations for low statistics.
460 const Double_t f = GetSpecNewIntegral(emin, fEnergyMax)/norm;
461 h.Scale(f*scale);
462 return;
463 }
464
465 const TAxis &axey = *h.GetYaxis();
466
467 // Find energy range between the minimum energy to be filles (emin)
468 // and the minimum energy corresponding to the data filled into
469 // this histogram (fEnergyMin)
470 const Int_t first = axey.FindFixBin(emin);
471 const Int_t last = axey.FindFixBin(fEnergyMin);
472 const Int_t max = axey.FindFixBin(fEnergyMax);
473
474 for (int x=1; x<=h.GetNbinsX(); x++)
475 {
476 // Ratio between the number of events in the zenith angle
477 // bin corresponding to x and the new spectrum.
478 const Double_t f = h.Integral(x, x, -1, 9999)/norm;
479
480 // Fill histogram with the "new spectrum" between
481 // emin and fEnergyMin.
482 if (emin<fEnergyMin)
483 for (int y=first; y<=last; y++)
484 {
485 // Check if the bin is only partly filled by the energy range
486 const Double_t lo = axey.GetBinLowEdge(y) <emin ? emin : axey.GetBinLowEdge(y);
487 const Double_t hi = axey.GetBinLowEdge(y+1)>fEnergyMin ? fEnergyMin : axey.GetBinLowEdge(y+1);
488
489 // Add the new spectrum extending the existing spectrum
490 h.AddBinContent(h.GetBin(x, y), f*GetSpecNewIntegral(lo, hi));
491 }
492
493 // If scale is >1 we also have to increse the statistics f the
494 // histogram according to scale.
495 if (scale>1)
496 for (int y=first; y<=max; y++)
497 {
498 // Check if the bin is only partly filled by the energy range
499 const Double_t lo = axey.GetBinLowEdge(y) <emin ? emin : axey.GetBinLowEdge(y);
500 const Double_t hi = axey.GetBinLowEdge(y+1)>fEnergyMax ? fEnergyMax : axey.GetBinLowEdge(y+1);
501
502 // Use the analytical solution to scale the histogram
503 h.AddBinContent(h.GetBin(x, y), f*GetSpecNewIntegral(lo, hi)*(scale-1));
504 }
505 }
506}
507
508// ---------------------------------------------------------------------------
509//
510// The current contants are printed
511//
512void MMcSpectrumWeight::Print(Option_t *o) const
513{
514 const TString opt(o);
515
516 const Bool_t hasnew = opt.Contains("new") || opt.IsNull();
517 const Bool_t hasold = opt.Contains("old") || opt.IsNull();
518
519 *fLog << all << GetDescriptor() << endl;
520
521 if (hasold)
522 {
523 *fLog << " Simulated energy range: " << fEnergyMin << "GeV - " << fEnergyMax << "GeV" << endl;
524 *fLog << " Simulated spectral slope: ";
525 if (fOldSlope==-99)
526 *fLog << "undefined" << endl;
527 else
528 *fLog << fOldSlope << endl;
529 }
530 if (hasnew)
531 {
532 *fLog << " New spectral slope: ";
533 if (fNewSlope==-99)
534 *fLog << "undefined/no change" << endl;
535 else
536 *fLog << fNewSlope << endl;
537 }
538 *fLog << " Additional user norm.: " << fNorm << endl;
539 *fLog << " Spectra are normalized: " << (fNormEnergy<0?"by integral":Form("at %.1fGeV", fNormEnergy)) << endl;
540 if (hasold)
541 {
542 *fLog << " Old Spectrum: ";
543 if (fNewSlope==-99)
544 *fLog << "undefined";
545 else
546 *fLog << GetFormulaSpecOldX();
547 if (fEnergyMin>=0 && fEnergyMax>0)
548 *fLog << " (I=" << GetSpecOldIntegral() << ")";
549 *fLog << endl;
550 }
551 if (hasnew)
552 {
553 *fLog << " New Spectrum: ";
554 if (fNewSlope==-99 && fFormula.IsNull())
555 *fLog << "undefined/no change";
556 else
557 *fLog << GetFormulaSpecNewX();
558 if (fEnergyMin>=0 && fEnergyMax>0)
559 *fLog << " (I=" << GetSpecNewIntegral() << ")";
560 *fLog << endl;
561 }
562 if (fFunc)
563 *fLog << " Weight func: " << fFunc->GetTitle() << endl;
564}
565
566// ----------------------------------------------------------------------------
567//
568// Executed each time a new root file is loaded
569// We will need fOldSlope and fE{Upp,Low}Lim to calculate the weights
570//
571Bool_t MMcSpectrumWeight::ReInit(MParList *plist)
572{
573 MMcCorsikaRunHeader *rh = (MMcCorsikaRunHeader*)plist->FindObject("MMcCorsikaRunHeader");
574 if (!rh)
575 {
576 *fLog << err << "MMcCorsikaRunHeader not found... abort." << endl;
577 return kFALSE;
578 }
579
580 return Set(*rh);
581}
582
583/*
584 * This could be used to improve the Zd-weighting within a bin.
585 * Another option is to use more bins, or both.
586 * Note that it seems unnecessary, because the shape within the
587 * theta-bins should be similar in data and Monte Carlo... hopefully.
588 *
589void MMcSpectrumWeight::InitZdWeights()
590{
591 TH2D w(*fWeightsZd);
592
593 for (int i=1; i<=w.GetNbinsX(); i++)
594 {
595 const Double_t tmin = w.GetBinLowEdge(i) *TMath::DegToRad();
596 const Double_t tmax = w.GetBinLowEdge(i+1)*TMath::DegToRad();
597
598 const Double_t wdth = tmax-tmin;
599 const Double_t integ = cos(tmin)-cos(tmax);
600
601 w.SetBinContent(i, w.GetBinContent(i)*wdth/integ);
602 }
603
604 // const Int_t i = fWeightsZd->GetXaxis()->FindFixBin(fPointing->GetZd());
605 // const Double_t theta = fPointing->GetZd()*TMath::DegToRad();
606 // w = sin(theta)*w.GetBinContent(i);
607}
608*/
609
610// ----------------------------------------------------------------------------
611//
612// Fill the result of the evaluation of fFunc at fEvEvt->GetEnergy
613// into the weights container.
614//
615Int_t MMcSpectrumWeight::Process()
616{
617 Double_t w = 1;
618
619 if (fWeightsZd)
620 {
621 const Int_t i = fWeightsZd->GetXaxis()->FindFixBin(fPointing->GetZd());
622 w = fWeightsZd->GetBinContent(i);
623 }
624 if (fWeightsSize)
625 {
626 const Int_t i = fWeightsSize->GetXaxis()->FindFixBin(fHillas->GetSize());
627 w *= fWeightsSize->GetBinContent(i);
628 // w *= fWeightsSize->Eval(TMath::Log10(fHillas->GetSize()));
629 }
630
631 const Double_t e = fMcEvt->GetEnergy();
632
633 fWeight->SetVal(fFunc->Eval(e)*w);
634
635 return kTRUE;
636}
637
638// --------------------------------------------------------------------------
639//
640// Read the setup from a TEnv, eg:
641//
642// MMcSpectrumWeight.NewSlope: -2.6
643// The new slope of the spectrum
644//
645// MMcSpectrumWeight.Norm: 1.0
646// An additional artificial scale factor
647//
648// MMcSpectrumWeight.NormEnergy: 200
649// To normalize at a given energy instead of the integral
650//
651// MMcSpectrumWeight.Formula: pow(X, -2.6)
652// A formula to which the spectrum is weighted (use a capital X for
653// the energy)
654//
655Int_t MMcSpectrumWeight::ReadEnv(const TEnv &env, TString prefix, Bool_t print)
656{
657 Bool_t rc = kFALSE;
658 if (IsEnvDefined(env, prefix, "NewSlope", print))
659 {
660 rc = kTRUE;
661 SetNewSlope(GetEnvValue(env, prefix, "NewSlope", fNewSlope));
662 }
663 if (IsEnvDefined(env, prefix, "Norm", print))
664 {
665 rc = kTRUE;
666 SetNorm(GetEnvValue(env, prefix, "Norm", fNorm));
667 }
668 if (IsEnvDefined(env, prefix, "NormEnergy", print))
669 {
670 rc = kTRUE;
671 SetNormEnergy(GetEnvValue(env, prefix, "NormEnergy", fNormEnergy));
672 }
673 if (IsEnvDefined(env, prefix, "Formula", print))
674 {
675 rc = kTRUE;
676 SetFormula(GetEnvValue(env, prefix, "Formula", fFormula));
677 }
678
679 return rc;
680}
Note: See TracBrowser for help on using the repository browser.