Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch Add support for Xilinx 7-series ADC.



details:   https://anonhg.NetBSD.org/src/rev/e9daaf8edd72
branches:  trunk
changeset: 372304:e9daaf8edd72
user:      jmcneill <jmcneill%NetBSD.org@localhost>
date:      Fri Nov 11 20:31:30 2022 +0000

description:
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

diffstat:

 sys/arch/arm/xilinx/files.zynq  |    7 +-
 sys/arch/arm/xilinx/zynq_xadc.c |  350 ++++++++++++++++++++++++++++++++++++++++
 sys/arch/evbarm/conf/GENERIC    |    3 +-
 3 files changed, 358 insertions(+), 2 deletions(-)

diffs (truncated from 390 to 300 lines):

diff -r 7fed4e756e1d -r e9daaf8edd72 sys/arch/arm/xilinx/files.zynq
--- a/sys/arch/arm/xilinx/files.zynq    Fri Nov 11 20:29:47 2022 +0000
+++ b/sys/arch/arm/xilinx/files.zynq    Fri Nov 11 20:31:30 2022 +0000
@@ -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 @@
 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
diff -r 7fed4e756e1d -r e9daaf8edd72 sys/arch/arm/xilinx/zynq_xadc.c
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sys/arch/arm/xilinx/zynq_xadc.c   Fri Nov 11 20:31:30 2022 +0000
@@ -0,0 +1,350 @@
+/* $NetBSD: zynq_xadc.c,v 1.1 2022/11/11 20:31:30 jmcneill Exp $ */
+
+/*-
+ * Copyright (c) 2022 Jared McNeill <jmcneill%invisible.ca@localhost>
+ * 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;



Home | Main Index | Thread Index | Old Index