Module Name: src
Committed By: jmcneill
Date: Fri Nov 11 20:31:30 UTC 2022
Modified Files:
src/sys/arch/arm/xilinx: files.zynq
src/sys/arch/evbarm/conf: GENERIC
Added Files:
src/sys/arch/arm/xilinx: zynq_xadc.c
Log Message:
Add support for Xilinx 7-series ADC.
The temperature and voltage sensors are exposed with sysmon:
# envstat -d zynqxadc0
Current CritMax WarnMax WarnMin CritMin Unit
temperature: 67.426 degC
vccint: 1.010 V
vccaux: 1.801 V
vp/vn: 0.011 V
vrefp: 1.250 V
vrefn: 0.004 V
vccbram: 1.009 V
vccpint: 1.009 V
vccpaux: 1.799 V
vcco_ddr: 1.500 V
To generate a diff of this commit:
cvs rdiff -u -r1.4 -r1.5 src/sys/arch/arm/xilinx/files.zynq
cvs rdiff -u -r0 -r1.1 src/sys/arch/arm/xilinx/zynq_xadc.c
cvs rdiff -u -r1.114 -r1.115 src/sys/arch/evbarm/conf/GENERIC
Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.
Modified files:
Index: src/sys/arch/arm/xilinx/files.zynq
diff -u src/sys/arch/arm/xilinx/files.zynq:1.4 src/sys/arch/arm/xilinx/files.zynq:1.5
--- src/sys/arch/arm/xilinx/files.zynq:1.4 Thu Oct 27 09:41:28 2022
+++ src/sys/arch/arm/xilinx/files.zynq Fri Nov 11 20:31:30 2022
@@ -1,4 +1,4 @@
-# $NetBSD: files.zynq,v 1.4 2022/10/27 09:41:28 jmcneill Exp $
+# $NetBSD: files.zynq,v 1.5 2022/11/11 20:31:30 jmcneill Exp $
#
# Configuration info for Xilinx Zynq-7000 SoC
#
@@ -37,3 +37,8 @@ file arch/arm/xilinx/zynq_cemac.c cemac
attach ehci at fdt with zynqusb
file arch/arm/xilinx/zynq_usb.c zynqusb
file arch/arm/xilinx/zynq7000_usb.c zynqusb
+
+# Xilinx 7 series ADC
+device zynqxadc: sysmon_envsys
+attach zynqxadc at fdt
+file arch/arm/xilinx/zynq_xadc.c zynqxadc
Index: src/sys/arch/evbarm/conf/GENERIC
diff -u src/sys/arch/evbarm/conf/GENERIC:1.114 src/sys/arch/evbarm/conf/GENERIC:1.115
--- src/sys/arch/evbarm/conf/GENERIC:1.114 Sat Nov 5 17:32:03 2022
+++ src/sys/arch/evbarm/conf/GENERIC Fri Nov 11 20:31:30 2022
@@ -1,5 +1,5 @@
#
-# $NetBSD: GENERIC,v 1.114 2022/11/05 17:32:03 jmcneill Exp $
+# $NetBSD: GENERIC,v 1.115 2022/11/11 20:31:30 jmcneill Exp $
#
# GENERIC ARM (aarch32) kernel
#
@@ -436,6 +436,7 @@ tegrartc* at fdt? # NVIDIA Tegra RTC
# Thermal sensor
sunxithermal* at fdt? # Thermal sensor controller
+zynqxadc* at fdt? # Xilinx 7 series ADC
# BCM2835 VCHIQ, etc
vchiq0 at fdt?
Added files:
Index: src/sys/arch/arm/xilinx/zynq_xadc.c
diff -u /dev/null src/sys/arch/arm/xilinx/zynq_xadc.c:1.1
--- /dev/null Fri Nov 11 20:31:30 2022
+++ src/sys/arch/arm/xilinx/zynq_xadc.c Fri Nov 11 20:31:30 2022
@@ -0,0 +1,350 @@
+/* $NetBSD: zynq_xadc.c,v 1.1 2022/11/11 20:31:30 jmcneill Exp $ */
+
+/*-
+ * Copyright (c) 2022 Jared McNeill <[email protected]>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
+ * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
+ * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+/*
+ * Xilinx 7 series ADC ("XADC")
+ *
+ * Documentation can be found on the Xilinx web site:
+ * - Zynq-7000 SoC Technical Reference Manual UG585 (v1.13)
+ * - XADC User Guide 3 UG480 (v1.11)
+ */
+
+#include <sys/cdefs.h>
+__KERNEL_RCSID(0, "$NetBSD: zynq_xadc.c,v 1.1 2022/11/11 20:31:30 jmcneill Exp $");
+
+#include <sys/param.h>
+#include <sys/bitops.h>
+#include <sys/bus.h>
+#include <sys/device.h>
+#include <sys/intr.h>
+#include <sys/kmem.h>
+#include <sys/lwp.h>
+#include <sys/mutex.h>
+#include <sys/systm.h>
+
+#include <dev/sysmon/sysmonvar.h>
+#include <dev/fdt/fdtvar.h>
+
+/* PS-XADC interface registers */
+#define XADCIF_CFG 0x00
+#define CFG_ENABLE __BIT(31)
+#define CFG_WEDGE __BIT(13)
+#define CFG_REDGE __BIT(12)
+#define CFG_TCKRATE __BITS(9,8)
+#define CFG_TCKRATE_DIV4 1
+#define XADCIF_INT_STS 0x04
+#define XADCIF_INT_MASK 0x08
+#define XADCIF_MSTS 0x0c
+#define MSTS_CFIFOE __BIT(10)
+#define XADCIF_CMDFIFO 0x10
+#define XADCIF_RDFIFO 0x14
+#define XADCIF_MCTL 0x18
+
+/* XADC registers */
+#define XADC_STATUS_TEMP 0x00
+#define XADC_STATUS_VCCINT 0x01
+#define XADC_STATUS_VCCAUX 0x02
+#define XADC_STATUS_VPVN 0x03
+#define XADC_STATUS_VREFP 0x04
+#define XADC_STATUS_VREFN 0x05
+#define XADC_STATUS_VCCBRAM 0x06
+#define XADC_STATUS_VCCPINT 0x0d
+#define XADC_STATUS_VCCPAUX 0x0e
+#define XADC_STATUS_VCCO_DDR 0x0f
+#define XADC_STATUS_FLAG 0x3f
+#define FLAG_OT __BIT(3)
+#define XADC_CONF(n) (0x40 + (n))
+#define CONF1_SEQ __BITS(15,12)
+#define CONF1_SEQ_CONT 2
+#define XADC_SEQ(n) (0x48 + (n))
+#define SEQ0_CALIB __BIT(0)
+#define SEQ0_ALL __BITS(5,14)
+#define SEQ4_VREFN __BIT(13)
+
+/* XADC commands */
+#define XADC_COMMAND_CMD __BITS(29,26)
+#define XADC_COMMAND_DRP_ADDR __BITS(25,16)
+#define XADC_COMMAND_DRP_DATA __BITS(15,0)
+#define XADC_COMMAND(cmd, addr, data) \
+ (__SHIFTIN(cmd, XADC_COMMAND_CMD) | \
+ __SHIFTIN(addr, XADC_COMMAND_DRP_ADDR) | \
+ __SHIFTIN(data, XADC_COMMAND_DRP_DATA))
+#define XADC_CMD_NOP 0
+#define XADC_CMD_READ 1
+#define XADC_CMD_WRITE 2
+
+enum {
+ XADC_SENSOR_TEMP,
+ XADC_SENSOR_VCCINT,
+ XADC_SENSOR_VCCAUX,
+ XADC_SENSOR_VPVN,
+ XADC_SENSOR_VREFP,
+ XADC_SENSOR_VREFN,
+ XADC_SENSOR_VCCBRAM,
+ XADC_SENSOR_VCCPINT,
+ XADC_SENSOR_VCCPAUX,
+ XADC_SENSOR_VCCO_DDR,
+ XADC_NSENSOR
+};
+
+static const struct {
+ const char *name;
+ uint32_t units;
+ uint16_t reg;
+} zynq_xadc_sensors[] = {
+ [XADC_SENSOR_TEMP] = { "temperature", ENVSYS_STEMP, XADC_STATUS_TEMP },
+ [XADC_SENSOR_VCCINT] = { "vccint", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCINT },
+ [XADC_SENSOR_VCCAUX] = { "vccaux", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCAUX },
+ [XADC_SENSOR_VPVN] = { "vp/vn", ENVSYS_SVOLTS_DC, XADC_STATUS_VPVN },
+ [XADC_SENSOR_VREFP] = { "vrefp", ENVSYS_SVOLTS_DC, XADC_STATUS_VREFP },
+ [XADC_SENSOR_VREFN] = { "vrefn", ENVSYS_SVOLTS_DC, XADC_STATUS_VREFN },
+ [XADC_SENSOR_VCCBRAM] = { "vccbram", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCBRAM },
+ [XADC_SENSOR_VCCPINT] = { "vccpint", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCPINT },
+ [XADC_SENSOR_VCCPAUX] = { "vccpaux", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCPAUX },
+ [XADC_SENSOR_VCCO_DDR] = { "vcco_ddr", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCO_DDR },
+};
+
+static const struct device_compatible_entry compat_data[] = {
+ { .compat = "xlnx,zynq-xadc-1.00.a" },
+ DEVICE_COMPAT_EOL
+};
+
+struct zynq_xadc_softc {
+ device_t sc_dev;
+ bus_space_tag_t sc_bst;
+ bus_space_handle_t sc_bsh;
+ kmutex_t sc_lock;
+
+ struct sysmon_envsys *sc_sme;
+ envsys_data_t sc_sensor[XADC_NSENSOR];
+};
+
+#define RD4(sc, reg) \
+ bus_space_read_4((sc)->sc_bst, (sc)->sc_bsh, (reg))
+#define WR4(sc, reg, val) \
+ bus_space_write_4((sc)->sc_bst, (sc)->sc_bsh, (reg), (val))
+
+static int zynq_xadc_match(device_t, cfdata_t, void *);
+static void zynq_xadc_attach(device_t, device_t, void *);
+
+CFATTACH_DECL_NEW(zynqxadc, sizeof(struct zynq_xadc_softc),
+ zynq_xadc_match, zynq_xadc_attach, NULL, NULL);
+
+static void
+zynq_xadc_write(struct zynq_xadc_softc *sc, uint16_t reg,
+ uint16_t data)
+{
+ int retry = 10000;
+
+ /*
+ * Write sequence is:
+ *
+ * 1. Prepare write command
+ * 2. Write data to Command FIFO
+ * 3. Wait until the Command FIFO becomes empty
+ */
+
+ WR4(sc, XADCIF_CMDFIFO, XADC_COMMAND(XADC_CMD_WRITE, reg, data));
+ while (--retry > 0) {
+ if ((RD4(sc, XADCIF_MSTS) & MSTS_CFIFOE) != 0) {
+ break;
+ }
+ delay(10);
+ }
+ if (retry == 0) {
+ device_printf(sc->sc_dev, "command FIFO timeout (write)\n");
+ }
+ /*
+ * Every write to Command FIFO shifts data into Read FIFO, so
+ * drain that after the command completes.
+ */
+ RD4(sc, XADCIF_RDFIFO);
+}
+
+static uint16_t
+zynq_xadc_read(struct zynq_xadc_softc *sc, uint16_t reg)
+{
+ int retry = 10000;
+ uint32_t val;
+
+ /*
+ * Read sequence is:
+ *
+ * 1. Prepare read command
+ * 2. Write data to Command FIFO
+ * 3. Wait until the Command FIFO becomes empty
+ * 4. Read dummy data from the Read Data FIFO
+ * 5. Prepare nop command
+ * 6. Write data to Command FIFO
+ * 7. Read the Read Data FIFO
+ */
+
+ WR4(sc, XADCIF_CMDFIFO, XADC_COMMAND(XADC_CMD_READ, reg, 0));
+ WR4(sc, XADCIF_CMDFIFO, XADC_COMMAND(XADC_CMD_NOP, 0, 0));
+ while (--retry > 0) {
+ if ((RD4(sc, XADCIF_MSTS) & MSTS_CFIFOE) != 0) {
+ break;
+ }
+ delay(10);
+ }
+ if (retry == 0) {
+ device_printf(sc->sc_dev, "command FIFO timeout (read)\n");
+ return 0xffff;
+ }
+ val = RD4(sc, XADCIF_RDFIFO);
+ val = RD4(sc, XADCIF_RDFIFO);
+
+ return val & 0xffff;
+}
+
+static void
+zynq_xadc_init(struct zynq_xadc_softc *sc, struct clk *clk)
+{
+ uint32_t val;
+
+ /* Enable the PS-XADC interface */
+ val = RD4(sc, XADCIF_CFG);
+ val |= CFG_ENABLE;
+ val &= ~CFG_TCKRATE;
+ val |= __SHIFTIN(CFG_TCKRATE_DIV4, CFG_TCKRATE);
+ val |= CFG_WEDGE | CFG_REDGE;
+ WR4(sc, XADCIF_CFG, val);
+ WR4(sc, XADCIF_MCTL, 0);
+
+ /* Turn on continuous sampling for all ADC channels we monitor */
+ zynq_xadc_write(sc, XADC_SEQ(0), SEQ0_CALIB | SEQ0_ALL);
+ zynq_xadc_write(sc, XADC_SEQ(4), SEQ4_VREFN);
+ zynq_xadc_write(sc, XADC_CONF(0), 0);
+ zynq_xadc_write(sc, XADC_CONF(1),
+ __SHIFTIN(CONF1_SEQ_CONT, CONF1_SEQ));
+
+}
+
+static void
+zynq_xadc_sensors_refresh(struct sysmon_envsys *sme, envsys_data_t *edata)
+{
+ struct zynq_xadc_softc *sc = sme->sme_cookie;
+ union {
+ uint16_t u16;
+ int16_t s16;
+ } val;
+ int64_t temp;
+
+
+ val.u16 = zynq_xadc_read(sc, zynq_xadc_sensors[edata->sensor].reg);
+ if (edata->units == ENVSYS_STEMP) {
+ if (val.u16 == 0) {
+ edata->state = ENVSYS_SINVALID;
+ } else {
+ temp = ((int64_t)(val.u16 >> 4) * 503975) / 4096;
+ edata->value_cur = 1000 * temp;
+ edata->state = ENVSYS_SVALID;
+ }
+
+ val.u16 = zynq_xadc_read(sc, XADC_STATUS_FLAG);
+ if ((val.u16 & FLAG_OT) != 0) {
+ edata->state = ENVSYS_SCRITOVER;
+ }
+ } else {
+ KASSERT(edata->units == ENVSYS_SVOLTS_DC);
+ switch (edata->sensor) {
+ case XADC_SENSOR_VPVN:
+ edata->value_cur = (((val.u16 >> 4) * 1000) / 4096) * 1000;
+ break;
+ case XADC_SENSOR_VREFN:
+ edata->value_cur = (((val.s16 >> 4) * 3000) / 4096) * 1000;
+ break;
+ default:
+ edata->value_cur = (((val.u16 >> 4) * 3000) / 4096) * 1000;
+ break;
+ }
+ edata->state = ENVSYS_SVALID;
+ }
+}
+
+static int
+zynq_xadc_match(device_t parent, cfdata_t cf, void *aux)
+{
+ struct fdt_attach_args * const faa = aux;
+
+ return of_compatible_match(faa->faa_phandle, compat_data);
+}
+
+static void
+zynq_xadc_attach(device_t parent, device_t self, void *aux)
+{
+ struct zynq_xadc_softc * const sc = device_private(self);
+ struct fdt_attach_args * const faa = aux;
+ const int phandle = faa->faa_phandle;
+ struct clk *clk;
+ bus_addr_t addr;
+ bus_size_t size;
+ u_int n;
+
+ if (fdtbus_get_reg(phandle, 0, &addr, &size) != 0) {
+ aprint_error(": couldn't get registers\n");
+ return;
+ }
+ clk = fdtbus_clock_get_index(phandle, 0);
+ if (clk == NULL || clk_enable(clk) != 0) {
+ aprint_error(": couldn't enable clock\n");
+ return;
+ }
+
+ sc->sc_dev = self;
+ sc->sc_bst = faa->faa_bst;
+ if (bus_space_map(sc->sc_bst, addr, size, 0, &sc->sc_bsh) != 0) {
+ aprint_error(": couldn't map registers\n");
+ return;
+ }
+ mutex_init(&sc->sc_lock, MUTEX_DEFAULT, IPL_VM);
+
+ aprint_naive("\n");
+ aprint_normal(": ADC\n");
+
+ zynq_xadc_init(sc, clk);
+
+ sc->sc_sme = sysmon_envsys_create();
+ sc->sc_sme->sme_name = device_xname(self);
+ sc->sc_sme->sme_cookie = sc;
+ sc->sc_sme->sme_refresh = zynq_xadc_sensors_refresh;
+
+ for (n = 0; n < XADC_NSENSOR; n++) {
+ sc->sc_sensor[n].units = zynq_xadc_sensors[n].units;
+ sc->sc_sensor[n].state = ENVSYS_SINVALID;
+ sc->sc_sensor[n].flags = ENVSYS_FHAS_ENTROPY;
+ if (zynq_xadc_sensors[n].units == ENVSYS_STEMP) {
+ sc->sc_sensor[n].flags |= ENVSYS_FMONCRITICAL;
+ }
+ strncpy(sc->sc_sensor[n].desc, zynq_xadc_sensors[n].name,
+ sizeof(sc->sc_sensor[n].desc));
+ sysmon_envsys_sensor_attach(sc->sc_sme, &sc->sc_sensor[n]);
+ }
+
+ sysmon_envsys_register(sc->sc_sme);
+}