base: Fix squares of stats
[gem5.git] / src / base / statistics.hh
1 /*
2 * Copyright (c) 2019 Arm Limited
3 * All rights reserved.
4 *
5 * The license below extends only to copyright in the software and shall
6 * not be construed as granting a license to any other intellectual
7 * property including but not limited to intellectual property relating
8 * to a hardware implementation of the functionality of the software
9 * licensed hereunder. You may use the software subject to the license
10 * terms below provided that you ensure that this notice is replicated
11 * unmodified and in its entirety in all distributions of the software,
12 * modified or unmodified, in source code or in binary form.
13 *
14 * Copyright (c) 2003-2005 The Regents of The University of Michigan
15 * Copyright (c) 2017, Centre National de la Recherche Scientifique
16 * All rights reserved.
17 *
18 * Redistribution and use in source and binary forms, with or without
19 * modification, are permitted provided that the following conditions are
20 * met: redistributions of source code must retain the above copyright
21 * notice, this list of conditions and the following disclaimer;
22 * redistributions in binary form must reproduce the above copyright
23 * notice, this list of conditions and the following disclaimer in the
24 * documentation and/or other materials provided with the distribution;
25 * neither the name of the copyright holders nor the names of its
26 * contributors may be used to endorse or promote products derived from
27 * this software without specific prior written permission.
28 *
29 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
30 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
31 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
32 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
33 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
34 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
35 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
36 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
37 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
38 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
39 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40 *
41 * Authors: Nathan Binkert
42 * Pierre-Yves Peneau
43 */
44
45 /** @file
46 * Declaration of Statistics objects.
47 */
48
49 /**
50 * @todo
51 *
52 * Generalized N-dimensinal vector
53 * documentation
54 * key stats
55 * interval stats
56 * -- these both can use the same function that prints out a
57 * specific set of stats
58 * VectorStandardDeviation totals
59 * Document Namespaces
60 */
61 #ifndef __BASE_STATISTICS_HH__
62 #define __BASE_STATISTICS_HH__
63
64 #include <algorithm>
65 #include <cassert>
66 #ifdef __SUNPRO_CC
67 #include <math.h>
68 #endif
69 #include <cmath>
70 #include <functional>
71 #include <iosfwd>
72 #include <list>
73 #include <map>
74 #include <memory>
75 #include <string>
76 #include <vector>
77
78 #include "base/stats/group.hh"
79 #include "base/stats/info.hh"
80 #include "base/stats/output.hh"
81 #include "base/stats/types.hh"
82 #include "base/cast.hh"
83 #include "base/cprintf.hh"
84 #include "base/intmath.hh"
85 #include "base/str.hh"
86 #include "base/types.hh"
87
88 class Callback;
89
90 /** The current simulated tick. */
91 extern Tick curTick();
92
93 /* A namespace for all of the Statistics */
94 namespace Stats {
95
96 template <class Stat, class Base>
97 class InfoProxy : public Base
98 {
99 protected:
100 Stat &s;
101
102 public:
103 InfoProxy(Stat &stat) : s(stat) {}
104
105 bool check() const { return s.check(); }
106 void prepare() { s.prepare(); }
107 void reset() { s.reset(); }
108 void
109 visit(Output &visitor)
110 {
111 visitor.visit(*static_cast<Base *>(this));
112 }
113 bool zero() const { return s.zero(); }
114 };
115
116 template <class Stat>
117 class ScalarInfoProxy : public InfoProxy<Stat, ScalarInfo>
118 {
119 public:
120 ScalarInfoProxy(Stat &stat) : InfoProxy<Stat, ScalarInfo>(stat) {}
121
122 Counter value() const { return this->s.value(); }
123 Result result() const { return this->s.result(); }
124 Result total() const { return this->s.total(); }
125 };
126
127 template <class Stat>
128 class VectorInfoProxy : public InfoProxy<Stat, VectorInfo>
129 {
130 protected:
131 mutable VCounter cvec;
132 mutable VResult rvec;
133
134 public:
135 VectorInfoProxy(Stat &stat) : InfoProxy<Stat, VectorInfo>(stat) {}
136
137 size_type size() const { return this->s.size(); }
138
139 VCounter &
140 value() const
141 {
142 this->s.value(cvec);
143 return cvec;
144 }
145
146 const VResult &
147 result() const
148 {
149 this->s.result(rvec);
150 return rvec;
151 }
152
153 Result total() const { return this->s.total(); }
154 };
155
156 template <class Stat>
157 class DistInfoProxy : public InfoProxy<Stat, DistInfo>
158 {
159 public:
160 DistInfoProxy(Stat &stat) : InfoProxy<Stat, DistInfo>(stat) {}
161 };
162
163 template <class Stat>
164 class VectorDistInfoProxy : public InfoProxy<Stat, VectorDistInfo>
165 {
166 public:
167 VectorDistInfoProxy(Stat &stat) : InfoProxy<Stat, VectorDistInfo>(stat) {}
168
169 size_type size() const { return this->s.size(); }
170 };
171
172 template <class Stat>
173 class Vector2dInfoProxy : public InfoProxy<Stat, Vector2dInfo>
174 {
175 public:
176 Vector2dInfoProxy(Stat &stat) : InfoProxy<Stat, Vector2dInfo>(stat) {}
177
178 Result total() const { return this->s.total(); }
179 };
180
181 struct StorageParams
182 {
183 virtual ~StorageParams();
184 };
185
186 class InfoAccess
187 {
188 private:
189 Info *_info;
190
191 protected:
192 /** Set up an info class for this statistic */
193 void setInfo(Group *parent, Info *info);
194 /** Save Storage class parameters if any */
195 void setParams(const StorageParams *params);
196 /** Save Storage class parameters if any */
197 void setInit();
198
199 /** Grab the information class for this statistic */
200 Info *info();
201 /** Grab the information class for this statistic */
202 const Info *info() const;
203
204 public:
205 InfoAccess()
206 : _info(nullptr) {};
207
208 /**
209 * Reset the stat to the default state.
210 */
211 void reset() { }
212
213 /**
214 * @return true if this stat has a value and satisfies its
215 * requirement as a prereq
216 */
217 bool zero() const { return true; }
218
219 /**
220 * Check that this stat has been set up properly and is ready for
221 * use
222 * @return true for success
223 */
224 bool check() const { return true; }
225 };
226
227 template <class Derived, template <class> class InfoProxyType>
228 class DataWrap : public InfoAccess
229 {
230 public:
231 typedef InfoProxyType<Derived> Info;
232
233 protected:
234 Derived &self() { return *static_cast<Derived *>(this); }
235
236 protected:
237 Info *
238 info()
239 {
240 return safe_cast<Info *>(InfoAccess::info());
241 }
242
243 public:
244 const Info *
245 info() const
246 {
247 return safe_cast<const Info *>(InfoAccess::info());
248 }
249
250 public:
251 DataWrap() = delete;
252 DataWrap(const DataWrap &) = delete;
253 DataWrap &operator=(const DataWrap &) = delete;
254
255
256 DataWrap(Group *parent, const char *name, const char *desc)
257 {
258 auto info = new Info(self());
259 this->setInfo(parent, info);
260
261 if (parent)
262 parent->addStat(info);
263
264 if (name) {
265 info->setName(parent, name);
266 info->flags.set(display);
267 }
268
269 if (desc)
270 info->desc = desc;
271 }
272
273 /**
274 * Set the name and marks this stat to print at the end of simulation.
275 * @param name The new name.
276 * @return A reference to this stat.
277 */
278 Derived &
279 name(const std::string &name)
280 {
281 Info *info = this->info();
282 info->setName(name);
283 info->flags.set(display);
284 return this->self();
285 }
286 const std::string &name() const { return this->info()->name; }
287
288 /**
289 * Set the character(s) used between the name and vector number
290 * on vectors, dist, etc.
291 * @param _sep The new separator string
292 * @return A reference to this stat.
293 */
294 Derived &
295 setSeparator(const std::string &_sep)
296 {
297 this->info()->setSeparator(_sep);
298 return this->self();
299 }
300 const std::string &setSeparator() const
301 {
302 return this->info()->separatorString;
303 }
304
305 /**
306 * Set the description and marks this stat to print at the end of
307 * simulation.
308 * @param desc The new description.
309 * @return A reference to this stat.
310 */
311 Derived &
312 desc(const std::string &_desc)
313 {
314 this->info()->desc = _desc;
315 return this->self();
316 }
317
318 /**
319 * Set the precision and marks this stat to print at the end of simulation.
320 * @param _precision The new precision
321 * @return A reference to this stat.
322 */
323 Derived &
324 precision(int _precision)
325 {
326 this->info()->precision = _precision;
327 return this->self();
328 }
329
330 /**
331 * Set the flags and marks this stat to print at the end of simulation.
332 * @param f The new flags.
333 * @return A reference to this stat.
334 */
335 Derived &
336 flags(Flags _flags)
337 {
338 this->info()->flags.set(_flags);
339 return this->self();
340 }
341
342 /**
343 * Set the prerequisite stat and marks this stat to print at the end of
344 * simulation.
345 * @param prereq The prerequisite stat.
346 * @return A reference to this stat.
347 */
348 template <class Stat>
349 Derived &
350 prereq(const Stat &prereq)
351 {
352 this->info()->prereq = prereq.info();
353 return this->self();
354 }
355 };
356
357 template <class Derived, template <class> class InfoProxyType>
358 class DataWrapVec : public DataWrap<Derived, InfoProxyType>
359 {
360 public:
361 typedef InfoProxyType<Derived> Info;
362
363 DataWrapVec(Group *parent = nullptr, const char *name = nullptr,
364 const char *desc = nullptr)
365 : DataWrap<Derived, InfoProxyType>(parent, name, desc)
366 {}
367
368 // The following functions are specific to vectors. If you use them
369 // in a non vector context, you will get a nice compiler error!
370
371 /**
372 * Set the subfield name for the given index, and marks this stat to print
373 * at the end of simulation.
374 * @param index The subfield index.
375 * @param name The new name of the subfield.
376 * @return A reference to this stat.
377 */
378 Derived &
379 subname(off_type index, const std::string &name)
380 {
381 Derived &self = this->self();
382 Info *info = self.info();
383
384 std::vector<std::string> &subn = info->subnames;
385 if (subn.size() <= index)
386 subn.resize(index + 1);
387 subn[index] = name;
388 return self;
389 }
390
391 // The following functions are specific to 2d vectors. If you use
392 // them in a non vector context, you will get a nice compiler
393 // error because info doesn't have the right variables.
394
395 /**
396 * Set the subfield description for the given index and marks this stat to
397 * print at the end of simulation.
398 * @param index The subfield index.
399 * @param desc The new description of the subfield
400 * @return A reference to this stat.
401 */
402 Derived &
403 subdesc(off_type index, const std::string &desc)
404 {
405 Info *info = this->info();
406
407 std::vector<std::string> &subd = info->subdescs;
408 if (subd.size() <= index)
409 subd.resize(index + 1);
410 subd[index] = desc;
411
412 return this->self();
413 }
414
415 void
416 prepare()
417 {
418 Derived &self = this->self();
419 Info *info = this->info();
420
421 size_t size = self.size();
422 for (off_type i = 0; i < size; ++i)
423 self.data(i)->prepare(info);
424 }
425
426 void
427 reset()
428 {
429 Derived &self = this->self();
430 Info *info = this->info();
431
432 size_t size = self.size();
433 for (off_type i = 0; i < size; ++i)
434 self.data(i)->reset(info);
435 }
436 };
437
438 template <class Derived, template <class> class InfoProxyType>
439 class DataWrapVec2d : public DataWrapVec<Derived, InfoProxyType>
440 {
441 public:
442 typedef InfoProxyType<Derived> Info;
443
444 DataWrapVec2d(Group *parent, const char *name, const char *desc)
445 : DataWrapVec<Derived, InfoProxyType>(parent, name, desc)
446 {
447 }
448
449 /**
450 * @warning This makes the assumption that if you're gonna subnames a 2d
451 * vector, you're subnaming across all y
452 */
453 Derived &
454 ysubnames(const char **names)
455 {
456 Derived &self = this->self();
457 Info *info = this->info();
458
459 info->y_subnames.resize(self.y);
460 for (off_type i = 0; i < self.y; ++i)
461 info->y_subnames[i] = names[i];
462 return self;
463 }
464
465 Derived &
466 ysubname(off_type index, const std::string &subname)
467 {
468 Derived &self = this->self();
469 Info *info = this->info();
470
471 assert(index < self.y);
472 info->y_subnames.resize(self.y);
473 info->y_subnames[index] = subname.c_str();
474 return self;
475 }
476
477 std::string
478 ysubname(off_type i) const
479 {
480 return this->info()->y_subnames[i];
481 }
482
483 };
484
485 //////////////////////////////////////////////////////////////////////
486 //
487 // Simple Statistics
488 //
489 //////////////////////////////////////////////////////////////////////
490
491 /**
492 * Templatized storage and interface for a simple scalar stat.
493 */
494 class StatStor
495 {
496 private:
497 /** The statistic value. */
498 Counter data;
499
500 public:
501 struct Params : public StorageParams {};
502
503 public:
504 /**
505 * Builds this storage element and calls the base constructor of the
506 * datatype.
507 */
508 StatStor(Info *info)
509 : data(Counter())
510 { }
511
512 /**
513 * The the stat to the given value.
514 * @param val The new value.
515 */
516 void set(Counter val) { data = val; }
517 /**
518 * Increment the stat by the given value.
519 * @param val The new value.
520 */
521 void inc(Counter val) { data += val; }
522 /**
523 * Decrement the stat by the given value.
524 * @param val The new value.
525 */
526 void dec(Counter val) { data -= val; }
527 /**
528 * Return the value of this stat as its base type.
529 * @return The value of this stat.
530 */
531 Counter value() const { return data; }
532 /**
533 * Return the value of this stat as a result type.
534 * @return The value of this stat.
535 */
536 Result result() const { return (Result)data; }
537 /**
538 * Prepare stat data for dumping or serialization
539 */
540 void prepare(Info *info) { }
541 /**
542 * Reset stat value to default
543 */
544 void reset(Info *info) { data = Counter(); }
545
546 /**
547 * @return true if zero value
548 */
549 bool zero() const { return data == Counter(); }
550 };
551
552 /**
553 * Templatized storage and interface to a per-tick average stat. This keeps
554 * a current count and updates a total (count * ticks) when this count
555 * changes. This allows the quick calculation of a per tick count of the item
556 * being watched. This is good for keeping track of residencies in structures
557 * among other things.
558 */
559 class AvgStor
560 {
561 private:
562 /** The current count. */
563 Counter current;
564 /** The tick of the last reset */
565 Tick lastReset;
566 /** The total count for all tick. */
567 mutable Result total;
568 /** The tick that current last changed. */
569 mutable Tick last;
570
571 public:
572 struct Params : public StorageParams {};
573
574 public:
575 /**
576 * Build and initializes this stat storage.
577 */
578 AvgStor(Info *info)
579 : current(0), lastReset(0), total(0), last(0)
580 { }
581
582 /**
583 * Set the current count to the one provided, update the total and last
584 * set values.
585 * @param val The new count.
586 */
587 void
588 set(Counter val)
589 {
590 total += current * (curTick() - last);
591 last = curTick();
592 current = val;
593 }
594
595 /**
596 * Increment the current count by the provided value, calls set.
597 * @param val The amount to increment.
598 */
599 void inc(Counter val) { set(current + val); }
600
601 /**
602 * Deccrement the current count by the provided value, calls set.
603 * @param val The amount to decrement.
604 */
605 void dec(Counter val) { set(current - val); }
606
607 /**
608 * Return the current count.
609 * @return The current count.
610 */
611 Counter value() const { return current; }
612
613 /**
614 * Return the current average.
615 * @return The current average.
616 */
617 Result
618 result() const
619 {
620 assert(last == curTick());
621 return (Result)(total + current) / (Result)(curTick() - lastReset + 1);
622 }
623
624 /**
625 * @return true if zero value
626 */
627 bool zero() const { return total == 0.0; }
628
629 /**
630 * Prepare stat data for dumping or serialization
631 */
632 void
633 prepare(Info *info)
634 {
635 total += current * (curTick() - last);
636 last = curTick();
637 }
638
639 /**
640 * Reset stat value to default
641 */
642 void
643 reset(Info *info)
644 {
645 total = 0.0;
646 last = curTick();
647 lastReset = curTick();
648 }
649
650 };
651
652 /**
653 * Implementation of a scalar stat. The type of stat is determined by the
654 * Storage template.
655 */
656 template <class Derived, class Stor>
657 class ScalarBase : public DataWrap<Derived, ScalarInfoProxy>
658 {
659 public:
660 typedef Stor Storage;
661 typedef typename Stor::Params Params;
662
663 protected:
664 /** The storage of this stat. */
665 char storage[sizeof(Storage)] __attribute__ ((aligned (8)));
666
667 protected:
668 /**
669 * Retrieve the storage.
670 * @param index The vector index to access.
671 * @return The storage object at the given index.
672 */
673 Storage *
674 data()
675 {
676 return reinterpret_cast<Storage *>(storage);
677 }
678
679 /**
680 * Retrieve a const pointer to the storage.
681 * for the given index.
682 * @param index The vector index to access.
683 * @return A const pointer to the storage object at the given index.
684 */
685 const Storage *
686 data() const
687 {
688 return reinterpret_cast<const Storage *>(storage);
689 }
690
691 void
692 doInit()
693 {
694 new (storage) Storage(this->info());
695 this->setInit();
696 }
697
698 public:
699 /**
700 * Return the current value of this stat as its base type.
701 * @return The current value.
702 */
703 Counter value() const { return data()->value(); }
704
705 public:
706 ScalarBase(Group *parent = nullptr, const char *name = nullptr,
707 const char *desc = nullptr)
708 : DataWrap<Derived, ScalarInfoProxy>(parent, name, desc)
709 {
710 this->doInit();
711 }
712
713 public:
714 // Common operators for stats
715 /**
716 * Increment the stat by 1. This calls the associated storage object inc
717 * function.
718 */
719 void operator++() { data()->inc(1); }
720 /**
721 * Decrement the stat by 1. This calls the associated storage object dec
722 * function.
723 */
724 void operator--() { data()->dec(1); }
725
726 /** Increment the stat by 1. */
727 void operator++(int) { ++*this; }
728 /** Decrement the stat by 1. */
729 void operator--(int) { --*this; }
730
731 /**
732 * Set the data value to the given value. This calls the associated storage
733 * object set function.
734 * @param v The new value.
735 */
736 template <typename U>
737 void operator=(const U &v) { data()->set(v); }
738
739 /**
740 * Increment the stat by the given value. This calls the associated
741 * storage object inc function.
742 * @param v The value to add.
743 */
744 template <typename U>
745 void operator+=(const U &v) { data()->inc(v); }
746
747 /**
748 * Decrement the stat by the given value. This calls the associated
749 * storage object dec function.
750 * @param v The value to substract.
751 */
752 template <typename U>
753 void operator-=(const U &v) { data()->dec(v); }
754
755 /**
756 * Return the number of elements, always 1 for a scalar.
757 * @return 1.
758 */
759 size_type size() const { return 1; }
760
761 Counter value() { return data()->value(); }
762
763 Result result() { return data()->result(); }
764
765 Result total() { return result(); }
766
767 bool zero() { return result() == 0.0; }
768
769 void reset() { data()->reset(this->info()); }
770 void prepare() { data()->prepare(this->info()); }
771 };
772
773 class ProxyInfo : public ScalarInfo
774 {
775 public:
776 std::string str() const { return std::to_string(value()); }
777 size_type size() const { return 1; }
778 bool check() const { return true; }
779 void prepare() { }
780 void reset() { }
781 bool zero() const { return value() == 0; }
782
783 void visit(Output &visitor) { visitor.visit(*this); }
784 };
785
786 template <class T>
787 class ValueProxy : public ProxyInfo
788 {
789 private:
790 T *scalar;
791
792 public:
793 ValueProxy(T &val) : scalar(&val) {}
794 Counter value() const { return *scalar; }
795 Result result() const { return *scalar; }
796 Result total() const { return *scalar; }
797 };
798
799 template <class T>
800 class FunctorProxy : public ProxyInfo
801 {
802 private:
803 T *functor;
804
805 public:
806 FunctorProxy(T &func) : functor(&func) {}
807 Counter value() const { return (*functor)(); }
808 Result result() const { return (*functor)(); }
809 Result total() const { return (*functor)(); }
810 };
811
812 /**
813 * A proxy similar to the FunctorProxy, but allows calling a method of a bound
814 * object, instead of a global free-standing function.
815 */
816 template <class T, class V>
817 class MethodProxy : public ProxyInfo
818 {
819 private:
820 T *object;
821 typedef V (T::*MethodPointer) () const;
822 MethodPointer method;
823
824 public:
825 MethodProxy(T *obj, MethodPointer meth) : object(obj), method(meth) {}
826 Counter value() const { return (object->*method)(); }
827 Result result() const { return (object->*method)(); }
828 Result total() const { return (object->*method)(); }
829 };
830
831 template <class Derived>
832 class ValueBase : public DataWrap<Derived, ScalarInfoProxy>
833 {
834 private:
835 ProxyInfo *proxy;
836
837 public:
838 ValueBase(Group *parent, const char *name, const char *desc)
839 : DataWrap<Derived, ScalarInfoProxy>(parent, name, desc),
840 proxy(NULL)
841 {
842 }
843
844 ~ValueBase() { if (proxy) delete proxy; }
845
846 template <class T>
847 Derived &
848 scalar(T &value)
849 {
850 proxy = new ValueProxy<T>(value);
851 this->setInit();
852 return this->self();
853 }
854
855 template <class T>
856 Derived &
857 functor(T &func)
858 {
859 proxy = new FunctorProxy<T>(func);
860 this->setInit();
861 return this->self();
862 }
863
864 /**
865 * Extended functor that calls the specified method of the provided object.
866 *
867 * @param obj Pointer to the object whose method should be called.
868 * @param method Pointer of the function / method of the object.
869 * @return Updated stats item.
870 */
871 template <class T, class V>
872 Derived &
873 method(T *obj, V (T::*method)() const)
874 {
875 proxy = new MethodProxy<T,V>(obj, method);
876 this->setInit();
877 return this->self();
878 }
879
880 Counter value() { return proxy->value(); }
881 Result result() const { return proxy->result(); }
882 Result total() const { return proxy->total(); };
883 size_type size() const { return proxy->size(); }
884
885 std::string str() const { return proxy->str(); }
886 bool zero() const { return proxy->zero(); }
887 bool check() const { return proxy != NULL; }
888 void prepare() { }
889 void reset() { }
890 };
891
892 //////////////////////////////////////////////////////////////////////
893 //
894 // Vector Statistics
895 //
896 //////////////////////////////////////////////////////////////////////
897
898 /**
899 * A proxy class to access the stat at a given index in a VectorBase stat.
900 * Behaves like a ScalarBase.
901 */
902 template <class Stat>
903 class ScalarProxy
904 {
905 private:
906 /** Pointer to the parent Vector. */
907 Stat &stat;
908
909 /** The index to access in the parent VectorBase. */
910 off_type index;
911
912 public:
913 /**
914 * Return the current value of this stat as its base type.
915 * @return The current value.
916 */
917 Counter value() const { return stat.data(index)->value(); }
918
919 /**
920 * Return the current value of this statas a result type.
921 * @return The current value.
922 */
923 Result result() const { return stat.data(index)->result(); }
924
925 public:
926 /**
927 * Create and initialize this proxy, do not register it with the database.
928 * @param i The index to access.
929 */
930 ScalarProxy(Stat &s, off_type i)
931 : stat(s), index(i)
932 {
933 }
934
935 /**
936 * Create a copy of the provided ScalarProxy.
937 * @param sp The proxy to copy.
938 */
939 ScalarProxy(const ScalarProxy &sp)
940 : stat(sp.stat), index(sp.index)
941 {}
942
943 /**
944 * Set this proxy equal to the provided one.
945 * @param sp The proxy to copy.
946 * @return A reference to this proxy.
947 */
948 const ScalarProxy &
949 operator=(const ScalarProxy &sp)
950 {
951 stat = sp.stat;
952 index = sp.index;
953 return *this;
954 }
955
956 public:
957 // Common operators for stats
958 /**
959 * Increment the stat by 1. This calls the associated storage object inc
960 * function.
961 */
962 void operator++() { stat.data(index)->inc(1); }
963 /**
964 * Decrement the stat by 1. This calls the associated storage object dec
965 * function.
966 */
967 void operator--() { stat.data(index)->dec(1); }
968
969 /** Increment the stat by 1. */
970 void operator++(int) { ++*this; }
971 /** Decrement the stat by 1. */
972 void operator--(int) { --*this; }
973
974 /**
975 * Set the data value to the given value. This calls the associated storage
976 * object set function.
977 * @param v The new value.
978 */
979 template <typename U>
980 void
981 operator=(const U &v)
982 {
983 stat.data(index)->set(v);
984 }
985
986 /**
987 * Increment the stat by the given value. This calls the associated
988 * storage object inc function.
989 * @param v The value to add.
990 */
991 template <typename U>
992 void
993 operator+=(const U &v)
994 {
995 stat.data(index)->inc(v);
996 }
997
998 /**
999 * Decrement the stat by the given value. This calls the associated
1000 * storage object dec function.
1001 * @param v The value to substract.
1002 */
1003 template <typename U>
1004 void
1005 operator-=(const U &v)
1006 {
1007 stat.data(index)->dec(v);
1008 }
1009
1010 /**
1011 * Return the number of elements, always 1 for a scalar.
1012 * @return 1.
1013 */
1014 size_type size() const { return 1; }
1015
1016 public:
1017 std::string
1018 str() const
1019 {
1020 return csprintf("%s[%d]", stat.info()->name, index);
1021 }
1022 };
1023
1024 /**
1025 * Implementation of a vector of stats. The type of stat is determined by the
1026 * Storage class. @sa ScalarBase
1027 */
1028 template <class Derived, class Stor>
1029 class VectorBase : public DataWrapVec<Derived, VectorInfoProxy>
1030 {
1031 public:
1032 typedef Stor Storage;
1033 typedef typename Stor::Params Params;
1034
1035 /** Proxy type */
1036 typedef ScalarProxy<Derived> Proxy;
1037 friend class ScalarProxy<Derived>;
1038 friend class DataWrapVec<Derived, VectorInfoProxy>;
1039
1040 protected:
1041 /** The storage of this stat. */
1042 Storage *storage;
1043 size_type _size;
1044
1045 protected:
1046 /**
1047 * Retrieve the storage.
1048 * @param index The vector index to access.
1049 * @return The storage object at the given index.
1050 */
1051 Storage *data(off_type index) { return &storage[index]; }
1052
1053 /**
1054 * Retrieve a const pointer to the storage.
1055 * @param index The vector index to access.
1056 * @return A const pointer to the storage object at the given index.
1057 */
1058 const Storage *data(off_type index) const { return &storage[index]; }
1059
1060 void
1061 doInit(size_type s)
1062 {
1063 assert(s > 0 && "size must be positive!");
1064 assert(!storage && "already initialized");
1065 _size = s;
1066
1067 char *ptr = new char[_size * sizeof(Storage)];
1068 storage = reinterpret_cast<Storage *>(ptr);
1069
1070 for (off_type i = 0; i < _size; ++i)
1071 new (&storage[i]) Storage(this->info());
1072
1073 this->setInit();
1074 }
1075
1076 public:
1077 void
1078 value(VCounter &vec) const
1079 {
1080 vec.resize(size());
1081 for (off_type i = 0; i < size(); ++i)
1082 vec[i] = data(i)->value();
1083 }
1084
1085 /**
1086 * Copy the values to a local vector and return a reference to it.
1087 * @return A reference to a vector of the stat values.
1088 */
1089 void
1090 result(VResult &vec) const
1091 {
1092 vec.resize(size());
1093 for (off_type i = 0; i < size(); ++i)
1094 vec[i] = data(i)->result();
1095 }
1096
1097 /**
1098 * Return a total of all entries in this vector.
1099 * @return The total of all vector entries.
1100 */
1101 Result
1102 total() const
1103 {
1104 Result total = 0.0;
1105 for (off_type i = 0; i < size(); ++i)
1106 total += data(i)->result();
1107 return total;
1108 }
1109
1110 /**
1111 * @return the number of elements in this vector.
1112 */
1113 size_type size() const { return _size; }
1114
1115 bool
1116 zero() const
1117 {
1118 for (off_type i = 0; i < size(); ++i)
1119 if (data(i)->zero())
1120 return false;
1121 return true;
1122 }
1123
1124 bool
1125 check() const
1126 {
1127 return storage != NULL;
1128 }
1129
1130 public:
1131 VectorBase(Group *parent, const char *name, const char *desc)
1132 : DataWrapVec<Derived, VectorInfoProxy>(parent, name, desc),
1133 storage(nullptr), _size(0)
1134 {}
1135
1136 ~VectorBase()
1137 {
1138 if (!storage)
1139 return;
1140
1141 for (off_type i = 0; i < _size; ++i)
1142 data(i)->~Storage();
1143 delete [] reinterpret_cast<char *>(storage);
1144 }
1145
1146 /**
1147 * Set this vector to have the given size.
1148 * @param size The new size.
1149 * @return A reference to this stat.
1150 */
1151 Derived &
1152 init(size_type size)
1153 {
1154 Derived &self = this->self();
1155 self.doInit(size);
1156 return self;
1157 }
1158
1159 /**
1160 * Return a reference (ScalarProxy) to the stat at the given index.
1161 * @param index The vector index to access.
1162 * @return A reference of the stat.
1163 */
1164 Proxy
1165 operator[](off_type index)
1166 {
1167 assert (index >= 0 && index < size());
1168 return Proxy(this->self(), index);
1169 }
1170 };
1171
1172 template <class Stat>
1173 class VectorProxy
1174 {
1175 private:
1176 Stat &stat;
1177 off_type offset;
1178 size_type len;
1179
1180 private:
1181 mutable VResult vec;
1182
1183 typename Stat::Storage *
1184 data(off_type index)
1185 {
1186 assert(index < len);
1187 return stat.data(offset + index);
1188 }
1189
1190 const typename Stat::Storage *
1191 data(off_type index) const
1192 {
1193 assert(index < len);
1194 return stat.data(offset + index);
1195 }
1196
1197 public:
1198 const VResult &
1199 result() const
1200 {
1201 vec.resize(size());
1202
1203 for (off_type i = 0; i < size(); ++i)
1204 vec[i] = data(i)->result();
1205
1206 return vec;
1207 }
1208
1209 Result
1210 total() const
1211 {
1212 Result total = 0.0;
1213 for (off_type i = 0; i < size(); ++i)
1214 total += data(i)->result();
1215 return total;
1216 }
1217
1218 public:
1219 VectorProxy(Stat &s, off_type o, size_type l)
1220 : stat(s), offset(o), len(l)
1221 {
1222 }
1223
1224 VectorProxy(const VectorProxy &sp)
1225 : stat(sp.stat), offset(sp.offset), len(sp.len)
1226 {
1227 }
1228
1229 const VectorProxy &
1230 operator=(const VectorProxy &sp)
1231 {
1232 stat = sp.stat;
1233 offset = sp.offset;
1234 len = sp.len;
1235 return *this;
1236 }
1237
1238 ScalarProxy<Stat>
1239 operator[](off_type index)
1240 {
1241 assert (index >= 0 && index < size());
1242 return ScalarProxy<Stat>(stat, offset + index);
1243 }
1244
1245 size_type size() const { return len; }
1246 };
1247
1248 template <class Derived, class Stor>
1249 class Vector2dBase : public DataWrapVec2d<Derived, Vector2dInfoProxy>
1250 {
1251 public:
1252 typedef Vector2dInfoProxy<Derived> Info;
1253 typedef Stor Storage;
1254 typedef typename Stor::Params Params;
1255 typedef VectorProxy<Derived> Proxy;
1256 friend class ScalarProxy<Derived>;
1257 friend class VectorProxy<Derived>;
1258 friend class DataWrapVec<Derived, Vector2dInfoProxy>;
1259 friend class DataWrapVec2d<Derived, Vector2dInfoProxy>;
1260
1261 protected:
1262 size_type x;
1263 size_type y;
1264 size_type _size;
1265 Storage *storage;
1266
1267 protected:
1268 Storage *data(off_type index) { return &storage[index]; }
1269 const Storage *data(off_type index) const { return &storage[index]; }
1270
1271 public:
1272 Vector2dBase(Group *parent, const char *name, const char *desc)
1273 : DataWrapVec2d<Derived, Vector2dInfoProxy>(parent, name, desc),
1274 x(0), y(0), _size(0), storage(nullptr)
1275 {}
1276
1277 ~Vector2dBase()
1278 {
1279 if (!storage)
1280 return;
1281
1282 for (off_type i = 0; i < _size; ++i)
1283 data(i)->~Storage();
1284 delete [] reinterpret_cast<char *>(storage);
1285 }
1286
1287 Derived &
1288 init(size_type _x, size_type _y)
1289 {
1290 assert(_x > 0 && _y > 0 && "sizes must be positive!");
1291 assert(!storage && "already initialized");
1292
1293 Derived &self = this->self();
1294 Info *info = this->info();
1295
1296 x = _x;
1297 y = _y;
1298 info->x = _x;
1299 info->y = _y;
1300 _size = x * y;
1301
1302 char *ptr = new char[_size * sizeof(Storage)];
1303 storage = reinterpret_cast<Storage *>(ptr);
1304
1305 for (off_type i = 0; i < _size; ++i)
1306 new (&storage[i]) Storage(info);
1307
1308 this->setInit();
1309
1310 return self;
1311 }
1312
1313 Proxy
1314 operator[](off_type index)
1315 {
1316 off_type offset = index * y;
1317 assert (index >= 0 && offset + y <= size());
1318 return Proxy(this->self(), offset, y);
1319 }
1320
1321
1322 size_type
1323 size() const
1324 {
1325 return _size;
1326 }
1327
1328 bool
1329 zero() const
1330 {
1331 return data(0)->zero();
1332 }
1333
1334 /**
1335 * Return a total of all entries in this vector.
1336 * @return The total of all vector entries.
1337 */
1338 Result
1339 total() const
1340 {
1341 Result total = 0.0;
1342 for (off_type i = 0; i < size(); ++i)
1343 total += data(i)->result();
1344 return total;
1345 }
1346
1347 void
1348 prepare()
1349 {
1350 Info *info = this->info();
1351 size_type size = this->size();
1352
1353 for (off_type i = 0; i < size; ++i)
1354 data(i)->prepare(info);
1355
1356 info->cvec.resize(size);
1357 for (off_type i = 0; i < size; ++i)
1358 info->cvec[i] = data(i)->value();
1359 }
1360
1361 /**
1362 * Reset stat value to default
1363 */
1364 void
1365 reset()
1366 {
1367 Info *info = this->info();
1368 size_type size = this->size();
1369 for (off_type i = 0; i < size; ++i)
1370 data(i)->reset(info);
1371 }
1372
1373 bool
1374 check() const
1375 {
1376 return storage != NULL;
1377 }
1378 };
1379
1380 //////////////////////////////////////////////////////////////////////
1381 //
1382 // Non formula statistics
1383 //
1384 //////////////////////////////////////////////////////////////////////
1385 /** The parameters for a distribution stat. */
1386 struct DistParams : public StorageParams
1387 {
1388 const DistType type;
1389 DistParams(DistType t) : type(t) {}
1390 };
1391
1392 /**
1393 * Templatized storage and interface for a distribution stat.
1394 */
1395 class DistStor
1396 {
1397 public:
1398 /** The parameters for a distribution stat. */
1399 struct Params : public DistParams
1400 {
1401 /** The minimum value to track. */
1402 Counter min;
1403 /** The maximum value to track. */
1404 Counter max;
1405 /** The number of entries in each bucket. */
1406 Counter bucket_size;
1407 /** The number of buckets. Equal to (max-min)/bucket_size. */
1408 size_type buckets;
1409
1410 Params() : DistParams(Dist), min(0), max(0), bucket_size(0),
1411 buckets(0) {}
1412 };
1413
1414 private:
1415 /** The minimum value to track. */
1416 Counter min_track;
1417 /** The maximum value to track. */
1418 Counter max_track;
1419 /** The number of entries in each bucket. */
1420 Counter bucket_size;
1421
1422 /** The smallest value sampled. */
1423 Counter min_val;
1424 /** The largest value sampled. */
1425 Counter max_val;
1426 /** The number of values sampled less than min. */
1427 Counter underflow;
1428 /** The number of values sampled more than max. */
1429 Counter overflow;
1430 /** The current sum. */
1431 Counter sum;
1432 /** The sum of squares. */
1433 Counter squares;
1434 /** The number of samples. */
1435 Counter samples;
1436 /** Counter for each bucket. */
1437 VCounter cvec;
1438
1439 public:
1440 DistStor(Info *info)
1441 : cvec(safe_cast<const Params *>(info->storageParams)->buckets)
1442 {
1443 reset(info);
1444 }
1445
1446 /**
1447 * Add a value to the distribution for the given number of times.
1448 * @param val The value to add.
1449 * @param number The number of times to add the value.
1450 */
1451 void
1452 sample(Counter val, int number)
1453 {
1454 if (val < min_track)
1455 underflow += number;
1456 else if (val > max_track)
1457 overflow += number;
1458 else {
1459 size_type index =
1460 (size_type)std::floor((val - min_track) / bucket_size);
1461 assert(index < size());
1462 cvec[index] += number;
1463 }
1464
1465 if (val < min_val)
1466 min_val = val;
1467
1468 if (val > max_val)
1469 max_val = val;
1470
1471 sum += val * number;
1472 squares += val * val * number;
1473 samples += number;
1474 }
1475
1476 /**
1477 * Return the number of buckets in this distribution.
1478 * @return the number of buckets.
1479 */
1480 size_type size() const { return cvec.size(); }
1481
1482 /**
1483 * Returns true if any calls to sample have been made.
1484 * @return True if any values have been sampled.
1485 */
1486 bool
1487 zero() const
1488 {
1489 return samples == Counter();
1490 }
1491
1492 void
1493 prepare(Info *info, DistData &data)
1494 {
1495 const Params *params = safe_cast<const Params *>(info->storageParams);
1496
1497 assert(params->type == Dist);
1498 data.type = params->type;
1499 data.min = params->min;
1500 data.max = params->max;
1501 data.bucket_size = params->bucket_size;
1502
1503 data.min_val = (min_val == CounterLimits::max()) ? 0 : min_val;
1504 data.max_val = (max_val == CounterLimits::min()) ? 0 : max_val;
1505 data.underflow = underflow;
1506 data.overflow = overflow;
1507
1508 data.cvec.resize(params->buckets);
1509 for (off_type i = 0; i < params->buckets; ++i)
1510 data.cvec[i] = cvec[i];
1511
1512 data.sum = sum;
1513 data.squares = squares;
1514 data.samples = samples;
1515 }
1516
1517 /**
1518 * Reset stat value to default
1519 */
1520 void
1521 reset(Info *info)
1522 {
1523 const Params *params = safe_cast<const Params *>(info->storageParams);
1524 min_track = params->min;
1525 max_track = params->max;
1526 bucket_size = params->bucket_size;
1527
1528 min_val = CounterLimits::max();
1529 max_val = CounterLimits::min();
1530 underflow = Counter();
1531 overflow = Counter();
1532
1533 size_type size = cvec.size();
1534 for (off_type i = 0; i < size; ++i)
1535 cvec[i] = Counter();
1536
1537 sum = Counter();
1538 squares = Counter();
1539 samples = Counter();
1540 }
1541 };
1542
1543 /**
1544 * Templatized storage and interface for a histogram stat.
1545 */
1546 class HistStor
1547 {
1548 public:
1549 /** The parameters for a distribution stat. */
1550 struct Params : public DistParams
1551 {
1552 /** The number of buckets.. */
1553 size_type buckets;
1554
1555 Params() : DistParams(Hist), buckets(0) {}
1556 };
1557
1558 private:
1559 /** The minimum value to track. */
1560 Counter min_bucket;
1561 /** The maximum value to track. */
1562 Counter max_bucket;
1563 /** The number of entries in each bucket. */
1564 Counter bucket_size;
1565
1566 /** The current sum. */
1567 Counter sum;
1568 /** The sum of logarithm of each sample, used to compute geometric mean. */
1569 Counter logs;
1570 /** The sum of squares. */
1571 Counter squares;
1572 /** The number of samples. */
1573 Counter samples;
1574 /** Counter for each bucket. */
1575 VCounter cvec;
1576
1577 public:
1578 HistStor(Info *info)
1579 : cvec(safe_cast<const Params *>(info->storageParams)->buckets)
1580 {
1581 reset(info);
1582 }
1583
1584 void grow_up();
1585 void grow_out();
1586 void grow_convert();
1587 void add(HistStor *);
1588
1589 /**
1590 * Add a value to the distribution for the given number of times.
1591 * @param val The value to add.
1592 * @param number The number of times to add the value.
1593 */
1594 void
1595 sample(Counter val, int number)
1596 {
1597 assert(min_bucket < max_bucket);
1598 if (val < min_bucket) {
1599 if (min_bucket == 0)
1600 grow_convert();
1601
1602 while (val < min_bucket)
1603 grow_out();
1604 } else if (val >= max_bucket + bucket_size) {
1605 if (min_bucket == 0) {
1606 while (val >= max_bucket + bucket_size)
1607 grow_up();
1608 } else {
1609 while (val >= max_bucket + bucket_size)
1610 grow_out();
1611 }
1612 }
1613
1614 size_type index =
1615 (int64_t)std::floor((val - min_bucket) / bucket_size);
1616
1617 assert(index < size());
1618 cvec[index] += number;
1619
1620 sum += val * number;
1621 squares += val * val * number;
1622 logs += log(val) * number;
1623 samples += number;
1624 }
1625
1626 /**
1627 * Return the number of buckets in this distribution.
1628 * @return the number of buckets.
1629 */
1630 size_type size() const { return cvec.size(); }
1631
1632 /**
1633 * Returns true if any calls to sample have been made.
1634 * @return True if any values have been sampled.
1635 */
1636 bool
1637 zero() const
1638 {
1639 return samples == Counter();
1640 }
1641
1642 void
1643 prepare(Info *info, DistData &data)
1644 {
1645 const Params *params = safe_cast<const Params *>(info->storageParams);
1646
1647 assert(params->type == Hist);
1648 data.type = params->type;
1649 data.min = min_bucket;
1650 data.max = max_bucket + bucket_size - 1;
1651 data.bucket_size = bucket_size;
1652
1653 data.min_val = min_bucket;
1654 data.max_val = max_bucket;
1655
1656 int buckets = params->buckets;
1657 data.cvec.resize(buckets);
1658 for (off_type i = 0; i < buckets; ++i)
1659 data.cvec[i] = cvec[i];
1660
1661 data.sum = sum;
1662 data.logs = logs;
1663 data.squares = squares;
1664 data.samples = samples;
1665 }
1666
1667 /**
1668 * Reset stat value to default
1669 */
1670 void
1671 reset(Info *info)
1672 {
1673 const Params *params = safe_cast<const Params *>(info->storageParams);
1674 min_bucket = 0;
1675 max_bucket = params->buckets - 1;
1676 bucket_size = 1;
1677
1678 size_type size = cvec.size();
1679 for (off_type i = 0; i < size; ++i)
1680 cvec[i] = Counter();
1681
1682 sum = Counter();
1683 squares = Counter();
1684 samples = Counter();
1685 logs = Counter();
1686 }
1687 };
1688
1689 /**
1690 * Templatized storage and interface for a distribution that calculates mean
1691 * and variance.
1692 */
1693 class SampleStor
1694 {
1695 public:
1696 struct Params : public DistParams
1697 {
1698 Params() : DistParams(Deviation) {}
1699 };
1700
1701 private:
1702 /** The current sum. */
1703 Counter sum;
1704 /** The sum of squares. */
1705 Counter squares;
1706 /** The number of samples. */
1707 Counter samples;
1708
1709 public:
1710 /**
1711 * Create and initialize this storage.
1712 */
1713 SampleStor(Info *info)
1714 : sum(Counter()), squares(Counter()), samples(Counter())
1715 { }
1716
1717 /**
1718 * Add a value the given number of times to this running average.
1719 * Update the running sum and sum of squares, increment the number of
1720 * values seen by the given number.
1721 * @param val The value to add.
1722 * @param number The number of times to add the value.
1723 */
1724 void
1725 sample(Counter val, int number)
1726 {
1727 sum += val * number;
1728 squares += val * val * number;
1729 samples += number;
1730 }
1731
1732 /**
1733 * Return the number of entries in this stat, 1
1734 * @return 1.
1735 */
1736 size_type size() const { return 1; }
1737
1738 /**
1739 * Return true if no samples have been added.
1740 * @return True if no samples have been added.
1741 */
1742 bool zero() const { return samples == Counter(); }
1743
1744 void
1745 prepare(Info *info, DistData &data)
1746 {
1747 const Params *params = safe_cast<const Params *>(info->storageParams);
1748
1749 assert(params->type == Deviation);
1750 data.type = params->type;
1751 data.sum = sum;
1752 data.squares = squares;
1753 data.samples = samples;
1754 }
1755
1756 /**
1757 * Reset stat value to default
1758 */
1759 void
1760 reset(Info *info)
1761 {
1762 sum = Counter();
1763 squares = Counter();
1764 samples = Counter();
1765 }
1766 };
1767
1768 /**
1769 * Templatized storage for distribution that calculates per tick mean and
1770 * variance.
1771 */
1772 class AvgSampleStor
1773 {
1774 public:
1775 struct Params : public DistParams
1776 {
1777 Params() : DistParams(Deviation) {}
1778 };
1779
1780 private:
1781 /** Current total. */
1782 Counter sum;
1783 /** Current sum of squares. */
1784 Counter squares;
1785
1786 public:
1787 /**
1788 * Create and initialize this storage.
1789 */
1790 AvgSampleStor(Info *info)
1791 : sum(Counter()), squares(Counter())
1792 {}
1793
1794 /**
1795 * Add a value to the distribution for the given number of times.
1796 * Update the running sum and sum of squares.
1797 * @param val The value to add.
1798 * @param number The number of times to add the value.
1799 */
1800 void
1801 sample(Counter val, int number)
1802 {
1803 sum += val * number;
1804 squares += val * val * number;
1805 }
1806
1807 /**
1808 * Return the number of entries, in this case 1.
1809 * @return 1.
1810 */
1811 size_type size() const { return 1; }
1812
1813 /**
1814 * Return true if no samples have been added.
1815 * @return True if the sum is zero.
1816 */
1817 bool zero() const { return sum == Counter(); }
1818
1819 void
1820 prepare(Info *info, DistData &data)
1821 {
1822 const Params *params = safe_cast<const Params *>(info->storageParams);
1823
1824 assert(params->type == Deviation);
1825 data.type = params->type;
1826 data.sum = sum;
1827 data.squares = squares;
1828 data.samples = curTick();
1829 }
1830
1831 /**
1832 * Reset stat value to default
1833 */
1834 void
1835 reset(Info *info)
1836 {
1837 sum = Counter();
1838 squares = Counter();
1839 }
1840 };
1841
1842 /**
1843 * Implementation of a distribution stat. The type of distribution is
1844 * determined by the Storage template. @sa ScalarBase
1845 */
1846 template <class Derived, class Stor>
1847 class DistBase : public DataWrap<Derived, DistInfoProxy>
1848 {
1849 public:
1850 typedef DistInfoProxy<Derived> Info;
1851 typedef Stor Storage;
1852 typedef typename Stor::Params Params;
1853
1854 protected:
1855 /** The storage for this stat. */
1856 char storage[sizeof(Storage)] __attribute__ ((aligned (8)));
1857
1858 protected:
1859 /**
1860 * Retrieve the storage.
1861 * @return The storage object for this stat.
1862 */
1863 Storage *
1864 data()
1865 {
1866 return reinterpret_cast<Storage *>(storage);
1867 }
1868
1869 /**
1870 * Retrieve a const pointer to the storage.
1871 * @return A const pointer to the storage object for this stat.
1872 */
1873 const Storage *
1874 data() const
1875 {
1876 return reinterpret_cast<const Storage *>(storage);
1877 }
1878
1879 void
1880 doInit()
1881 {
1882 new (storage) Storage(this->info());
1883 this->setInit();
1884 }
1885
1886 public:
1887 DistBase(Group *parent, const char *name, const char *desc)
1888 : DataWrap<Derived, DistInfoProxy>(parent, name, desc)
1889 {
1890 }
1891
1892 /**
1893 * Add a value to the distribtion n times. Calls sample on the storage
1894 * class.
1895 * @param v The value to add.
1896 * @param n The number of times to add it, defaults to 1.
1897 */
1898 template <typename U>
1899 void sample(const U &v, int n = 1) { data()->sample(v, n); }
1900
1901 /**
1902 * Return the number of entries in this stat.
1903 * @return The number of entries.
1904 */
1905 size_type size() const { return data()->size(); }
1906 /**
1907 * Return true if no samples have been added.
1908 * @return True if there haven't been any samples.
1909 */
1910 bool zero() const { return data()->zero(); }
1911
1912 void
1913 prepare()
1914 {
1915 Info *info = this->info();
1916 data()->prepare(info, info->data);
1917 }
1918
1919 /**
1920 * Reset stat value to default
1921 */
1922 void
1923 reset()
1924 {
1925 data()->reset(this->info());
1926 }
1927
1928 /**
1929 * Add the argument distribution to the this distribution.
1930 */
1931 void add(DistBase &d) { data()->add(d.data()); }
1932
1933 };
1934
1935 template <class Stat>
1936 class DistProxy;
1937
1938 template <class Derived, class Stor>
1939 class VectorDistBase : public DataWrapVec<Derived, VectorDistInfoProxy>
1940 {
1941 public:
1942 typedef VectorDistInfoProxy<Derived> Info;
1943 typedef Stor Storage;
1944 typedef typename Stor::Params Params;
1945 typedef DistProxy<Derived> Proxy;
1946 friend class DistProxy<Derived>;
1947 friend class DataWrapVec<Derived, VectorDistInfoProxy>;
1948
1949 protected:
1950 Storage *storage;
1951 size_type _size;
1952
1953 protected:
1954 Storage *
1955 data(off_type index)
1956 {
1957 return &storage[index];
1958 }
1959
1960 const Storage *
1961 data(off_type index) const
1962 {
1963 return &storage[index];
1964 }
1965
1966 void
1967 doInit(size_type s)
1968 {
1969 assert(s > 0 && "size must be positive!");
1970 assert(!storage && "already initialized");
1971 _size = s;
1972
1973 char *ptr = new char[_size * sizeof(Storage)];
1974 storage = reinterpret_cast<Storage *>(ptr);
1975
1976 Info *info = this->info();
1977 for (off_type i = 0; i < _size; ++i)
1978 new (&storage[i]) Storage(info);
1979
1980 this->setInit();
1981 }
1982
1983 public:
1984 VectorDistBase(Group *parent, const char *name, const char *desc)
1985 : DataWrapVec<Derived, VectorDistInfoProxy>(parent, name, desc),
1986 storage(NULL)
1987 {}
1988
1989 ~VectorDistBase()
1990 {
1991 if (!storage)
1992 return ;
1993
1994 for (off_type i = 0; i < _size; ++i)
1995 data(i)->~Storage();
1996 delete [] reinterpret_cast<char *>(storage);
1997 }
1998
1999 Proxy operator[](off_type index)
2000 {
2001 assert(index >= 0 && index < size());
2002 return Proxy(this->self(), index);
2003 }
2004
2005 size_type
2006 size() const
2007 {
2008 return _size;
2009 }
2010
2011 bool
2012 zero() const
2013 {
2014 for (off_type i = 0; i < size(); ++i)
2015 if (!data(i)->zero())
2016 return false;
2017 return true;
2018 }
2019
2020 void
2021 prepare()
2022 {
2023 Info *info = this->info();
2024 size_type size = this->size();
2025 info->data.resize(size);
2026 for (off_type i = 0; i < size; ++i)
2027 data(i)->prepare(info, info->data[i]);
2028 }
2029
2030 bool
2031 check() const
2032 {
2033 return storage != NULL;
2034 }
2035 };
2036
2037 template <class Stat>
2038 class DistProxy
2039 {
2040 private:
2041 Stat &stat;
2042 off_type index;
2043
2044 protected:
2045 typename Stat::Storage *data() { return stat.data(index); }
2046 const typename Stat::Storage *data() const { return stat.data(index); }
2047
2048 public:
2049 DistProxy(Stat &s, off_type i)
2050 : stat(s), index(i)
2051 {}
2052
2053 DistProxy(const DistProxy &sp)
2054 : stat(sp.stat), index(sp.index)
2055 {}
2056
2057 const DistProxy &
2058 operator=(const DistProxy &sp)
2059 {
2060 stat = sp.stat;
2061 index = sp.index;
2062 return *this;
2063 }
2064
2065 public:
2066 template <typename U>
2067 void
2068 sample(const U &v, int n = 1)
2069 {
2070 data()->sample(v, n);
2071 }
2072
2073 size_type
2074 size() const
2075 {
2076 return 1;
2077 }
2078
2079 bool
2080 zero() const
2081 {
2082 return data()->zero();
2083 }
2084
2085 /**
2086 * Proxy has no state. Nothing to reset.
2087 */
2088 void reset() { }
2089 };
2090
2091 //////////////////////////////////////////////////////////////////////
2092 //
2093 // Formula Details
2094 //
2095 //////////////////////////////////////////////////////////////////////
2096
2097 /**
2098 * Base class for formula statistic node. These nodes are used to build a tree
2099 * that represents the formula.
2100 */
2101 class Node
2102 {
2103 public:
2104 /**
2105 * Return the number of nodes in the subtree starting at this node.
2106 * @return the number of nodes in this subtree.
2107 */
2108 virtual size_type size() const = 0;
2109 /**
2110 * Return the result vector of this subtree.
2111 * @return The result vector of this subtree.
2112 */
2113 virtual const VResult &result() const = 0;
2114 /**
2115 * Return the total of the result vector.
2116 * @return The total of the result vector.
2117 */
2118 virtual Result total() const = 0;
2119
2120 /**
2121 *
2122 */
2123 virtual std::string str() const = 0;
2124
2125 virtual ~Node() {};
2126 };
2127
2128 /** Shared pointer to a function Node. */
2129 typedef std::shared_ptr<Node> NodePtr;
2130
2131 class ScalarStatNode : public Node
2132 {
2133 private:
2134 const ScalarInfo *data;
2135 mutable VResult vresult;
2136
2137 public:
2138 ScalarStatNode(const ScalarInfo *d) : data(d), vresult(1) {}
2139
2140 const VResult &
2141 result() const
2142 {
2143 vresult[0] = data->result();
2144 return vresult;
2145 }
2146
2147 Result total() const { return data->result(); };
2148
2149 size_type size() const { return 1; }
2150
2151 /**
2152 *
2153 */
2154 std::string str() const { return data->name; }
2155 };
2156
2157 template <class Stat>
2158 class ScalarProxyNode : public Node
2159 {
2160 private:
2161 const ScalarProxy<Stat> proxy;
2162 mutable VResult vresult;
2163
2164 public:
2165 ScalarProxyNode(const ScalarProxy<Stat> &p)
2166 : proxy(p), vresult(1)
2167 { }
2168
2169 const VResult &
2170 result() const
2171 {
2172 vresult[0] = proxy.result();
2173 return vresult;
2174 }
2175
2176 Result
2177 total() const
2178 {
2179 return proxy.result();
2180 }
2181
2182 size_type
2183 size() const
2184 {
2185 return 1;
2186 }
2187
2188 /**
2189 *
2190 */
2191 std::string
2192 str() const
2193 {
2194 return proxy.str();
2195 }
2196 };
2197
2198 class VectorStatNode : public Node
2199 {
2200 private:
2201 const VectorInfo *data;
2202
2203 public:
2204 VectorStatNode(const VectorInfo *d) : data(d) { }
2205 const VResult &result() const { return data->result(); }
2206 Result total() const { return data->total(); };
2207
2208 size_type size() const { return data->size(); }
2209
2210 std::string str() const { return data->name; }
2211 };
2212
2213 template <class T>
2214 class ConstNode : public Node
2215 {
2216 private:
2217 VResult vresult;
2218
2219 public:
2220 ConstNode(T s) : vresult(1, (Result)s) {}
2221 const VResult &result() const { return vresult; }
2222 Result total() const { return vresult[0]; };
2223 size_type size() const { return 1; }
2224 std::string str() const { return std::to_string(vresult[0]); }
2225 };
2226
2227 template <class T>
2228 class ConstVectorNode : public Node
2229 {
2230 private:
2231 VResult vresult;
2232
2233 public:
2234 ConstVectorNode(const T &s) : vresult(s.begin(), s.end()) {}
2235 const VResult &result() const { return vresult; }
2236
2237 Result
2238 total() const
2239 {
2240 size_type size = this->size();
2241 Result tmp = 0;
2242 for (off_type i = 0; i < size; i++)
2243 tmp += vresult[i];
2244 return tmp;
2245 }
2246
2247 size_type size() const { return vresult.size(); }
2248 std::string
2249 str() const
2250 {
2251 size_type size = this->size();
2252 std::string tmp = "(";
2253 for (off_type i = 0; i < size; i++)
2254 tmp += csprintf("%s ", std::to_string(vresult[i]));
2255 tmp += ")";
2256 return tmp;
2257 }
2258 };
2259
2260 template <class Op>
2261 struct OpString;
2262
2263 template<>
2264 struct OpString<std::plus<Result> >
2265 {
2266 static std::string str() { return "+"; }
2267 };
2268
2269 template<>
2270 struct OpString<std::minus<Result> >
2271 {
2272 static std::string str() { return "-"; }
2273 };
2274
2275 template<>
2276 struct OpString<std::multiplies<Result> >
2277 {
2278 static std::string str() { return "*"; }
2279 };
2280
2281 template<>
2282 struct OpString<std::divides<Result> >
2283 {
2284 static std::string str() { return "/"; }
2285 };
2286
2287 template<>
2288 struct OpString<std::modulus<Result> >
2289 {
2290 static std::string str() { return "%"; }
2291 };
2292
2293 template<>
2294 struct OpString<std::negate<Result> >
2295 {
2296 static std::string str() { return "-"; }
2297 };
2298
2299 template <class Op>
2300 class UnaryNode : public Node
2301 {
2302 public:
2303 NodePtr l;
2304 mutable VResult vresult;
2305
2306 public:
2307 UnaryNode(NodePtr &p) : l(p) {}
2308
2309 const VResult &
2310 result() const
2311 {
2312 const VResult &lvec = l->result();
2313 size_type size = lvec.size();
2314
2315 assert(size > 0);
2316
2317 vresult.resize(size);
2318 Op op;
2319 for (off_type i = 0; i < size; ++i)
2320 vresult[i] = op(lvec[i]);
2321
2322 return vresult;
2323 }
2324
2325 Result
2326 total() const
2327 {
2328 const VResult &vec = this->result();
2329 Result total = 0.0;
2330 for (off_type i = 0; i < size(); i++)
2331 total += vec[i];
2332 return total;
2333 }
2334
2335 size_type size() const { return l->size(); }
2336
2337 std::string
2338 str() const
2339 {
2340 return OpString<Op>::str() + l->str();
2341 }
2342 };
2343
2344 template <class Op>
2345 class BinaryNode : public Node
2346 {
2347 public:
2348 NodePtr l;
2349 NodePtr r;
2350 mutable VResult vresult;
2351
2352 public:
2353 BinaryNode(NodePtr &a, NodePtr &b) : l(a), r(b) {}
2354
2355 const VResult &
2356 result() const override
2357 {
2358 Op op;
2359 const VResult &lvec = l->result();
2360 const VResult &rvec = r->result();
2361
2362 assert(lvec.size() > 0 && rvec.size() > 0);
2363
2364 if (lvec.size() == 1 && rvec.size() == 1) {
2365 vresult.resize(1);
2366 vresult[0] = op(lvec[0], rvec[0]);
2367 } else if (lvec.size() == 1) {
2368 size_type size = rvec.size();
2369 vresult.resize(size);
2370 for (off_type i = 0; i < size; ++i)
2371 vresult[i] = op(lvec[0], rvec[i]);
2372 } else if (rvec.size() == 1) {
2373 size_type size = lvec.size();
2374 vresult.resize(size);
2375 for (off_type i = 0; i < size; ++i)
2376 vresult[i] = op(lvec[i], rvec[0]);
2377 } else if (rvec.size() == lvec.size()) {
2378 size_type size = rvec.size();
2379 vresult.resize(size);
2380 for (off_type i = 0; i < size; ++i)
2381 vresult[i] = op(lvec[i], rvec[i]);
2382 }
2383
2384 return vresult;
2385 }
2386
2387 Result
2388 total() const override
2389 {
2390 const VResult &vec = this->result();
2391 const VResult &lvec = l->result();
2392 const VResult &rvec = r->result();
2393 Result total = 0.0;
2394 Result lsum = 0.0;
2395 Result rsum = 0.0;
2396 Op op;
2397
2398 assert(lvec.size() > 0 && rvec.size() > 0);
2399 assert(lvec.size() == rvec.size() ||
2400 lvec.size() == 1 || rvec.size() == 1);
2401
2402 /** If vectors are the same divide their sums (x0+x1)/(y0+y1) */
2403 if (lvec.size() == rvec.size() && lvec.size() > 1) {
2404 for (off_type i = 0; i < size(); ++i) {
2405 lsum += lvec[i];
2406 rsum += rvec[i];
2407 }
2408 return op(lsum, rsum);
2409 }
2410
2411 /** Otherwise divide each item by the divisor */
2412 for (off_type i = 0; i < size(); ++i) {
2413 total += vec[i];
2414 }
2415
2416 return total;
2417 }
2418
2419 size_type
2420 size() const override
2421 {
2422 size_type ls = l->size();
2423 size_type rs = r->size();
2424 if (ls == 1) {
2425 return rs;
2426 } else if (rs == 1) {
2427 return ls;
2428 } else {
2429 assert(ls == rs && "Node vector sizes are not equal");
2430 return ls;
2431 }
2432 }
2433
2434 std::string
2435 str() const override
2436 {
2437 return csprintf("(%s %s %s)", l->str(), OpString<Op>::str(), r->str());
2438 }
2439 };
2440
2441 template <class Op>
2442 class SumNode : public Node
2443 {
2444 public:
2445 NodePtr l;
2446 mutable VResult vresult;
2447
2448 public:
2449 SumNode(NodePtr &p) : l(p), vresult(1) {}
2450
2451 const VResult &
2452 result() const
2453 {
2454 const VResult &lvec = l->result();
2455 size_type size = lvec.size();
2456 assert(size > 0);
2457
2458 vresult[0] = 0.0;
2459
2460 Op op;
2461 for (off_type i = 0; i < size; ++i)
2462 vresult[0] = op(vresult[0], lvec[i]);
2463
2464 return vresult;
2465 }
2466
2467 Result
2468 total() const
2469 {
2470 const VResult &lvec = l->result();
2471 size_type size = lvec.size();
2472 assert(size > 0);
2473
2474 Result result = 0.0;
2475
2476 Op op;
2477 for (off_type i = 0; i < size; ++i)
2478 result = op(result, lvec[i]);
2479
2480 return result;
2481 }
2482
2483 size_type size() const { return 1; }
2484
2485 std::string
2486 str() const
2487 {
2488 return csprintf("total(%s)", l->str());
2489 }
2490 };
2491
2492
2493 //////////////////////////////////////////////////////////////////////
2494 //
2495 // Visible Statistics Types
2496 //
2497 //////////////////////////////////////////////////////////////////////
2498 /**
2499 * @defgroup VisibleStats "Statistic Types"
2500 * These are the statistics that are used in the simulator.
2501 * @{
2502 */
2503
2504 /**
2505 * This is a simple scalar statistic, like a counter.
2506 * @sa Stat, ScalarBase, StatStor
2507 */
2508 class Scalar : public ScalarBase<Scalar, StatStor>
2509 {
2510 public:
2511 using ScalarBase<Scalar, StatStor>::operator=;
2512
2513 Scalar(Group *parent = nullptr, const char *name = nullptr,
2514 const char *desc = nullptr)
2515 : ScalarBase<Scalar, StatStor>(parent, name, desc)
2516 {
2517 }
2518 };
2519
2520 /**
2521 * A stat that calculates the per tick average of a value.
2522 * @sa Stat, ScalarBase, AvgStor
2523 */
2524 class Average : public ScalarBase<Average, AvgStor>
2525 {
2526 public:
2527 using ScalarBase<Average, AvgStor>::operator=;
2528
2529 Average(Group *parent = nullptr, const char *name = nullptr,
2530 const char *desc = nullptr)
2531 : ScalarBase<Average, AvgStor>(parent, name, desc)
2532 {
2533 }
2534 };
2535
2536 class Value : public ValueBase<Value>
2537 {
2538 public:
2539 Value(Group *parent = nullptr, const char *name = nullptr,
2540 const char *desc = nullptr)
2541 : ValueBase<Value>(parent, name, desc)
2542 {
2543 }
2544 };
2545
2546 /**
2547 * A vector of scalar stats.
2548 * @sa Stat, VectorBase, StatStor
2549 */
2550 class Vector : public VectorBase<Vector, StatStor>
2551 {
2552 public:
2553 Vector(Group *parent = nullptr, const char *name = nullptr,
2554 const char *desc = nullptr)
2555 : VectorBase<Vector, StatStor>(parent, name, desc)
2556 {
2557 }
2558 };
2559
2560 /**
2561 * A vector of Average stats.
2562 * @sa Stat, VectorBase, AvgStor
2563 */
2564 class AverageVector : public VectorBase<AverageVector, AvgStor>
2565 {
2566 public:
2567 AverageVector(Group *parent = nullptr, const char *name = nullptr,
2568 const char *desc = nullptr)
2569 : VectorBase<AverageVector, AvgStor>(parent, name, desc)
2570 {
2571 }
2572 };
2573
2574 /**
2575 * A 2-Dimensional vecto of scalar stats.
2576 * @sa Stat, Vector2dBase, StatStor
2577 */
2578 class Vector2d : public Vector2dBase<Vector2d, StatStor>
2579 {
2580 public:
2581 Vector2d(Group *parent = nullptr, const char *name = nullptr,
2582 const char *desc = nullptr)
2583 : Vector2dBase<Vector2d, StatStor>(parent, name, desc)
2584 {
2585 }
2586 };
2587
2588 /**
2589 * A simple distribution stat.
2590 * @sa Stat, DistBase, DistStor
2591 */
2592 class Distribution : public DistBase<Distribution, DistStor>
2593 {
2594 public:
2595 Distribution(Group *parent = nullptr, const char *name = nullptr,
2596 const char *desc = nullptr)
2597 : DistBase<Distribution, DistStor>(parent, name, desc)
2598 {
2599 }
2600
2601 /**
2602 * Set the parameters of this distribution. @sa DistStor::Params
2603 * @param min The minimum value of the distribution.
2604 * @param max The maximum value of the distribution.
2605 * @param bkt The number of values in each bucket.
2606 * @return A reference to this distribution.
2607 */
2608 Distribution &
2609 init(Counter min, Counter max, Counter bkt)
2610 {
2611 DistStor::Params *params = new DistStor::Params;
2612 params->min = min;
2613 params->max = max;
2614 params->bucket_size = bkt;
2615 // Division by zero is especially serious in an Aarch64 host,
2616 // where it gets rounded to allocate 32GiB RAM.
2617 assert(bkt > 0);
2618 params->buckets = (size_type)ceil((max - min + 1.0) / bkt);
2619 this->setParams(params);
2620 this->doInit();
2621 return this->self();
2622 }
2623 };
2624
2625 /**
2626 * A simple histogram stat.
2627 * @sa Stat, DistBase, HistStor
2628 */
2629 class Histogram : public DistBase<Histogram, HistStor>
2630 {
2631 public:
2632 Histogram(Group *parent = nullptr, const char *name = nullptr,
2633 const char *desc = nullptr)
2634 : DistBase<Histogram, HistStor>(parent, name, desc)
2635 {
2636 }
2637
2638 /**
2639 * Set the parameters of this histogram. @sa HistStor::Params
2640 * @param size The number of buckets in the histogram
2641 * @return A reference to this histogram.
2642 */
2643 Histogram &
2644 init(size_type size)
2645 {
2646 HistStor::Params *params = new HistStor::Params;
2647 params->buckets = size;
2648 this->setParams(params);
2649 this->doInit();
2650 return this->self();
2651 }
2652 };
2653
2654 /**
2655 * Calculates the mean and variance of all the samples.
2656 * @sa DistBase, SampleStor
2657 */
2658 class StandardDeviation : public DistBase<StandardDeviation, SampleStor>
2659 {
2660 public:
2661 /**
2662 * Construct and initialize this distribution.
2663 */
2664 StandardDeviation(Group *parent = nullptr, const char *name = nullptr,
2665 const char *desc = nullptr)
2666 : DistBase<StandardDeviation, SampleStor>(parent, name, desc)
2667 {
2668 SampleStor::Params *params = new SampleStor::Params;
2669 this->doInit();
2670 this->setParams(params);
2671 }
2672 };
2673
2674 /**
2675 * Calculates the per tick mean and variance of the samples.
2676 * @sa DistBase, AvgSampleStor
2677 */
2678 class AverageDeviation : public DistBase<AverageDeviation, AvgSampleStor>
2679 {
2680 public:
2681 /**
2682 * Construct and initialize this distribution.
2683 */
2684 AverageDeviation(Group *parent = nullptr, const char *name = nullptr,
2685 const char *desc = nullptr)
2686 : DistBase<AverageDeviation, AvgSampleStor>(parent, name, desc)
2687 {
2688 AvgSampleStor::Params *params = new AvgSampleStor::Params;
2689 this->doInit();
2690 this->setParams(params);
2691 }
2692 };
2693
2694 /**
2695 * A vector of distributions.
2696 * @sa VectorDistBase, DistStor
2697 */
2698 class VectorDistribution : public VectorDistBase<VectorDistribution, DistStor>
2699 {
2700 public:
2701 VectorDistribution(Group *parent = nullptr, const char *name = nullptr,
2702 const char *desc = nullptr)
2703 : VectorDistBase<VectorDistribution, DistStor>(parent, name, desc)
2704 {
2705 }
2706
2707 /**
2708 * Initialize storage and parameters for this distribution.
2709 * @param size The size of the vector (the number of distributions).
2710 * @param min The minimum value of the distribution.
2711 * @param max The maximum value of the distribution.
2712 * @param bkt The number of values in each bucket.
2713 * @return A reference to this distribution.
2714 */
2715 VectorDistribution &
2716 init(size_type size, Counter min, Counter max, Counter bkt)
2717 {
2718 DistStor::Params *params = new DistStor::Params;
2719 params->min = min;
2720 params->max = max;
2721 params->bucket_size = bkt;
2722 params->buckets = (size_type)ceil((max - min + 1.0) / bkt);
2723 this->setParams(params);
2724 this->doInit(size);
2725 return this->self();
2726 }
2727 };
2728
2729 /**
2730 * This is a vector of StandardDeviation stats.
2731 * @sa VectorDistBase, SampleStor
2732 */
2733 class VectorStandardDeviation
2734 : public VectorDistBase<VectorStandardDeviation, SampleStor>
2735 {
2736 public:
2737 VectorStandardDeviation(Group *parent = nullptr, const char *name = nullptr,
2738 const char *desc = nullptr)
2739 : VectorDistBase<VectorStandardDeviation, SampleStor>(parent, name,
2740 desc)
2741 {
2742 }
2743
2744 /**
2745 * Initialize storage for this distribution.
2746 * @param size The size of the vector.
2747 * @return A reference to this distribution.
2748 */
2749 VectorStandardDeviation &
2750 init(size_type size)
2751 {
2752 SampleStor::Params *params = new SampleStor::Params;
2753 this->doInit(size);
2754 this->setParams(params);
2755 return this->self();
2756 }
2757 };
2758
2759 /**
2760 * This is a vector of AverageDeviation stats.
2761 * @sa VectorDistBase, AvgSampleStor
2762 */
2763 class VectorAverageDeviation
2764 : public VectorDistBase<VectorAverageDeviation, AvgSampleStor>
2765 {
2766 public:
2767 VectorAverageDeviation(Group *parent = nullptr, const char *name = nullptr,
2768 const char *desc = nullptr)
2769 : VectorDistBase<VectorAverageDeviation, AvgSampleStor>(parent, name,
2770 desc)
2771 {
2772 }
2773
2774 /**
2775 * Initialize storage for this distribution.
2776 * @param size The size of the vector.
2777 * @return A reference to this distribution.
2778 */
2779 VectorAverageDeviation &
2780 init(size_type size)
2781 {
2782 AvgSampleStor::Params *params = new AvgSampleStor::Params;
2783 this->doInit(size);
2784 this->setParams(params);
2785 return this->self();
2786 }
2787 };
2788
2789 template <class Stat>
2790 class FormulaInfoProxy : public InfoProxy<Stat, FormulaInfo>
2791 {
2792 protected:
2793 mutable VResult vec;
2794 mutable VCounter cvec;
2795
2796 public:
2797 FormulaInfoProxy(Stat &stat) : InfoProxy<Stat, FormulaInfo>(stat) {}
2798
2799 size_type size() const { return this->s.size(); }
2800
2801 const VResult &
2802 result() const
2803 {
2804 this->s.result(vec);
2805 return vec;
2806 }
2807 Result total() const { return this->s.total(); }
2808 VCounter &value() const { return cvec; }
2809
2810 std::string str() const { return this->s.str(); }
2811 };
2812
2813 template <class Stat>
2814 class SparseHistInfoProxy : public InfoProxy<Stat, SparseHistInfo>
2815 {
2816 public:
2817 SparseHistInfoProxy(Stat &stat) : InfoProxy<Stat, SparseHistInfo>(stat) {}
2818 };
2819
2820 /**
2821 * Implementation of a sparse histogram stat. The storage class is
2822 * determined by the Storage template.
2823 */
2824 template <class Derived, class Stor>
2825 class SparseHistBase : public DataWrap<Derived, SparseHistInfoProxy>
2826 {
2827 public:
2828 typedef SparseHistInfoProxy<Derived> Info;
2829 typedef Stor Storage;
2830 typedef typename Stor::Params Params;
2831
2832 protected:
2833 /** The storage for this stat. */
2834 char storage[sizeof(Storage)];
2835
2836 protected:
2837 /**
2838 * Retrieve the storage.
2839 * @return The storage object for this stat.
2840 */
2841 Storage *
2842 data()
2843 {
2844 return reinterpret_cast<Storage *>(storage);
2845 }
2846
2847 /**
2848 * Retrieve a const pointer to the storage.
2849 * @return A const pointer to the storage object for this stat.
2850 */
2851 const Storage *
2852 data() const
2853 {
2854 return reinterpret_cast<const Storage *>(storage);
2855 }
2856
2857 void
2858 doInit()
2859 {
2860 new (storage) Storage(this->info());
2861 this->setInit();
2862 }
2863
2864 public:
2865 SparseHistBase(Group *parent, const char *name, const char *desc)
2866 : DataWrap<Derived, SparseHistInfoProxy>(parent, name, desc)
2867 {
2868 }
2869
2870 /**
2871 * Add a value to the distribtion n times. Calls sample on the storage
2872 * class.
2873 * @param v The value to add.
2874 * @param n The number of times to add it, defaults to 1.
2875 */
2876 template <typename U>
2877 void sample(const U &v, int n = 1) { data()->sample(v, n); }
2878
2879 /**
2880 * Return the number of entries in this stat.
2881 * @return The number of entries.
2882 */
2883 size_type size() const { return data()->size(); }
2884 /**
2885 * Return true if no samples have been added.
2886 * @return True if there haven't been any samples.
2887 */
2888 bool zero() const { return data()->zero(); }
2889
2890 void
2891 prepare()
2892 {
2893 Info *info = this->info();
2894 data()->prepare(info, info->data);
2895 }
2896
2897 /**
2898 * Reset stat value to default
2899 */
2900 void
2901 reset()
2902 {
2903 data()->reset(this->info());
2904 }
2905 };
2906
2907 /**
2908 * Templatized storage and interface for a sparse histogram stat.
2909 */
2910 class SparseHistStor
2911 {
2912 public:
2913 /** The parameters for a sparse histogram stat. */
2914 struct Params : public DistParams
2915 {
2916 Params() : DistParams(Hist) {}
2917 };
2918
2919 private:
2920 /** Counter for number of samples */
2921 Counter samples;
2922 /** Counter for each bucket. */
2923 MCounter cmap;
2924
2925 public:
2926 SparseHistStor(Info *info)
2927 {
2928 reset(info);
2929 }
2930
2931 /**
2932 * Add a value to the distribution for the given number of times.
2933 * @param val The value to add.
2934 * @param number The number of times to add the value.
2935 */
2936 void
2937 sample(Counter val, int number)
2938 {
2939 cmap[val] += number;
2940 samples += number;
2941 }
2942
2943 /**
2944 * Return the number of buckets in this distribution.
2945 * @return the number of buckets.
2946 */
2947 size_type size() const { return cmap.size(); }
2948
2949 /**
2950 * Returns true if any calls to sample have been made.
2951 * @return True if any values have been sampled.
2952 */
2953 bool
2954 zero() const
2955 {
2956 return samples == Counter();
2957 }
2958
2959 void
2960 prepare(Info *info, SparseHistData &data)
2961 {
2962 MCounter::iterator it;
2963 data.cmap.clear();
2964 for (it = cmap.begin(); it != cmap.end(); it++) {
2965 data.cmap[(*it).first] = (*it).second;
2966 }
2967
2968 data.samples = samples;
2969 }
2970
2971 /**
2972 * Reset stat value to default
2973 */
2974 void
2975 reset(Info *info)
2976 {
2977 cmap.clear();
2978 samples = 0;
2979 }
2980 };
2981
2982 class SparseHistogram : public SparseHistBase<SparseHistogram, SparseHistStor>
2983 {
2984 public:
2985 SparseHistogram(Group *parent = nullptr, const char *name = nullptr,
2986 const char *desc = nullptr)
2987 : SparseHistBase<SparseHistogram, SparseHistStor>(parent, name, desc)
2988 {
2989 }
2990
2991 /**
2992 * Set the parameters of this histogram. @sa HistStor::Params
2993 * @param size The number of buckets in the histogram
2994 * @return A reference to this histogram.
2995 */
2996 SparseHistogram &
2997 init(size_type size)
2998 {
2999 SparseHistStor::Params *params = new SparseHistStor::Params;
3000 this->setParams(params);
3001 this->doInit();
3002 return this->self();
3003 }
3004 };
3005
3006 class Temp;
3007 /**
3008 * A formula for statistics that is calculated when printed. A formula is
3009 * stored as a tree of Nodes that represent the equation to calculate.
3010 * @sa Stat, ScalarStat, VectorStat, Node, Temp
3011 */
3012 class Formula : public DataWrapVec<Formula, FormulaInfoProxy>
3013 {
3014 protected:
3015 /** The root of the tree which represents the Formula */
3016 NodePtr root;
3017 friend class Temp;
3018
3019 public:
3020 /**
3021 * Create and initialize thie formula, and register it with the database.
3022 */
3023 Formula(Group *parent = nullptr, const char *name = nullptr,
3024 const char *desc = nullptr);
3025
3026 Formula(Group *parent, const char *name, const char *desc,
3027 const Temp &r);
3028
3029 /**
3030 * Set an unitialized Formula to the given root.
3031 * @param r The root of the expression tree.
3032 * @return a reference to this formula.
3033 */
3034 const Formula &operator=(const Temp &r);
3035
3036 template<typename T>
3037 const Formula &operator=(const T &v)
3038 {
3039 *this = Temp(v);
3040 return *this;
3041 }
3042
3043 /**
3044 * Add the given tree to the existing one.
3045 * @param r The root of the expression tree.
3046 * @return a reference to this formula.
3047 */
3048 const Formula &operator+=(Temp r);
3049
3050 /**
3051 * Divide the existing tree by the given one.
3052 * @param r The root of the expression tree.
3053 * @return a reference to this formula.
3054 */
3055 const Formula &operator/=(Temp r);
3056
3057 /**
3058 * Return the result of the Fomula in a vector. If there were no Vector
3059 * components to the Formula, then the vector is size 1. If there were,
3060 * like x/y with x being a vector of size 3, then the result returned will
3061 * be x[0]/y, x[1]/y, x[2]/y, respectively.
3062 * @return The result vector.
3063 */
3064 void result(VResult &vec) const;
3065
3066 /**
3067 * Return the total Formula result. If there is a Vector
3068 * component to this Formula, then this is the result of the
3069 * Formula if the formula is applied after summing all the
3070 * components of the Vector. For example, if Formula is x/y where
3071 * x is size 3, then total() will return (x[1]+x[2]+x[3])/y. If
3072 * there is no Vector component, total() returns the same value as
3073 * the first entry in the VResult val() returns.
3074 * @return The total of the result vector.
3075 */
3076 Result total() const;
3077
3078 /**
3079 * Return the number of elements in the tree.
3080 */
3081 size_type size() const;
3082
3083 void prepare() { }
3084
3085 /**
3086 * Formulas don't need to be reset
3087 */
3088 void reset();
3089
3090 /**
3091 *
3092 */
3093 bool zero() const;
3094
3095 std::string str() const;
3096 };
3097
3098 class FormulaNode : public Node
3099 {
3100 private:
3101 const Formula &formula;
3102 mutable VResult vec;
3103
3104 public:
3105 FormulaNode(const Formula &f) : formula(f) {}
3106
3107 size_type size() const { return formula.size(); }
3108 const VResult &result() const { formula.result(vec); return vec; }
3109 Result total() const { return formula.total(); }
3110
3111 std::string str() const { return formula.str(); }
3112 };
3113
3114 /**
3115 * Helper class to construct formula node trees.
3116 */
3117 class Temp
3118 {
3119 protected:
3120 /**
3121 * Pointer to a Node object.
3122 */
3123 NodePtr node;
3124
3125 public:
3126 /**
3127 * Copy the given pointer to this class.
3128 * @param n A pointer to a Node object to copy.
3129 */
3130 Temp(const NodePtr &n) : node(n) { }
3131
3132 Temp(NodePtr &&n) : node(std::move(n)) { }
3133
3134 /**
3135 * Return the node pointer.
3136 * @return the node pointer.
3137 */
3138 operator NodePtr&() { return node; }
3139
3140 /**
3141 * Makde gcc < 4.6.3 happy and explicitly get the underlying node.
3142 */
3143 NodePtr getNodePtr() const { return node; }
3144
3145 public:
3146 /**
3147 * Create a new ScalarStatNode.
3148 * @param s The ScalarStat to place in a node.
3149 */
3150 Temp(const Scalar &s)
3151 : node(new ScalarStatNode(s.info()))
3152 { }
3153
3154 /**
3155 * Create a new ScalarStatNode.
3156 * @param s The ScalarStat to place in a node.
3157 */
3158 Temp(const Value &s)
3159 : node(new ScalarStatNode(s.info()))
3160 { }
3161
3162 /**
3163 * Create a new ScalarStatNode.
3164 * @param s The ScalarStat to place in a node.
3165 */
3166 Temp(const Average &s)
3167 : node(new ScalarStatNode(s.info()))
3168 { }
3169
3170 /**
3171 * Create a new VectorStatNode.
3172 * @param s The VectorStat to place in a node.
3173 */
3174 Temp(const Vector &s)
3175 : node(new VectorStatNode(s.info()))
3176 { }
3177
3178 Temp(const AverageVector &s)
3179 : node(new VectorStatNode(s.info()))
3180 { }
3181
3182 /**
3183 *
3184 */
3185 Temp(const Formula &f)
3186 : node(new FormulaNode(f))
3187 { }
3188
3189 /**
3190 * Create a new ScalarProxyNode.
3191 * @param p The ScalarProxy to place in a node.
3192 */
3193 template <class Stat>
3194 Temp(const ScalarProxy<Stat> &p)
3195 : node(new ScalarProxyNode<Stat>(p))
3196 { }
3197
3198 /**
3199 * Create a ConstNode
3200 * @param value The value of the const node.
3201 */
3202 Temp(signed char value)
3203 : node(new ConstNode<signed char>(value))
3204 { }
3205
3206 /**
3207 * Create a ConstNode
3208 * @param value The value of the const node.
3209 */
3210 Temp(unsigned char value)
3211 : node(new ConstNode<unsigned char>(value))
3212 { }
3213
3214 /**
3215 * Create a ConstNode
3216 * @param value The value of the const node.
3217 */
3218 Temp(signed short value)
3219 : node(new ConstNode<signed short>(value))
3220 { }
3221
3222 /**
3223 * Create a ConstNode
3224 * @param value The value of the const node.
3225 */
3226 Temp(unsigned short value)
3227 : node(new ConstNode<unsigned short>(value))
3228 { }
3229
3230 /**
3231 * Create a ConstNode
3232 * @param value The value of the const node.
3233 */
3234 Temp(signed int value)
3235 : node(new ConstNode<signed int>(value))
3236 { }
3237
3238 /**
3239 * Create a ConstNode
3240 * @param value The value of the const node.
3241 */
3242 Temp(unsigned int value)
3243 : node(new ConstNode<unsigned int>(value))
3244 { }
3245
3246 /**
3247 * Create a ConstNode
3248 * @param value The value of the const node.
3249 */
3250 Temp(signed long value)
3251 : node(new ConstNode<signed long>(value))
3252 { }
3253
3254 /**
3255 * Create a ConstNode
3256 * @param value The value of the const node.
3257 */
3258 Temp(unsigned long value)
3259 : node(new ConstNode<unsigned long>(value))
3260 { }
3261
3262 /**
3263 * Create a ConstNode
3264 * @param value The value of the const node.
3265 */
3266 Temp(signed long long value)
3267 : node(new ConstNode<signed long long>(value))
3268 { }
3269
3270 /**
3271 * Create a ConstNode
3272 * @param value The value of the const node.
3273 */
3274 Temp(unsigned long long value)
3275 : node(new ConstNode<unsigned long long>(value))
3276 { }
3277
3278 /**
3279 * Create a ConstNode
3280 * @param value The value of the const node.
3281 */
3282 Temp(float value)
3283 : node(new ConstNode<float>(value))
3284 { }
3285
3286 /**
3287 * Create a ConstNode
3288 * @param value The value of the const node.
3289 */
3290 Temp(double value)
3291 : node(new ConstNode<double>(value))
3292 { }
3293 };
3294
3295
3296 /**
3297 * @}
3298 */
3299
3300 inline Temp
3301 operator+(Temp l, Temp r)
3302 {
3303 return Temp(std::make_shared<BinaryNode<std::plus<Result> > >(l, r));
3304 }
3305
3306 inline Temp
3307 operator-(Temp l, Temp r)
3308 {
3309 return Temp(std::make_shared<BinaryNode<std::minus<Result> > >(l, r));
3310 }
3311
3312 inline Temp
3313 operator*(Temp l, Temp r)
3314 {
3315 return Temp(std::make_shared<BinaryNode<std::multiplies<Result> > >(l, r));
3316 }
3317
3318 inline Temp
3319 operator/(Temp l, Temp r)
3320 {
3321 return Temp(std::make_shared<BinaryNode<std::divides<Result> > >(l, r));
3322 }
3323
3324 inline Temp
3325 operator-(Temp l)
3326 {
3327 return Temp(std::make_shared<UnaryNode<std::negate<Result> > >(l));
3328 }
3329
3330 template <typename T>
3331 inline Temp
3332 constant(T val)
3333 {
3334 return Temp(std::make_shared<ConstNode<T> >(val));
3335 }
3336
3337 template <typename T>
3338 inline Temp
3339 constantVector(T val)
3340 {
3341 return Temp(std::make_shared<ConstVectorNode<T> >(val));
3342 }
3343
3344 inline Temp
3345 sum(Temp val)
3346 {
3347 return Temp(std::make_shared<SumNode<std::plus<Result> > >(val));
3348 }
3349
3350 /** Dump all statistics data to the registered outputs */
3351 void dump();
3352 void reset();
3353 void enable();
3354 bool enabled();
3355
3356 /**
3357 * Register reset and dump handlers. These are the functions which
3358 * will actually perform the whole statistics reset/dump actions
3359 * including processing the reset/dump callbacks
3360 */
3361 typedef void (*Handler)();
3362
3363 void registerHandlers(Handler reset_handler, Handler dump_handler);
3364
3365 /**
3366 * Register a callback that should be called whenever statistics are
3367 * reset
3368 */
3369 void registerResetCallback(Callback *cb);
3370
3371 /**
3372 * Register a callback that should be called whenever statistics are
3373 * about to be dumped
3374 */
3375 void registerDumpCallback(Callback *cb);
3376
3377 /**
3378 * Process all the callbacks in the reset callbacks queue
3379 */
3380 void processResetQueue();
3381
3382 /**
3383 * Process all the callbacks in the dump callbacks queue
3384 */
3385 void processDumpQueue();
3386
3387 std::list<Info *> &statsList();
3388
3389 typedef std::map<const void *, Info *> MapType;
3390 MapType &statsMap();
3391
3392 typedef std::map<std::string, Info *> NameMapType;
3393 NameMapType &nameMap();
3394
3395 bool validateStatName(const std::string &name);
3396
3397 } // namespace Stats
3398
3399 void debugDumpStats();
3400
3401 #endif // __BASE_STATISTICS_HH__