analog.c 2.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758
  1. /* Copyright 2015 Jack Humbert
  2. *
  3. * This program is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. // Simple analog to digitial conversion
  17. #include <avr/io.h>
  18. #include <avr/pgmspace.h>
  19. #include <stdint.h>
  20. #include "analog.h"
  21. static uint8_t aref = (1 << REFS0); // default to AREF = Vcc
  22. void analogReference(uint8_t mode) { aref = mode & 0xC0; }
  23. // Arduino compatible pin input
  24. int16_t analogRead(uint8_t pin) {
  25. #if defined(__AVR_ATmega32U4__)
  26. static const uint8_t PROGMEM pin_to_mux[] = {0x00, 0x01, 0x04, 0x05, 0x06, 0x07, 0x25, 0x24, 0x23, 0x22, 0x21, 0x20};
  27. if (pin >= 12) return 0;
  28. return adc_read(pgm_read_byte(pin_to_mux + pin));
  29. #elif defined(__AVR_AT90USB646__) || defined(__AVR_AT90USB1286__)
  30. if (pin >= 8) return 0;
  31. return adc_read(pin);
  32. #else
  33. return 0;
  34. #endif
  35. }
  36. // Mux input
  37. int16_t adc_read(uint8_t mux) {
  38. #if defined(__AVR_AT90USB162__)
  39. return 0;
  40. #else
  41. uint8_t low;
  42. ADCSRA = (1 << ADEN) | ADC_PRESCALER; // enable ADC
  43. ADCSRB = (1 << ADHSM) | (mux & 0x20); // high speed mode
  44. ADMUX = aref | (mux & 0x1F); // configure mux input
  45. ADCSRA = (1 << ADEN) | ADC_PRESCALER | (1 << ADSC); // start the conversion
  46. while (ADCSRA & (1 << ADSC))
  47. ; // wait for result
  48. low = ADCL; // must read LSB first
  49. return (ADCH << 8) | low; // must read MSB only once!
  50. #endif
  51. }