2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/ACS712/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
2021-10-16 11:40:09 +02:00
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/ACS712/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/ACS712/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/ACS712/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/ACS712/actions/workflows/jsoncheck.yml)
|
2021-01-29 12:31:58 +01:00
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/ACS712/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/ACS712.svg?maxAge=3600)](https://github.com/RobTillaart/ACS712/releases)
|
|
|
|
|
2021-10-16 11:40:09 +02:00
|
|
|
|
2020-11-27 11:10:47 +01:00
|
|
|
# ACS712
|
2020-03-19 15:16:52 +01:00
|
|
|
|
2021-12-09 15:42:38 +01:00
|
|
|
Library for the ACS712 Current Sensor - 5A, 20A, 30A.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2020-03-19 15:16:52 +01:00
|
|
|
|
|
|
|
## Description
|
|
|
|
|
2020-11-27 11:10:47 +01:00
|
|
|
The ACS712 is a chip to measure current, both AC or DC. The chip has an
|
2021-10-16 11:40:09 +02:00
|
|
|
analogue output that provides a voltage that is linear with the current.
|
|
|
|
The ACS712 library supports only a built in ADC by means of **analogRead()**.
|
2020-03-19 15:16:52 +01:00
|
|
|
There are 2 core functions:
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
- **int mA_DC()**
|
2021-10-16 11:40:09 +02:00
|
|
|
- **int mA_AC(float freq = 50)** The frequency can be set to typically 50 or 60 Hz
|
|
|
|
however other values e.g. 50.1 or 40 or 123.456 are possible.
|
2020-03-19 15:16:52 +01:00
|
|
|
|
2021-12-09 15:42:38 +01:00
|
|
|
To measure DC current a single **analogRead()** with conversion maths is sufficient to get
|
2021-10-16 11:40:09 +02:00
|
|
|
a value. To stabilize the signal **analogRead()** is called twice.
|
2020-03-19 15:16:52 +01:00
|
|
|
|
2021-10-16 11:40:09 +02:00
|
|
|
To measure AC current **a blocking loop for 20 milliseconds** is run to determine the
|
2020-11-27 11:10:47 +01:00
|
|
|
peak to peak value which is converted to the RMS value. To convert the peak2peak
|
|
|
|
value to RMS one need the so called crest or form factor. This factor depends heavily
|
2021-10-16 11:40:09 +02:00
|
|
|
on the signal form. For a perfect sinus the value is sqrt(2)/2 == 1/sqrt(2).
|
2021-12-09 15:42:38 +01:00
|
|
|
See Form factor below.
|
2020-03-19 15:16:52 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
## Interface
|
|
|
|
|
|
|
|
#### Base
|
|
|
|
|
2021-12-01 14:20:22 +01:00
|
|
|
- **ACS712(uint8_t analogPin, float volts = 5.0, uint16_t maxADC = 1023, uint8_t mVperA = 100)** constructor.
|
|
|
|
It defaults a 20 A type sensor, which is defined by the default value of mVperA. See below.
|
2021-10-16 11:40:09 +02:00
|
|
|
- **int mA_AC(float freq = 50)** blocks ~21 ms (depending on the freq) to sample a whole 50 or 60 Hz period.
|
2021-12-01 14:20:22 +01:00
|
|
|
Since version 0.2.2 frequencies other integer values than 50 and 60 are supported, the lower the frequency,
|
|
|
|
the longer the blocking period.
|
2021-10-16 11:40:09 +02:00
|
|
|
Since version 0.2.3 floating point frequencies are supported to tune optimally.
|
2021-12-09 15:42:38 +01:00
|
|
|
- **int mA_DC()** blocks < 1 ms (Arduino UNO) as it just needs one **analogRead()**.
|
2021-06-24 14:41:36 +02:00
|
|
|
|
|
|
|
|
|
|
|
| type sensor | mVperA | LSB 5V-10bit |
|
|
|
|
|:-------------|:------:|:------------:|
|
|
|
|
| 5 A | 185 | ~25 mA |
|
|
|
|
| 20 A | 100 | ~50 mA |
|
|
|
|
| 30 A | 66 | ~75 mA |
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
#### Midpoint
|
|
|
|
|
2021-10-16 11:40:09 +02:00
|
|
|
- **void setMidPoint(uint16_t mp)** sets midpoint for the ADC conversion.
|
2021-12-01 14:20:22 +01:00
|
|
|
- **void autoMidPoint(float freq = 50)** Auto midPoint, assuming zero DC current or any AC current.
|
|
|
|
Note it will block for 2 periods. Since version 0.2.2 frequencies other than 50 and 60 are supported.
|
2021-10-16 11:40:09 +02:00
|
|
|
By setting the frequency to e.g 1, the code will sample for 2 seconds, possibly getting a better average.
|
|
|
|
- **uint16_t getMidPoint()** read the value set / determined.
|
|
|
|
- **void incMidPoint()** manual increase midpoint, e.g. useful to manually adjust the midPoint in an interactive application.
|
|
|
|
- **void decMidPoint()** manual decrease midpoint.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
|
2021-12-01 14:20:22 +01:00
|
|
|
#### Form factor
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
Also known as crest factor; affects AC signals only.
|
|
|
|
|
2021-10-16 11:40:09 +02:00
|
|
|
- **void setFormFactor(float ff = ACS712_FF_SINUS)** manually sets form factor, must be between 0.0 and 1.0
|
|
|
|
- **float getFormFactor()** returns current form factor.
|
2021-06-24 14:41:36 +02:00
|
|
|
|
2021-12-09 15:42:38 +01:00
|
|
|
The library has a number of predefined form factors:
|
2021-06-24 14:41:36 +02:00
|
|
|
|
|
|
|
| definition | value | approx | notes |
|
|
|
|
|:---------------------|:--------------|:------:|:--------|
|
|
|
|
| ACS712_FF_SQUARE | 1.0 | 1.000 | |
|
2021-10-16 11:40:09 +02:00
|
|
|
| ACS712_FF_SINUS | 1.0 / sqrt(2) | 0.707 | default |
|
2021-06-24 14:41:36 +02:00
|
|
|
| ACS712_FF_TRIANGLE | 1.0 / sqrt(3) | 0.577 | |
|
|
|
|
| | | | |
|
|
|
|
|
|
|
|
It is important to measure the current with a calibrated multimeter
|
2021-12-09 15:42:38 +01:00
|
|
|
and determine / verify the form factor of the signal.
|
|
|
|
This can help to improve the quality of your measurements.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
|
|
|
|
#### Noise
|
|
|
|
|
|
|
|
Default = 21 mV.
|
2021-06-24 14:41:36 +02:00
|
|
|
|
2021-12-09 15:42:38 +01:00
|
|
|
- **void setNoisemV(uint8_t noisemV = 21)** set noise level,
|
|
|
|
is used to determine zero level e.g. in AC measurements.
|
2021-10-16 11:40:09 +02:00
|
|
|
- **uint8_t getNoisemV()** returns the set value.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
#### mV per Ampere
|
|
|
|
|
2021-12-09 15:42:38 +01:00
|
|
|
Both for AC and DC. Is defined in the constructor and depends on sensor used.
|
2021-06-24 14:41:36 +02:00
|
|
|
|
2021-10-16 11:40:09 +02:00
|
|
|
- **void setmVperAmp(uint8_t mva)** sets the milliVolt per Ampere measured.
|
|
|
|
- **uint8_t getmVperAmp()** returns the set value.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
Typical values see constructor above.
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-12-01 14:20:22 +01:00
|
|
|
#### Experimental
|
|
|
|
|
2021-12-02 19:29:13 +01:00
|
|
|
- **float detectFrequency(float minimalFrequency = 40)** Detect the frequency of the AC signal.
|
|
|
|
- **void setMicrosAdjust(float factor = 1.0)** adjusts the timing of micros in **detectFrequency()**.
|
2021-12-01 14:20:22 +01:00
|
|
|
Values are typical around 1.0 ± 1%
|
2021-12-02 19:29:13 +01:00
|
|
|
- **float getMicrosAdjust()** returns the set factor.
|
2021-12-01 14:20:22 +01:00
|
|
|
|
|
|
|
The minimum frequency of 40 Hz is used to sample enough time to find the minimum and maximum for 50 and 60 Hz signals.
|
|
|
|
Thereafter the signal is sampled 10 cycles to minimize the variation of the frequency.
|
|
|
|
|
|
|
|
The **microsAdjust()** is to adjust the timing of **micros()**.
|
|
|
|
It is only useful if one has a good source like a calibrated function generator to find the factor
|
|
|
|
to adjust. Testing with my UNO I got a factor 0.9986.
|
|
|
|
|
|
|
|
Current version is not performance optimized.
|
|
|
|
|
|
|
|
|
2020-03-19 15:16:52 +01:00
|
|
|
## Test
|
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
The library is tested with the RobotDyn ACS712 20 A breakout and an Arduino UNO.
|
2020-03-19 15:16:52 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2020-03-19 15:16:52 +01:00
|
|
|
## Operation
|
|
|
|
|
2020-11-27 11:10:47 +01:00
|
|
|
With the constructor the parameters **volts** and **maxADC (steps)** of the ADC are set
|
|
|
|
together with the **milliVolt per Ampere** value. The last parameter can be adjusted
|
2021-06-24 14:41:36 +02:00
|
|
|
afterwards, e.g. to calibrate this value runtime. Note this parameter affects both
|
2020-03-19 15:16:52 +01:00
|
|
|
AC and DC measurements.
|
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
To calibrate the zero level for DC measurements, 5 functions are available to
|
2020-03-19 15:16:52 +01:00
|
|
|
adjust the midPoint.
|
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
To calibrate the RMS value for AC measurements, 2 functions are available to
|
2020-03-19 15:16:52 +01:00
|
|
|
get and set the formFactor.
|
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
To calibrate the noise level (used for AC measurements), 2 functions are available to
|
2020-11-27 11:10:47 +01:00
|
|
|
get and set the noise in mV.
|
2020-03-19 15:16:52 +01:00
|
|
|
|
2020-11-27 11:10:47 +01:00
|
|
|
The examples show the basic working of the functions.
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-06-24 14:41:36 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
## Future
|
|
|
|
|
2021-10-16 11:40:09 +02:00
|
|
|
- mA_AC blocks 20 ms so might affect task scheduling on a ESP32.
|
2021-06-24 14:41:36 +02:00
|
|
|
This needs to be investigated. Probably need a separate thread that wakes up when new analogRead is available.
|
2021-12-09 15:42:38 +01:00
|
|
|
- **detectFrequency** also blocks pretty long.
|
2021-10-16 11:40:09 +02:00
|
|
|
- int point2point(float freq) function for AC. Is part of mA_AC() already.
|
|
|
|
Needs extra global variables, which are slower than local ones
|
|
|
|
Or just cache the last p2p value?
|
2021-06-24 14:41:36 +02:00
|
|
|
- external analogue read support? separate class?
|