1*5754a6ffSjmcneill /* $NetBSD: zynq_xadc.c,v 1.1 2022/11/11 20:31:30 jmcneill Exp $ */
2*5754a6ffSjmcneill
3*5754a6ffSjmcneill /*-
4*5754a6ffSjmcneill * Copyright (c) 2022 Jared McNeill <jmcneill@invisible.ca>
5*5754a6ffSjmcneill * All rights reserved.
6*5754a6ffSjmcneill *
7*5754a6ffSjmcneill * Redistribution and use in source and binary forms, with or without
8*5754a6ffSjmcneill * modification, are permitted provided that the following conditions
9*5754a6ffSjmcneill * are met:
10*5754a6ffSjmcneill * 1. Redistributions of source code must retain the above copyright
11*5754a6ffSjmcneill * notice, this list of conditions and the following disclaimer.
12*5754a6ffSjmcneill * 2. Redistributions in binary form must reproduce the above copyright
13*5754a6ffSjmcneill * notice, this list of conditions and the following disclaimer in the
14*5754a6ffSjmcneill * documentation and/or other materials provided with the distribution.
15*5754a6ffSjmcneill *
16*5754a6ffSjmcneill * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17*5754a6ffSjmcneill * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18*5754a6ffSjmcneill * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19*5754a6ffSjmcneill * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20*5754a6ffSjmcneill * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21*5754a6ffSjmcneill * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22*5754a6ffSjmcneill * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23*5754a6ffSjmcneill * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24*5754a6ffSjmcneill * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25*5754a6ffSjmcneill * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26*5754a6ffSjmcneill * SUCH DAMAGE.
27*5754a6ffSjmcneill */
28*5754a6ffSjmcneill
29*5754a6ffSjmcneill /*
30*5754a6ffSjmcneill * Xilinx 7 series ADC ("XADC")
31*5754a6ffSjmcneill *
32*5754a6ffSjmcneill * Documentation can be found on the Xilinx web site:
33*5754a6ffSjmcneill * - Zynq-7000 SoC Technical Reference Manual UG585 (v1.13)
34*5754a6ffSjmcneill * - XADC User Guide 3 UG480 (v1.11)
35*5754a6ffSjmcneill */
36*5754a6ffSjmcneill
37*5754a6ffSjmcneill #include <sys/cdefs.h>
38*5754a6ffSjmcneill __KERNEL_RCSID(0, "$NetBSD: zynq_xadc.c,v 1.1 2022/11/11 20:31:30 jmcneill Exp $");
39*5754a6ffSjmcneill
40*5754a6ffSjmcneill #include <sys/param.h>
41*5754a6ffSjmcneill #include <sys/bitops.h>
42*5754a6ffSjmcneill #include <sys/bus.h>
43*5754a6ffSjmcneill #include <sys/device.h>
44*5754a6ffSjmcneill #include <sys/intr.h>
45*5754a6ffSjmcneill #include <sys/kmem.h>
46*5754a6ffSjmcneill #include <sys/lwp.h>
47*5754a6ffSjmcneill #include <sys/mutex.h>
48*5754a6ffSjmcneill #include <sys/systm.h>
49*5754a6ffSjmcneill
50*5754a6ffSjmcneill #include <dev/sysmon/sysmonvar.h>
51*5754a6ffSjmcneill #include <dev/fdt/fdtvar.h>
52*5754a6ffSjmcneill
53*5754a6ffSjmcneill /* PS-XADC interface registers */
54*5754a6ffSjmcneill #define XADCIF_CFG 0x00
55*5754a6ffSjmcneill #define CFG_ENABLE __BIT(31)
56*5754a6ffSjmcneill #define CFG_WEDGE __BIT(13)
57*5754a6ffSjmcneill #define CFG_REDGE __BIT(12)
58*5754a6ffSjmcneill #define CFG_TCKRATE __BITS(9,8)
59*5754a6ffSjmcneill #define CFG_TCKRATE_DIV4 1
60*5754a6ffSjmcneill #define XADCIF_INT_STS 0x04
61*5754a6ffSjmcneill #define XADCIF_INT_MASK 0x08
62*5754a6ffSjmcneill #define XADCIF_MSTS 0x0c
63*5754a6ffSjmcneill #define MSTS_CFIFOE __BIT(10)
64*5754a6ffSjmcneill #define XADCIF_CMDFIFO 0x10
65*5754a6ffSjmcneill #define XADCIF_RDFIFO 0x14
66*5754a6ffSjmcneill #define XADCIF_MCTL 0x18
67*5754a6ffSjmcneill
68*5754a6ffSjmcneill /* XADC registers */
69*5754a6ffSjmcneill #define XADC_STATUS_TEMP 0x00
70*5754a6ffSjmcneill #define XADC_STATUS_VCCINT 0x01
71*5754a6ffSjmcneill #define XADC_STATUS_VCCAUX 0x02
72*5754a6ffSjmcneill #define XADC_STATUS_VPVN 0x03
73*5754a6ffSjmcneill #define XADC_STATUS_VREFP 0x04
74*5754a6ffSjmcneill #define XADC_STATUS_VREFN 0x05
75*5754a6ffSjmcneill #define XADC_STATUS_VCCBRAM 0x06
76*5754a6ffSjmcneill #define XADC_STATUS_VCCPINT 0x0d
77*5754a6ffSjmcneill #define XADC_STATUS_VCCPAUX 0x0e
78*5754a6ffSjmcneill #define XADC_STATUS_VCCO_DDR 0x0f
79*5754a6ffSjmcneill #define XADC_STATUS_FLAG 0x3f
80*5754a6ffSjmcneill #define FLAG_OT __BIT(3)
81*5754a6ffSjmcneill #define XADC_CONF(n) (0x40 + (n))
82*5754a6ffSjmcneill #define CONF1_SEQ __BITS(15,12)
83*5754a6ffSjmcneill #define CONF1_SEQ_CONT 2
84*5754a6ffSjmcneill #define XADC_SEQ(n) (0x48 + (n))
85*5754a6ffSjmcneill #define SEQ0_CALIB __BIT(0)
86*5754a6ffSjmcneill #define SEQ0_ALL __BITS(5,14)
87*5754a6ffSjmcneill #define SEQ4_VREFN __BIT(13)
88*5754a6ffSjmcneill
89*5754a6ffSjmcneill /* XADC commands */
90*5754a6ffSjmcneill #define XADC_COMMAND_CMD __BITS(29,26)
91*5754a6ffSjmcneill #define XADC_COMMAND_DRP_ADDR __BITS(25,16)
92*5754a6ffSjmcneill #define XADC_COMMAND_DRP_DATA __BITS(15,0)
93*5754a6ffSjmcneill #define XADC_COMMAND(cmd, addr, data) \
94*5754a6ffSjmcneill (__SHIFTIN(cmd, XADC_COMMAND_CMD) | \
95*5754a6ffSjmcneill __SHIFTIN(addr, XADC_COMMAND_DRP_ADDR) | \
96*5754a6ffSjmcneill __SHIFTIN(data, XADC_COMMAND_DRP_DATA))
97*5754a6ffSjmcneill #define XADC_CMD_NOP 0
98*5754a6ffSjmcneill #define XADC_CMD_READ 1
99*5754a6ffSjmcneill #define XADC_CMD_WRITE 2
100*5754a6ffSjmcneill
101*5754a6ffSjmcneill enum {
102*5754a6ffSjmcneill XADC_SENSOR_TEMP,
103*5754a6ffSjmcneill XADC_SENSOR_VCCINT,
104*5754a6ffSjmcneill XADC_SENSOR_VCCAUX,
105*5754a6ffSjmcneill XADC_SENSOR_VPVN,
106*5754a6ffSjmcneill XADC_SENSOR_VREFP,
107*5754a6ffSjmcneill XADC_SENSOR_VREFN,
108*5754a6ffSjmcneill XADC_SENSOR_VCCBRAM,
109*5754a6ffSjmcneill XADC_SENSOR_VCCPINT,
110*5754a6ffSjmcneill XADC_SENSOR_VCCPAUX,
111*5754a6ffSjmcneill XADC_SENSOR_VCCO_DDR,
112*5754a6ffSjmcneill XADC_NSENSOR
113*5754a6ffSjmcneill };
114*5754a6ffSjmcneill
115*5754a6ffSjmcneill static const struct {
116*5754a6ffSjmcneill const char *name;
117*5754a6ffSjmcneill uint32_t units;
118*5754a6ffSjmcneill uint16_t reg;
119*5754a6ffSjmcneill } zynq_xadc_sensors[] = {
120*5754a6ffSjmcneill [XADC_SENSOR_TEMP] = { "temperature", ENVSYS_STEMP, XADC_STATUS_TEMP },
121*5754a6ffSjmcneill [XADC_SENSOR_VCCINT] = { "vccint", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCINT },
122*5754a6ffSjmcneill [XADC_SENSOR_VCCAUX] = { "vccaux", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCAUX },
123*5754a6ffSjmcneill [XADC_SENSOR_VPVN] = { "vp/vn", ENVSYS_SVOLTS_DC, XADC_STATUS_VPVN },
124*5754a6ffSjmcneill [XADC_SENSOR_VREFP] = { "vrefp", ENVSYS_SVOLTS_DC, XADC_STATUS_VREFP },
125*5754a6ffSjmcneill [XADC_SENSOR_VREFN] = { "vrefn", ENVSYS_SVOLTS_DC, XADC_STATUS_VREFN },
126*5754a6ffSjmcneill [XADC_SENSOR_VCCBRAM] = { "vccbram", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCBRAM },
127*5754a6ffSjmcneill [XADC_SENSOR_VCCPINT] = { "vccpint", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCPINT },
128*5754a6ffSjmcneill [XADC_SENSOR_VCCPAUX] = { "vccpaux", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCPAUX },
129*5754a6ffSjmcneill [XADC_SENSOR_VCCO_DDR] = { "vcco_ddr", ENVSYS_SVOLTS_DC, XADC_STATUS_VCCO_DDR },
130*5754a6ffSjmcneill };
131*5754a6ffSjmcneill
132*5754a6ffSjmcneill static const struct device_compatible_entry compat_data[] = {
133*5754a6ffSjmcneill { .compat = "xlnx,zynq-xadc-1.00.a" },
134*5754a6ffSjmcneill DEVICE_COMPAT_EOL
135*5754a6ffSjmcneill };
136*5754a6ffSjmcneill
137*5754a6ffSjmcneill struct zynq_xadc_softc {
138*5754a6ffSjmcneill device_t sc_dev;
139*5754a6ffSjmcneill bus_space_tag_t sc_bst;
140*5754a6ffSjmcneill bus_space_handle_t sc_bsh;
141*5754a6ffSjmcneill kmutex_t sc_lock;
142*5754a6ffSjmcneill
143*5754a6ffSjmcneill struct sysmon_envsys *sc_sme;
144*5754a6ffSjmcneill envsys_data_t sc_sensor[XADC_NSENSOR];
145*5754a6ffSjmcneill };
146*5754a6ffSjmcneill
147*5754a6ffSjmcneill #define RD4(sc, reg) \
148*5754a6ffSjmcneill bus_space_read_4((sc)->sc_bst, (sc)->sc_bsh, (reg))
149*5754a6ffSjmcneill #define WR4(sc, reg, val) \
150*5754a6ffSjmcneill bus_space_write_4((sc)->sc_bst, (sc)->sc_bsh, (reg), (val))
151*5754a6ffSjmcneill
152*5754a6ffSjmcneill static int zynq_xadc_match(device_t, cfdata_t, void *);
153*5754a6ffSjmcneill static void zynq_xadc_attach(device_t, device_t, void *);
154*5754a6ffSjmcneill
155*5754a6ffSjmcneill CFATTACH_DECL_NEW(zynqxadc, sizeof(struct zynq_xadc_softc),
156*5754a6ffSjmcneill zynq_xadc_match, zynq_xadc_attach, NULL, NULL);
157*5754a6ffSjmcneill
158*5754a6ffSjmcneill static void
zynq_xadc_write(struct zynq_xadc_softc * sc,uint16_t reg,uint16_t data)159*5754a6ffSjmcneill zynq_xadc_write(struct zynq_xadc_softc *sc, uint16_t reg,
160*5754a6ffSjmcneill uint16_t data)
161*5754a6ffSjmcneill {
162*5754a6ffSjmcneill int retry = 10000;
163*5754a6ffSjmcneill
164*5754a6ffSjmcneill /*
165*5754a6ffSjmcneill * Write sequence is:
166*5754a6ffSjmcneill *
167*5754a6ffSjmcneill * 1. Prepare write command
168*5754a6ffSjmcneill * 2. Write data to Command FIFO
169*5754a6ffSjmcneill * 3. Wait until the Command FIFO becomes empty
170*5754a6ffSjmcneill */
171*5754a6ffSjmcneill
172*5754a6ffSjmcneill WR4(sc, XADCIF_CMDFIFO, XADC_COMMAND(XADC_CMD_WRITE, reg, data));
173*5754a6ffSjmcneill while (--retry > 0) {
174*5754a6ffSjmcneill if ((RD4(sc, XADCIF_MSTS) & MSTS_CFIFOE) != 0) {
175*5754a6ffSjmcneill break;
176*5754a6ffSjmcneill }
177*5754a6ffSjmcneill delay(10);
178*5754a6ffSjmcneill }
179*5754a6ffSjmcneill if (retry == 0) {
180*5754a6ffSjmcneill device_printf(sc->sc_dev, "command FIFO timeout (write)\n");
181*5754a6ffSjmcneill }
182*5754a6ffSjmcneill /*
183*5754a6ffSjmcneill * Every write to Command FIFO shifts data into Read FIFO, so
184*5754a6ffSjmcneill * drain that after the command completes.
185*5754a6ffSjmcneill */
186*5754a6ffSjmcneill RD4(sc, XADCIF_RDFIFO);
187*5754a6ffSjmcneill }
188*5754a6ffSjmcneill
189*5754a6ffSjmcneill static uint16_t
zynq_xadc_read(struct zynq_xadc_softc * sc,uint16_t reg)190*5754a6ffSjmcneill zynq_xadc_read(struct zynq_xadc_softc *sc, uint16_t reg)
191*5754a6ffSjmcneill {
192*5754a6ffSjmcneill int retry = 10000;
193*5754a6ffSjmcneill uint32_t val;
194*5754a6ffSjmcneill
195*5754a6ffSjmcneill /*
196*5754a6ffSjmcneill * Read sequence is:
197*5754a6ffSjmcneill *
198*5754a6ffSjmcneill * 1. Prepare read command
199*5754a6ffSjmcneill * 2. Write data to Command FIFO
200*5754a6ffSjmcneill * 3. Wait until the Command FIFO becomes empty
201*5754a6ffSjmcneill * 4. Read dummy data from the Read Data FIFO
202*5754a6ffSjmcneill * 5. Prepare nop command
203*5754a6ffSjmcneill * 6. Write data to Command FIFO
204*5754a6ffSjmcneill * 7. Read the Read Data FIFO
205*5754a6ffSjmcneill */
206*5754a6ffSjmcneill
207*5754a6ffSjmcneill WR4(sc, XADCIF_CMDFIFO, XADC_COMMAND(XADC_CMD_READ, reg, 0));
208*5754a6ffSjmcneill WR4(sc, XADCIF_CMDFIFO, XADC_COMMAND(XADC_CMD_NOP, 0, 0));
209*5754a6ffSjmcneill while (--retry > 0) {
210*5754a6ffSjmcneill if ((RD4(sc, XADCIF_MSTS) & MSTS_CFIFOE) != 0) {
211*5754a6ffSjmcneill break;
212*5754a6ffSjmcneill }
213*5754a6ffSjmcneill delay(10);
214*5754a6ffSjmcneill }
215*5754a6ffSjmcneill if (retry == 0) {
216*5754a6ffSjmcneill device_printf(sc->sc_dev, "command FIFO timeout (read)\n");
217*5754a6ffSjmcneill return 0xffff;
218*5754a6ffSjmcneill }
219*5754a6ffSjmcneill val = RD4(sc, XADCIF_RDFIFO);
220*5754a6ffSjmcneill val = RD4(sc, XADCIF_RDFIFO);
221*5754a6ffSjmcneill
222*5754a6ffSjmcneill return val & 0xffff;
223*5754a6ffSjmcneill }
224*5754a6ffSjmcneill
225*5754a6ffSjmcneill static void
zynq_xadc_init(struct zynq_xadc_softc * sc,struct clk * clk)226*5754a6ffSjmcneill zynq_xadc_init(struct zynq_xadc_softc *sc, struct clk *clk)
227*5754a6ffSjmcneill {
228*5754a6ffSjmcneill uint32_t val;
229*5754a6ffSjmcneill
230*5754a6ffSjmcneill /* Enable the PS-XADC interface */
231*5754a6ffSjmcneill val = RD4(sc, XADCIF_CFG);
232*5754a6ffSjmcneill val |= CFG_ENABLE;
233*5754a6ffSjmcneill val &= ~CFG_TCKRATE;
234*5754a6ffSjmcneill val |= __SHIFTIN(CFG_TCKRATE_DIV4, CFG_TCKRATE);
235*5754a6ffSjmcneill val |= CFG_WEDGE | CFG_REDGE;
236*5754a6ffSjmcneill WR4(sc, XADCIF_CFG, val);
237*5754a6ffSjmcneill WR4(sc, XADCIF_MCTL, 0);
238*5754a6ffSjmcneill
239*5754a6ffSjmcneill /* Turn on continuous sampling for all ADC channels we monitor */
240*5754a6ffSjmcneill zynq_xadc_write(sc, XADC_SEQ(0), SEQ0_CALIB | SEQ0_ALL);
241*5754a6ffSjmcneill zynq_xadc_write(sc, XADC_SEQ(4), SEQ4_VREFN);
242*5754a6ffSjmcneill zynq_xadc_write(sc, XADC_CONF(0), 0);
243*5754a6ffSjmcneill zynq_xadc_write(sc, XADC_CONF(1),
244*5754a6ffSjmcneill __SHIFTIN(CONF1_SEQ_CONT, CONF1_SEQ));
245*5754a6ffSjmcneill
246*5754a6ffSjmcneill }
247*5754a6ffSjmcneill
248*5754a6ffSjmcneill static void
zynq_xadc_sensors_refresh(struct sysmon_envsys * sme,envsys_data_t * edata)249*5754a6ffSjmcneill zynq_xadc_sensors_refresh(struct sysmon_envsys *sme, envsys_data_t *edata)
250*5754a6ffSjmcneill {
251*5754a6ffSjmcneill struct zynq_xadc_softc *sc = sme->sme_cookie;
252*5754a6ffSjmcneill union {
253*5754a6ffSjmcneill uint16_t u16;
254*5754a6ffSjmcneill int16_t s16;
255*5754a6ffSjmcneill } val;
256*5754a6ffSjmcneill int64_t temp;
257*5754a6ffSjmcneill
258*5754a6ffSjmcneill
259*5754a6ffSjmcneill val.u16 = zynq_xadc_read(sc, zynq_xadc_sensors[edata->sensor].reg);
260*5754a6ffSjmcneill if (edata->units == ENVSYS_STEMP) {
261*5754a6ffSjmcneill if (val.u16 == 0) {
262*5754a6ffSjmcneill edata->state = ENVSYS_SINVALID;
263*5754a6ffSjmcneill } else {
264*5754a6ffSjmcneill temp = ((int64_t)(val.u16 >> 4) * 503975) / 4096;
265*5754a6ffSjmcneill edata->value_cur = 1000 * temp;
266*5754a6ffSjmcneill edata->state = ENVSYS_SVALID;
267*5754a6ffSjmcneill }
268*5754a6ffSjmcneill
269*5754a6ffSjmcneill val.u16 = zynq_xadc_read(sc, XADC_STATUS_FLAG);
270*5754a6ffSjmcneill if ((val.u16 & FLAG_OT) != 0) {
271*5754a6ffSjmcneill edata->state = ENVSYS_SCRITOVER;
272*5754a6ffSjmcneill }
273*5754a6ffSjmcneill } else {
274*5754a6ffSjmcneill KASSERT(edata->units == ENVSYS_SVOLTS_DC);
275*5754a6ffSjmcneill switch (edata->sensor) {
276*5754a6ffSjmcneill case XADC_SENSOR_VPVN:
277*5754a6ffSjmcneill edata->value_cur = (((val.u16 >> 4) * 1000) / 4096) * 1000;
278*5754a6ffSjmcneill break;
279*5754a6ffSjmcneill case XADC_SENSOR_VREFN:
280*5754a6ffSjmcneill edata->value_cur = (((val.s16 >> 4) * 3000) / 4096) * 1000;
281*5754a6ffSjmcneill break;
282*5754a6ffSjmcneill default:
283*5754a6ffSjmcneill edata->value_cur = (((val.u16 >> 4) * 3000) / 4096) * 1000;
284*5754a6ffSjmcneill break;
285*5754a6ffSjmcneill }
286*5754a6ffSjmcneill edata->state = ENVSYS_SVALID;
287*5754a6ffSjmcneill }
288*5754a6ffSjmcneill }
289*5754a6ffSjmcneill
290*5754a6ffSjmcneill static int
zynq_xadc_match(device_t parent,cfdata_t cf,void * aux)291*5754a6ffSjmcneill zynq_xadc_match(device_t parent, cfdata_t cf, void *aux)
292*5754a6ffSjmcneill {
293*5754a6ffSjmcneill struct fdt_attach_args * const faa = aux;
294*5754a6ffSjmcneill
295*5754a6ffSjmcneill return of_compatible_match(faa->faa_phandle, compat_data);
296*5754a6ffSjmcneill }
297*5754a6ffSjmcneill
298*5754a6ffSjmcneill static void
zynq_xadc_attach(device_t parent,device_t self,void * aux)299*5754a6ffSjmcneill zynq_xadc_attach(device_t parent, device_t self, void *aux)
300*5754a6ffSjmcneill {
301*5754a6ffSjmcneill struct zynq_xadc_softc * const sc = device_private(self);
302*5754a6ffSjmcneill struct fdt_attach_args * const faa = aux;
303*5754a6ffSjmcneill const int phandle = faa->faa_phandle;
304*5754a6ffSjmcneill struct clk *clk;
305*5754a6ffSjmcneill bus_addr_t addr;
306*5754a6ffSjmcneill bus_size_t size;
307*5754a6ffSjmcneill u_int n;
308*5754a6ffSjmcneill
309*5754a6ffSjmcneill if (fdtbus_get_reg(phandle, 0, &addr, &size) != 0) {
310*5754a6ffSjmcneill aprint_error(": couldn't get registers\n");
311*5754a6ffSjmcneill return;
312*5754a6ffSjmcneill }
313*5754a6ffSjmcneill clk = fdtbus_clock_get_index(phandle, 0);
314*5754a6ffSjmcneill if (clk == NULL || clk_enable(clk) != 0) {
315*5754a6ffSjmcneill aprint_error(": couldn't enable clock\n");
316*5754a6ffSjmcneill return;
317*5754a6ffSjmcneill }
318*5754a6ffSjmcneill
319*5754a6ffSjmcneill sc->sc_dev = self;
320*5754a6ffSjmcneill sc->sc_bst = faa->faa_bst;
321*5754a6ffSjmcneill if (bus_space_map(sc->sc_bst, addr, size, 0, &sc->sc_bsh) != 0) {
322*5754a6ffSjmcneill aprint_error(": couldn't map registers\n");
323*5754a6ffSjmcneill return;
324*5754a6ffSjmcneill }
325*5754a6ffSjmcneill mutex_init(&sc->sc_lock, MUTEX_DEFAULT, IPL_VM);
326*5754a6ffSjmcneill
327*5754a6ffSjmcneill aprint_naive("\n");
328*5754a6ffSjmcneill aprint_normal(": ADC\n");
329*5754a6ffSjmcneill
330*5754a6ffSjmcneill zynq_xadc_init(sc, clk);
331*5754a6ffSjmcneill
332*5754a6ffSjmcneill sc->sc_sme = sysmon_envsys_create();
333*5754a6ffSjmcneill sc->sc_sme->sme_name = device_xname(self);
334*5754a6ffSjmcneill sc->sc_sme->sme_cookie = sc;
335*5754a6ffSjmcneill sc->sc_sme->sme_refresh = zynq_xadc_sensors_refresh;
336*5754a6ffSjmcneill
337*5754a6ffSjmcneill for (n = 0; n < XADC_NSENSOR; n++) {
338*5754a6ffSjmcneill sc->sc_sensor[n].units = zynq_xadc_sensors[n].units;
339*5754a6ffSjmcneill sc->sc_sensor[n].state = ENVSYS_SINVALID;
340*5754a6ffSjmcneill sc->sc_sensor[n].flags = ENVSYS_FHAS_ENTROPY;
341*5754a6ffSjmcneill if (zynq_xadc_sensors[n].units == ENVSYS_STEMP) {
342*5754a6ffSjmcneill sc->sc_sensor[n].flags |= ENVSYS_FMONCRITICAL;
343*5754a6ffSjmcneill }
344*5754a6ffSjmcneill strncpy(sc->sc_sensor[n].desc, zynq_xadc_sensors[n].name,
345*5754a6ffSjmcneill sizeof(sc->sc_sensor[n].desc));
346*5754a6ffSjmcneill sysmon_envsys_sensor_attach(sc->sc_sme, &sc->sc_sensor[n]);
347*5754a6ffSjmcneill }
348*5754a6ffSjmcneill
349*5754a6ffSjmcneill sysmon_envsys_register(sc->sc_sme);
350*5754a6ffSjmcneill }
351