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
|
"""
Implements partial support for Joy-IT JDS6600 function generator.
"""
from lab_control.function_generator import FunctionGenerator
from lab_control.connection.serial_connection import SerialConnection
def _checkChannel(channel: int):
assert channel in JDS6600.AVAILABLE_CHANNELS, f"JDS6600: Invalid channel {channel}"
def _checkBounds(value: float, lowerBound: float, upperBound: float):
valid = value is not None
if valid:
valid &= value >= lowerBound
valid &= value <= upperBound
assert valid, f"JDS6600: Invalid argument {value}"
class JDS6600(FunctionGenerator):
"""
Instances of this class bind to the JDS6600 serial port
and offer an API to control the device.
"""
AVAILABLE_CHANNELS = [1, 2]
# TODO type hints
def __init__(self, portName: str, overrideConnection=None):
super().__init__()
config = {
"baudrate" : 115200,
"bytesize" : 8,
"stopbits" : 1,
"parity" : "N"
}
if overrideConnection is not None:
self._connection = overrideConnection
else:
self._connection = SerialConnection(portName)
self._connection.configure(config)
def closePort(self) -> None:
"""
Close the serial port. Instances of this class
are no longer usable after this is called.
"""
self._connection.close()
def _sendRequest(self, opcode: str, args: str="") -> str:
request = f":{opcode}={args}.\r\n"
response = self._connection.send(request)
return response.strip()
def _query(self, opcode: str) -> list[str]:
# response format: ":{opcode}={v1},{v2}."
response = self._sendRequest(opcode)
return response[5:-1].split(",")
def setOn(self, channel: int) -> None:
_checkChannel(channel)
state = self._query("r20")
state[channel - 1] = "1"
self._sendRequest("w20", f"{state[0]},{state[1]}")
def setOff(self, channel: int) -> None:
_checkChannel(channel)
state = self._query("r20")
state[channel - 1] = "0"
self._sendRequest("w20", f"{state[0]},{state[1]}")
def setFrequency(self, channel: int, frequency: float) -> None:
_checkChannel(channel)
_checkBounds(frequency, 0.0, 60e6)
opcode = 23 + channel - 1
arg = int(frequency * 100.0)
self._sendRequest(f"w{opcode}", f"{arg},0")
# JDS6600 has been observed to sometimes incorrectly
# ignore the frequency and set to 0 instead
actual = self._query(f"r{opcode}")[0]
if actual != arg:
self._sendRequest(f"w{opcode}", f"{arg},0")
def setAmplitude(self, channel: int, amplitude: float) -> None:
_checkChannel(channel)
_checkBounds(amplitude, 0.0, 20.0)
opcode = f"w{25 + channel - 1}"
arg = int(amplitude * 1000.0)
self._sendRequest(opcode, str(arg))
def setFunction(self, channel: int, function: int) -> None:
_checkChannel(channel)
_checkBounds(function, 0, 16)
opcode = f"w{21 + channel - 1}"
self._sendRequest(opcode, str(function))
|