diff options
Diffstat (limited to 'OpenSim/Region/Physics/BulletSPlugin/BSMotors.cs')
-rwxr-xr-x | OpenSim/Region/Physics/BulletSPlugin/BSMotors.cs | 590 |
1 files changed, 486 insertions, 104 deletions
diff --git a/OpenSim/Region/Physics/BulletSPlugin/BSMotors.cs b/OpenSim/Region/Physics/BulletSPlugin/BSMotors.cs index bc6e4c4..9501e2d 100755 --- a/OpenSim/Region/Physics/BulletSPlugin/BSMotors.cs +++ b/OpenSim/Region/Physics/BulletSPlugin/BSMotors.cs | |||
@@ -1,104 +1,486 @@ | |||
1 | using System; | 1 | /* |
2 | using System.Collections.Generic; | 2 | * Copyright (c) Contributors, http://opensimulator.org/ |
3 | using System.Text; | 3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. |
4 | using OpenMetaverse; | 4 | * |
5 | 5 | * Redistribution and use in source and binary forms, with or without | |
6 | namespace OpenSim.Region.Physics.BulletSPlugin | 6 | * modification, are permitted provided that the following conditions are met: |
7 | { | 7 | * * Redistributions of source code must retain the above copyright |
8 | public abstract class BSMotor | 8 | * notice, this list of conditions and the following disclaimer. |
9 | { | 9 | * * Redistributions in binary form must reproduce the above copyright |
10 | public virtual void Reset() { } | 10 | * notice, this list of conditions and the following disclaimer in the |
11 | public virtual void Zero() { } | 11 | * documentation and/or other materials provided with the distribution. |
12 | } | 12 | * * Neither the name of the OpenSimulator Project nor the |
13 | // Can all the incremental stepping be replaced with motor classes? | 13 | * names of its contributors may be used to endorse or promote products |
14 | public class BSVMotor : BSMotor | 14 | * derived from this software without specific prior written permission. |
15 | { | 15 | * |
16 | public Vector3 FrameOfReference { get; set; } | 16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY |
17 | public Vector3 Offset { get; set; } | 17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
18 | 18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |
19 | public float TimeScale { get; set; } | 19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY |
20 | public float TargetValueDecayTimeScale { get; set; } | 20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
21 | public Vector3 CurrentValueReductionTimescale { get; set; } | 21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
22 | public float Efficiency { get; set; } | 22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
23 | 23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
24 | public Vector3 TargetValue { get; private set; } | 24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
25 | public Vector3 CurrentValue { get; private set; } | 25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
26 | 26 | * | |
27 | 27 | */ | |
28 | 28 | using System; | |
29 | BSVMotor(float timeScale, float decayTimeScale, Vector3 frictionTimeScale, float efficiency) | 29 | using System.Collections.Generic; |
30 | { | 30 | using System.Text; |
31 | TimeScale = timeScale; | 31 | using OpenMetaverse; |
32 | TargetValueDecayTimeScale = decayTimeScale; | 32 | using OpenSim.Framework; |
33 | CurrentValueReductionTimescale = frictionTimeScale; | 33 | |
34 | Efficiency = efficiency; | 34 | namespace OpenSim.Region.Physics.BulletSPlugin |
35 | } | 35 | { |
36 | public void SetCurrent(Vector3 current) | 36 | public abstract class BSMotor |
37 | { | 37 | { |
38 | CurrentValue = current; | 38 | // Timescales and other things can be turned off by setting them to 'infinite'. |
39 | } | 39 | public const float Infinite = 12345.6f; |
40 | public void SetTarget(Vector3 target) | 40 | public readonly static Vector3 InfiniteVector = new Vector3(BSMotor.Infinite, BSMotor.Infinite, BSMotor.Infinite); |
41 | { | 41 | |
42 | TargetValue = target; | 42 | public BSMotor(string useName) |
43 | } | 43 | { |
44 | public Vector3 Step(float timeStep) | 44 | UseName = useName; |
45 | { | 45 | PhysicsScene = null; |
46 | if (CurrentValue.LengthSquared() > 0.001f) | 46 | Enabled = true; |
47 | { | 47 | } |
48 | // Vector3 origDir = Target; // DEBUG | 48 | public virtual bool Enabled { get; set; } |
49 | // Vector3 origVel = CurrentValue; // DEBUG | 49 | public virtual void Reset() { } |
50 | 50 | public virtual void Zero() { } | |
51 | // Add (desiredVelocity - currentAppliedVelocity) / howLongItShouldTakeToComplete | 51 | public virtual void GenerateTestOutput(float timeStep) { } |
52 | Vector3 addAmount = (TargetValue - CurrentValue)/(TargetValue) * timeStep; | 52 | |
53 | CurrentValue += addAmount; | 53 | // A name passed at motor creation for easily identifyable debugging messages. |
54 | 54 | public string UseName { get; private set; } | |
55 | float decayFactor = (1.0f / TargetValueDecayTimeScale) * timeStep; | 55 | |
56 | TargetValue *= (1f - decayFactor); | 56 | // Used only for outputting debug information. Might not be set so check for null. |
57 | 57 | public BSScene PhysicsScene { get; set; } | |
58 | Vector3 frictionFactor = (Vector3.One / CurrentValueReductionTimescale) * timeStep; | 58 | protected void MDetailLog(string msg, params Object[] parms) |
59 | CurrentValue *= (Vector3.One - frictionFactor); | 59 | { |
60 | } | 60 | if (PhysicsScene != null) |
61 | else | 61 | { |
62 | { | 62 | PhysicsScene.DetailLog(msg, parms); |
63 | // if what remains of direction is very small, zero it. | 63 | } |
64 | TargetValue = Vector3.Zero; | 64 | } |
65 | CurrentValue = Vector3.Zero; | 65 | } |
66 | 66 | ||
67 | // VDetailLog("{0},MoveLinear,zeroed", Prim.LocalID); | 67 | // Motor which moves CurrentValue to TargetValue over TimeScale seconds. |
68 | } | 68 | // The TargetValue decays in TargetValueDecayTimeScale and |
69 | return CurrentValue; | 69 | // the CurrentValue will be held back by FrictionTimeScale. |
70 | } | 70 | // This motor will "zero itself" over time in that the targetValue will |
71 | } | 71 | // decay to zero and the currentValue will follow it to that zero. |
72 | 72 | // The overall effect is for the returned correction value to go from large | |
73 | public class BSFMotor : BSMotor | 73 | // values (the total difference between current and target minus friction) |
74 | { | 74 | // to small and eventually zero values. |
75 | public float TimeScale { get; set; } | 75 | // TimeScale and TargetDelayTimeScale may be 'infinite' which means no decay. |
76 | public float DecayTimeScale { get; set; } | 76 | |
77 | public float Friction { get; set; } | 77 | // For instance, if something is moving at speed X and the desired speed is Y, |
78 | public float Efficiency { get; set; } | 78 | // CurrentValue is X and TargetValue is Y. As the motor is stepped, new |
79 | 79 | // values of CurrentValue are returned that approach the TargetValue. | |
80 | public float Target { get; private set; } | 80 | // The feature of decaying TargetValue is so vehicles will eventually |
81 | public float CurrentValue { get; private set; } | 81 | // come to a stop rather than run forever. This can be disabled by |
82 | 82 | // setting TargetValueDecayTimescale to 'infinite'. | |
83 | BSFMotor(float timeScale, float decayTimescale, float friction, float efficiency) | 83 | // The change from CurrentValue to TargetValue is linear over TimeScale seconds. |
84 | { | 84 | public class BSVMotor : BSMotor |
85 | } | 85 | { |
86 | public void SetCurrent(float target) | 86 | // public Vector3 FrameOfReference { get; set; } |
87 | { | 87 | // public Vector3 Offset { get; set; } |
88 | } | 88 | |
89 | public void SetTarget(float target) | 89 | public virtual float TimeScale { get; set; } |
90 | { | 90 | public virtual float TargetValueDecayTimeScale { get; set; } |
91 | } | 91 | public virtual Vector3 FrictionTimescale { get; set; } |
92 | public float Step(float timeStep) | 92 | public virtual float Efficiency { get; set; } |
93 | { | 93 | |
94 | return 0f; | 94 | public virtual float ErrorZeroThreshold { get; set; } |
95 | } | 95 | |
96 | } | 96 | public virtual Vector3 TargetValue { get; protected set; } |
97 | public class BSPIDMotor : BSMotor | 97 | public virtual Vector3 CurrentValue { get; protected set; } |
98 | { | 98 | public virtual Vector3 LastError { get; protected set; } |
99 | // TODO: write and use this one | 99 | |
100 | BSPIDMotor() | 100 | public virtual bool ErrorIsZero() |
101 | { | 101 | { |
102 | } | 102 | return ErrorIsZero(LastError); |
103 | } | 103 | } |
104 | } | 104 | public virtual bool ErrorIsZero(Vector3 err) |
105 | { | ||
106 | return (err == Vector3.Zero || err.ApproxEquals(Vector3.Zero, ErrorZeroThreshold)); | ||
107 | } | ||
108 | |||
109 | public BSVMotor(string useName) | ||
110 | : base(useName) | ||
111 | { | ||
112 | TimeScale = TargetValueDecayTimeScale = BSMotor.Infinite; | ||
113 | Efficiency = 1f; | ||
114 | FrictionTimescale = BSMotor.InfiniteVector; | ||
115 | CurrentValue = TargetValue = Vector3.Zero; | ||
116 | ErrorZeroThreshold = 0.001f; | ||
117 | } | ||
118 | public BSVMotor(string useName, float timeScale, float decayTimeScale, Vector3 frictionTimeScale, float efficiency) | ||
119 | : this(useName) | ||
120 | { | ||
121 | TimeScale = timeScale; | ||
122 | TargetValueDecayTimeScale = decayTimeScale; | ||
123 | FrictionTimescale = frictionTimeScale; | ||
124 | Efficiency = efficiency; | ||
125 | CurrentValue = TargetValue = Vector3.Zero; | ||
126 | } | ||
127 | public void SetCurrent(Vector3 current) | ||
128 | { | ||
129 | CurrentValue = current; | ||
130 | } | ||
131 | public void SetTarget(Vector3 target) | ||
132 | { | ||
133 | TargetValue = target; | ||
134 | } | ||
135 | public override void Zero() | ||
136 | { | ||
137 | base.Zero(); | ||
138 | CurrentValue = TargetValue = Vector3.Zero; | ||
139 | } | ||
140 | |||
141 | // Compute the next step and return the new current value. | ||
142 | // Returns the correction needed to move 'current' to 'target'. | ||
143 | public virtual Vector3 Step(float timeStep) | ||
144 | { | ||
145 | if (!Enabled) return TargetValue; | ||
146 | |||
147 | Vector3 origTarget = TargetValue; // DEBUG | ||
148 | Vector3 origCurrVal = CurrentValue; // DEBUG | ||
149 | |||
150 | Vector3 correction = Vector3.Zero; | ||
151 | Vector3 error = TargetValue - CurrentValue; | ||
152 | LastError = error; | ||
153 | if (!ErrorIsZero(error)) | ||
154 | { | ||
155 | correction = StepError(timeStep, error); | ||
156 | |||
157 | CurrentValue += correction; | ||
158 | |||
159 | // The desired value reduces to zero which also reduces the difference with current. | ||
160 | // If the decay time is infinite, don't decay at all. | ||
161 | float decayFactor = 0f; | ||
162 | if (TargetValueDecayTimeScale != BSMotor.Infinite) | ||
163 | { | ||
164 | decayFactor = (1.0f / TargetValueDecayTimeScale) * timeStep; | ||
165 | TargetValue *= (1f - decayFactor); | ||
166 | } | ||
167 | |||
168 | // The amount we can correct the error is reduced by the friction | ||
169 | Vector3 frictionFactor = Vector3.Zero; | ||
170 | if (FrictionTimescale != BSMotor.InfiniteVector) | ||
171 | { | ||
172 | // frictionFactor = (Vector3.One / FrictionTimescale) * timeStep; | ||
173 | // Individual friction components can be 'infinite' so compute each separately. | ||
174 | frictionFactor.X = (FrictionTimescale.X == BSMotor.Infinite) ? 0f : (1f / FrictionTimescale.X); | ||
175 | frictionFactor.Y = (FrictionTimescale.Y == BSMotor.Infinite) ? 0f : (1f / FrictionTimescale.Y); | ||
176 | frictionFactor.Z = (FrictionTimescale.Z == BSMotor.Infinite) ? 0f : (1f / FrictionTimescale.Z); | ||
177 | frictionFactor *= timeStep; | ||
178 | CurrentValue *= (Vector3.One - frictionFactor); | ||
179 | } | ||
180 | |||
181 | MDetailLog("{0}, BSVMotor.Step,nonZero,{1},origCurr={2},origTarget={3},timeStep={4},err={5},corr={6}", | ||
182 | BSScene.DetailLogZero, UseName, origCurrVal, origTarget, | ||
183 | timeStep, error, correction); | ||
184 | MDetailLog("{0}, BSVMotor.Step,nonZero,{1},tgtDecayTS={2},decayFact={3},frictTS={4},frictFact={5},tgt={6},curr={7}", | ||
185 | BSScene.DetailLogZero, UseName, | ||
186 | TargetValueDecayTimeScale, decayFactor, FrictionTimescale, frictionFactor, | ||
187 | TargetValue, CurrentValue); | ||
188 | } | ||
189 | else | ||
190 | { | ||
191 | // Difference between what we have and target is small. Motor is done. | ||
192 | if (TargetValue.ApproxEquals(Vector3.Zero, ErrorZeroThreshold)) | ||
193 | { | ||
194 | // The target can step down to nearly zero but not get there. If close to zero | ||
195 | // it is really zero. | ||
196 | TargetValue = Vector3.Zero; | ||
197 | } | ||
198 | CurrentValue = TargetValue; | ||
199 | MDetailLog("{0}, BSVMotor.Step,zero,{1},origTgt={2},origCurr={3},currTgt={4},currCurr={5}", | ||
200 | BSScene.DetailLogZero, UseName, origCurrVal, origTarget, TargetValue, CurrentValue); | ||
201 | } | ||
202 | |||
203 | return correction; | ||
204 | } | ||
205 | // version of step that sets the current value before doing the step | ||
206 | public virtual Vector3 Step(float timeStep, Vector3 current) | ||
207 | { | ||
208 | CurrentValue = current; | ||
209 | return Step(timeStep); | ||
210 | } | ||
211 | public virtual Vector3 StepError(float timeStep, Vector3 error) | ||
212 | { | ||
213 | if (!Enabled) return Vector3.Zero; | ||
214 | |||
215 | Vector3 returnCorrection = Vector3.Zero; | ||
216 | if (!ErrorIsZero(error)) | ||
217 | { | ||
218 | // correction = error / secondsItShouldTakeToCorrect | ||
219 | Vector3 correctionAmount; | ||
220 | if (TimeScale == 0f || TimeScale == BSMotor.Infinite) | ||
221 | correctionAmount = error * timeStep; | ||
222 | else | ||
223 | correctionAmount = error / TimeScale * timeStep; | ||
224 | |||
225 | returnCorrection = correctionAmount; | ||
226 | MDetailLog("{0}, BSVMotor.Step,nonZero,{1},timeStep={2},timeScale={3},err={4},corr={5}", | ||
227 | BSScene.DetailLogZero, UseName, timeStep, TimeScale, error, correctionAmount); | ||
228 | } | ||
229 | return returnCorrection; | ||
230 | } | ||
231 | |||
232 | // The user sets all the parameters and calls this which outputs values until error is zero. | ||
233 | public override void GenerateTestOutput(float timeStep) | ||
234 | { | ||
235 | // maximum number of outputs to generate. | ||
236 | int maxOutput = 50; | ||
237 | MDetailLog("{0},BSVMotor.Test,{1},===================================== BEGIN Test Output", BSScene.DetailLogZero, UseName); | ||
238 | MDetailLog("{0},BSVMotor.Test,{1},timeScale={2},targDlyTS={3},frictTS={4},eff={5},curr={6},tgt={7}", | ||
239 | BSScene.DetailLogZero, UseName, | ||
240 | TimeScale, TargetValueDecayTimeScale, FrictionTimescale, Efficiency, | ||
241 | CurrentValue, TargetValue); | ||
242 | |||
243 | LastError = BSMotor.InfiniteVector; | ||
244 | while (maxOutput-- > 0 && !LastError.ApproxEquals(Vector3.Zero, ErrorZeroThreshold)) | ||
245 | { | ||
246 | Vector3 lastStep = Step(timeStep); | ||
247 | MDetailLog("{0},BSVMotor.Test,{1},cur={2},tgt={3},lastError={4},lastStep={5}", | ||
248 | BSScene.DetailLogZero, UseName, CurrentValue, TargetValue, LastError, lastStep); | ||
249 | } | ||
250 | MDetailLog("{0},BSVMotor.Test,{1},===================================== END Test Output", BSScene.DetailLogZero, UseName); | ||
251 | |||
252 | |||
253 | } | ||
254 | |||
255 | public override string ToString() | ||
256 | { | ||
257 | return String.Format("<{0},curr={1},targ={2},lastErr={3},decayTS={4},frictTS={5}>", | ||
258 | UseName, CurrentValue, TargetValue, LastError, TargetValueDecayTimeScale, FrictionTimescale); | ||
259 | } | ||
260 | } | ||
261 | |||
262 | // ============================================================================ | ||
263 | // ============================================================================ | ||
264 | public class BSFMotor : BSMotor | ||
265 | { | ||
266 | public virtual float TimeScale { get; set; } | ||
267 | public virtual float TargetValueDecayTimeScale { get; set; } | ||
268 | public virtual float FrictionTimescale { get; set; } | ||
269 | public virtual float Efficiency { get; set; } | ||
270 | |||
271 | public virtual float ErrorZeroThreshold { get; set; } | ||
272 | |||
273 | public virtual float TargetValue { get; protected set; } | ||
274 | public virtual float CurrentValue { get; protected set; } | ||
275 | public virtual float LastError { get; protected set; } | ||
276 | |||
277 | public virtual bool ErrorIsZero() | ||
278 | { | ||
279 | return ErrorIsZero(LastError); | ||
280 | } | ||
281 | public virtual bool ErrorIsZero(float err) | ||
282 | { | ||
283 | return (err >= -ErrorZeroThreshold && err <= ErrorZeroThreshold); | ||
284 | } | ||
285 | |||
286 | public BSFMotor(string useName, float timeScale, float decayTimescale, float friction, float efficiency) | ||
287 | : base(useName) | ||
288 | { | ||
289 | TimeScale = TargetValueDecayTimeScale = BSMotor.Infinite; | ||
290 | Efficiency = 1f; | ||
291 | FrictionTimescale = BSMotor.Infinite; | ||
292 | CurrentValue = TargetValue = 0f; | ||
293 | ErrorZeroThreshold = 0.01f; | ||
294 | } | ||
295 | public void SetCurrent(float current) | ||
296 | { | ||
297 | CurrentValue = current; | ||
298 | } | ||
299 | public void SetTarget(float target) | ||
300 | { | ||
301 | TargetValue = target; | ||
302 | } | ||
303 | public override void Zero() | ||
304 | { | ||
305 | base.Zero(); | ||
306 | CurrentValue = TargetValue = 0f; | ||
307 | } | ||
308 | |||
309 | public virtual float Step(float timeStep) | ||
310 | { | ||
311 | if (!Enabled) return TargetValue; | ||
312 | |||
313 | float origTarget = TargetValue; // DEBUG | ||
314 | float origCurrVal = CurrentValue; // DEBUG | ||
315 | |||
316 | float correction = 0f; | ||
317 | float error = TargetValue - CurrentValue; | ||
318 | LastError = error; | ||
319 | if (!ErrorIsZero(error)) | ||
320 | { | ||
321 | correction = StepError(timeStep, error); | ||
322 | |||
323 | CurrentValue += correction; | ||
324 | |||
325 | // The desired value reduces to zero which also reduces the difference with current. | ||
326 | // If the decay time is infinite, don't decay at all. | ||
327 | float decayFactor = 0f; | ||
328 | if (TargetValueDecayTimeScale != BSMotor.Infinite) | ||
329 | { | ||
330 | decayFactor = (1.0f / TargetValueDecayTimeScale) * timeStep; | ||
331 | TargetValue *= (1f - decayFactor); | ||
332 | } | ||
333 | |||
334 | // The amount we can correct the error is reduced by the friction | ||
335 | float frictionFactor = 0f; | ||
336 | if (FrictionTimescale != BSMotor.Infinite) | ||
337 | { | ||
338 | // frictionFactor = (Vector3.One / FrictionTimescale) * timeStep; | ||
339 | // Individual friction components can be 'infinite' so compute each separately. | ||
340 | frictionFactor = 1f / FrictionTimescale; | ||
341 | frictionFactor *= timeStep; | ||
342 | CurrentValue *= (1f - frictionFactor); | ||
343 | } | ||
344 | |||
345 | MDetailLog("{0}, BSFMotor.Step,nonZero,{1},origCurr={2},origTarget={3},timeStep={4},err={5},corr={6}", | ||
346 | BSScene.DetailLogZero, UseName, origCurrVal, origTarget, | ||
347 | timeStep, error, correction); | ||
348 | MDetailLog("{0}, BSFMotor.Step,nonZero,{1},tgtDecayTS={2},decayFact={3},frictTS={4},frictFact={5},tgt={6},curr={7}", | ||
349 | BSScene.DetailLogZero, UseName, | ||
350 | TargetValueDecayTimeScale, decayFactor, FrictionTimescale, frictionFactor, | ||
351 | TargetValue, CurrentValue); | ||
352 | } | ||
353 | else | ||
354 | { | ||
355 | // Difference between what we have and target is small. Motor is done. | ||
356 | if (Util.InRange<float>(TargetValue, -ErrorZeroThreshold, ErrorZeroThreshold)) | ||
357 | { | ||
358 | // The target can step down to nearly zero but not get there. If close to zero | ||
359 | // it is really zero. | ||
360 | TargetValue = 0f; | ||
361 | } | ||
362 | CurrentValue = TargetValue; | ||
363 | MDetailLog("{0}, BSFMotor.Step,zero,{1},origTgt={2},origCurr={3},ret={4}", | ||
364 | BSScene.DetailLogZero, UseName, origCurrVal, origTarget, CurrentValue); | ||
365 | } | ||
366 | |||
367 | return CurrentValue; | ||
368 | } | ||
369 | |||
370 | public virtual float StepError(float timeStep, float error) | ||
371 | { | ||
372 | if (!Enabled) return 0f; | ||
373 | |||
374 | float returnCorrection = 0f; | ||
375 | if (!ErrorIsZero(error)) | ||
376 | { | ||
377 | // correction = error / secondsItShouldTakeToCorrect | ||
378 | float correctionAmount; | ||
379 | if (TimeScale == 0f || TimeScale == BSMotor.Infinite) | ||
380 | correctionAmount = error * timeStep; | ||
381 | else | ||
382 | correctionAmount = error / TimeScale * timeStep; | ||
383 | |||
384 | returnCorrection = correctionAmount; | ||
385 | MDetailLog("{0}, BSFMotor.Step,nonZero,{1},timeStep={2},timeScale={3},err={4},corr={5}", | ||
386 | BSScene.DetailLogZero, UseName, timeStep, TimeScale, error, correctionAmount); | ||
387 | } | ||
388 | return returnCorrection; | ||
389 | } | ||
390 | |||
391 | public override string ToString() | ||
392 | { | ||
393 | return String.Format("<{0},curr={1},targ={2},lastErr={3},decayTS={4},frictTS={5}>", | ||
394 | UseName, CurrentValue, TargetValue, LastError, TargetValueDecayTimeScale, FrictionTimescale); | ||
395 | } | ||
396 | |||
397 | } | ||
398 | |||
399 | // ============================================================================ | ||
400 | // ============================================================================ | ||
401 | // Proportional, Integral, Derivitive Motor | ||
402 | // Good description at http://www.answers.com/topic/pid-controller . Includes processes for choosing p, i and d factors. | ||
403 | public class BSPIDVMotor : BSVMotor | ||
404 | { | ||
405 | // Larger makes more overshoot, smaller means converge quicker. Range of 0.1 to 10. | ||
406 | public Vector3 proportionFactor { get; set; } | ||
407 | public Vector3 integralFactor { get; set; } | ||
408 | public Vector3 derivFactor { get; set; } | ||
409 | |||
410 | // The factors are vectors for the three dimensions. This is the proportional of each | ||
411 | // that is applied. This could be multiplied through the actual factors but it | ||
412 | // is sometimes easier to manipulate the factors and their mix separately. | ||
413 | // to | ||
414 | public Vector3 FactorMix; | ||
415 | |||
416 | // Arbritrary factor range. | ||
417 | // EfficiencyHigh means move quickly to the correct number. EfficiencyLow means might over correct. | ||
418 | public float EfficiencyHigh = 0.4f; | ||
419 | public float EfficiencyLow = 4.0f; | ||
420 | |||
421 | // Running integration of the error | ||
422 | Vector3 RunningIntegration { get; set; } | ||
423 | |||
424 | public BSPIDVMotor(string useName) | ||
425 | : base(useName) | ||
426 | { | ||
427 | proportionFactor = new Vector3(1.00f, 1.00f, 1.00f); | ||
428 | integralFactor = new Vector3(1.00f, 1.00f, 1.00f); | ||
429 | derivFactor = new Vector3(1.00f, 1.00f, 1.00f); | ||
430 | FactorMix = new Vector3(0.5f, 0.25f, 0.25f); | ||
431 | RunningIntegration = Vector3.Zero; | ||
432 | LastError = Vector3.Zero; | ||
433 | } | ||
434 | |||
435 | public override void Zero() | ||
436 | { | ||
437 | base.Zero(); | ||
438 | } | ||
439 | |||
440 | public override float Efficiency | ||
441 | { | ||
442 | get { return base.Efficiency; } | ||
443 | set | ||
444 | { | ||
445 | base.Efficiency = Util.Clamp(value, 0f, 1f); | ||
446 | |||
447 | // Compute factors based on efficiency. | ||
448 | // If efficiency is high (1f), use a factor value that moves the error value to zero with little overshoot. | ||
449 | // If efficiency is low (0f), use a factor value that overcorrects. | ||
450 | // TODO: might want to vary contribution of different factor depending on efficiency. | ||
451 | float factor = ((1f - this.Efficiency) * EfficiencyHigh + EfficiencyLow) / 3f; | ||
452 | // float factor = (1f - this.Efficiency) * EfficiencyHigh + EfficiencyLow; | ||
453 | |||
454 | proportionFactor = new Vector3(factor, factor, factor); | ||
455 | integralFactor = new Vector3(factor, factor, factor); | ||
456 | derivFactor = new Vector3(factor, factor, factor); | ||
457 | |||
458 | MDetailLog("{0},BSPIDVMotor.setEfficiency,eff={1},factor={2}", BSScene.DetailLogZero, Efficiency, factor); | ||
459 | } | ||
460 | } | ||
461 | |||
462 | // Advance the PID computation on this error. | ||
463 | public override Vector3 StepError(float timeStep, Vector3 error) | ||
464 | { | ||
465 | if (!Enabled) return Vector3.Zero; | ||
466 | |||
467 | // Add up the error so we can integrate over the accumulated errors | ||
468 | RunningIntegration += error * timeStep; | ||
469 | |||
470 | // A simple derivitive is the rate of change from the last error. | ||
471 | Vector3 derivitive = (error - LastError) * timeStep; | ||
472 | LastError = error; | ||
473 | |||
474 | // Correction = (proportionOfPresentError + accumulationOfPastError + rateOfChangeOfError) | ||
475 | Vector3 ret = error * timeStep * proportionFactor * FactorMix.X | ||
476 | + RunningIntegration * integralFactor * FactorMix.Y | ||
477 | + derivitive * derivFactor * FactorMix.Z | ||
478 | ; | ||
479 | |||
480 | MDetailLog("{0},BSPIDVMotor.step,ts={1},err={2},runnInt={3},deriv={4},ret={5}", | ||
481 | BSScene.DetailLogZero, timeStep, error, RunningIntegration, derivitive, ret); | ||
482 | |||
483 | return ret; | ||
484 | } | ||
485 | } | ||
486 | } | ||