forked from parallaxinc/spin-standard-library
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsensor.power.ina260.spin
413 lines (355 loc) · 13.6 KB
/
sensor.power.ina260.spin
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
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
{
---------------------------------------------------------------------------------------------------
Filename: sensor.power.ina260.spin
Description: Driver for the TI INA260 Precision Current and Power Monitor IC
Author: Jesse Burt
Started: Nov 13, 2019
Updated: Feb 27, 2024
Copyright (c) 2024 - See end of file for terms of use.
---------------------------------------------------------------------------------------------------
}
#include "sensor.power.common.spinh"
CON
{ default I/O settings; these can be overridden in the parent object }
SCL = DEF_SCL
SDA = DEF_SDA
I2C_FREQ = DEF_HZ
I2C_ADDR = DEF_ADDR
SLAVE_WR = core.SLAVE_ADDR
SLAVE_RD = core.SLAVE_ADDR|1
DEF_SCL = 28
DEF_SDA = 29
DEF_HZ = 100_000
DEF_ADDR = %0000
I2C_MAX_FREQ = core.I2C_MAX_FREQ
' Address pins vs slave addresses
' A1 A0 SLAVE ADDRESS
' GND GND 100_0000
' GND VS+ 100_0001
' GND SDA 100_0010
' GND SCL 100_0011
' VS+ GND 100_0100
' VS+ VS+ 100_0101
' VS+ SDA 100_0110
' VS+ SCL 100_0111
' SDA GND 100_1000
' SDA VS+ 100_1001
' SDA SDA 100_1010
' SDA SCL 100_1011
' SCL GND 100_1100
' SCL VS+ 100_1101
' SCL SDA 100_1110
' SCL SCL 100_1111
' Operating modes
POWERDN = %000
CURR_TRIGD = %001
VOLT_TRIGD = %010
CURR_VOLT_TRIGD = %011
POWERDN2 = %100
CURR_CONT = %101
VOLT_CONT = %110
CURR_VOLT_CONT = %111
' Interrupt/alert pin sources
INT_CURRENT_HI = 1 << 5
INT_CURRENT_LO = 1 << 4
INT_BUSVOLT_HI = 1 << 3
INT_BUSVOLT_LO = 1 << 2
INT_POWER_HI = 1 << 1
INT_CONV_READY = 1 << 0
' Interrupt/alert pin level/polarity
INTLVL_LO = 0
INTLVL_HI = 1
VAR
long _addr_bits
OBJ
#ifdef INA260_I2C_BC
i2c: "com.i2c.nocog"
#else
i2c: "com.i2c"
#endif
core: "core.con.ina260"
time: "time"
PUB null()
' This is not a top-level object
PUB start(): status
' Start using default I/O settings
return startx(SCL, SDA, I2C_FREQ, I2C_ADDR)
PUB startx(SCL_PIN, SDA_PIN, I2C_HZ, ADDR_BITS): status
' Start using custom settings
if ( lookdown(SCL_PIN: 0..31) and lookdown(SDA_PIN: 0..31) and ...
lookdown(ADDR_BITS: %0000..%1111) )
if ( status := i2c.init(SCL_PIN, SDA_PIN, I2C_HZ) )
time.usleep(core.T_POR)
_addr_bits := (ADDR_BITS << 1)
reset()
if ( dev_id() == core.DEVID_RESP )
return
' if this point is reached, something above failed
' Double check I/O pin assignments, connections, power
' Lastly - make sure you have at least one free core/cog
return FALSE
PUB stop()
' Stop the driver
i2c.deinit()
_addr_bits := 0
PUB adc2amps(adc_word): a
' Convert current ADC word to amperage
' Returns: current in microamperes
return (~~adc_word) * 1_250
PUB adc2volts(adc_word): v
' Convert bus voltage ADC word to voltage
' Returns: voltage in microvolts
return (adc_word & $7fff) * 1_250
PUB adc2watts(adc_word): w
' Convert power ADC word to wattage
' Returns: power in microwatts
return (adc_word * 10_000)
PUB current_data(): a
' Read the measured current, in microamperes
' NOTE: If averaging is enabled, this will return the averaged value
a := 0
readreg(core.CURRENT, 2, @a)
PUB current_conv_time(ctime): curr_set
' Set conversion time for shunt current measurement, in microseconds
' Valid values: 140, 204, 332, 588, *1100, 2116, 4156, 8244
' Any other value polls the chip and returns the current setting
curr_set := 0
readreg(core.CONFIG, 2, @curr_set)
case ctime
140, 204, 332, 588, 1100, 2116, 4156, 8244:
ctime := lookdownz(ctime: 140, 204, 332, 588, 1100, 2116, 4156, 8244) << core.ISHCT
other:
curr_set := (curr_set >> core.ISHCT) & core.ISHCT_BITS
return lookupz(curr_set: 140, 204, 332, 588, 1100, 2116, 4156, 8244)
ctime := ((curr_set & core.ISHCT_MASK) | ctime) & core.CONFIG_MASK
writereg(core.CONFIG, 2, @ctime)
PUB dev_id(): id
' Read device ID
' Returns:
' Most-significant word: Die ID
' Least-significant word: Mfr ID
id := 0
return (die_id() << 16) | mfr_id()
PUB die_id(): id
' Read the Die ID from the chip
' Returns: $2270
id := 0
readreg(core.DIE_ID, 2, @id)
PUB int_clear() | tmp
' Clear active interrupt
' NOTE: This method is only effective when interrupts are latched (int_latch_ena() == true)
readreg(core.ENABLE, 1, @tmp) ' simply reading this reg clears a latched int
PUB int_polarity(state): curr_state
' Set interrupt active level/polarity
' Valid values:
' *INTLVL_LO (0) Active low
' INTLVL_HI (1) Active high
' Any other value polls the chip and returns the current setting
' NOTE: The ALERT pin is open collector
curr_state := 0
readreg(core.ENABLE, 2, @curr_state)
case state
INTLVL_LO, INTLVL_HI:
state <<= core.APOL
other:
return ((curr_state >> core.APOL) & 1)
state := ((curr_state & core.APOL_MASK) | state) & core.ENABLE_MASK
writereg(core.ENABLE, 2, @state)
PUB int_mask(mask): curr_mask
' Set interrupt mask
' Valid values:
' Bits: 5..0
' 5: Over current limit
' 4: Under current limit
' 3: Bus voltage over-voltage
' 2: Bus voltage under-voltage
' 1: Power over-limit
' 0: Conversion ready
' Any other value polls the chip and returns the current setting
' NOTE: Only one source may be enabled at a time. If more than one source is enabled,
' only the function in the higher significant bit position will be used.
' NOTE: INT_ symbols can optionally be used to define sources
' (defined near top of this file)
curr_mask := 0
readreg(core.ENABLE, 2, @curr_mask)
case mask
%000000..%111111:
mask <<= core.ALERTS
other:
return (curr_mask >> core.ALERTS) & core.ALERTS_BITS
mask := ((curr_mask & core.ALERTS_MASK) | mask) & core.ENABLE_MASK
writereg(core.ENABLE, 2, @mask)
PUB int_latch_ena(state): curr_state
' Enable latching of interrupts
' Valid values:
' TRUE (-1 or 1): Active interrupts remain asserted until cleared manually
' FALSE (0): Active interrupts clear when the fault has been cleared
curr_state := 0
readreg(core.ENABLE, 2, @curr_state)
case ||(state)
0, 1:
state := ||(state) & 1
other:
return ((curr_state & 1) == 1)
state := ((curr_state & core.LEN_MASK) | state) & core.ENABLE_MASK
writereg(core.ENABLE, 2, @state)
PUB int_set_thresh(thresh)
' Set interrupt/alert threshold
' Valid values:
' Current: 0..15_000_000 (microamps; clamped to range)
' Voltage: 0..36_000_000 (microvolts; clamped to range)
' Power: 540_000_000 (microwatts; clamped to range)
case int_mask(-2) ' determine scaling based on interrupt type set
INT_CURRENT_HI, INT_CURRENT_LO:
{ current low or high threshold }
thresh := (0 #> thresh <# 15_000000) / 1_250
INT_BUSVOLT_HI, INT_BUSVOLT_LO:
{ voltage low or high threshold }
thresh := (0 #> thresh <# 36_000000) / 1_250
INT_POWER_HI:
{ power high threshold }
thresh := (0 #> thresh <# 540_000000) / 10_000
writereg(core.ALERT_LIMIT, 2, @thresh)
PUB int_thresh(): thresh
' Get interrupt/alert threshold
thresh := 0
readreg(core.ALERT_LIMIT, 2, @thresh)
case int_mask(-2) ' determine scaling based on interrupt type set
INT_CURRENT_HI, INT_CURRENT_LO, INT_BUSVOLT_HI, INT_BUSVOLT_LO:
{ current or voltage, low or high threshold }
return (thresh * 1_250)
INT_POWER_HI:
{ power high threshold }
return (thresh * 10_000)
other:
{ no interrupt mask set; just return the reg value unscaled }
return thresh
PUB mfr_id(): id
' Read the Manufacturer ID from the chip
' Returns: $5449
id := 0
readreg(core.MFR_ID, 2, @id)
PUB opmode(mode): curr_mode
' Set operation mode
' Valid values:
' POWERDN (0): Power-down/shutdown
' CURR_TRIGD (1): Shunt current, triggered
' VOLT_TRIGD (2): Bus voltage, triggered
' CURR_VOLT_TRIGD (3): Shunt current and bus voltage, triggered
' POWERDN2 (4): Power-down/shutdown
' CURR_CONT (5): Shunt current, continuous
' VOLT_CONT (6): Bus voltage, continuous
' *CURR_VOLT_CONT (7): Shunt current and bus voltage, continuous
' Any other value polls the chip and returns the current setting
curr_mode := 0
readreg(core.CONFIG, 2, @curr_mode)
case mode
POWERDN, CURR_TRIGD, VOLT_TRIGD, CURR_VOLT_TRIGD, POWERDN2, CURR_CONT, VOLT_CONT, ...
CURR_VOLT_CONT:
mode := lookdownz(mode: POWERDN, CURR_TRIGD, VOLT_TRIGD, CURR_VOLT_TRIGD, POWERDN2, ...
CURR_CONT, VOLT_CONT, CURR_VOLT_CONT)
other:
curr_mode &= core.MODE_BITS
return curr_mode
mode := ((curr_mode & core.MODE_MASK) | mode) & core.CONFIG_MASK
writereg(core.CONFIG, 2, @mode)
PUB power_data(): p
' Read the power measured by the chip, in microwatts
' NOTE: If averaging is enabled, this will return the averaged value
' NOTE: The maximum value returned is 419_430_000
p := 0
readreg(core.POWER, 2, @p)
PUB power_data_rdy(): flag
' Flag indicating data from the last conversion is available for reading
' Returns: TRUE (-1) if data available, FALSE (0) otherwise
flag := 0
readreg(core.ENABLE, 2, @flag)
return ((flag & core.DRDY) <> 0)
PUB power_overflowed(): flag
' Flag indicating power data exceeded the maximum measurable value (419_430_000uW or 419.43W)
flag := 0
readreg(core.ENABLE, 2, @flag)
return ((flag & core.OVERFL) <> 0)
PUB reset() | tmp
' Reset the chip
' NOTE: Equivalent to Power-On Reset
tmp := core.SOFT_RES
writereg(core.CONFIG, 2, @tmp)
PUB samples_avg(samples=-2): curr_smp
' Set number of samples used for averaging measurements
' Valid values: *1, 4, 16, 64, 128, 256, 512, 1024
' Any other value polls the chip and returns the current setting
curr_smp := 0
readreg(core.CONFIG, 2, @curr_smp)
case samples
1, 4, 16, 64, 128, 256, 512, 1024:
samples := lookdownz(samples: 1, 4, 16, 64, 128, 256, 512, 1024) << core.AVG
other:
curr_smp := (curr_smp >> core.AVG) & core.AVG_BITS
return lookupz(curr_smp: 1, 4, 16, 64, 128, 256, 512, 1024)
samples := ((curr_smp & core.AVG_MASK) | samples) & core.CONFIG_MASK
writereg(core.CONFIG, 2, @samples)
PUB voltage_data(): v
' Read the measured bus voltage, in microvolts
' NOTE: If averaging is enabled, this will return the averaged value
' NOTE: Full-scale range is 40_960_000uV
v := 0
readreg(core.BUS_VOLTAGE, 2, @v)
PUB voltage_conv_time(ctime): curr_time
' Set conversion time for bus voltage measurement, in microseconds
' Valid values: 140, 204, 332, 588, *1100, 2116, 4156, 8244
' Any other value polls the chip and returns the current setting
curr_time := 0
readreg(core.CONFIG, 2, @curr_time)
case ctime
140, 204, 332, 588, 1100, 2116, 4156, 8244:
ctime := lookdownz(ctime: 140, 204, 332, 588, 1100, 2116, 4156, 8244) << core.VBUSCT
other:
curr_time := (curr_time >> core.VBUSCT) & core.VBUSCT_BITS
return lookupz(curr_time: 140, 204, 332, 588, 1100, 2116, 4156, 8244)
ctime := ((curr_time & core.VBUSCT_MASK) | ctime) & core.CONFIG_MASK
writereg(core.CONFIG, 2, @ctime)
PRI readreg(reg_nr, nr_bytes, ptr_buff) | cmd_pkt
' Read nr_bytes from the slave device into ptr_buff
case reg_nr
$00..$03, $06, $07, $fe, $ff:
cmd_pkt.byte[0] := SLAVE_WR | _addr_bits
cmd_pkt.byte[1] := reg_nr
i2c.start()
i2c.wrblock_lsbf(@cmd_pkt, 2)
i2c.start()
i2c.write(SLAVE_RD | _addr_bits)
i2c.rdblock_msbf(ptr_buff, nr_bytes, i2c.NAK)
i2c.stop()
other:
return
PRI writereg(reg_nr, nr_bytes, ptr_buff) | cmd_pkt
' Write nr_bytes from ptr_buff to the slave device
case reg_nr
$00:
word[ptr_buff][0] |= core.RSVD_BITS
$06, $07:
other:
return
cmd_pkt.byte[0] := SLAVE_WR | _addr_bits
cmd_pkt.byte[1] := reg_nr
i2c.start()
i2c.wrblock_lsbf(@cmd_pkt, 2)
i2c.wrblock_msbf(ptr_buff, nr_bytes)
i2c.stop()
DAT
{
Copyright (c) 2024 Jesse Burt
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.
}