-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathspo2.go
49 lines (40 loc) · 878 Bytes
/
spo2.go
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
package max3010x
import (
"errors"
"fmt"
)
// SpO2 returns the SpO2 value in 100%.
func (d *Device) SpO2() (float64, error) {
r, err := d.rValue()
if errors.Is(err, errLowValue) {
d.spo2.reset()
return 0, fmt.Errorf("max3010x: could not get SpO2: %w", ErrNotDetected)
} else if err != nil {
d.spo2.reset()
return 0, fmt.Errorf("max3010x: could not get R value: %w", err)
}
spo2 := 104 - 17*r
if spo2 <= 0 {
return 0, nil
}
// if first measurement, pre-fill values.
if d.spo2.mean == 0 {
d.spo2.mean = spo2
}
d.spo2.add(spo2)
return d.spo2.mean, nil
}
func (d *Device) rValue() (float64, error) {
err := d.leds()
if err != nil {
return 0, err
}
if d.redLED.last() < threshold || d.irLED.last() < threshold {
return 0, errLowValue
}
irACDC := d.irLED.acdc()
if irACDC == 0 {
return 0, nil
}
return d.redLED.acdc() / irACDC, nil
}