summaryrefslogtreecommitdiffstats
path: root/watch-library/watch/watch_adc.h
blob: a4f9edad198e69e777d23c9af8af99027e0c4835 (plain)
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
148
149
150
151
152
153
154
155
156
157
/*
 * MIT License
 *
 * Copyright (c) 2020 Joey Castillo
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */
#ifndef _WATCH_ADC_H_INCLUDED
#define _WATCH_ADC_H_INCLUDED
////< @file watch_adc.h

#include "watch.h"

/** @addtogroup adc Analog Input
  * @brief This section covers functions related to the SAM L22's analog-to-digital converter,
  *        as well as configuring and reading values from the five analog-capable pins on the
  *        9-pin connector.
  */
/// @{
/** @brief Enables the ADC peripheral. You must call this before attempting to read a value
  *        from an analog pin.
  */
void watch_enable_adc();

/** @brief Configures the selected pin for analog input.
  * @param pin One of pins A0-A4.
  */
void watch_enable_analog_input(const uint8_t pin);

/** @brief Reads an analog value from one of the pins.
  * @param pin One of pins A0-A4.
  * @return a 16-bit unsigned integer from 0-65535 representing the sampled value, unless you
  *         have changed the number of samples. @see watch_set_num_analog_samples for details
  *         on how that function changes the values returned from this one.
  **/
uint16_t watch_get_analog_pin_level(const uint8_t pin);

/** @brief Sets the number of samples to accumulate when measuring a pin level. Default is 16.
  * @param samples A power of 2 <= 1024. Specifically: 1, 2, 4, 8, 16, 32, 64, 128, 256, 512
                   or 1024. Any other value will be ignored.
  * @details The SAM L22's ADC has a resolution of 12 bits. By default, the watch configures
  *          the ADC to take 16 samples of the analog input and accumulate them in the result
  *          register; this effectively gives us a 16-bit resolution, at the cost of taking 16
  *          ADC cycles to complete a measurement. If you are measuring a slowly changing signal
  *          like a thermistor output or an ambient light sensor this is probably fine, even
  *          desirable. If you are measuring something a bit more fast-paced, like an analog
  *          accelerometer, you may wish to exchange precision for speed. In this case you may
  *          call this function to configure the ADC to accumulate fewer samples. HOWEVER! Note
  *          that this may change the range of values returned from watch_get_analog_pin_level:
  *            - For watch_set_num_analog_samples(1), the returned value will be 12 bits (0-4095).
  *            - For watch_set_num_analog_samples(2), the returned value will be 13 bits (0-8191).
  *            - For watch_set_num_analog_samples(4), the returned value will be 14 bits (0-16383).
  *            - For watch_set_num_analog_samples(8), the returned value will be 15 bits (0-32767).
  *         For sampling values over 16, the returned value will still be 16 bits (0-65535); the
  *         ADC will automatically divide the measured value by whatever factor is necessary to fit
  *         the result in 16 bits.
  * @see watch_get_analog_pin_level
  **/
void watch_set_analog_num_samples(uint16_t samples);

/** @brief Sets the length of time spent sampling, which allows measurement of higher impedance inputs.
  *        Default is 1.
  * @param cycles The number of ADC cycles to sample, between 1 and 64.
  * @see this article by Thea Flowers: https://blog.thea.codes/getting-the-most-out-of-the-samd21-adc/
  *      which is where I learned all of this.
  * @details To measure an analog value, the SAM L22 must charge a capacitor to the analog voltage
  *          presented at the input. This takes time. Importantly, the higher the input impedance,
  *          the more time this takes. As a basic example: if you are using a thermistor tied to
  *          VCC to measure temperature, the capacitor has to charge through the thermistor. The
  *          higher the resistor value, the higher the input impedance, and the more time we need
  *          to allow for the measurement. By default, the ADC is configured to run on a 500 kHz
  *          clock with a sample time of one cycle. This is appropriate for an input impedance up
  *          to about 28kΩ. Setting the sampling time to 4 cycles allows for an input impedance up
  *          to 123kΩ. Setting the sampling time to the maximum of 64 cycles theoretically allows
  *          for input impedance up to 2 MΩ. (I based these numbers on the calculator in the linked
  *          blog post; it also has a ton of great info on the SAM D21 ADC, which is similar to the
  *          SAM L22's).
  **/
void watch_set_analog_sampling_length(uint8_t cycles);

typedef enum {
    ADC_REFERENCE_INTREF = ADC_REFCTRL_REFSEL_INTREF_Val,
    ADC_REFERENCE_VCC_DIV1POINT6 = ADC_REFCTRL_REFSEL_INTVCC0_Val,
    ADC_REFERENCE_VCC_DIV2 = ADC_REFCTRL_REFSEL_INTVCC1_Val,
    ADC_REFERENCE_VCC = ADC_REFCTRL_REFSEL_INTVCC2_Val,
} watch_adc_reference_voltage;


/** @brief Selects the reference voltage to use for analog readings. Default is ADC_REFERENCE_VCC.
  * @param reference One of ADC_REFERENCE_VCC, ADC_REFERENCE_VCC_DIV1POINT6, ADC_REFERENCE_VCC_DIV2
  *                  or ADC_REFERENCE_INTREF.
  * @details In order to turn an analog voltage into a 16-bit integer, the ADC needs to compare the
  *          measured voltage to a reference point. For example, if you were powering the watch with
  *          VCC == 3.0V and you had two 10K resistors connected in series from 3V to GND, you could
  *          expect to get 3 volts when you measure the top of the voltage divider, 0 volts at the
  *          bottom, and 1.5 volts in the middle. If you read these values uising a reference voltage
  *          of ADC_REFERENCE_VCC, the top value would be about 65535, the bottom about 0, and the
  *          middle about 32768. However! If we used ADC_REFERENCE_VCC_DIV2 as our reference, we would
  *          expect to get 65535 both at the top and the middle, because the largest value the ADC can
  *          measure in this configutation is 1.5V (VCC / 2).
  *
  *          By changing the reference voltage from ADC_REFERENCE_VCC to ADC_REFERENCE_VCC_DIV1POINT6
  *          or ADC_REFERENCE_VCC_DIV2, you can get more resolution when measuring small voltages (i.e.
  *          a phototransistor circuit in low light).
  *
  *          There is also a special reference voltage called ADC_REFERENCE_INTREF. The SAM L22's
  *          Supply Controller provides a selectable voltage reference (by default, 1.024 V) that you
  *          can select as a reference voltage for ADC conversions. Unlike the three references we
  *          talked about in the last paragraph, this reference voltage does not depend on VCC, which
  *          makes it very useful for measuring the battery voltage (since you can't really compare
  *          VCC to itself). You can change the INTREF voltage to 2.048 or 4.096 V by poking at the
  *          supply controller's VREF register, but the watch library does not support this use case.
  **/
void watch_set_analog_reference_voltage(watch_adc_reference_voltage reference);

/** @brief Returns the voltage of the VCC supply in millivolts (i.e. 3000 mV == 3.0 V). If running on
  *        a coin cell, this will be the battery voltage.
  * @details Unlike other ADC functions, this function does not return a raw value from the ADC, but
  *          rather scales it to an actual number of millivolts. This is because the ADC doesn't let
  *          us measure VCC per se; it instead lets us measure VCC / 4, and we choose to measure it
  *          against the internal reference voltage of 1.024 V. In short, the ADC gives us a number
  *          that's complicated to deal with, so we just turn it into a useful number for you :)
  * @note This function depends on INTREF being 1.024V. If you have changed it by poking at the supply
  *       controller's VREF.SEL bits, this function will return inaccurate values.
  */
uint16_t watch_get_vcc_voltage();

/** @brief Disables the analog circuitry on the selected pin.
  * @param pin One of pins A0-A4.
  */
void watch_disable_analog_input(const uint8_t pin);

/** @brief Disables the ADC peripheral.
  * @note You will need to call watch_enable_adc to re-enable the ADC peripheral. When you do, it will
  *       have the default settings of 16 samples and 1 measurement cycle; if you customized these
  *       parameters, you will need to set them up again.
  **/
void watch_disable_adc();

/// @}
#endif