-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathProgress.h
147 lines (119 loc) · 2.92 KB
/
Progress.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
//
// Copyright (C) Wojciech Jarosz <wjarosz@gmail.com>. All rights reserved.
// Use of this source code is governed by a BSD-style license that can
// be found in the LICENSE.txt file.
//
#pragma once
#include <atomic>
#include <cstdint>
#include <cmath>
#include <memory>
/*!
* A fixed-point fractional number stored using an std::atomic
*/
template <typename Fixed, typename BigFixed, int FractionBits>
class AtomicFixed
{
public:
static const Fixed ScalingFactor = (1 << FractionBits);
static Fixed float2fixed(float b)
{
return std::round(b * ScalingFactor);
}
static float fixed2float(Fixed f)
{
return float(f) / ScalingFactor;
}
std::atomic<Fixed> f;
AtomicFixed() = default;
explicit AtomicFixed(float d) :
f(float2fixed(d))
{
// empty
}
explicit operator float() const
{
return fixed2float(f);
}
Fixed operator=(float b)
{
return (f = float2fixed(b));
}
Fixed operator+=(float b)
{
return (f += float2fixed(b));
}
Fixed operator-=(float b)
{
return (f -= float2fixed(b));
}
/// This operator is *NOT* atomic
Fixed operator*=(float b)
{
return (f = Fixed(BigFixed(f) * BigFixed(float2fixed(b))) / ScalingFactor);
}
/// This operator is *NOT* atomic
Fixed operator/=(float b)
{
return (f = Fixed((BigFixed(f) * ScalingFactor) / float2fixed(b)));
}
bool operator<(float b) const
{
return f < float2fixed(b);
}
bool operator<=(float b) const
{
return f <= float2fixed(b);
}
bool operator>(float b) const
{
return f > float2fixed(b);
}
bool operator>=(float b) const
{
return f >= float2fixed(b);
}
bool operator==(float b) const
{
return f == float2fixed(b);
}
bool operator!=(float b) const
{
return f != float2fixed(b);
}
};
using AtomicFixed16 = AtomicFixed<std::int16_t, std::int32_t, 8>;
using AtomicFixed32 = AtomicFixed<std::int32_t, std::int64_t, 16>;
/*!
* Helper object to manage the progress display.
* {
* AtomicProgress p1(true);
* p1.setNumSteps(10);
* for (int i = 0; i < 10; ++i, ++p1)
* {
* // do something
* }
* } // end progress p1
*
*/
class AtomicProgress
{
public:
using AtomicPercent32 = AtomicFixed<std::int32_t, std::int64_t, 30>;
explicit AtomicProgress(bool createState = false, float totalPercentage = 1.f);
AtomicProgress(const AtomicProgress & parent, float percentageOfParent = 1.f);
// access to the atomic internal storage
void resetProgress(float p = 0.f);
float progress() const;
void setDone() {resetProgress(1.f);}
void setBusy() {resetProgress(-1.f);}
// access to the discrete stepping
void setAvailablePercent(float percent);
void setNumSteps(int numSteps);
AtomicProgress& operator+=(int steps);
AtomicProgress& operator++() {return ((*this)+=1);}
private:
int m_numSteps;
float m_percentageOfParent, m_stepPercent;
std::shared_ptr<AtomicPercent32> m_atomicState; ///< Atomic internal state of progress
};