-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathPTN5150.py
379 lines (319 loc) · 10.3 KB
/
PTN5150.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
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
import subprocess
from enum import IntEnum
from pydantic import BaseModel
class RpSelection(IntEnum):
"""
Bits [4:3] in the Control register (0x02), controlling Rp (in DFP mode).
00 => 80μA
01 => 180μA
10 => 330μA
11 => Reserved
"""
X80_uA = 0
X180_uA = 1
X330_uA = 2
RESERVED = 3
class PortState(IntEnum):
"""
Bits [2:1] in the Control register (0x02), controlling PORT pin / Mode selection.
00 => UFP
01 => DFP
10 => DRP
11 => Reserved
"""
UFP = 0
DFP = 1
DRP = 2
RESERVED = 3
class InterruptMask(IntEnum):
"""
Bit [0] in the Control register (0x02), controlling the attach/detach interrupt mask.
0 => Does not mask interrupts
1 => Masks interrupts
"""
NOT_MASKED = 0
MASKED = 1
class VBUSDetect(IntEnum):
"""
Bit [7] in CC Status register (0x04).
0 => VBUS not detected
1 => VBUS detected
"""
NOT_DETECTED = 0
DETECTED = 1
class RpDetect(IntEnum):
"""
Bits [6:5] in CC Status register (0x04), valid in UFP mode.
00 => Standby
01 => Rp = Std USB (default 500mA)
10 => Rp = 1.5A
11 => Rp = 3.0A
"""
STANDBY = 0
STD_USB = 1
A_1_5 = 2
A_3_0 = 3
class PortAttachment(IntEnum):
"""
Bits [4:2] in CC Status register (0x04).
000 => Not Connected
001 => DFP attached
010 => UFP attached
011 => Analog Audio Accessory
100 => Debug Accessory
101 => Reserved
110 => Reserved
111 => Reserved
"""
NOT_CONNECTED = 0
DFP_ATTACHED = 1
UFP_ATTACHED = 2
ANALOG_AUDIO = 3
DEBUG_ACCESSORY = 4
RESERVED_5 = 5
RESERVED_6 = 6
RESERVED_7 = 7
class CCPolarity(IntEnum):
"""
Bits [1:0] in CC Status register (0x04).
00 => Cable Not Attached
01 => CC1 is connected (normal orientation)
10 => CC2 is connected (reversed orientation)
11 => Reserved
"""
NOT_ATTACHED = 0
CC1_NORMAL = 1
CC2_REVERSED = 2
RESERVED = 3
#
# Optionally, a Pydantic model to hold CC status as a typed structure
#
class CCStatus(BaseModel):
vbus_detect: VBUSDetect
rp_detect: RpDetect
port_attachment: PortAttachment
cc_polarity: CCPolarity
class PTN5150H:
"""
A simple class to access PTN5150H registers over I2C using i2cget/i2cset.
Datasheet: https://web.archive.org/web/20240515225630/http://www.nxp.com/docs/en/data-sheet/PTN5150H.pdf
"""
def __init__(self, i2cbus=1, address=0x3D):
"""
:param i2cbus: The I2C bus number (e.g. 1 on a Raspberry Pi).
:param address: The 7-bit address of the PTN5150H (default = 0x3D).
"""
self.i2cbus = i2cbus
self.address = address
def read_register(self, reg_addr: int) -> int:
"""
Reads an 8-bit value from the given register address using i2cget.
:param reg_addr: The register address to read (0x00..0xFF).
:return: The register value (0..255).
"""
cmd = [
"i2cget",
"-f",
"-y",
str(self.i2cbus),
f"0x{self.address:02x}",
f"0x{reg_addr:02x}",
]
try:
output = subprocess.check_output(cmd).strip()
except subprocess.CalledProcessError as e:
print(f"Error reading register {reg_addr}: {e}")
return -1
# i2cget outputs a string like "0x1f"
return int(output, 16)
def write_register(self, reg_addr: int, value: int) -> None:
"""
Writes an 8-bit value to the given register address using i2cset.
:param reg_addr: The register address to write (0x00..0xFF).
:param value: The value to write (0..255).
"""
cmd = [
"i2cset",
"-f",
"-y",
str(self.i2cbus),
f"0x{self.address:02x}",
f"0x{reg_addr:02x}",
f"0x{value:02x}",
]
try:
subprocess.check_call(cmd)
except subprocess.CalledProcessError as e:
print(f"Error writing register {reg_addr}: {e}")
# ------------------------------------------------------------------------
# Register 0x01: Version / Vendor (Read Only)
#
# Bits [7:3] = Version ID (5 bits)
# Bits [2:0] = Vendor ID (3 bits)
# ------------------------------------------------------------------------
def get_version_id(self) -> int:
"""
Bits [7:3] = version ID
"""
reg_val = self.read_register(0x01)
if reg_val == -1:
return None
version_id = (reg_val >> 3) & 0x1F
return version_id
def get_vendor_id(self) -> int:
"""
Bits [2:0] = vendor ID
"""
reg_val = self.read_register(0x01)
if reg_val == -1:
return None
vendor_id = reg_val & 0x07
return vendor_id
# ------------------------------------------------------------------------
# Register 0x02: Control (Read/Write)
#
# Bits [4:3] = Rp Selection
# Bits [2:1] = PORT pin state / Mode Selection
# Bit [0] = Interrupt Mask
# ------------------------------------------------------------------------
def get_control(self) -> int:
"""Returns the entire 8-bit control register (0x02)."""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
return reg_val
def set_control(self, value: int) -> None:
"""Writes the entire 8-bit control register (0x02)."""
self.write_register(0x02, value)
#
# Rp Selection
#
def get_rp_selection(self) -> RpSelection:
"""
Extract bits [4:3] from the control register and return an RpSelection enum.
"""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
raw = (reg_val >> 3) & 0x03
return RpSelection(raw)
def set_rp_selection(self, rp_sel: RpSelection) -> None:
"""
Set bits [4:3] in control register to the given RpSelection enum.
"""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
# Clear bits [4:3], then set them
reg_val = (reg_val & 0xE7) | ((rp_sel.value & 0x03) << 3)
self.write_register(0x02, reg_val)
#
# Port State (Mode Selection)
#
def get_port_state(self) -> PortState:
"""
Extract bits [2:1] from the control register and return a PortState enum.
"""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
raw = (reg_val >> 1) & 0x03
return PortState(raw)
def set_port_state(self, port_mode: PortState) -> None:
"""
Set bits [2:1] in control register to the given PortState enum.
"""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
# Clear bits [2:1], then set them
reg_val = (reg_val & 0xF9) | ((port_mode.value & 0x03) << 1)
self.write_register(0x02, reg_val)
#
# Interrupt Mask (bit [0])
#
def get_interrupt_mask(self) -> InterruptMask:
"""
Extract bit [0] from the control register and return an InterruptMask enum.
"""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
raw = reg_val & 0x01
return InterruptMask(raw)
def set_interrupt_mask(self, mask_bit: InterruptMask) -> None:
"""
Set bit [0] in the control register to the given InterruptMask enum.
"""
reg_val = self.read_register(0x02)
if reg_val == -1:
return None
reg_val = (reg_val & 0xFE) | (mask_bit.value & 0x01)
self.write_register(0x02, reg_val)
# ------------------------------------------------------------------------
# Register 0x03: Interrupt Status (Read-Only, Clear on Read)
#
# Bits [1] = Cable Detach Interrupt
# Bits [0] = Cable Attach Interrupt
# ------------------------------------------------------------------------
def get_interrupt_status(self) -> dict:
"""
Reads the interrupt status at 0x03 (clears on read).
Returns a dict with 'cable_detach' and 'cable_attach' flags (booleans).
"""
reg_val = self.read_register(0x03)
return {
"cable_detach": bool((reg_val >> 1) & 0x01),
"cable_attach": bool(reg_val & 0x01),
}
# ------------------------------------------------------------------------
# Register 0x04: CC Status (Read-Only)
#
# [7] = VBUS Detection (0=not detected, 1=detected)
# [6:5] = Rp Detection (when UFP)
# [4:2] = Port Attachment Status
# [1:0] = CC Polarity
# ------------------------------------------------------------------------
def get_cc_status(self) -> CCStatus:
"""
Reads and parses register 0x04
"""
reg_val = self.read_register(0x04)
vbus = VBUSDetect((reg_val >> 7) & 0x01)
rp_d = RpDetect((reg_val >> 5) & 0x03)
attach = PortAttachment((reg_val >> 2) & 0x07)
polarity = CCPolarity(reg_val & 0x03)
return CCStatus(
vbus_detect=vbus,
rp_detect=rp_d,
port_attachment=attach,
cc_polarity=polarity,
)
#
# Example usage:
#
if __name__ == "__main__":
# Instantiate PTN5150H at bus=1, address=0x3D
ptn = PTN5150H(i2cbus=1, address=0x3D)
# Read vendor/version
version = ptn.get_version_id()
vendor = ptn.get_vendor_id()
print("Version ID:", version)
print("Vendor ID:", vendor)
# Read and parse control register bitfields
rp_sel = ptn.get_rp_selection()
port_mode = ptn.get_port_state()
intr_mask = ptn.get_interrupt_mask()
print(f"Current RpSelection: {rp_sel.name} ({rp_sel.value})")
print(f"Current PortState: {port_mode.name} ({port_mode.value})")
print(f"Current InterruptMask: {intr_mask.name} ({intr_mask.value})")
# Update some fields
ptn.set_rp_selection(RpSelection.X180_uA) # 180μA
ptn.set_port_state(PortState.DRP) # DRP
ptn.set_interrupt_mask(InterruptMask.MASKED)
# Check (and clear) interrupt status
interrupts = ptn.get_interrupt_status()
print("Interrupts:", interrupts)
# Read CC status as a typed Pydantic model
cc_stat = ptn.get_cc_status()
print("CC Status model:", cc_stat.model_dump())