你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32开发项目:ADS1115的驱动与使用

[复制链接]
STMCU小助手 发布时间:2022-4-13 18:00
ADS1115介绍
& M8 A9 ]4 A8 C" r% G& u1 WADS1115是具有 PGA、振荡器、电压基准、比较器的 16 位、860SPS、4 通道 Δ-Σ ADC,数据通过一个 I2C 兼容型串行接口进行传输。有关它的详细说明可以参考官方数据手册。$ m1 ^5 w6 B* e5 _* |& }* V+ G; `
8 W6 l! L) z% S% T
驱动源码2 I' j8 w5 j! `3 w
头文件

/ a" z- B& \! m% d% E
  1. #ifndef __ADS1115_H__  I" r3 h) M& g, P# y2 i" k5 M
  2. #define __ADS1115_H__
    ( X& u0 T. T) |0 h; [: I

  3. 7 [8 g. a% f1 X8 T3 i! |
  4. #include "stm32f4xx.h"
    " A2 d3 m  K  s$ r
  5. #include "stm32f4xx_conf.h"
    % |/ E  K$ E/ H# b% ?! t( r  s4 e
  6. 3 `6 g3 {, a: n' n& r
  7. #ifdef ADS1115- t+ H4 k/ d( f' ^
  8. #include "i2c_virtual.h"
    6 p; x+ [+ c- _: A

  9. ( g! r& b# ^; p
  10. #define ADS1115_SCL_PORT        'A'
    # u+ Y2 W: l' l, _6 w$ g) y
  11. #define ADS1115_SCL_PIN                6
    ; N! {& J/ g% l  A( C
  12. 8 _6 t+ K" i; B  q% o
  13. #define ADS1115_SDA_PORT        'A'4 i* S3 O$ e5 f0 y4 T2 w, a. [
  14. #define ADS1115_SDA_PIN                57 o+ d; I4 |: R  y

  15. . O5 p) R! t1 n% ^6 C6 V. c
  16. #define Accuracy                                                                         32768        //ADC单端输入15位精度# ]3 Q7 E, o- j: u9 u
  17. #define ADS1115_ADDRESS_0                                        0x90        //ADDR PIN ->GND( X" j% w9 z% q, c" m5 ?
  18. #define ADS1115_ADDRESS_1                                                0x92        //ADDR PIN ->VDD
    8 `( O# J- m$ [) w
  19. #define ADS1115_ADDRESS_2                                        0x94        //ADDR PIN ->SDA$ N9 q" v& D( O4 m3 J
  20. #define ADS1115_ADDRESS_3                                        0x96        //ADDR PIN ->SCL& O- d# A! N& @0 S: {& c
  21. & C8 S& r6 b  R) l# B8 V" s' G
  22. //*************ADDR Initial********************/2 L2 p6 S, N: R; s" z8 T2 N
  23. #define ADS1115_ADDRESS                ADS1115_ADDRESS_0                //ADDR PIN ->GND
    & t" Q: a# z: z6 D
  24. #define ADS1115_ADDRESS_W        ADS1115_ADDRESS|0x00        //write address
    % h& X" R4 q! Q8 ?2 Q3 Z  y
  25. #define ADS1115_ADDRESS_R        ADS1115_ADDRESS|0x01        //read address4 ^0 M% [) x' x- @9 q1 j) \
  26. / F; u* R& \0 m" l- R1 E5 ]; V; n
  27. /************POINTER REGISTER*****************/
    6 r0 `! d! V# t7 W' ~
  28. #define ADS1115_Pointer_ConverReg                        0x00        //Convertion register5 X- t& `) H0 P9 p! o( K  C
  29. #define ADS1115_Pointer_ConfigReg                        0x01        //Config register
      d0 K  L6 V/ f
  30. #define ADS1115_Pointer_LoThreshReg                        0x02        //Lo_thresh register
    - e* d4 ?# ]- O
  31. #define ADS1115_Pointer_HiThreshReg                        0x03        //Hi_thresh register
    ! @: x/ v7 T# t$ w6 g7 |3 b6 C
  32. 7 V$ c* m( m; k& w  ^; j
  33. /************CONFIG REGISTER*****************/
    . J$ I/ c# n  a: g% E: C1 h+ B

  34. 9 |0 U. B, B/ h- B5 ?
  35. //Bit[15]7 i7 r0 [7 ]) Q
  36. #define ADS1115_OS_OperationalStatus                0x0000                //No Effect6 r- q2 ~% @- R0 d6 y
  37. #define ADS1115_OS_SingleConverStart                0x8000                //Begin a single conversion6 p" R* ]* ?7 S: v1 T6 _
  38. //Bits[14:12]
    " H# y0 Q- l$ c
  39. #define ADS1115_MUX_Differ_01                                0x0000                //AINp=AIN0, AINn=AIN1(default)# y- p9 T- g1 v6 O
  40. #define ADS1115_MUX_Differ_03                                0x1000                //AINp=AIN0, AINn=AIN3& ?! k8 n- z& E. S5 V: A, s, m2 X
  41. #define ADS1115_MUX_Differ_13                                0x2000                //AINp=AIN1, AINn=AIN3
    # w: a8 P2 M, W7 P. {0 t3 k
  42. #define ADS1115_MUX_Differ_23                                0x3000                //AINp=AIN2, AINn=AIN3/ q( m; ^  j; }% y# s- Q" k6 j
  43. #define ADS1115_MUX_Channel_0                                0x4000                //AINp=AIN0, AINn=GND
    : }  m) z7 e5 J8 I( @& L2 a
  44. #define ADS1115_MUX_Channel_1                                0x5000                //AINp=AIN1, AINn=GND
    & n4 L8 D  z; G" }5 N, J4 B* b, N
  45. #define ADS1115_MUX_Channel_2                                0x6000                //AINp=AIN2, AINn=GND
    . P! c+ A! f- ^8 m$ z7 q" v
  46. #define ADS1115_MUX_Channel_3                                0x7000                //AINp=AIN3, AINn=GND; A2 B3 S9 s7 P- ?
  47. //Bits[11:9]
    " b' y9 N6 J, ]( L
  48. #define ADS1115_PGA_6144                                        0x0000                //FS=6.144V; L8 A% O" c' {6 \. [
  49. #define ADS1115_PGA_4096                                        0x0200                //FS=4.096V
    ' H; @: j" I2 f5 a$ M5 O9 N1 f
  50. #define ADS1115_PGA_2048                                        0x0400                //FS=2.048V(default)
    * y5 j9 `" a, o3 A
  51. #define ADS1115_PGA_1024                                        0x0600                //FS=1.024V2 M/ Q9 y" J9 m: b
  52. #define ADS1115_PGA_0512                                        0x0800                //FS=0.512V6 J4 z" ?. ?  h9 j+ M9 ]. E
  53. #define ADS1115_PGA_0256                                        0x0A00                //FS=0.256V: h: G2 o, i' s3 F
  54. //Bit[8]
    . k) G# u, U: o9 j7 R4 |! p/ j0 J( j
  55. #define ADS1115_MODE_ContinuConver                        0x0000                //Continuous conversion mode
    ) l* s; I6 T. ?7 c9 s
  56. #define ADS1115_MODE_SingleConver                        0x0100                //Power-down single-shot mode(default)0 n# v( W! |2 X! c0 P
  57. //Bits[7:5]6 M' U3 W0 ^- z& [+ B
  58. #define ADS1115_DataRate_8                                        0x0000                //Data Rate = 8# |. ~6 Y0 {7 [' r5 _" L& g
  59. #define ADS1115_DataRate_16                                        0x0020                //Data Rate = 16
    7 f% y% K3 B! Y3 a7 h6 N) `
  60. #define ADS1115_DataRate_32                                        0x0040                //Data Rate = 32
    # |% z+ a9 v$ n+ k8 B4 t/ C% A
  61. #define ADS1115_DataRate_64                                        0x0060                //Data Rate = 64
    , Z( H6 C- S. m! N4 L# p
  62. #define ADS1115_DataRate_128                                0x0080                //Data Rate = 128(default)$ u4 a6 S1 x& z% N5 w
  63. #define ADS1115_DataRate_250                                0x00A0                //Data Rate = 2503 z* R! r: U1 A2 n# Y
  64. #define ADS1115_DataRate_475                                0x00C0                //Data Rate = 475
    % ^9 {2 _  c& x: g' M
  65. #define ADS1115_DataRate_860                                0x00E0                //Data Rate = 860
    * U6 ^! b7 O, [1 x$ q8 ]  [' H
  66. //Bit[4]
    + Y- U) V/ b  d7 Q3 A: N. J& r/ p6 d, {
  67. #define ADS1115_COMP_MODE_0                                        0x0000                //Traditional comparator with hysteresis
    5 b3 [& c8 i% ?, q
  68. #define ADS1115_COMP_MODE_1                                        0x0010                //Window comparator& L3 C/ c) a& P$ k5 ~- `& V$ |) Z
  69. //Bit[3]6 L+ m" m/ M! F. ]+ y5 a
  70. #define ADS1115_COMP_POL_0                                        0x0000                //Active low
    / _* [6 d+ N! n# h8 J
  71. #define ADS1115_COMP_POL_1                                        0x0008                //Active high( M8 f+ x: \  I: J( K. s" d% q
  72. //Bit[2]
    ; g: \% O) g  q# a; Q% P0 q  ]0 {! j
  73. #define ADS1115_COMP_LAT_0                                        0x0000                //Non-latching comparator7 ?! \: j- P1 |/ V7 x. b# p
  74. #define ADS1115_COMP_LAT_1                                        0x0004                //Latching comparator/ ?( |3 z8 Z% f# K) e" b+ F8 Q: J
  75. //Bits[1:0]
      D' o' b7 |6 U
  76. #define ADS1115_COMP_QUE_0                                        0x0000                //Assert after one conversion! h$ D" L4 T; a' \  m
  77. #define ADS1115_COMP_QUE_1                                        0x0001                //Assert after two conversion
    7 _# W7 g2 {% ^8 p2 \  q0 ^
  78. #define ADS1115_COMP_QUE_2                                        0x0002                //Assert after four conversion
    " g/ b: w3 |. U# Z2 [* ]& d
  79. #define ADS1115_COMP_QUE_3                                        0x0003                //Disable Comparator; A. q0 p  r6 A
  80. + k& W1 \: h% B6 J+ u& v
  81. typedef struct  n0 ~$ h5 x- K$ ]( [; A: J/ y
  82. {1 Q6 }6 I6 e0 a/ F5 @4 X. r
  83.     u16 OS;
    , A& @4 K+ x0 e- G( R
  84.     u16 MUX;& s2 `. z5 K1 {8 ]2 z3 R6 |" @# _
  85.     u16 PGA;
    . i) U0 O( ~% f* s% r
  86.     u16 MODE;/ y4 t* f4 h. `5 m& L: y5 Y
  87.     u16 DataRate;2 p( ?* I; ]; S& P- x
  88.     u16 COMP_MODE;
    ( d6 D7 A  E" P7 ]
  89.     u16 COMP_POL;3 U1 ~+ `0 [; H% `
  90.     u16 COMP_LAT;  W) n9 M. x8 w
  91.     u16 COMP_QUE;
    , a' @- r. x. L! n& B4 {# Q
  92. } ADS1115_InitTypeDefine;
    $ t7 R, K: g' w+ ~

  93. ; w) x* d# N! f3 F; ?
  94. extern int16_t ADS1115_RawData[4];
      t/ f7 N4 [" f; q. f$ y
  95. ! z8 {9 ^/ q& E. Y( v1 E
  96. void ADS1115_Init();
    , Z# U( B0 m% e1 C) U' X
  97. void ADS1115_UserConfig1();: }( d0 ?! r# c$ H: s. n4 X
  98. void ADS1115_UserConfig2();: ~% w9 `7 Q! W1 W. N
  99. u8 ADS1115_Config(ADS1115_InitTypeDefine* ADS1115_InitStruct);# J1 g7 z& W1 x. g; h' K

  100. : T1 L" a2 Z/ X4 E- S3 m( j' h3 Y% n1 @. D
  101. u8 ADS1115_ReadRawData(int16_t* rawData);2 \0 R( R0 @' p! h. m. J
  102. void ADS1115_ScanChannel(uint8_t channel);
    & Y2 ~- ~+ c% a) e; w) m
  103. float ADS1115_RawDataToVoltage(int16_t rawData);
      Q2 X+ q+ r9 q$ Z9 b7 o

  104. + C, |& b) s  b+ ^& P5 `
  105. float ADS1115_GetVoltage();) ^' b3 x# c0 n3 {8 B7 ~) c
  106. float ADS1115_GetAverageVoltage(uint16_t num);
    " e  e3 P5 T  K. M; ^
  107. & X. Q5 Q  P# K
  108. void ADS1115_RefreshAllChannel();
    5 s% @( |1 m* n- V8 ~: u
  109. 0 W9 y  Y: M, ~* F5 e6 y
  110. #endif
    ) K& {3 y, l5 G9 K2 Z3 d& T! o8 J
  111. #endif$ H5 j$ r! E0 ~# \- F! ]# U! Z5 m7 l2 a8 N
复制代码

0 I- z' V8 p7 b5 r, X1 h源文件1 o1 A9 M$ l- L8 q4 m9 M3 [
  1. #include "ads1115.h"9 Z2 w# l9 P) c/ r. h" b

  2. 0 h2 {9 k1 i8 y7 R" ~
  3. #ifdef ADS11155 A* D0 `& H/ z! c* W0 c

  4. / I; }% a4 c' X) N/ I
  5. static ADS1115_InitTypeDefine ADS1115_InitType;
    1 ~. ]0 A) z. p$ P/ n

  6. . x0 y" I' m8 w- h1 y" V  O; \
  7. /**
    3 ~' d( c2 e  |
  8. * 可供外部调用的全局变量,记录了ADS1115采样的原始16位数据。调用void ADS1115_RefreshAllChannel( )函数可以刷新这个变量。) p1 D9 N8 x2 G/ @2 {4 p
  9. * 通过定义宏ADS1115_USE_FILTER,可以将ADS1115的轮询采样数据经过滑动滤波后,保存到ADS1115_RawData[]中。0 ]. z: v" b# |$ b
  10. * 通过float ADS1115_RawDataToVoltage(int16_t rawData)函数可以将ADS1115_RawData[]换算成对应的电压值。
    . y, d6 }# Q6 R8 {7 }# A' v; S
  11. */. k' p5 G# K0 M9 h+ z
  12. int16_t ADS1115_RawData[4] = {0};- g; F, d' N% D' `. W  S
  13. ! [: Q3 M; R& ~
  14. 2 p% t  d& C( V; D. N- m+ P
  15. /**% Y2 A$ W/ M6 F2 B; o9 N; J7 a
  16. * @brief 完成芯片控制端口初始化,并设置初始状态) W  z- f% F7 ?6 ]
  17. */
    ) H% j* w% c+ ?: G0 Y2 L/ }
  18. void ADS1115_Init()
    1 ]: s7 ~7 y8 S; G& v
  19. {
    ! i: W6 I! |! |$ z* K
  20.         I2C_Virtual_ConfigPort(ADS1115_SDA_PORT, ADS1115_SDA_PIN, ADS1115_SCL_PORT, ADS1115_SCL_PIN);" V5 l  U. m  E1 W
  21. }
    6 O, F2 _3 h1 o  D
  22. 1 L/ y/ _4 F2 d, p: r* R$ A. ^
  23. " F( o  y% [0 g
  24. /**) m2 t0 ~+ {  E7 U( g% Q0 M) Z. J. Q
  25. * @brief Configuration of ADS1115, single-shot: r; p7 t: U+ g; e
  26. */
    7 H6 T. E5 o0 a: o9 r
  27. void ADS1115_UserConfig1()( O1 g: W9 W* @* d7 F) }
  28. {
    * t: ?1 q: M2 r1 M5 d/ C7 t
  29.         ADS1115_InitType.COMP_LAT = ADS1115_COMP_LAT_0;
    ( u( g1 m0 U% ~' {
  30.         ADS1115_InitType.COMP_MODE = ADS1115_COMP_MODE_0;
    5 u) u  m+ x! o% L- D$ i: w/ c
  31.         ADS1115_InitType.COMP_POL = ADS1115_COMP_POL_0;; g: L5 j% Q' V% V" ~6 C2 s% f2 @
  32.         ADS1115_InitType.DataRate = ADS1115_DataRate_475;. n# l' `5 i- N9 S  Q
  33.         ADS1115_InitType.MODE = ADS1115_MODE_SingleConver;
    . m; X) l" `1 [# U) u. r4 }* J
  34.         ADS1115_InitType.MUX = ADS1115_MUX_Channel_0;
    9 ^; ^4 c/ f$ A7 ]8 ~5 e, G  p. ~
  35.         ADS1115_InitType.OS = ADS1115_OS_SingleConverStart;1 B* A% g( f9 I! m  F3 I
  36.         ADS1115_InitType.PGA = ADS1115_PGA_4096;
    5 Q, W0 L6 H; ~
  37. ' A  R) E7 f( D/ b+ O
  38.         ADS1115_Config(&ADS1115_InitType);& ~4 B8 J7 s9 o" A$ X  `. `
  39. }7 k% F6 p, Y7 p2 T$ \/ ^, W

  40. / i# @) i& }0 @6 ^0 M

  41. 8 ]: N/ g) l0 c; I/ _% {; D
  42. /**0 Y; F9 E. S6 N6 ~5 i% \
  43. * @brief Configuration of ADS1115, continuous conversion
    & ~2 ]; M) V1 e7 q! h$ ?! k4 {
  44. */3 `5 j% L: f. m, D2 y+ b  z9 h$ v3 u
  45. void ADS1115_UserConfig2()9 d! S2 N) m- m2 e* i' @5 S# g
  46. {
    # W& f# ]+ s: p9 \5 v- A+ ?! Y
  47.         ADS1115_InitType.COMP_LAT = ADS1115_COMP_LAT_0;
    ) f& z! Q% f0 X$ d4 w7 M9 q/ g
  48.         ADS1115_InitType.COMP_MODE = ADS1115_COMP_MODE_0;8 ~3 H# X7 s. m3 L9 Z
  49.         ADS1115_InitType.COMP_POL = ADS1115_COMP_POL_0;8 z: Q  B, r: }* [, p6 E+ L
  50.         ADS1115_InitType.DataRate = ADS1115_DataRate_475;! c) |) l# Y9 s/ K. G# r0 k. ]
  51.         ADS1115_InitType.MODE = ADS1115_MODE_ContinuConver;( |1 r8 I  |4 a
  52.         ADS1115_InitType.MUX = ADS1115_MUX_Channel_0;8 Z8 f9 o1 g0 P" ?
  53.         ADS1115_InitType.OS = ADS1115_OS_OperationalStatus;* i6 b- P0 |7 b+ Q* M
  54.         ADS1115_InitType.PGA = ADS1115_PGA_4096;5 L0 O1 D( S9 H/ @! T& @: h

  55. ) q* L3 b& t0 U% d+ Q
  56.         ADS1115_Config(&ADS1115_InitType);6 J1 d: o! ?+ m/ c5 s
  57. }
    # m2 S! @! ]! N
  58. 3 j5 @5 V6 ]$ V- d# p' g

  59. ! @3 o* f3 T- s- F$ R  h
  60. /**2 y& D3 W# U) u) K  M
  61. * @brief 配置ADS1115
    ! S' ~- b. v# ^( m( [8 e
  62. * @param ADS1115_InitStruct: 用来配置ADS1115的结构体变量指针
    " E+ ?$ }9 B' D7 Y
  63. * @return 配置结果/ j0 d0 A7 q6 G9 ~& ^8 s7 h
  64. *                 @arg: fail
    9 A" x! j* K. p- W+ R
  65. *                 @arg: success
    $ h. P+ R  a& D) B1 u
  66. */
    7 s9 S, V% H5 n- }5 ^+ A- T
  67. u8 ADS1115_Config(ADS1115_InitTypeDefine *ADS1115_InitStruct)# G9 ~6 _6 f  z5 G# C
  68. {
    1 d) S6 y+ d' Q5 a& M  w
  69.         u16 Config;7 T* w( X* {$ O" B
  70.         u8 Writebuff[2];
    % H! n5 }# R9 f2 \& k
  71. 7 g4 \( }  v8 m( \* W
  72.         Config = ADS1115_InitStruct->OS + ADS1115_InitStruct->MUX + ADS1115_InitStruct->PGA + ADS1115_InitStruct->MODE, d/ C; R( \& D- R, o' I: s
  73.                         + ADS1115_InitStruct->DataRate + ADS1115_InitStruct->COMP_MODE + ADS1115_InitStruct->COMP_POL
    9 I# C. `$ k$ W2 N
  74.                         + ADS1115_InitStruct->COMP_LAT + ADS1115_InitStruct->COMP_QUE;$ p6 Z5 u6 I6 ]% C

  75. # c, @, D1 c& Y6 m1 i4 k
  76.         Writebuff[0] = (unsigned char) ((Config >> 8) & 0xFF);$ X8 O4 X3 A' C
  77.         Writebuff[1] = (unsigned char) (Config & 0xFF);) m: I+ P9 D) u  t* k! O
  78. ( @3 j# T# a% z& G
  79.         I2C_Virtual_Start();                                        //启动总线4 g' ^/ _  F' Q6 ^9 O' L) n
  80.         I2C_Virtual_SendByte(ADS1115_ADDRESS_W);                //发送器件地址(写)
    ! s. }# j* L3 Q& p4 m

  81. 9 O  q4 D6 K; ^9 @
  82.         if (I2C_Virtual_ack == 0)
    ( e* B7 v& g6 W: _
  83.                 return (0);1 {8 u/ x1 A/ w) U% Y& T

  84. 7 f! a9 b' ~+ J  E; {- p) Z
  85.         I2C_Virtual_SendByte(ADS1115_Pointer_ConfigReg);                //发送寄存器地址
    + s0 q4 c5 M- ]! w7 G4 q% F

  86. . p0 q+ ]. I) I5 \/ \& C
  87.         if (I2C_Virtual_ack == 0)
    8 h! e3 D- ~2 d$ b( F+ h
  88.                 return (0);
    $ P6 J, ~" F* [2 r

  89. 0 G6 C! w( a3 J
  90.         I2C_Virtual_SendByte(Writebuff[0]);                //发送数据% a7 U. \" @/ y% }* _. P; J

  91. 4 Z2 T* b& i! j+ g" I4 K& y
  92.         if (I2C_Virtual_ack == 0)
    9 B  H' j. d6 r1 }# o
  93.                 return (0);
    ' k( i5 H  Q$ O6 Y  l
  94. " q7 r1 M. N; V- n
  95.         I2C_Virtual_SendByte(Writebuff[1]);                //发送数据% E& i4 |" t- V2 T4 b  D' ~" ^
  96. # f+ d7 o9 l0 A  s* S
  97.         if (I2C_Virtual_ack == 0)
    $ S& e2 n% D6 j; n1 a
  98.                 return (0);
    ( }7 ~) y: x0 O" |' ^+ s$ E' W
  99. ' ]) x% k$ C! N0 n
  100.         I2C_Virtual_Stop();0 U- d# e# J% T6 E7 t

  101. % K2 R: N% h6 N( x- {+ }7 z" }
  102.         return (1);
    % R. m) K, J. {) C2 A1 V! U0 Z  L* L
  103. }
    ; \3 k2 f. G) G5 A/ V& J
  104. . r, ], B+ h% |
  105. % d/ q: `' c9 F, E( C
  106. + r' a; C" _! D  Y( [( \" b
  107. /**
    9 O: G6 n/ V5 _4 z) D
  108. * @brief 读取ADS1115当前通道下的原始数据
    0 I* i0 m! R3 J9 Y
  109. * @param rawData: 传入一个int16_t整型变量的指针,ADS1115的原始数据将保存在这个变量中
    1 @/ C) B# a# u+ M, Y/ y' t
  110. * @return 读取结果
    5 E  q- ~3 t. r( ^' C
  111. *                 @arg 0: fail" U" G) c6 b& |; u7 L9 [# C
  112. *                 @arg 1: success. U' [  X, |0 O
  113. */
    5 W5 S! o0 d/ l, _
  114. uint8_t ADS1115_ReadRawData(int16_t *rawData)
    " C0 H9 p5 o$ V. r
  115. {
    3 u8 S+ q, n' r! x) v
  116.         unsigned char Result[2];
    $ y: y% ]8 q( J5 p& z! p
  117. 6 \+ h, m' F3 ?; i1 S* ?$ R
  118.         I2C_Virtual_Start();                                                                                                                        //启动总线5 V. e/ Z2 R! f8 ]
  119.         I2C_Virtual_SendByte(ADS1115_ADDRESS_W);                                        //发送器件地址(写)
    * t/ S( s3 o1 C; B- A
  120. 4 |) J+ @  w- w; m8 k
  121.         if (I2C_Virtual_ack == 0)
    # [6 P; T6 [8 ]+ d
  122.                 return (0);( m9 ?. f9 E( i! o% {

  123. ) `3 @0 ]4 U* M! P% ~. H' W8 c
  124.         I2C_Virtual_SendByte(ADS1115_Pointer_ConverReg);         //发送寄存器地址
    7 W3 \! y* b. Q  y
  125. 6 _' h5 l: t7 ]1 ]* C. f/ u
  126.         if (I2C_Virtual_ack == 0)5 E) F; w9 Z" S7 R! `* m
  127.                 return (0);, t, r& f; j4 I7 f( \
  128. 5 j1 J, o- y. L
  129.         I2C_Virtual_Stop();0 j1 B) }2 g' P4 u
  130.         delay_us(10);
    ; Y* V' o0 @2 c3 \
  131.         I2C_Virtual_Start();                                                                                        //写寄存器之后需要重新启动总线
    3 F: D, |9 ]6 M: j* C  m  }
  132. + z& C, q" D, z/ w1 O$ ?
  133.         I2C_Virtual_SendByte(ADS1115_ADDRESS_R);                                                //发送器件地址(读)
    2 C- q) \! y9 ?* H9 p+ P. M
  134. . i5 s$ N0 s. I/ T# z9 s1 T
  135.         if (I2C_Virtual_ack == 0)& N% a# K& A* [5 ^. B' W3 p
  136.                 return (0);
    7 Y7 Q  u. U* b% D  C) M" c( c

  137. # E- n) c: O. Q7 [9 t& i
  138.         Result[0] = I2C_Virtual_RcvByte();                //接收数据: S' H7 D0 o' y; k$ q
  139.         I2C_Virtual_Ack();                        //发送就答位
    1 r% T# Q4 ?! I2 V6 A- d. u$ L
  140.         Result[1] = I2C_Virtual_RcvByte();
    . g$ d, W, B5 I
  141.         I2C_Virtual_NoAck();                        //发送非应位* S$ G9 D3 B  K+ @4 G( X6 z
  142.         I2C_Virtual_Stop();                         //结束总线% y1 a6 E- ?& x% P3 H2 x3 q4 s7 c
  143. # U2 }+ N1 P% k% t1 _) u4 c( W# ^
  144.         *rawData = (int16_t) (((Result[0] << 8) & 0xFF00) | (Result[1] & 0xFF));
    , {3 u0 o, |0 X  L% M. {! [

  145. & e2 f+ ?  g" x7 A- E$ ?( b
  146.         return 1;
    , f! b. {, d, c( b+ j# f. K+ C
  147. }
    & o) o  c% t  x9 q, x. U; T

  148. & z$ M) H: i1 z$ P( u

  149. . H6 A- |' {, `5 b. }' {1 l+ Y/ X# g
  150. /**
    ' r5 ]6 ~$ U2 c. \9 Q4 T7 I
  151. * @brief Switch the channel of ADS1115
    + T1 L, h3 {  B) Q/ l$ N( F
  152. * @param channel/ S1 h5 T( k: s5 s' D+ b
  153. */$ u4 W9 ?) z% b* d
  154. void ADS1115_ScanChannel(uint8_t channel)
    8 x6 M- {# ^5 A) e9 P
  155. {
    ; h) e( U% o4 `' I* d. K3 f
  156.         switch (channel)
    1 D4 K3 Q9 r- l
  157.         {
    & K/ L6 h6 u8 i8 J
  158.         case 0:6 s4 }7 L7 |# O+ j
  159.                 ADS1115_InitType.MUX = ADS1115_MUX_Channel_0;% X3 P5 Z4 }: N" `6 m
  160.                 break;
    ! ]6 m6 B. Q" o$ V: S4 U# v' n/ p. ~
  161.         case 1:
    - r/ J/ {4 n1 f( A, G4 g! O
  162.                 ADS1115_InitType.MUX = ADS1115_MUX_Channel_1;
    % m8 t1 x7 O; g1 {8 h
  163.                 break;
    $ y; |* @3 L- u& B& P; z; w: _
  164.         case 2:
    $ c- `+ f0 i9 o6 f* K2 m( I
  165.                 ADS1115_InitType.MUX = ADS1115_MUX_Channel_2;
    5 d+ T. b$ G, s9 g, k
  166.                 break;. [% @7 x' p! }: I# [. O' a
  167.         case 3:
    7 j* f, N& [% ?
  168.                 ADS1115_InitType.MUX = ADS1115_MUX_Channel_3;
    3 }8 E/ H" e7 E
  169.                 break;
    * _) @9 I$ }) u
  170.         default:* C+ W9 z1 \# W. g
  171.                 break;
    ( @3 R& |+ o  j( u) {  ~& J) P: h
  172.         }
    0 s* @( T# ~/ _

  173. & ?3 |0 G( a; i+ H& F% J/ p
  174.         ADS1115_Config(&ADS1115_InitType);; _% N( u5 D: P" T: i
  175. }/ S' j1 ~3 z! J6 s
  176. 3 m' L. `9 x% g! R
  177. : u. v) @( q) ]9 z0 a
  178. /**
    " m0 L7 b) w2 x) i( V4 a3 H' @
  179. * @brief 将传感器的原始采样数据转化为电压数据,8 K. b! Q( H( J
  180. *                         根据ADS1115_InitType结构体中包含的增益信息计算
    + k7 r' i/ H2 e! t# |! H
  181. * @param rawData: 待转换的原始数据  P: x7 x( `) u+ d
  182. * @retval 返回经过计算的电压值7 g; l8 B) _2 q) w* d* x# |( k
  183. */
    , ]# g! W5 o' {
  184. float ADS1115_RawDataToVoltage(int16_t rawData)
    $ u  m% n! q: c, H6 O. J8 z
  185. {" o+ F% q0 e4 f& p! k+ K
  186.         float voltage;
    9 K, X5 K! Q* a" ?7 {1 K
  187. ( P- L% j9 h: R7 I* h
  188.         switch (ADS1115_InitType.PGA)
    2 T" {* Q) W9 e  e* ~* p/ r
  189.         {
    * a$ ?' b. S! u1 {1 d3 \
  190.         case ADS1115_PGA_0256:
    7 \/ P- y0 n! |5 ]7 a6 J1 N
  191.                 voltage = rawData * 0.0078125;" v/ W4 ~: c* s# e# X* o. D) @
  192.                 break;" I/ M( o/ ^9 N' T" O* B
  193. & h' V# W5 L1 w; K0 J
  194.         case ADS1115_PGA_0512:
      Y* a- [0 b- d! V
  195.                 voltage = rawData * 0.015625;
    5 ^' M" D8 e5 H: L8 s
  196.                 break;% H6 Y, ]! P0 M; X

  197. - ]6 |- [: F& z4 L, b+ L! w* ]
  198.         case ADS1115_PGA_1024:  H- t, z" K% o, m/ l$ E  }+ ?& Q
  199.                 voltage = rawData * 0.03125;# p0 {/ Z7 C( D& e; f& J
  200.                 break;
    ' F; Q" H7 `- m0 Y$ J* T) F
  201. : j' l  J# A) c  A+ x/ P, z" r
  202.         case ADS1115_PGA_2048:
    5 F- S2 E6 P5 X4 L
  203.                 voltage = rawData * 0.0625;
    " m' T# L  N# x& P2 `% T! W; A
  204.                 break;
    $ t1 I4 \8 T, X% o0 O( `. e* I" C

  205. ' d! G. U- A7 y$ g
  206.         case ADS1115_PGA_4096:, U/ N2 S& F; ~6 w0 \1 |
  207.                 voltage = rawData * 0.125;
    7 ]. N0 r7 i1 T. O0 J- L( _
  208.                 break;, H$ O' o+ O" P' n* l/ ~, y
  209. 4 |2 s8 C! A' \4 T8 ^
  210.         case ADS1115_PGA_6144:
    # |4 Q6 `  Z" Z
  211.                 voltage = rawData * 0.1875;
    # i7 ~; b- n% I8 Q/ o
  212.                 break;9 N! b. ?8 W; e7 u1 M

  213. " V- D* V! a9 B# A+ N1 ~* \
  214.         default:3 G) @' ^5 c0 g5 s7 v' _
  215.                 voltage = 0;! l1 K) `( z0 v" V$ ^6 T
  216.                 break;
    0 A+ F+ \- W: N" R5 V
  217.         }# I+ O- y( q1 Q5 J. ~
  218. / g( ]  t: G# }" D7 H
  219.         return voltage;
      E7 D& b3 U) q/ z3 q0 D& i
  220. }
    + P, q3 b  t, z* s$ @

  221. " D6 a* T) A4 g0 e+ C! K: Z
  222. & `4 }7 l3 k/ Y
  223. /**
    * K' K# Q+ s2 P6 x8 M
  224. * @brief 直接获取ADS1115当前通道的电压采样值  ?1 r  f- X+ j* _7 G  s: G
  225. * @return 电压采样值
    3 s, ]; V# ^% z/ N/ _0 T
  226. */! v$ {5 @+ t5 q$ G% U1 S
  227. float ADS1115_GetVoltage()
    , J. L: P$ p: U& K5 `
  228. {
    ; q/ z0 T; }* N% l1 O: S
  229.         int16_t rawData;
    5 K: `" b0 ~; n7 W- a* V

  230. ( r* {4 r+ n5 |& N5 z1 T
  231.         ADS1115_ReadRawData(&rawData);
    . Z) ]$ h9 e; P% Q. T' F
  232. 2 r% }. t5 ^" m
  233.         return ADS1115_RawDataToVoltage(rawData);
    + P8 }* i# b# A6 \
  234. }
    , m4 J& c) Q# ^# o7 u# D4 x* h
  235. 1 e$ R9 ?' @: n

  236. - g" D! H8 \; J9 A
  237. /**
    ) i7 j% F) Z7 |3 w1 Z) k2 q- _
  238. * @brief 获取并计算ADC采样的平均电压值
    * M( P* W8 i; o' n, C
  239. * @param num: 计算平均值的数量
    , `7 A3 {) h1 C5 ]9 j( a/ m$ }- |
  240. * @retval 电压采样的平均值9 W+ ?" V: ~2 z8 e/ }
  241. */
    $ s7 R; C7 ?2 C
  242. float ADS1115_GetAverageVoltage(uint16_t num)  L( {5 f" a% W8 K0 l: Y( F
  243. {
    ) o: z8 I5 p4 K
  244.         int32_t sum = 0;+ d/ \0 O+ u2 o% U; G% G7 F
  245.         int16_t rawData;: q: e8 P0 c8 Q5 A

  246. 9 _7 x1 s, {: Q0 u; S
  247.         if(num == 0)3 k& W! F, j9 A$ x- W: b
  248.         {7 k4 t& u& x2 e  D, y( K
  249.                 return ADS1115_GetVoltage( );  t* u8 E0 e4 [3 G7 n9 e6 T5 @( X, X
  250.         }, N9 H; l- }1 S0 J3 [; u

  251. : v* f9 L; m- V% Y
  252.         for(uint16_t i =0; i< num;i++)
    1 U5 b- J# A( W* T: L) e
  253.         {' ^) L# w: q( E! n
  254.                 ADS1115_ReadRawData(&rawData);0 |0 K& b4 ?, c4 l/ J  q  @9 o2 ]
  255.                 sum += rawData;
    ) m1 f$ [' E; t& ?% H2 g0 I& F/ V( A) ]
  256.         }) M) F1 x. \% i5 w! H
  257. % A& {3 b; {) Y1 W) ]( K% y
  258.         return ADS1115_RawDataToVoltage(sum/num);
    ) }7 V$ X" E( C4 g1 v$ X8 b
  259. }$ T2 B& O( E1 C6 f

  260. 8 z' l% I* Z/ {
  261. /**# j: z' E3 G+ J6 E
  262. * @brief 刷新ADS1115全部通道的采样数据
    1 [# m6 B: O. Q4 ^4 C, I! G
  263. *                 由于ADS1115通道切换后需要等待较长时间数据才能够稳定,1 h' o; r6 _+ t8 G  z
  264. *                 在进行多路数据采集的时候,切换通道后延时阻塞等待切换完成会占用过多的系统时间,  r; v5 x8 @3 j) W! C. \6 f
  265. *                 因此需要在一个定时器中轮询采集ADS1115数据,每次采集完成后,切换到下一个通道0 b1 j/ i9 {" X5 @$ e: f# d& ]
  266. *                 大幅度提高了系统工作的效率。
      G$ N8 e' `) c4 y
  267. *
      g: i/ A7 Q$ ]9 F9 c+ y6 i6 x
  268. *                 调用此函数可以刷新全局变量ADS1115_RawData[4]的值。& H* F% _4 O2 x% E- E6 H
  269. *) t* d; \* Y4 s' u8 h
  270. *                 应当在一个定时器更新中断服务函数中周期性的调用此函数,更新周期最好小于200Hz" P: @  O8 l- `3 q8 t6 Q* \
  271. */. d: p' {3 `1 p
  272. void ADS1115_RefreshAllChannel()
    8 A6 i# {9 O, h  u. m
  273. {7 y6 P) B# j: A' t: [2 `- ^
  274.         static uint8_t channel = 0;
    1 T: g3 k+ j3 S2 \. T; U& R
  275.         int16_t adcDataTemp = 0;7 a$ K$ ?# R' x( f. p: V

  276. " W/ J$ P+ \2 G/ h  M
  277.         //通道切换时可能有不确定的数据读出,因此需要将前1~2次读出的数据舍弃
    ( [* ~/ k) ]7 O7 l$ Y3 m
  278.         ADS1115_ReadRawData(&adcDataTemp);
    / c6 H; b4 @/ M- a
  279.         ADS1115_ReadRawData(&adcDataTemp);
      ]" A5 w6 j! |* T% g) b" P- h; ?

  280. 6 a1 a/ E* C, I  W6 b/ \0 k* W# B7 J
  281.         //读取数据返回正确,则将读到的数据写入ADS1115_RawData数组中
    0 d# C/ G! s3 X2 _+ \, ]( l3 ]- d
  282.         if( ADS1115_ReadRawData(&adcDataTemp) !=0 )
    2 L& _# J$ q5 _$ E; b* J
  283.         {- U$ z7 D7 E% ^3 N( H& S2 e
  284.                 ADS1115_RawData[channel] = adcDataTemp;4 Y$ n1 f+ z4 K2 f
  285.         }( b2 T5 w# i: ]
  286.         
    # M/ o9 J4 V* |0 i/ Z
  287.         //ADS1115总共4个通道: b. d7 Y3 [' ]9 ~( t9 {
  288.         channel++;
    , k" D8 j6 S7 a( m- I
  289.         
    / g- Q) g+ {# f9 R
  290.         if(channel>ADS1115_MAX_CHANNEL-1)
    7 r" x1 ?" ]9 N+ I& G$ ?
  291.                 channel = 0;
      E, ?; n! X# t. k0 U1 _, Z4 g
  292. 9 F& I8 y0 j" w
  293.         //结束采样后切换至下一通道
    3 q( W3 Q8 r4 |, W
  294.         ADS1115_ScanChannel(channel);
    5 C9 U' |# ?2 I$ q* p
  295. }8 e0 I$ I) @( v( l9 n' h5 b6 J
  296. #endif  \* j4 S  m$ x
复制代码

" x0 @* @6 ~6 E: ^使用指南, r2 u( L2 i2 Z% z; o! c& `
基本步骤
* L3 L: Q# Q5 n. m0 z: E2 d
初始化软件模拟I2C或者硬件I2C外设(以笔者编写的软件模拟I2C库为例)5 M$ K9 B5 P1 T! A$ v
  1.         I2C_Virtual_ConfigPort(ADS1115_SDA_PORT, ADS1115_SDA_PIN, ADS1115_SCL_PORT, ADS1115_SCL_PIN);( u* ?0 C, y* M: ]2 T4 m8 H5 Y
  2.         I2C_Virtual_SwitchBus(ADS1115_SDA_PORT, ADS1115_SDA_PIN, ADS1115_SCL_PORT, ADS1115_SCL_PIN);
复制代码

: z# r4 _9 D; k% M; r! I& c初始化ADS1115芯片的配置。笔者在驱动库中提供了两种初始化配置函数void ADS1115_UserConfig1()与void ADS1115_UserConfig2(),可以修改这两个函数中的配置参数后直接调用。
- i/ B# A/ X, S! [
  1.         ADS1115_UserConfig2();
复制代码
6 N4 x: ?+ q: K9 c/ Z, q
设置或者切换ADS1115的通道(配置完ADS1115之后,可以设置一个初始通道)。
7 N" {" ?" K# R( W, j) y
  1.         ADS1115_ScanChannel(1);
复制代码
) g: ]( f. s" X" x' Y+ g
根据项目的需要,选择采集原始数据然后在其他地方换算成电压数据,或者直接采集电压数据以及滤波后的电压数据。" [# f# A6 c! l: O" y
  1.         int16_t rawData = 0;" K( R0 Y9 v% g5 s& R
  2.         float voltage = 0;
    : s) n1 r. P/ J) O
  3.         & o+ P- A" E- S3 ]+ Q6 I% R1 G
  4.         //获取ADS1115采集的原始16位数据  F1 |+ p2 _. Z+ q+ l
  5.         ADS1115_ReadRawData(&rawDAta);
    9 }3 k( A6 `$ ~
  6.         //将ADS1115采集的原始16位数据换算为实际电压值
    % I% Y' I1 Z( `# I' O, |& i$ w/ O
  7.         voltage = ADS1115_RawDataToVoltage(rawData);# O3 |7 O8 L! D

  8. & w6 q  A! C; t" J
  9.         //直接获取ADS1115采集的电压数据
    4 c" o+ z, w! l0 e5 y7 e* O' c
  10.         voltage = ADS1115_GetVoltage();
    - t# K" Q+ @5 M2 a$ x8 `
  11.         ( |* r# W- {: [$ \8 z8 ]
  12.         //直接获取ADS1115采集的电压数据(经过多次采样计算平均值)( m! `- A/ j+ I& u$ @
  13.         voltage = ADS1115_GetAverageVoltage(10);9 T) s, M. N" x% E. N0 H
复制代码
3 n* n) j2 C, W& Z* B7 k
注意事项
7 ?: B" S; h% |不使用实时操作系统

: c% |- A4 t0 ~- s对于多通道采样,每次通道切换时,应当等待几毫秒的时间后再进行采样,否则采样的数据可能不稳定或者发生通道间干扰。7 }( E1 ?! u9 v2 q+ [
  1.         int32_t ADC_Temp_Filter[4] = {0};
    6 T$ t* {% H8 J
  2.         float ADC_Temp[4] = {0};3 D' @# i5 t; O& X, k8 j6 D8 P
  3.         ; [! ~# t) M3 a% x' V
  4.         for (uint8_t chan = 0; chan < 4; chan++)
    : Z* j2 ?. m* K1 v% U6 N
  5.         {9 t3 M* g# i' E* F/ [1 G- R- v
  6.                 ADS1115_ScanChannel(chan);9 g# ^1 ?) p& [& m6 K# D& x
  7.                 delay_ms(5);: z- X+ |# [- f: l3 J
  8.                 ADS1115_ReadRawData(&dataTemp);1 P  U9 k! L9 M4 r% |0 f, w/ N
  9.                 //Keep 3 decimal places and filter
    ' E! o$ B/ F3 P+ i- y
  10.                 ADC_Temp_Filter[chan] = (int32_t)(ADS1115_GetVoltage()*1000);
    - E0 h9 a. `# y: k8 L! R8 V
  11.                 ADC_Temp[chan] = Filter_MovingAverage(ADC_Temp_Filter[chan],chan)/1000.0;
    % d; t6 ?- Y( w/ K
  12.         }
复制代码
( N  q( u7 x4 J7 x/ d3 I
由于多通道切换的通道稳定等待时间的存在,上面的代码耗时将会很长(>20ms)。如果在中断服务函数中执行了这些代码,将会导致单片机的性能严重下降。因此,需要考虑在定时器更新中断服务函数中周期性的调用void ADS1115_RefreshAllChannel()。每次调用将刷新一个通道的数据,四次调用便可以将全部通道的数据都刷新一遍,大大提高了代码的运行效率。示例如下:
' q+ l' Z- S# b# m( K% p5 ?" h$ ~! y+ o: d
软件定时器1以40Hz的频率周期性的调用void ADS1115_RefreshAllChannel(),以获得10Hz的全通道数据刷新速度。
; ^3 {- R( A1 S3 b" P) k- c, t: ?2 h9 I3 q软件定时器0以10Hz的频率处理ADS1115的原始采样数据(换算成电压、滤波、校正等)与其他传感器的数据。9 e, I' I: l6 `* C# Q# G
通过多个更新周期不同的定时器的配合,可以在无代码延时的情况下解决不同传感器数据采集速度不一致的问题,周期性的调用代替了延时等待。& n* s+ X* e- Y4 ~( {% F$ S
  1. /**: }" q4 l4 N( c7 [9 V7 t, _; e
  2. * @brief 设置软件定时器,主要包括初始化与开启各个软件定时器+ J% r# J; A+ t# Z5 U
  3. */
    . }+ R% H* o: k+ g" h' C
  4. void User_SetupTimer()
    $ |+ W1 p1 d) I2 u8 E" M6 S; Z& P
  5. {
    9 H  Y& h5 ?0 Q
  6.         //Poll frequency of software timer is 1KHz
    ) A' Y, @: E; [/ u  P& k
  7. ) k7 U& E6 Q) b* I2 Z2 m
  8.         //SW_Timer0 for refresh sensor data and modbus action.         Freq. = 10Hz
    * B0 \' n& s: o: S$ O
  9.         SoftwareTimer_Init(0, 100, User_SoftwareTimer0_Handler, -1);
    $ O* F( Y. e3 s0 _

  10. . a: `" x: c; v7 Q& r/ \
  11.         //SW_Timer1 for ads1115 data acquisition.        Freq. = 40Hz
    3 F4 D3 j1 }) J
  12.         SoftwareTimer_Init(1, 25, User_SoftwareTimer1_Handler, -1);. U0 W( `) z$ u8 U
  13. & z0 m  K: Q* I1 D- a. `$ y9 q. a! z
  14.         //SW_Timer2 for RS485 scan.                Freq. = 10Hz
    5 @3 @7 c# B! d  @$ Q" g
  15.         SoftwareTimer_Init(2, 100, User_SoftwareTimer2_Handler, -1);4 ~) V& v- K: o; ]1 @( O5 j  Y
  16. ; S  [8 `4 H  I
  17.         //SW_Timer3 for MTSICS.                Freq. = 10Hz
    7 n* [; I5 \7 w9 R% @9 o
  18.         SoftwareTimer_Init(3, 100, User_SoftwareTimer3_Handler, -1);3 J9 X& q- d$ z) `
  19. , w! p3 J) r: F4 {5 a, n+ P
  20.         //SW_Timer4 for RS100 (Printer has not been used).                Freq. = 10Hz
    $ ?9 J0 ^: K/ L( g& v% Y" U" K# @
  21.         SoftwareTimer_Init(4, 100, User_SoftwareTimer4_Handler, -1);
    6 R. t- Z2 D: v2 d
  22. ; U8 b3 ?" O6 [. {: p8 Z/ X
  23.         SoftwareTimer_Enable(0);
    5 O6 y& G7 _+ z1 Y  o
  24.         SoftwareTimer_Enable(1);8 |* {& q: G$ {# ~; K. G* J- p5 N
  25.         SoftwareTimer_Enable(2);
    , O! Q  Z; D: i5 S% F6 c* T" f7 R
  26.         SoftwareTimer_Enable(3);
    5 e. C, [) N" O; }; s4 V; ]
  27.         SoftwareTimer_Enable(4);  Q9 `1 d1 ]" T6 U$ D% S& p
  28. }
    4 b. S/ _/ u+ B0 H  N0 s5 M

  29. 7 j8 {7 e7 i. j
  30. # Q4 s* D2 D3 p/ Y
  31. /**
    0 u; E9 R; a% n3 }* s2 {
  32. * @brief 软件定时器1的更新服务函数. p! N+ B- D& ^5 z, W9 h  e* {- U* y
  33. */
    + a8 b% F, r* M  R
  34. void User_SoftwareTimer0_Handler()
    4 U; {" \' o3 Y5 Z% s
  35. {
    2 b' r& }5 o) `
  36.         User_RefreshData();: U( A' T: |& f  G0 E
  37.         User_RefreshAction();. M6 y. M3 ^, R4 k3 j/ z+ J% ]8 p# y
  38. }$ P  e8 b) ^; M* E
  39. 9 K" J+ ]3 Q; {# I) @
  40. 8 h0 i/ c6 A# w* i, W) ~7 I( B
  41. /**
    0 g8 J  ?" P6 g7 A( l
  42. * @brief 软件定时器2的更新服务函数2 z! [# n4 U, z1 W" T+ S
  43. */: c! n& @9 j1 v% u6 B2 `; @& x
  44. void User_SoftwareTimer1_Handler()
      e" w1 g' `) j9 y7 F( I
  45. {! C( k0 f1 J1 z
  46.         ADS1115_RefreshAllChannel();1 y1 U/ X- }* t! ?, n
  47. }% c# m9 W9 ~5 u& M6 t# a, x

  48. + _5 f- U# u% B- B- R; {
  49. /**. K/ O8 i: d9 e% W( H( n3 Z/ }4 G* o' N
  50. * @brief This function is called by timer update interrupt handler, which will be executed periodically.
    $ S4 q) O- T' X0 h
  51. */
    ' r' a, k  z- _) O- t
  52. void User_RefreshData()- N# C7 l6 G  M5 d9 K
  53. {( K$ A4 i7 X7 G$ T
  54.         for (uint8_t chan = 0; chan < 4; chan++)
    ! k/ W% G- C# b% H) i8 ^, W, A9 z
  55.         {
    & Y1 k, S4 y: n: T$ |
  56.                 ADC_Voltage[chan] = Filter_MovingAverage(ADS1115_Data[chan]*10,chan)/10.0;                        //unit: mV, resolution: 0.1 mV/ m& D2 T! Y' ^" _
  57.         }# _: n8 M3 _2 G

  58. 4 P' V* f, V$ G
  59.         SensorData[0].value = OmronEC55_PV;
    1 G& W' R% t6 ]! w9 r0 a" n  F% g' H
  60.         SensorData[1].value = SGDAQ_PV;7 m1 |  z' m: W1 L: H7 x$ |0 j
  61.         SensorData[2].value = (float)RS100_Count*0.0005;
    ) \( q6 F* j" a6 t. c- D: h
  62.         SensorData[3].value = MTSICS_Weight;
    - h" D5 `4 j0 w, O8 S" a

  63. 0 T8 {% Y' U! G+ @; L5 A
  64.         //Pt100 - 1 PV (℃)
    . i7 r6 x: O$ R! Q% C* o. \
  65.         SensorData[4].value = Pt100_RtoT(User_VoltageToResistance(ADC_Voltage[0]));
    * Z! q! P* e. c; A; T9 b( l
  66.         SensorData[5].value = ADC_Voltage[1];
    ) K4 _; t* }3 F$ v7 z- [
  67.         SensorData[6].value = ADC_Voltage[2];
    8 y  Z. G$ f2 O8 W6 E- i9 T" q
  68.         SensorData[7].value = ADC_Voltage[3];6 b) x8 Z6 {; a$ _6 t! p

  69. 9 x6 E1 Q& L) y2 q
  70.         //校正后的SGDAQ_PV
    0 g) H% a) ]' `/ N6 c. e" o
  71.         SensorData[8].value = SensorData[1].value * HoldingReg_GetData(30) + HoldingReg_GetData(31);
    " p4 _; v, ]" d1 u+ `9 q
  72. }
复制代码
8 ^! d) W3 \1 ?# {
在两个或者多个ADS1115组合使用的情况下,对于多通道数据的采样处理应该遵循与上面相似的策略。以两个ADS1115通过软件模拟I2C组成的8通道数据采集功能为例:7 P# f" ?/ u7 Z, ?
volatile int16_t User_ADS1115Data[8];
; F( _4 O& v, m4 e: h) m& q, l$ ]/ j) {
  1. /**$ I2 P* z! o& ~) s* f
  2. * @brief 软件定时器2的服务函数
    ; y" ?3 c9 W* ^) b8 C% g! t
  3. *                 主要实现了两个ADS1115(总共8路)的数据采集功能。
    * t; {/ P3 q6 a' ~' [5 L! `' ]
  4. */
    " G2 s: M/ L1 U( Z9 H
  5. void User_SoftwareTimer2_Handler()
    ) C) h0 y" O: Y# K) ~' C
  6. {
    ; P$ M7 c  l: F4 E+ h" H
  7.         static uint8_t chan = 0;
    4 [4 o3 V, c# e5 l% X$ O
  8.         static uint8_t isData1Copyed = 0;2 ?* a0 s) X$ M: C
  9.         static uint8_t isData2Copyed = 0;! \2 N$ j1 R1 |9 Q
  10.   t) w1 M2 f* p3 c1 G4 C
  11.         //Operate ADS1115-19 S, Q+ q! A; r( G1 ]4 D
  12.         if(chan < 4)
    2 L5 }8 {3 F/ D) E; j0 Q+ p
  13.         {
    ! K, m$ H$ E0 X+ A& H6 b4 h5 [
  14.                 if(isData1Copyed == 0)8 V2 q0 f# a/ m; @4 z! S
  15.                 {
    - q; B: }3 `7 a7 s- s, @' ^0 Q7 {
  16.                         //将ADS1115-2采样的数据复制到User_ADS1115Data[]中% N8 I4 N: o3 m$ g. R
  17.                         for(uint8_t i = 0; i <4; i++)
    . f8 u4 Y+ s9 q
  18.                         {; u7 l$ k$ H0 \. }
  19.                                 User_ADS1115Data[i+4] = ADS1115_RawData<i>;
    ! |! p; R* z# s- }
  20.                         }0 i/ s2 n+ j. q- i7 |/ B
  21.                         isData1Copyed = 1;
    7 d) o3 I. Q+ g
  22.                         isData2Copyed = 0;' s% g' x1 \( p$ n; Z+ I4 P, m7 a
  23.                 }
    ! m' _& l; Q$ b0 p6 A9 P
  24. - j7 X# m( B! D, z; u9 Z
  25.                 //切换I2C总线至ADS1115-19 ?+ N4 [  N5 W+ \; q1 Q6 @
  26.                 I2C_Virtual_SwitchBus(ADS1115_SDA_PORT, ADS1115_SDA_PIN, ADS1115_SCL_PORT, ADS1115_SCL_PIN);
    - e+ N: S2 X- N/ M. d, l) M

  27. . b. t. n6 {8 m
  28.                 ADS1115_RefreshAllChannel();# u2 c+ Y+ N( a* e& K! N* i( E# b( n( q
  29.         }3 H2 v) E( F3 x0 R! a
  30.         //Operate ADS1115-2
    ( a" I4 j3 i9 X5 D" f, O
  31.         else  i$ m7 _. W; a* G- q1 f- z
  32.         {2 n  ]$ E/ u' q% g( O) x
  33.                 if(isData2Copyed == 0)
    ) I# E7 J# {# [) _9 H1 j
  34.                 {
    ' B7 l/ R4 R- i. ~% c- ~6 C
  35.                         //将ADS1115-1采样的数据复制到User_ADS1115Data[]中9 X2 J  d3 P* y+ X' E7 }* O7 o# ^
  36.                         for(uint8_t i = 0; i <4; i++)5 l2 \& H$ p3 P
  37.                         {4 n9 U5 f+ }/ {
  38.                                 User_ADS1115Data<i> = ADS1115_RawData<i>;
    % E) e% a  P5 g' H
  39.                         }0 F9 t( p" D  n. u
  40.                         isData2Copyed = 1;* Q! f. S% S. V
  41.                         isData1Copyed = 0;
    " u/ m" J2 u7 Y3 J4 O) V
  42.                 }
    - E6 C; M6 h3 H/ ~5 g

  43. " u) b7 U0 V- l* W3 y
  44.                 //切换I2C总线至ADS1115-21 J/ l$ f/ t3 k! O& {9 D2 t$ }
  45.                 I2C_Virtual_SwitchBus(ADS1115_SDA_PORT_2, ADS1115_SDA_PIN_2, ADS1115_SCL_PORT_2, ADS1115_SCL_PIN_2);; u% r# q; f5 l% G8 v0 Z  Z9 z+ k

  46. * s/ K, }- K9 o: x7 b6 d
  47.                 ADS1115_RefreshAllChannel();
    + E, E4 d5 o$ K2 D% I9 T
  48.         }# L6 p% l* o' S5 m: |. e3 z6 l
  49. - ~, s3 a. `$ I# B3 M8 ^
  50.         chan++;% t* Y' h" t% E* g
  51.         if (chan > 7)2 J( {% n5 o) ?1 @/ F
  52.         {/ B$ M1 W& i" n  H& s$ B
  53.                 chan = 0;
    5 m! e; C( Z7 C$ u1 j; j5 n
  54.         }" S6 z5 Q/ K9 A7 S/ U+ z2 J6 k
  55. }</i></i></i>
复制代码
2 t' W$ H; T) _% M5 L
使用实时操作系统
0 I3 t: g" M' X" d8 |在实时操作系统中使用延时函数时,任务调度器会自动切换执行低优先级任务,因此延时函数(由操作系统提供的API)不存在浪费CPU资源的问题。此时一般的做法是建立一个ADS1115的数据采集任务,在这个任务中轮询采集需要的数据。
$ l" Q6 p" T+ O6 i) f, @. [$ x( i& m4 [* \+ z/ k; P# [( _* T6 G
  1. void ADS1115Daq_task(void *pvParameters)  K, W" g1 w* V/ M' a( |4 N& p& f
  2. {
    ) y; G5 b% H" ]& l
  3.         int16_t adcDataTemp[4] = {0};
    ) V" Y( g, b9 t: o. @7 ~
  4.         TickType_t ticks = xTaskGetTickCount();
    0 N4 \# s, l, j- q& f) r& ?6 m

  5. ' d7 `9 C. u: h" p0 S. A5 c) V
  6.         //配置ADS1115端口,由于采用了软件I2C通讯,因此可以直接调用虚拟I2C中的配置函数完成配置
      M" A/ d* A+ ?/ Q) M. v9 d$ S' O
  7.         I2C_Virtual_ConfigPort(ADS1115_SDA_PORT, ADS1115_SDA_PIN, ADS1115_SCL_PORT, ADS1115_SCL_PIN);, U$ W) ?# M( q! h1 c% i1 c
  8.         I2C_Virtual_SwitchBus(ADS1115_SDA_PORT, ADS1115_SDA_PIN, ADS1115_SCL_PORT, ADS1115_SCL_PIN);
    0 v: Z9 G# f- l5 w( \
  9.         ' y0 B3 ^* L2 u# d
  10.         //使用内置的快速配置模板完成ADS1115的配置
    3 S1 M9 A; k' w
  11.         ADS1115_UserConfig2();
    ( e: b1 m" j/ t
  12. ; a3 Z  O& O' ~- |9 c  ]  f$ ?4 [
  13.         while (1)% D* U4 w" w6 }0 H/ r9 p% \
  14.         {5 P! w5 g0 N$ [
  15.                 for (uint8_t chan = 0; chan < 4; chan++)9 e+ m0 p! A. h3 k  r( V" v2 m
  16.                 {
    ) j' P2 S, r4 v- f3 J  a, \4 ?0 z/ ?
  17.                         //设置ADS1115的采样通道
    4 {3 V/ L% [& S% r0 s9 j# _5 b
  18.                         ADS1115_ScanChannel(chan);
    1 q" L  X) [2 b) w! a+ [
  19.                         //调用RTOS提供的API延时10ms
    2 W: S1 ~& w1 q% E# k0 f
  20.                         vTaskDelay(10);$ {# h( X4 D1 C2 \

  21. 1 u0 {! G! O, L& h$ Q) r0 M7 v
  22.                         if(ADS1115_ReadRawData(&adcDataTemp[chan])!=0)
    % A* |5 M2 n# L3 \9 ^8 j8 Q' G
  23.                         {& P) r/ R! B% O& f
  24.                                 //保留小数点后三位精度3 y" n; f' A) T0 m
  25.                                 SensorData[chan].value = (float)(ADS1115_RawDataToVoltage(adcDataTemp[chan])*1000)/1000.0;$ k0 P' f, h1 T; s6 i  k7 l4 Z
  26.                         }
    : m2 ]% b: |1 r
  27.                 }* Q6 w* ^! Q: P  i: P( T5 {3 |( s0 U
  28.                 //100ms 一个处理周期
    , w, O& u8 ]( y2 l  d0 G. Q
  29.                 vTaskDelayUntil( &ticks, 100);
      Y5 y% t, r) d, ]! F
  30.         }
    : q5 p0 x3 o' l5 {$ k6 R  G  z
  31. }
    ! I6 O: H, p# j/ H

  32. * v. Y4 j; u! r+ C. W+ s
  33. 9 c$ s+ c, i. v- A5 P4 A6 p$ P0 ]
复制代码
3 T4 I& c3 L0 o* e; z
收藏 1 评论0 发布时间:2022-4-13 18:00

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版