mirror of https://github.com/m-labs/artiq.git
move korad_ka3005p out of tree (#887)
This commit is contained in:
parent
73d6078883
commit
4c1fb0c2a1
|
@ -1,152 +0,0 @@
|
||||||
# Written by Joe Britton, 2016
|
|
||||||
|
|
||||||
import logging
|
|
||||||
import asyncio
|
|
||||||
import asyncserial
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class UnexpectedResponse(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class KoradKA3005P:
|
|
||||||
"""The Korad KA3005P is a 1-channel programmable power supply
|
|
||||||
(0-30V/0-5A) with both USB/serial and RS232 connectivity.
|
|
||||||
|
|
||||||
All amplitudes are in volts.
|
|
||||||
All currents are in amperes.
|
|
||||||
"""
|
|
||||||
|
|
||||||
# Serial interface gleaned from the following.
|
|
||||||
# https://github.com/starforgelabs/py-korad-serial
|
|
||||||
# https://sigrok.org/wiki/Korad_KAxxxxP_series
|
|
||||||
|
|
||||||
def __init__(self, serial_dev):
|
|
||||||
if serial_dev is None:
|
|
||||||
self.simulation = True
|
|
||||||
else:
|
|
||||||
self.simulation = False
|
|
||||||
self.port = asyncserial.AsyncSerial(serial_dev, baudrate=9600)
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
"""Close the serial port."""
|
|
||||||
if not self.simulation:
|
|
||||||
self.port.close()
|
|
||||||
|
|
||||||
async def _ser_read(self, fixed_length=None):
|
|
||||||
""" strings returned by firmware are zero-terminated or fixed length
|
|
||||||
"""
|
|
||||||
r = ""
|
|
||||||
if self.simulation:
|
|
||||||
logger.info("simulation _ser_read()")
|
|
||||||
else:
|
|
||||||
c = (await self.port.read(1)).decode()
|
|
||||||
r = c
|
|
||||||
while len(c) > 0 and ord(c) != 0 and not len(r) == fixed_length:
|
|
||||||
c = (await self.port.read(1)).decode().rstrip('\0')
|
|
||||||
r += c
|
|
||||||
logger.debug("_read %s: ", r)
|
|
||||||
return r
|
|
||||||
|
|
||||||
async def _ser_write(self, cmd):
|
|
||||||
if self.simulation:
|
|
||||||
logger.info("simulation _ser_write(\"%s\")", cmd)
|
|
||||||
else:
|
|
||||||
logger.debug("_ser_write(\"%s\")", cmd)
|
|
||||||
await asyncio.sleep(0.1)
|
|
||||||
await self.port.write(cmd.encode("ascii"))
|
|
||||||
|
|
||||||
async def setup(self):
|
|
||||||
"""Configure in known state."""
|
|
||||||
await self.set_output(False)
|
|
||||||
await self.set_v(0)
|
|
||||||
await self.set_ovp(False)
|
|
||||||
await self.set_i(0)
|
|
||||||
await self.set_ocp(False)
|
|
||||||
|
|
||||||
async def get_id(self):
|
|
||||||
"""Request identification from device.
|
|
||||||
"""
|
|
||||||
if self.simulation:
|
|
||||||
return "KORADKA3005PV2.0"
|
|
||||||
await self._ser_write("*IDN?")
|
|
||||||
return await self._ser_read()
|
|
||||||
|
|
||||||
async def set_output(self, b):
|
|
||||||
"""Enable/disable the power output.
|
|
||||||
"""
|
|
||||||
if b:
|
|
||||||
await self._ser_write("OUT1")
|
|
||||||
else:
|
|
||||||
await self._ser_write("OUT0")
|
|
||||||
|
|
||||||
async def set_v(self, v):
|
|
||||||
"""Set the maximum output voltage."""
|
|
||||||
await self._ser_write("VSET1:{0:05.2f}".format(v))
|
|
||||||
|
|
||||||
async def get_v(self):
|
|
||||||
"""Request the voltage as set by the user."""
|
|
||||||
await self._ser_write("VSET1?")
|
|
||||||
return float(await self._ser_read(fixed_length=5))
|
|
||||||
|
|
||||||
async def measure_v(self):
|
|
||||||
"""Request the actual voltage output."""
|
|
||||||
await self._ser_write("VOUT1?")
|
|
||||||
return float(await self._ser_read(fixed_length=5))
|
|
||||||
|
|
||||||
async def set_ovp(self, b):
|
|
||||||
"""Enable/disable the "Over Voltage Protection", the PS will switch off the
|
|
||||||
output when the voltage rises above the actual level."""
|
|
||||||
if b:
|
|
||||||
await self._ser_write("OVP1")
|
|
||||||
else:
|
|
||||||
await self._ser_write("OVP0")
|
|
||||||
|
|
||||||
async def set_i(self, v):
|
|
||||||
"""Set the maximum output current."""
|
|
||||||
await self._ser_write("ISET1:{0:05.3f}".format(v))
|
|
||||||
|
|
||||||
async def get_i(self):
|
|
||||||
"""Request the current as set by the user. """
|
|
||||||
# Expected behavior of ISET1? is to return 5 bytes.
|
|
||||||
# However, if *IDN? has been previously called, ISET1? replies
|
|
||||||
# with a sixth byte 'K' which should be discarded. For consistency,
|
|
||||||
# always call *IDN? before calling ISET1?.
|
|
||||||
self.get_id()
|
|
||||||
await self._ser_write("ISET1?")
|
|
||||||
r = (await self._ser_read(fixed_length=6)).rstrip('K')
|
|
||||||
return float(r)
|
|
||||||
|
|
||||||
async def measure_i(self):
|
|
||||||
"""Request the actual output current."""
|
|
||||||
await self._ser_write("IOUT1?")
|
|
||||||
r = await self._ser_read(fixed_length=5)
|
|
||||||
if r[0] == "K":
|
|
||||||
r = r[1:-1]
|
|
||||||
return float(r)
|
|
||||||
|
|
||||||
async def set_ocp(self, b):
|
|
||||||
"""Enable/disable the "Over Current Protection", the PS will switch off
|
|
||||||
the output when the current rises above the actual level."""
|
|
||||||
if b:
|
|
||||||
await self._ser_write("OCP1")
|
|
||||||
else:
|
|
||||||
await self._ser_write("OCP0")
|
|
||||||
|
|
||||||
async def ping(self):
|
|
||||||
"""Check if device is responding."""
|
|
||||||
if self.simulation:
|
|
||||||
return True
|
|
||||||
try:
|
|
||||||
id = await self.get_id()
|
|
||||||
except asyncio.CancelledError:
|
|
||||||
raise
|
|
||||||
except:
|
|
||||||
return False
|
|
||||||
if id == "KORADKA3005PV2.0":
|
|
||||||
logger.debug("ping successful")
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
|
@ -1,54 +0,0 @@
|
||||||
#!/usr/bin/env python3
|
|
||||||
|
|
||||||
# Written by Joe Britton, 2016
|
|
||||||
|
|
||||||
import argparse
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
import asyncio
|
|
||||||
import os
|
|
||||||
|
|
||||||
from artiq.devices.korad_ka3005p.driver import KoradKA3005P
|
|
||||||
from artiq.protocols.pc_rpc import simple_server_loop
|
|
||||||
from artiq.tools import *
|
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def get_argparser():
|
|
||||||
parser = argparse.ArgumentParser(
|
|
||||||
description="ARTIQ controller for the Korad KA3005P programmable DC power supply")
|
|
||||||
simple_network_args(parser, 3256)
|
|
||||||
parser.add_argument(
|
|
||||||
"-d", "--device", default=None,
|
|
||||||
help="serial port.")
|
|
||||||
parser.add_argument(
|
|
||||||
"--simulation", action="store_true",
|
|
||||||
help="Put the driver in simulation mode, even if --device is used.")
|
|
||||||
add_common_args(parser)
|
|
||||||
return parser
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
args = get_argparser().parse_args()
|
|
||||||
init_logger(args)
|
|
||||||
|
|
||||||
if os.name == "nt":
|
|
||||||
asyncio.set_event_loop(asyncio.ProactorEventLoop())
|
|
||||||
|
|
||||||
if not args.simulation and args.device is None:
|
|
||||||
print("You need to specify either --simulation or -d/--device "
|
|
||||||
"argument. Use --help for more information.")
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
dev = KoradKA3005P(args.device if not args.simulation else None)
|
|
||||||
asyncio.get_event_loop().run_until_complete(dev.setup())
|
|
||||||
try:
|
|
||||||
simple_server_loop(
|
|
||||||
{"korad_ka3005p": dev}, bind_address_from_args(args), args.port)
|
|
||||||
finally:
|
|
||||||
dev.close()
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
|
@ -10,7 +10,7 @@ class TestFrontends(unittest.TestCase):
|
||||||
# Skip tests for GUI programs on headless CI environments.
|
# Skip tests for GUI programs on headless CI environments.
|
||||||
commands = {
|
commands = {
|
||||||
"aqctl": [
|
"aqctl": [
|
||||||
"corelog", "korad_ka3005p", "lda", "novatech409b",
|
"corelog", "lda", "novatech409b",
|
||||||
"thorlabs_tcube"
|
"thorlabs_tcube"
|
||||||
],
|
],
|
||||||
"artiq": [
|
"artiq": [
|
||||||
|
|
|
@ -1,38 +0,0 @@
|
||||||
import sys
|
|
||||||
import unittest
|
|
||||||
|
|
||||||
from artiq.test.hardware_testbench import GenericControllerCase, ControllerCase
|
|
||||||
|
|
||||||
|
|
||||||
class GenericKoradKA3005PTest:
|
|
||||||
def test_parameters_readback(self):
|
|
||||||
|
|
||||||
# check device ID baked into firmware
|
|
||||||
ids = self.driver.get_id()
|
|
||||||
self.assertEqual(ids, "KORADKA3005PV2.0")
|
|
||||||
|
|
||||||
|
|
||||||
class TestKoradKA3005P(GenericKoradKA3005PTest, ControllerCase):
|
|
||||||
def setUp(self):
|
|
||||||
ControllerCase.setUp(self)
|
|
||||||
self.start_controller("koradka3005p")
|
|
||||||
self.driver = self.device_mgr.get("koradka3005p")
|
|
||||||
|
|
||||||
|
|
||||||
class TestKoradKA3005P(GenericKoradKA3005PTest, GenericControllerCase):
|
|
||||||
def get_device_db(self):
|
|
||||||
return {
|
|
||||||
"korad_ka3005p": {
|
|
||||||
"type": "controller",
|
|
||||||
"host": "::1",
|
|
||||||
"port": 3256,
|
|
||||||
"command": (sys.executable.replace("\\", "\\\\")
|
|
||||||
+ " -m artiq.frontend.aqctl_korad_ka3005p "
|
|
||||||
+ "-p {port} --simulation")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
GenericControllerCase.setUp(self)
|
|
||||||
self.start_controller("korad_ka3005p")
|
|
||||||
self.driver = self.device_mgr.get("korad_ka3005p")
|
|
|
@ -46,22 +46,6 @@ You can choose the LDA model with the ``-P`` parameter. The default is LDA-102.
|
||||||
:ref: artiq.frontend.aqctl_lda.get_argparser
|
:ref: artiq.frontend.aqctl_lda.get_argparser
|
||||||
:prog: aqctl_lda
|
:prog: aqctl_lda
|
||||||
|
|
||||||
Korad KA3005P
|
|
||||||
-------------
|
|
||||||
|
|
||||||
Driver
|
|
||||||
++++++
|
|
||||||
|
|
||||||
.. automodule:: artiq.devices.korad_ka3005p.driver
|
|
||||||
:members:
|
|
||||||
|
|
||||||
Controller
|
|
||||||
++++++++++
|
|
||||||
|
|
||||||
.. argparse::
|
|
||||||
:ref: artiq.frontend.aqctl_korad_ka3005p.get_argparser
|
|
||||||
:prog: aqctl_korad_ka3005p
|
|
||||||
|
|
||||||
Novatech 409B
|
Novatech 409B
|
||||||
-------------
|
-------------
|
||||||
|
|
||||||
|
|
1
setup.py
1
setup.py
|
@ -37,7 +37,6 @@ console_scripts = [
|
||||||
"artiq_flash = artiq.frontend.artiq_flash:main",
|
"artiq_flash = artiq.frontend.artiq_flash:main",
|
||||||
|
|
||||||
"aqctl_corelog = artiq.frontend.aqctl_corelog:main",
|
"aqctl_corelog = artiq.frontend.aqctl_corelog:main",
|
||||||
"aqctl_korad_ka3005p = artiq.frontend.aqctl_korad_ka3005p:main",
|
|
||||||
"aqctl_lda = artiq.frontend.aqctl_lda:main",
|
"aqctl_lda = artiq.frontend.aqctl_lda:main",
|
||||||
"aqctl_novatech409b = artiq.frontend.aqctl_novatech409b:main",
|
"aqctl_novatech409b = artiq.frontend.aqctl_novatech409b:main",
|
||||||
"aqctl_thorlabs_tcube = artiq.frontend.aqctl_thorlabs_tcube:main",
|
"aqctl_thorlabs_tcube = artiq.frontend.aqctl_thorlabs_tcube:main",
|
||||||
|
|
Loading…
Reference in New Issue