-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathfnirsi_logger.py
executable file
·353 lines (288 loc) · 11.4 KB
/
fnirsi_logger.py
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
#!/usr/bin/env python3
import sys
import os.path
import time
import usb.core
import usb.util
import argparse
from typing import Union
# FNB48
# Bus 001 Device 020: ID 0483:003a STMicroelectronics FNB-48
VID = 0x0483
PID_FNB48 = 0x003A
# iManufacturer 1 FNIRSI
# iProduct 2 FNB-48
# iSerial 3 0001A0000000
# C1
# Bus 001 Device 029: ID 0483:003b STMicroelectronics USB Tester
PID_C1 = 0x003B
# FNB58
VID_FNB58 = 0x2E3C
PID_FNB58 = 0x5558
# FNB48S
# Bus 001 Device 003: ID 2e3c:0049 FNIRSI USB Tester
VID_FNB48S = 0x2E3C
PID_FNB48S = 0x0049
def setup_crc():
import crc
# $ ./reveng -w 8 -s $(shuf -n 100 /tmp/dump2.txt)
# width=8 poly=0x39 init=0x42 refin=false refout=false xorout=0x00 check=0x4b residue=0x00 name=(none)
width = 8
poly = 0x39
init_value = 0x42
final_xor_value = 0x00
reverse_input = False
reverse_output = False
configuration = crc.Configuration(width, poly, init_value, final_xor_value, reverse_input, reverse_output)
if hasattr(crc, "CrcCalculator"): # crc 1.x
crc_calculator = crc.CrcCalculator(configuration, use_table=True)
return crc_calculator.calculate_checksum
else: # crc 2.x+
calculator = crc.Calculator(configuration, optimized=True)
return calculator.checksum
def find_device():
is_fnb58_or_fnb48s = False
dev = usb.core.find(idVendor=VID, idProduct=PID_FNB48)
if dev is None:
dev = usb.core.find(idVendor=VID, idProduct=PID_C1)
if dev is None:
dev = usb.core.find(idVendor=VID_FNB58, idProduct=PID_FNB58)
if dev:
is_fnb58_or_fnb48s = True
if dev is None:
dev = usb.core.find(idVendor=VID_FNB48S, idProduct=PID_FNB48S)
if dev:
is_fnb58_or_fnb48s = True
return dev, is_fnb58_or_fnb48s
def find_hid_interface_num(dev):
# https://github.com/pyusb/pyusb/issues/76#issuecomment-118460796
for cfg in dev:
for interface in cfg:
if interface.bInterfaceClass == 0x03: # HID class
return interface.bInterfaceNumber
def ensure_interface_not_busy(dev, interface):
if dev.is_kernel_driver_active(interface.bInterfaceNumber):
try:
dev.detach_kernel_driver(interface.bInterfaceNumber)
except usb.core.USBError as e:
print(f"Could not detatch kernel driver from interface({interface.bInterfaceNumber}): {e}", file=sys.stderr)
sys.exit(1)
def ensure_all_interfaces_not_busy(dev):
for cfg in dev:
for interface in cfg:
ensure_interface_not_busy(dev, interface)
def print_configs(dev):
for cfg in dev:
print("cfg", cfg.bConfigurationValue, file=sys.stderr)
for interface in cfg:
print(
" ",
"interface",
interface.bInterfaceNumber,
interface.bAlternateSetting,
file=sys.stderr,
)
for ep in interface:
print(" ", "ep", ep.bEndpointAddress, file=sys.stderr)
def print_configs_overview(dev):
for cfg in dev:
for interface in cfg:
print(interface)
# State for accumulating in decodee
energy = 0.0
capacity = 0.0
def decode(data, calculate_crc, time_interval, alpha):
global energy
global capacity
temp_ema = None
# Data is 64 bytes (64 bytes of HID data minus vendor constant 0xaa)
# First byte is HID vendor constant 0xaa
# Second byte is payload type:
# 0x04 is data packet
# Other types (0x03 and maybe other ones) is unknown
# Next 4 samples each 15 bytes. 60 bytes total.
# At the end 2 bytes:
# 1 byte is semi constant with unknown purpose.
# 1 byte (last) is a 8-bit CRC checksum
packet_type = data[1]
if packet_type != 0x04:
# ignore all non-data packets
# print("Ignoring")
return
if calculate_crc:
actual_checksum = data[-1]
expected_checksum = calculate_crc(bytearray(data[1:-1]))
if actual_checksum != expected_checksum:
print(
f"Ignoring packet of length {len(data)} with unexpected checksum. "
f"Expected: {expected_checksum:02x} Actual: {actual_checksum:02x}",
file=sys.stderr,
)
return
t0 = time.time() - 4 * time_interval
for i in range(4):
offset = 2 + 15 * i
# 4 + 4 + 2 + 2 + 2 + 1 (unknown constant 1) = 15 bytes
voltage = (
data[offset + 3] * 256 * 256 * 256
+ data[offset + 2] * 256 * 256
+ data[offset + 1] * 256
+ data[offset + 0]
) / 100000
current = (
data[offset + 7] * 256 * 256 * 256
+ data[offset + 6] * 256 * 256
+ data[offset + 5] * 256
+ data[offset + 4]
) / 100000
dp = (data[offset + 8] + data[offset + 9] * 256) / 1000
dn = (data[offset + 10] + data[offset + 11] * 256) / 1000
# unknown12 = data[offset + 12] # ? constant 1 # some PD info?
# It does not look to be a sign of current. I tried reversing
# USB-C-In and USB-C-Out, which does reversed orientation of the
# blue arrow for current on device screen, but unknown12 remains 1.
# print(f"unknown{offset+12} {unknown12:02x}")
temp_C = (data[offset + 13] + data[offset + 14] * 256) / 10.0
if temp_ema is not None:
temp_ema = temp_C * (1.0 - alpha) + temp_ema * alpha
else:
temp_ema = temp_C
power = voltage * current
# TODO(baryluk): This might be slightly inaccurate, if there is sampling jitter.
energy += power * time_interval
capacity += current * time_interval
t = t0 + i * time_interval
print(
f"{t:.3f} {i} {voltage:7.5f} {current:7.5f} {dp:5.3f} {dn:5.3f} "
f"{temp_ema:6.3f} {energy:.6f} {capacity:.6f}"
)
# unknown62 = data[62] # data[-2]
# print(f"unknown62 {unknown:02x}")
# print()
def request_data(is_fnb58_or_fnb48s, ep_out):
# Setup communication with power meter
ep_out.write(b"\xaa\x81" + b"\x00" * 61 + b"\x8e")
ep_out.write(b"\xaa\x82" + b"\x00" * 61 + b"\x96")
if is_fnb58_or_fnb48s:
ep_out.write(b"\xaa\x82" + b"\x00" * 61 + b"\x96")
else:
ep_out.write(b"\xaa\x83" + b"\x00" * 61 + b"\x9e")
# Argument handling
def str2bool(v: str) -> bool:
vl = v.lower()
if vl in ("yes", "true", "t", "1"):
return True
if vl in ("no", "false", "f", "0"):
return False
assert False, "Argument must be a boolean (true or false)"
def main():
parser = argparse.ArgumentParser(formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument("--crc", type=str2bool, help="Enable CRC checks", default=False)
parser.add_argument("--verbose", type=str2bool, help="Show some extra initalization information", default=False)
parser.add_argument("--alpha", type=float, help="Set temperature smoothing factor", default=0.9)
args = parser.parse_args()
crc_calculator = None
if args.crc:
try:
crc_calculator = setup_crc() # can be None
print("CRC checks enabled", file=sys.stderr)
except ModuleNotFoundError:
print("Warning: crc package not found. crc checks will not be performed", file=sys.stderr)
crc_calculator = None
except Exception as e:
print("Warning: crc calculator got exception: {e}, disabling crc checks", file=sys.stderr)
crc_calculator = None
# At the moment only 100 sps is supported
sps = 100
time_interval = 1.0 / sps
dev, is_fnb58_or_fnb48s = find_device()
assert dev, "Device not found"
if args.verbose:
if is_fnb58_or_fnb48s:
print("Found FNB48S or FNB58 device", file=sys.stderr)
else:
print("Found FNB48 or C1 device", file=sys.stderr)
dev.reset()
if args.verbose:
print("Configs:", file=sys.stderr)
print_configs(dev)
print("", file=sys.stderr)
if args.verbose:
print("Configs overview:", file=sys.stderr)
print_configs_overview(dev)
print("", file=sys.stderr)
interface_hid_num = find_hid_interface_num(dev)
if args.verbose:
print(f"Using interface_hid_num={interface_hid_num}", file=sys.stderr)
if args.verbose:
print(f"Ensuring all interfaces not busy and detaching kernel driver, if needed …", file=sys.stderr)
ensure_all_interfaces_not_busy(dev)
# if args.verbose:
# print(f"Claining interface …", file=sys.stderr)
# usb.util.claim_interface(dev, 0)
if args.verbose:
print(f"Setting configuration …", file=sys.stderr)
# Set the active configuration. With no arguments, the first
# configuration will be the active one
dev.set_configuration()
# Get an endpoint instance
cfg = dev.get_active_configuration()
intf = cfg[(interface_hid_num, 0)]
ep_out = usb.util.find_descriptor(
intf,
# match the first OUT endpoint
custom_match=lambda e: usb.util.endpoint_direction(e.bEndpointAddress) == usb.util.ENDPOINT_OUT,
)
ep_in = usb.util.find_descriptor(
intf,
# match the first IN endpoint
custom_match=lambda e: usb.util.endpoint_direction(e.bEndpointAddress) == usb.util.ENDPOINT_IN,
)
assert ep_in
assert ep_out
if args.verbose:
print(f"Starting data request …", file=sys.stderr)
request_data(is_fnb58_or_fnb48s, ep_out)
print() # Extra line so concatenation work better in gnuplot.
print("timestamp sample_in_packet voltage_V current_A dp_V dn_V temp_C_ema energy_Ws capacity_As")
time.sleep(0.1)
refresh = 1.0 if is_fnb58_or_fnb48s else 0.003 # 1 s for FNB58 / FNB48S, 3 ms for others
continue_time = time.time() + refresh
alpha = args.alpha
stop = False
while not stop:
try:
data = ep_in.read(size_or_buffer=64, timeout=5000)
# print("".join([f"{x:02x}" for x in data]))
decode(data, crc_calculator, time_interval, alpha)
if time.time() >= continue_time:
continue_time = time.time() + refresh
ep_out.write(b"\xaa\x83" + b"\x00" * 61 + b"\x9e")
except KeyboardInterrupt:
print(
"Terminating due to the keyboard interrupt, please wait until draining is complete …", file=sys.stderr
)
stop = True
if os.path.exists("fnirsi_stop"):
stop = True
try:
while True:
# Exhaust data in descriptor
if args.verbose:
print("Please wait until draining is finished …", file=sys.stderr)
# We do not know exactly how many bytes to read (even if we track things like refresh
# and continue_time it might be tricky), so just read with timeout.
data = ep_in.read(size_or_buffer=64, timeout=1000)
if data:
if args.verbose:
print(f"During termination drained {len(data)} bytes", file=sys.stderr)
except usb.core.USBTimeoutError as e:
# Exception [Errno 110] Operation timed out
# This is expected during draining, and in fact indicates a success.
if args.verbose:
print("Exiting …", file=sys.stderr)
except Exception as e:
print(f"Exception {type(e)} {e}", file=sys.stderr)
raise
if __name__ == "__main__":
main()