xref: /openbsd-src/sys/dev/ipmi.c (revision 5f3b3788e991ae1c9b79d51dcc9b88892de3ac45)
1*5f3b3788Sgkoehler /*	$OpenBSD: ipmi.c,v 1.119 2024/04/03 18:32:47 gkoehler Exp $ */
2fbca7c3fSmarco 
3fbca7c3fSmarco /*
48f330473Suebayasi  * Copyright (c) 2015 Masao Uebayashi
5fbca7c3fSmarco  * Copyright (c) 2005 Jordan Hargrave
6fbca7c3fSmarco  * All rights reserved.
7fbca7c3fSmarco  *
8fbca7c3fSmarco  * Redistribution and use in source and binary forms, with or without
9fbca7c3fSmarco  * modification, are permitted provided that the following conditions
10fbca7c3fSmarco  * are met:
11fbca7c3fSmarco  * 1. Redistributions of source code must retain the above copyright
12fbca7c3fSmarco  *    notice, this list of conditions and the following disclaimer.
13fbca7c3fSmarco  * 2. Redistributions in binary form must reproduce the above copyright
14fbca7c3fSmarco  *    notice, this list of conditions and the following disclaimer in the
15fbca7c3fSmarco  *    documentation and/or other materials provided with the distribution.
16fbca7c3fSmarco  *
17fbca7c3fSmarco  * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
18fbca7c3fSmarco  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19fbca7c3fSmarco  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20fbca7c3fSmarco  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR
21fbca7c3fSmarco  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22fbca7c3fSmarco  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23fbca7c3fSmarco  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24fbca7c3fSmarco  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25fbca7c3fSmarco  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26fbca7c3fSmarco  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27fbca7c3fSmarco  * SUCH DAMAGE.
28fbca7c3fSmarco  */
29fbca7c3fSmarco 
30fbca7c3fSmarco #include <sys/param.h>
31fbca7c3fSmarco #include <sys/systm.h>
32fbca7c3fSmarco #include <sys/kernel.h>
33fbca7c3fSmarco #include <sys/device.h>
348f330473Suebayasi #include <sys/ioctl.h>
35fbca7c3fSmarco #include <sys/extent.h>
36fbca7c3fSmarco #include <sys/sensors.h>
37fbca7c3fSmarco #include <sys/malloc.h>
381d18187bSjordan #include <sys/kthread.h>
3966590f60Suebayasi #include <sys/task.h>
40fbca7c3fSmarco 
41fbca7c3fSmarco #include <machine/bus.h>
42de8bfd7dSgwk #include <machine/smbiosvar.h>
43fbca7c3fSmarco 
44fbca7c3fSmarco #include <dev/ipmivar.h>
458f330473Suebayasi #include <dev/ipmi.h>
46fbca7c3fSmarco 
47fbca7c3fSmarco struct ipmi_sensor {
48fbca7c3fSmarco 	u_int8_t	*i_sdr;
49fbca7c3fSmarco 	int		i_num;
50bb816960Sjordan 	int		stype;
51bb816960Sjordan 	int		etype;
52275cbf62Sderaadt 	struct		ksensor i_sensor;
53fbca7c3fSmarco 	SLIST_ENTRY(ipmi_sensor) list;
54fbca7c3fSmarco };
55fbca7c3fSmarco 
56bd7ba471Smarco int	ipmi_enabled = 0;
57fbca7c3fSmarco 
5854277de9Scheloha #define SENSOR_REFRESH_RATE 5	/* seconds */
59fbca7c3fSmarco 
601d18187bSjordan #define DEVNAME(s)  ((s)->sc_dev.dv_xname)
611d18187bSjordan 
62fbca7c3fSmarco #define IPMI_BTMSG_LEN			0
63fbca7c3fSmarco #define IPMI_BTMSG_NFLN			1
64fbca7c3fSmarco #define IPMI_BTMSG_SEQ			2
65fbca7c3fSmarco #define IPMI_BTMSG_CMD			3
66fbca7c3fSmarco #define IPMI_BTMSG_CCODE		4
67bb816960Sjordan #define IPMI_BTMSG_DATASND		4
68bb816960Sjordan #define IPMI_BTMSG_DATARCV		5
69fbca7c3fSmarco 
7016dc177fSsthen /* IPMI 2.0, Table 42-3: Sensor Type Codes */
71bfec59d3Sjordan #define IPMI_SENSOR_TYPE_TEMP		0x0101
72bfec59d3Sjordan #define IPMI_SENSOR_TYPE_VOLT		0x0102
7316dc177fSsthen #define IPMI_SENSOR_TYPE_CURRENT	0x0103
74bfec59d3Sjordan #define IPMI_SENSOR_TYPE_FAN		0x0104
75bfec59d3Sjordan #define IPMI_SENSOR_TYPE_INTRUSION	0x6F05
76bfec59d3Sjordan #define IPMI_SENSOR_TYPE_PWRSUPPLY	0x6F08
77bb816960Sjordan 
7816dc177fSsthen /* IPMI 2.0, Table 43-15: Sensor Unit Type Codes */
7916dc177fSsthen #define IPMI_UNIT_TYPE_DEGREE_C		1
8016dc177fSsthen #define IPMI_UNIT_TYPE_DEGREE_F		2
8116dc177fSsthen #define IPMI_UNIT_TYPE_DEGREE_K		3
8216dc177fSsthen #define IPMI_UNIT_TYPE_VOLTS		4
8316dc177fSsthen #define IPMI_UNIT_TYPE_AMPS		5
8416dc177fSsthen #define IPMI_UNIT_TYPE_WATTS		6
8516dc177fSsthen #define IPMI_UNIT_TYPE_RPM		18
8616dc177fSsthen 
878cdc17caSjordan #define IPMI_NAME_UNICODE		0x00
888cdc17caSjordan #define IPMI_NAME_BCDPLUS		0x01
898cdc17caSjordan #define IPMI_NAME_ASCII6BIT		0x02
908cdc17caSjordan #define IPMI_NAME_ASCII8BIT		0x03
918cdc17caSjordan 
92bb816960Sjordan #define IPMI_ENTITY_PWRSUPPLY		0x0A
93fbca7c3fSmarco 
94bfec59d3Sjordan #define IPMI_INVALID_SENSOR		(1L << 5)
958866e14aSuebayasi #define IPMI_DISABLED_SENSOR		(1L << 6)
96bfec59d3Sjordan 
978cdc17caSjordan #define IPMI_SDR_TYPEFULL		1
988cdc17caSjordan #define IPMI_SDR_TYPECOMPACT		2
998cdc17caSjordan 
100fbca7c3fSmarco #define byteof(x) ((x) >> 3)
101fbca7c3fSmarco #define bitof(x)  (1L << ((x) & 0x7))
102fbca7c3fSmarco #define TB(b,m)	  (data[2+byteof(b)] & bitof(b))
103fbca7c3fSmarco 
10427dce37dSreyk #ifdef IPMI_DEBUG
10527dce37dSreyk int	ipmi_dbg = 0;
1064dd6fa05Sderaadt #define dbg_printf(lvl, fmt...) \
1074dd6fa05Sderaadt 	if (ipmi_dbg >= lvl) \
1084dd6fa05Sderaadt 		printf(fmt);
1094dd6fa05Sderaadt #define dbg_dump(lvl, msg, len, buf) \
1104dd6fa05Sderaadt 	if (len && ipmi_dbg >= lvl) \
1114dd6fa05Sderaadt 		dumpb(msg, len, (const u_int8_t *)(buf));
11227dce37dSreyk #else
11327dce37dSreyk #define dbg_printf(lvl, fmt...)
11427dce37dSreyk #define dbg_dump(lvl, msg, len, buf)
11527dce37dSreyk #endif
116fbca7c3fSmarco 
1177a3397e7Sjordan long signextend(unsigned long, int);
1187a3397e7Sjordan 
119fbca7c3fSmarco SLIST_HEAD(ipmi_sensors_head, ipmi_sensor);
120fbca7c3fSmarco struct ipmi_sensors_head ipmi_sensor_list =
1217887b06bSguenther     SLIST_HEAD_INITIALIZER(ipmi_sensor_list);
122fbca7c3fSmarco 
123fbca7c3fSmarco void	dumpb(const char *, int, const u_int8_t *);
124fbca7c3fSmarco 
125fbca7c3fSmarco int	read_sensor(struct ipmi_softc *, struct ipmi_sensor *);
1262409b544Suebayasi int	add_sdr_sensor(struct ipmi_softc *, u_int8_t *, int);
127fbca7c3fSmarco int	get_sdr_partial(struct ipmi_softc *, u_int16_t, u_int16_t,
128fbca7c3fSmarco 	    u_int8_t, u_int8_t, void *, u_int16_t *);
129fbca7c3fSmarco int	get_sdr(struct ipmi_softc *, u_int16_t, u_int16_t *);
130fbca7c3fSmarco 
1310fe7a73cSuebayasi int	ipmi_sendcmd(struct ipmi_cmd *);
1320fe7a73cSuebayasi int	ipmi_recvcmd(struct ipmi_cmd *);
133f640b71bSuebayasi void	ipmi_cmd(struct ipmi_cmd *);
134f640b71bSuebayasi void	ipmi_cmd_poll(struct ipmi_cmd *);
135f640b71bSuebayasi void	ipmi_cmd_wait(struct ipmi_cmd *);
136f640b71bSuebayasi void	ipmi_cmd_wait_cb(void *);
137fbca7c3fSmarco 
1387a3397e7Sjordan int	ipmi_watchdog(void *, int);
139e824610bSuebayasi void	ipmi_watchdog_tickle(void *);
140e824610bSuebayasi void	ipmi_watchdog_set(void *);
1417a3397e7Sjordan 
1428f330473Suebayasi struct ipmi_softc *ipmilookup(dev_t dev);
1438f330473Suebayasi 
1448f330473Suebayasi int	ipmiopen(dev_t, int, int, struct proc *);
1458f330473Suebayasi int	ipmiclose(dev_t, int, int, struct proc *);
1468f330473Suebayasi int	ipmiioctl(dev_t, u_long, caddr_t, int, struct proc *);
147fbca7c3fSmarco 
148fbca7c3fSmarco long	ipow(long, int);
149b6d6d087Smarco long	ipmi_convert(u_int8_t, struct sdrtype1 *, long);
1502409b544Suebayasi int	ipmi_sensor_name(char *, int, u_int8_t, u_int8_t *, int);
151fbca7c3fSmarco 
152fbca7c3fSmarco /* BMC Helper Functions */
153bb816960Sjordan u_int8_t bmc_read(struct ipmi_softc *, int);
154bb816960Sjordan void	bmc_write(struct ipmi_softc *, int, u_int8_t);
1558f330473Suebayasi int	bmc_io_wait(struct ipmi_softc *, struct ipmi_iowait *);
156fbca7c3fSmarco 
157235dcbf6Suebayasi void	bt_buildmsg(struct ipmi_cmd *);
158235dcbf6Suebayasi void	cmn_buildmsg(struct ipmi_cmd *);
159fbca7c3fSmarco 
160fbca7c3fSmarco int	getbits(u_int8_t *, int, int);
16116dc177fSsthen int	ipmi_sensor_type(int, int, int, int);
162fbca7c3fSmarco 
163fbca7c3fSmarco void	ipmi_refresh_sensors(struct ipmi_softc *sc);
164fbca7c3fSmarco int	ipmi_map_regs(struct ipmi_softc *sc, struct ipmi_attach_args *ia);
1654b532468Smarco void	ipmi_unmap_regs(struct ipmi_softc *);
166fbca7c3fSmarco 
167bb816960Sjordan int	ipmi_sensor_status(struct ipmi_softc *, struct ipmi_sensor *,
168bb816960Sjordan     u_int8_t *);
169bb816960Sjordan 
170bb816960Sjordan int	 add_child_sensors(struct ipmi_softc *, u_int8_t *, int, int, int,
171bb816960Sjordan     int, int, int, const char *);
172bb816960Sjordan 
173c08dc278Skettenis void	ipmi_create_thread(void *);
174c08dc278Skettenis void	ipmi_poll_thread(void *);
175c08dc278Skettenis 
176c08dc278Skettenis int	kcs_probe(struct ipmi_softc *);
177c08dc278Skettenis int	kcs_reset(struct ipmi_softc *);
178c08dc278Skettenis int	kcs_sendmsg(struct ipmi_cmd *);
179c08dc278Skettenis int	kcs_recvmsg(struct ipmi_cmd *);
180c08dc278Skettenis 
181c08dc278Skettenis int	bt_probe(struct ipmi_softc *);
182c08dc278Skettenis int	bt_reset(struct ipmi_softc *);
183c08dc278Skettenis int	bt_sendmsg(struct ipmi_cmd *);
184c08dc278Skettenis int	bt_recvmsg(struct ipmi_cmd *);
185c08dc278Skettenis 
186c08dc278Skettenis int	smic_probe(struct ipmi_softc *);
187c08dc278Skettenis int	smic_reset(struct ipmi_softc *);
188c08dc278Skettenis int	smic_sendmsg(struct ipmi_cmd *);
189c08dc278Skettenis int	smic_recvmsg(struct ipmi_cmd *);
190c08dc278Skettenis 
191fbca7c3fSmarco struct ipmi_if kcs_if = {
19235701cd5Sderaadt 	"KCS",
193fbca7c3fSmarco 	IPMI_IF_KCS_NREGS,
194fbca7c3fSmarco 	cmn_buildmsg,
195fbca7c3fSmarco 	kcs_sendmsg,
196fbca7c3fSmarco 	kcs_recvmsg,
197fbca7c3fSmarco 	kcs_reset,
198fbca7c3fSmarco 	kcs_probe,
199779678c0Suebayasi 	IPMI_MSG_DATASND,
200779678c0Suebayasi 	IPMI_MSG_DATARCV,
201fbca7c3fSmarco };
202fbca7c3fSmarco 
203fbca7c3fSmarco struct ipmi_if smic_if = {
20435701cd5Sderaadt 	"SMIC",
205fbca7c3fSmarco 	IPMI_IF_SMIC_NREGS,
206fbca7c3fSmarco 	cmn_buildmsg,
207fbca7c3fSmarco 	smic_sendmsg,
208fbca7c3fSmarco 	smic_recvmsg,
209fbca7c3fSmarco 	smic_reset,
210fbca7c3fSmarco 	smic_probe,
211779678c0Suebayasi 	IPMI_MSG_DATASND,
212779678c0Suebayasi 	IPMI_MSG_DATARCV,
213fbca7c3fSmarco };
214fbca7c3fSmarco 
215fbca7c3fSmarco struct ipmi_if bt_if = {
21635701cd5Sderaadt 	"BT",
217fbca7c3fSmarco 	IPMI_IF_BT_NREGS,
218fbca7c3fSmarco 	bt_buildmsg,
219fbca7c3fSmarco 	bt_sendmsg,
220fbca7c3fSmarco 	bt_recvmsg,
221fbca7c3fSmarco 	bt_reset,
222fbca7c3fSmarco 	bt_probe,
223779678c0Suebayasi 	IPMI_BTMSG_DATASND,
224779678c0Suebayasi 	IPMI_BTMSG_DATARCV,
225fbca7c3fSmarco };
226fbca7c3fSmarco 
227fbca7c3fSmarco struct ipmi_if *ipmi_get_if(int);
228c5c15106Sderaadt 
229fbca7c3fSmarco struct ipmi_if *
ipmi_get_if(int iftype)230fbca7c3fSmarco ipmi_get_if(int iftype)
231fbca7c3fSmarco {
232fbca7c3fSmarco 	switch (iftype) {
233fbca7c3fSmarco 	case IPMI_IF_KCS:
234fbca7c3fSmarco 		return (&kcs_if);
235fbca7c3fSmarco 	case IPMI_IF_SMIC:
236fbca7c3fSmarco 		return (&smic_if);
237fbca7c3fSmarco 	case IPMI_IF_BT:
238fbca7c3fSmarco 		return (&bt_if);
239fbca7c3fSmarco 	}
240fbca7c3fSmarco 
241fbca7c3fSmarco 	return (NULL);
242fbca7c3fSmarco }
243fbca7c3fSmarco 
244fbca7c3fSmarco /*
245fbca7c3fSmarco  * BMC Helper Functions
246fbca7c3fSmarco  */
247bb816960Sjordan u_int8_t
bmc_read(struct ipmi_softc * sc,int offset)248fbca7c3fSmarco bmc_read(struct ipmi_softc *sc, int offset)
249fbca7c3fSmarco {
250cda7789cSkettenis 	if (sc->sc_if_iosize == 4)
251cda7789cSkettenis 		return (bus_space_read_4(sc->sc_iot, sc->sc_ioh,
252cda7789cSkettenis 		    offset * sc->sc_if_iospacing));
253cda7789cSkettenis 	else
254fbca7c3fSmarco 		return (bus_space_read_1(sc->sc_iot, sc->sc_ioh,
255fbca7c3fSmarco 		    offset * sc->sc_if_iospacing));
256fbca7c3fSmarco }
257fbca7c3fSmarco 
258fbca7c3fSmarco void
bmc_write(struct ipmi_softc * sc,int offset,u_int8_t val)259bb816960Sjordan bmc_write(struct ipmi_softc *sc, int offset, u_int8_t val)
260fbca7c3fSmarco {
261cda7789cSkettenis 	if (sc->sc_if_iosize == 4)
262cda7789cSkettenis 		bus_space_write_4(sc->sc_iot, sc->sc_ioh,
263cda7789cSkettenis 		    offset * sc->sc_if_iospacing, val);
264cda7789cSkettenis 	else
265fbca7c3fSmarco 		bus_space_write_1(sc->sc_iot, sc->sc_ioh,
266fbca7c3fSmarco 		    offset * sc->sc_if_iospacing, val);
267fbca7c3fSmarco }
268fbca7c3fSmarco 
269fbca7c3fSmarco int
bmc_io_wait(struct ipmi_softc * sc,struct ipmi_iowait * a)2708f330473Suebayasi bmc_io_wait(struct ipmi_softc *sc, struct ipmi_iowait *a)
271b2bc3525Smarco {
272b2bc3525Smarco 	volatile u_int8_t	v;
273987084d8Smarco 	int			count = 5000000; /* == 5s XXX can be shorter */
274b2bc3525Smarco 
275fbca7c3fSmarco 	while (count--) {
2768f330473Suebayasi 		v = bmc_read(sc, a->offset);
2778f330473Suebayasi 		if ((v & a->mask) == a->value)
278fbca7c3fSmarco 			return v;
279987084d8Smarco 
280987084d8Smarco 		delay(1);
281fbca7c3fSmarco 	}
282b2bc3525Smarco 
2838f330473Suebayasi 	dbg_printf(1, "%s: bmc_io_wait fails : *v=%.2x m=%.2x b=%.2x %s\n",
2848f330473Suebayasi 	    DEVNAME(sc), v, a->mask, a->value, a->lbl);
285fbca7c3fSmarco 	return (-1);
286b2bc3525Smarco 
287fbca7c3fSmarco }
288fbca7c3fSmarco 
2898f330473Suebayasi #define RSSA_MASK 0xff
2908f330473Suebayasi #define LUN_MASK 0x3
2918f330473Suebayasi #define NETFN_LUN(nf,ln) (((nf) << 2) | ((ln) & LUN_MASK))
292fbca7c3fSmarco 
293fbca7c3fSmarco /*
294fbca7c3fSmarco  * BT interface
295fbca7c3fSmarco  */
296fbca7c3fSmarco #define _BT_CTRL_REG			0
297bb816960Sjordan #define	  BT_CLR_WR_PTR			(1L << 0)
298bb816960Sjordan #define	  BT_CLR_RD_PTR			(1L << 1)
299bb816960Sjordan #define	  BT_HOST2BMC_ATN		(1L << 2)
300bb816960Sjordan #define	  BT_BMC2HOST_ATN		(1L << 3)
301bb816960Sjordan #define	  BT_EVT_ATN			(1L << 4)
302bb816960Sjordan #define	  BT_HOST_BUSY			(1L << 6)
303bb816960Sjordan #define	  BT_BMC_BUSY			(1L << 7)
304bb816960Sjordan 
305bb816960Sjordan #define	  BT_READY	(BT_HOST_BUSY|BT_HOST2BMC_ATN|BT_BMC2HOST_ATN)
306fbca7c3fSmarco 
307fbca7c3fSmarco #define _BT_DATAIN_REG			1
308fbca7c3fSmarco #define _BT_DATAOUT_REG			1
309bb816960Sjordan 
310fbca7c3fSmarco #define _BT_INTMASK_REG			2
311bb816960Sjordan #define	 BT_IM_HIRQ_PEND		(1L << 1)
312bb816960Sjordan #define	 BT_IM_SCI_EN			(1L << 2)
313bb816960Sjordan #define	 BT_IM_SMI_EN			(1L << 3)
314bb816960Sjordan #define	 BT_IM_NMI2SMI			(1L << 4)
315fbca7c3fSmarco 
3166bb54963Sjordan int bt_read(struct ipmi_softc *, int);
3176bb54963Sjordan int bt_write(struct ipmi_softc *, int, uint8_t);
3186bb54963Sjordan 
3196bb54963Sjordan int
bt_read(struct ipmi_softc * sc,int reg)3206bb54963Sjordan bt_read(struct ipmi_softc *sc, int reg)
3216bb54963Sjordan {
3226bb54963Sjordan 	return bmc_read(sc, reg);
3236bb54963Sjordan }
3246bb54963Sjordan 
3256bb54963Sjordan int
bt_write(struct ipmi_softc * sc,int reg,uint8_t data)3266bb54963Sjordan bt_write(struct ipmi_softc *sc, int reg, uint8_t data)
3276bb54963Sjordan {
3288f330473Suebayasi 	struct ipmi_iowait a;
3298f330473Suebayasi 
3308f330473Suebayasi 	a.offset = _BT_CTRL_REG;
3318f330473Suebayasi 	a.mask = BT_BMC_BUSY;
3328f330473Suebayasi 	a.value = 0;
3338f330473Suebayasi 	a.lbl = "bt_write";
3348f330473Suebayasi 	if (bmc_io_wait(sc, &a) < 0)
3356bb54963Sjordan 		return (-1);
3366bb54963Sjordan 
3376bb54963Sjordan 	bmc_write(sc, reg, data);
3386bb54963Sjordan 	return (0);
3396bb54963Sjordan }
3406bb54963Sjordan 
341fbca7c3fSmarco int
bt_sendmsg(struct ipmi_cmd * c)342779678c0Suebayasi bt_sendmsg(struct ipmi_cmd *c)
343fbca7c3fSmarco {
344779678c0Suebayasi 	struct ipmi_softc *sc = c->c_sc;
3458f330473Suebayasi 	struct ipmi_iowait a;
346fbca7c3fSmarco 	int i;
347fbca7c3fSmarco 
3486bb54963Sjordan 	bt_write(sc, _BT_CTRL_REG, BT_CLR_WR_PTR);
349779678c0Suebayasi 	for (i = 0; i < c->c_txlen; i++)
350779678c0Suebayasi 		bt_write(sc, _BT_DATAOUT_REG, sc->sc_buf[i]);
351fbca7c3fSmarco 
3526bb54963Sjordan 	bt_write(sc, _BT_CTRL_REG, BT_HOST2BMC_ATN);
3538f330473Suebayasi 	a.offset = _BT_CTRL_REG;
3548f330473Suebayasi 	a.mask = BT_HOST2BMC_ATN | BT_BMC_BUSY;
3558f330473Suebayasi 	a.value = 0;
3568f330473Suebayasi 	a.lbl = "bt_sendwait";
3578f330473Suebayasi 	if (bmc_io_wait(sc, &a) < 0)
358b2bc3525Smarco 		return (-1);
359bb816960Sjordan 
360fbca7c3fSmarco 	return (0);
361fbca7c3fSmarco }
362fbca7c3fSmarco 
363fbca7c3fSmarco int
bt_recvmsg(struct ipmi_cmd * c)364779678c0Suebayasi bt_recvmsg(struct ipmi_cmd *c)
365fbca7c3fSmarco {
366779678c0Suebayasi 	struct ipmi_softc *sc = c->c_sc;
3678f330473Suebayasi 	struct ipmi_iowait a;
368779678c0Suebayasi 	u_int8_t len, v, i, j;
369fbca7c3fSmarco 
3708f330473Suebayasi 	a.offset = _BT_CTRL_REG;
3718f330473Suebayasi 	a.mask = BT_BMC2HOST_ATN;
3728f330473Suebayasi 	a.value = BT_BMC2HOST_ATN;
3738f330473Suebayasi 	a.lbl = "bt_recvwait";
3748f330473Suebayasi 	if (bmc_io_wait(sc, &a) < 0)
3756bb54963Sjordan 		return (-1);
3766bb54963Sjordan 
3776bb54963Sjordan 	bt_write(sc, _BT_CTRL_REG, BT_HOST_BUSY);
3786bb54963Sjordan 	bt_write(sc, _BT_CTRL_REG, BT_BMC2HOST_ATN);
3796bb54963Sjordan 	bt_write(sc, _BT_CTRL_REG, BT_CLR_RD_PTR);
3806bb54963Sjordan 	len = bt_read(sc, _BT_DATAIN_REG);
381779678c0Suebayasi 	for (i = IPMI_BTMSG_NFLN, j = 0; i <= len; i++) {
3826bb54963Sjordan 		v = bt_read(sc, _BT_DATAIN_REG);
383fbca7c3fSmarco 		if (i != IPMI_BTMSG_SEQ)
384779678c0Suebayasi 			*(sc->sc_buf + j++) = v;
385fbca7c3fSmarco 	}
3866bb54963Sjordan 	bt_write(sc, _BT_CTRL_REG, BT_HOST_BUSY);
387779678c0Suebayasi 	c->c_rxlen = len - 1;
388fbca7c3fSmarco 
389fbca7c3fSmarco 	return (0);
390fbca7c3fSmarco }
391fbca7c3fSmarco 
392fbca7c3fSmarco int
bt_reset(struct ipmi_softc * sc)393fbca7c3fSmarco bt_reset(struct ipmi_softc *sc)
394fbca7c3fSmarco {
395fbca7c3fSmarco 	return (-1);
396fbca7c3fSmarco }
397fbca7c3fSmarco 
398fbca7c3fSmarco int
bt_probe(struct ipmi_softc * sc)399fbca7c3fSmarco bt_probe(struct ipmi_softc *sc)
400fbca7c3fSmarco {
401bb816960Sjordan 	u_int8_t rv;
402fbca7c3fSmarco 
403bb816960Sjordan 	rv = bmc_read(sc, _BT_CTRL_REG);
404bb816960Sjordan 	rv &= BT_HOST_BUSY;
405bb816960Sjordan 	rv |= BT_CLR_WR_PTR|BT_CLR_RD_PTR|BT_BMC2HOST_ATN|BT_HOST2BMC_ATN;
406bb816960Sjordan 	bmc_write(sc, _BT_CTRL_REG, rv);
407bb816960Sjordan 
408bb816960Sjordan 	rv = bmc_read(sc, _BT_INTMASK_REG);
409bb816960Sjordan 	rv &= BT_IM_SCI_EN|BT_IM_SMI_EN|BT_IM_NMI2SMI;
410bb816960Sjordan 	rv |= BT_IM_HIRQ_PEND;
411bb816960Sjordan 	bmc_write(sc, _BT_INTMASK_REG, rv);
412bb816960Sjordan 
413fbca7c3fSmarco #if 0
414fbca7c3fSmarco 	printf("bt_probe: %2x\n", v);
415fbca7c3fSmarco 	printf(" WR    : %2x\n", v & BT_CLR_WR_PTR);
416fbca7c3fSmarco 	printf(" RD    : %2x\n", v & BT_CLR_RD_PTR);
417fbca7c3fSmarco 	printf(" H2B   : %2x\n", v & BT_HOST2BMC_ATN);
418fbca7c3fSmarco 	printf(" B2H   : %2x\n", v & BT_BMC2HOST_ATN);
419fbca7c3fSmarco 	printf(" EVT   : %2x\n", v & BT_EVT_ATN);
420fbca7c3fSmarco 	printf(" HBSY  : %2x\n", v & BT_HOST_BUSY);
421fbca7c3fSmarco 	printf(" BBSY  : %2x\n", v & BT_BMC_BUSY);
422fbca7c3fSmarco #endif
423bb816960Sjordan 	return (0);
424fbca7c3fSmarco }
425fbca7c3fSmarco 
426fbca7c3fSmarco /*
427fbca7c3fSmarco  * SMIC interface
428fbca7c3fSmarco  */
429fbca7c3fSmarco #define _SMIC_DATAIN_REG		0
430fbca7c3fSmarco #define _SMIC_DATAOUT_REG		0
431fbca7c3fSmarco 
432fbca7c3fSmarco #define _SMIC_CTRL_REG			1
433b2d63a69Sjordan #define	  SMS_CC_GET_STATUS		 0x40
434b2d63a69Sjordan #define	  SMS_CC_START_TRANSFER		 0x41
435b2d63a69Sjordan #define	  SMS_CC_NEXT_TRANSFER		 0x42
436b2d63a69Sjordan #define	  SMS_CC_END_TRANSFER		 0x43
437b2d63a69Sjordan #define	  SMS_CC_START_RECEIVE		 0x44
438b2d63a69Sjordan #define	  SMS_CC_NEXT_RECEIVE		 0x45
439b2d63a69Sjordan #define	  SMS_CC_END_RECEIVE		 0x46
440b2d63a69Sjordan #define	  SMS_CC_TRANSFER_ABORT		 0x47
441b2d63a69Sjordan 
442b2d63a69Sjordan #define	  SMS_SC_READY			 0xc0
443b2d63a69Sjordan #define	  SMS_SC_WRITE_START		 0xc1
444b2d63a69Sjordan #define	  SMS_SC_WRITE_NEXT		 0xc2
445b2d63a69Sjordan #define	  SMS_SC_WRITE_END		 0xc3
446b2d63a69Sjordan #define	  SMS_SC_READ_START		 0xc4
447b2d63a69Sjordan #define	  SMS_SC_READ_NEXT		 0xc5
448b2d63a69Sjordan #define	  SMS_SC_READ_END		 0xc6
449fbca7c3fSmarco 
450fbca7c3fSmarco #define _SMIC_FLAG_REG			2
451bb816960Sjordan #define	  SMIC_BUSY			(1L << 0)
452bb816960Sjordan #define	  SMIC_SMS_ATN			(1L << 2)
453bb816960Sjordan #define	  SMIC_EVT_ATN			(1L << 3)
454bb816960Sjordan #define	  SMIC_SMI			(1L << 4)
455bb816960Sjordan #define	  SMIC_TX_DATA_RDY		(1L << 6)
456bb816960Sjordan #define	  SMIC_RX_DATA_RDY		(1L << 7)
457fbca7c3fSmarco 
458b2d63a69Sjordan int	smic_wait(struct ipmi_softc *, u_int8_t, u_int8_t, const char *);
459b2d63a69Sjordan int	smic_write_cmd_data(struct ipmi_softc *, u_int8_t, const u_int8_t *);
460b2d63a69Sjordan int	smic_read_data(struct ipmi_softc *, u_int8_t *);
461b2d63a69Sjordan 
462fbca7c3fSmarco int
smic_wait(struct ipmi_softc * sc,u_int8_t mask,u_int8_t val,const char * lbl)463fbca7c3fSmarco smic_wait(struct ipmi_softc *sc, u_int8_t mask, u_int8_t val, const char *lbl)
464fbca7c3fSmarco {
4658f330473Suebayasi 	struct ipmi_iowait a;
466b2d63a69Sjordan 	int v;
467b2d63a69Sjordan 
468b2d63a69Sjordan 	/* Wait for expected flag bits */
4698f330473Suebayasi 	a.offset = _SMIC_FLAG_REG;
4708f330473Suebayasi 	a.mask = mask;
4718f330473Suebayasi 	a.value = val;
4728f330473Suebayasi 	a.lbl = "smicwait";
4738f330473Suebayasi 	v = bmc_io_wait(sc, &a);
474b2d63a69Sjordan 	if (v < 0)
475b2d63a69Sjordan 		return (-1);
476b2d63a69Sjordan 
477b2d63a69Sjordan 	/* Return current status */
478b2d63a69Sjordan 	v = bmc_read(sc, _SMIC_CTRL_REG);
479b2d63a69Sjordan 	dbg_printf(99, "smic_wait = %.2x\n", v);
480b2d63a69Sjordan 	return (v);
481fbca7c3fSmarco }
482fbca7c3fSmarco 
483fbca7c3fSmarco int
smic_write_cmd_data(struct ipmi_softc * sc,u_int8_t cmd,const u_int8_t * data)484fbca7c3fSmarco smic_write_cmd_data(struct ipmi_softc *sc, u_int8_t cmd, const u_int8_t *data)
485fbca7c3fSmarco {
486b2d63a69Sjordan 	int	sts, v;
487fbca7c3fSmarco 
488b2d63a69Sjordan 	dbg_printf(50, "smic_wcd: %.2x %.2x\n", cmd, data ? *data : -1);
489fbca7c3fSmarco 	sts = smic_wait(sc, SMIC_TX_DATA_RDY | SMIC_BUSY, SMIC_TX_DATA_RDY,
490fbca7c3fSmarco 	    "smic_write_cmd_data ready");
491b2d63a69Sjordan 	if (sts < 0)
492fbca7c3fSmarco 		return (sts);
493fbca7c3fSmarco 
494fbca7c3fSmarco 	bmc_write(sc, _SMIC_CTRL_REG, cmd);
495fbca7c3fSmarco 	if (data)
496fbca7c3fSmarco 		bmc_write(sc, _SMIC_DATAOUT_REG, *data);
497fbca7c3fSmarco 
498b2d63a69Sjordan 	/* Toggle BUSY bit, then wait for busy bit to clear */
499fbca7c3fSmarco 	v = bmc_read(sc, _SMIC_FLAG_REG);
500fbca7c3fSmarco 	bmc_write(sc, _SMIC_FLAG_REG, v | SMIC_BUSY);
501fbca7c3fSmarco 
502fbca7c3fSmarco 	return (smic_wait(sc, SMIC_BUSY, 0, "smic_write_cmd_data busy"));
503fbca7c3fSmarco }
504fbca7c3fSmarco 
505fbca7c3fSmarco int
smic_read_data(struct ipmi_softc * sc,u_int8_t * data)506fbca7c3fSmarco smic_read_data(struct ipmi_softc *sc, u_int8_t *data)
507fbca7c3fSmarco {
508b2d63a69Sjordan 	int sts;
509b2d63a69Sjordan 
510fbca7c3fSmarco 	sts = smic_wait(sc, SMIC_RX_DATA_RDY | SMIC_BUSY, SMIC_RX_DATA_RDY,
511fbca7c3fSmarco 	    "smic_read_data");
512b2d63a69Sjordan 	if (sts >= 0) {
513fbca7c3fSmarco 		*data = bmc_read(sc, _SMIC_DATAIN_REG);
514b2d63a69Sjordan 		dbg_printf(50, "smic_readdata: %.2x\n", *data);
515b2d63a69Sjordan 	}
516fbca7c3fSmarco 	return (sts);
517fbca7c3fSmarco }
518b2d63a69Sjordan 
5194dd6fa05Sderaadt #define ErrStat(a,b) if (a) printf(b);
520fbca7c3fSmarco 
521fbca7c3fSmarco int
smic_sendmsg(struct ipmi_cmd * c)522779678c0Suebayasi smic_sendmsg(struct ipmi_cmd *c)
523fbca7c3fSmarco {
524779678c0Suebayasi 	struct ipmi_softc *sc = c->c_sc;
525fbca7c3fSmarco 	int sts, idx;
526fbca7c3fSmarco 
527779678c0Suebayasi 	sts = smic_write_cmd_data(sc, SMS_CC_START_TRANSFER, &sc->sc_buf[0]);
528b2d63a69Sjordan 	ErrStat(sts != SMS_SC_WRITE_START, "wstart");
529779678c0Suebayasi 	for (idx = 1; idx < c->c_txlen - 1; idx++) {
530b2d63a69Sjordan 		sts = smic_write_cmd_data(sc, SMS_CC_NEXT_TRANSFER,
531779678c0Suebayasi 		    &sc->sc_buf[idx]);
532b2d63a69Sjordan 		ErrStat(sts != SMS_SC_WRITE_NEXT, "write");
533bb816960Sjordan 	}
534779678c0Suebayasi 	sts = smic_write_cmd_data(sc, SMS_CC_END_TRANSFER, &sc->sc_buf[idx]);
535b2d63a69Sjordan 	if (sts != SMS_SC_WRITE_END) {
536779678c0Suebayasi 		dbg_printf(50, "smic_sendmsg %d/%d = %.2x\n", idx, c->c_txlen, sts);
537fbca7c3fSmarco 		return (-1);
538fbca7c3fSmarco 	}
539fbca7c3fSmarco 
540b2d63a69Sjordan 	return (0);
541b2d63a69Sjordan }
542b2d63a69Sjordan 
543fbca7c3fSmarco int
smic_recvmsg(struct ipmi_cmd * c)544779678c0Suebayasi smic_recvmsg(struct ipmi_cmd *c)
545fbca7c3fSmarco {
546779678c0Suebayasi 	struct ipmi_softc *sc = c->c_sc;
547fbca7c3fSmarco 	int sts, idx;
548fbca7c3fSmarco 
549779678c0Suebayasi 	c->c_rxlen = 0;
550b2d63a69Sjordan 	sts = smic_wait(sc, SMIC_RX_DATA_RDY, SMIC_RX_DATA_RDY, "smic_recvmsg");
5514dd6fa05Sderaadt 	if (sts < 0)
552b2d63a69Sjordan 		return (-1);
553b2d63a69Sjordan 
554fbca7c3fSmarco 	sts = smic_write_cmd_data(sc, SMS_CC_START_RECEIVE, NULL);
555b2d63a69Sjordan 	ErrStat(sts != SMS_SC_READ_START, "rstart");
556b2d63a69Sjordan 	for (idx = 0;; ) {
557779678c0Suebayasi 		sts = smic_read_data(sc, &sc->sc_buf[idx++]);
5584dd6fa05Sderaadt 		if (sts != SMS_SC_READ_START && sts != SMS_SC_READ_NEXT)
559b2d63a69Sjordan 			break;
560fbca7c3fSmarco 		smic_write_cmd_data(sc, SMS_CC_NEXT_RECEIVE, NULL);
561fbca7c3fSmarco 	}
562b2d63a69Sjordan 	ErrStat(sts != SMS_SC_READ_END, "rend");
563b2d63a69Sjordan 
564779678c0Suebayasi 	c->c_rxlen = idx;
565b2d63a69Sjordan 
566b2d63a69Sjordan 	sts = smic_write_cmd_data(sc, SMS_CC_END_RECEIVE, NULL);
567b2d63a69Sjordan 	if (sts != SMS_SC_READY) {
568779678c0Suebayasi 		dbg_printf(50, "smic_recvmsg %d/%d = %.2x\n", idx, c->c_maxrxlen, sts);
569fbca7c3fSmarco 		return (-1);
570fbca7c3fSmarco 	}
571fbca7c3fSmarco 
572b2d63a69Sjordan 	return (0);
573b2d63a69Sjordan }
574b2d63a69Sjordan 
575fbca7c3fSmarco int
smic_reset(struct ipmi_softc * sc)576fbca7c3fSmarco smic_reset(struct ipmi_softc *sc)
577fbca7c3fSmarco {
578fbca7c3fSmarco 	return (-1);
579fbca7c3fSmarco }
580fbca7c3fSmarco 
581fbca7c3fSmarco int
smic_probe(struct ipmi_softc * sc)582fbca7c3fSmarco smic_probe(struct ipmi_softc *sc)
583fbca7c3fSmarco {
584b2d63a69Sjordan 	/* Flag register should not be 0xFF on a good system */
585b2d63a69Sjordan 	if (bmc_read(sc, _SMIC_FLAG_REG) == 0xFF)
586fbca7c3fSmarco 		return (-1);
587b2d63a69Sjordan 
588b2d63a69Sjordan 	return (0);
589fbca7c3fSmarco }
590fbca7c3fSmarco 
591fbca7c3fSmarco /*
592fbca7c3fSmarco  * KCS interface
593fbca7c3fSmarco  */
594fbca7c3fSmarco #define _KCS_DATAIN_REGISTER		0
595fbca7c3fSmarco #define _KCS_DATAOUT_REGISTER		0
596fbca7c3fSmarco #define	  KCS_READ_NEXT			0x68
597fbca7c3fSmarco 
598fbca7c3fSmarco #define _KCS_COMMAND_REGISTER		1
599fbca7c3fSmarco #define	  KCS_GET_STATUS		0x60
600fbca7c3fSmarco #define	  KCS_WRITE_START		0x61
601fbca7c3fSmarco #define	  KCS_WRITE_END			0x62
602fbca7c3fSmarco 
603fbca7c3fSmarco #define _KCS_STATUS_REGISTER		1
604bb816960Sjordan #define	  KCS_OBF			(1L << 0)
605bb816960Sjordan #define	  KCS_IBF			(1L << 1)
606bb816960Sjordan #define	  KCS_SMS_ATN			(1L << 2)
607bb816960Sjordan #define	  KCS_CD			(1L << 3)
608bb816960Sjordan #define	  KCS_OEM1			(1L << 4)
609bb816960Sjordan #define	  KCS_OEM2			(1L << 5)
610fbca7c3fSmarco #define	  KCS_STATE_MASK		0xc0
611fbca7c3fSmarco #define	    KCS_IDLE_STATE		0x00
612fbca7c3fSmarco #define	    KCS_READ_STATE		0x40
613fbca7c3fSmarco #define	    KCS_WRITE_STATE		0x80
614fbca7c3fSmarco #define	    KCS_ERROR_STATE		0xC0
615fbca7c3fSmarco 
616fbca7c3fSmarco int	kcs_wait(struct ipmi_softc *, u_int8_t, u_int8_t, const char *);
617fbca7c3fSmarco int	kcs_write_cmd(struct ipmi_softc *, u_int8_t);
618fbca7c3fSmarco int	kcs_write_data(struct ipmi_softc *, u_int8_t);
619fbca7c3fSmarco int	kcs_read_data(struct ipmi_softc *, u_int8_t *);
620fbca7c3fSmarco 
621fbca7c3fSmarco int
kcs_wait(struct ipmi_softc * sc,u_int8_t mask,u_int8_t value,const char * lbl)622fbca7c3fSmarco kcs_wait(struct ipmi_softc *sc, u_int8_t mask, u_int8_t value, const char *lbl)
623fbca7c3fSmarco {
6248f330473Suebayasi 	struct ipmi_iowait a;
625fbca7c3fSmarco 	int v;
626fbca7c3fSmarco 
6278f330473Suebayasi 	a.offset = _KCS_STATUS_REGISTER;
6288f330473Suebayasi 	a.mask = mask;
6298f330473Suebayasi 	a.value = value;
6308f330473Suebayasi 	a.lbl = lbl;
6318f330473Suebayasi 	v = bmc_io_wait(sc, &a);
632fbca7c3fSmarco 	if (v < 0)
633fbca7c3fSmarco 		return (v);
634fbca7c3fSmarco 
635fbca7c3fSmarco 	/* Check if output buffer full, read dummy byte	 */
636b2bc3525Smarco 	if ((v & (KCS_OBF | KCS_STATE_MASK)) == (KCS_OBF | KCS_WRITE_STATE))
637fbca7c3fSmarco 		bmc_read(sc, _KCS_DATAIN_REGISTER);
638fbca7c3fSmarco 
639fbca7c3fSmarco 	/* Check for error state */
640fbca7c3fSmarco 	if ((v & KCS_STATE_MASK) == KCS_ERROR_STATE) {
641fbca7c3fSmarco 		bmc_write(sc, _KCS_COMMAND_REGISTER, KCS_GET_STATUS);
6420d17b928Sderaadt 		while (bmc_read(sc, _KCS_STATUS_REGISTER) & KCS_IBF)
64342fa0292Stedu 			continue;
6448deac126Smarco 		printf("%s: error code: %x\n", DEVNAME(sc),
6458deac126Smarco 		    bmc_read(sc, _KCS_DATAIN_REGISTER));
646fbca7c3fSmarco 	}
647fbca7c3fSmarco 
648fbca7c3fSmarco 	return (v & KCS_STATE_MASK);
649fbca7c3fSmarco }
650fbca7c3fSmarco 
651fbca7c3fSmarco int
kcs_write_cmd(struct ipmi_softc * sc,u_int8_t cmd)652fbca7c3fSmarco kcs_write_cmd(struct ipmi_softc *sc, u_int8_t cmd)
653fbca7c3fSmarco {
654fbca7c3fSmarco 	/* ASSERT: IBF and OBF are clear */
655f89aadebSmarco 	dbg_printf(50, "kcswritecmd: %.2x\n", cmd);
656fbca7c3fSmarco 	bmc_write(sc, _KCS_COMMAND_REGISTER, cmd);
657fbca7c3fSmarco 
658fbca7c3fSmarco 	return (kcs_wait(sc, KCS_IBF, 0, "write_cmd"));
659fbca7c3fSmarco }
660fbca7c3fSmarco 
661fbca7c3fSmarco int
kcs_write_data(struct ipmi_softc * sc,u_int8_t data)662fbca7c3fSmarco kcs_write_data(struct ipmi_softc *sc, u_int8_t data)
663fbca7c3fSmarco {
664fbca7c3fSmarco 	/* ASSERT: IBF and OBF are clear */
665f89aadebSmarco 	dbg_printf(50, "kcswritedata: %.2x\n", data);
666fbca7c3fSmarco 	bmc_write(sc, _KCS_DATAOUT_REGISTER, data);
667fbca7c3fSmarco 
668fbca7c3fSmarco 	return (kcs_wait(sc, KCS_IBF, 0, "write_data"));
669fbca7c3fSmarco }
670fbca7c3fSmarco 
671fbca7c3fSmarco int
kcs_read_data(struct ipmi_softc * sc,u_int8_t * data)672fbca7c3fSmarco kcs_read_data(struct ipmi_softc *sc, u_int8_t * data)
673fbca7c3fSmarco {
674fbca7c3fSmarco 	int sts;
675fbca7c3fSmarco 
676fbca7c3fSmarco 	sts = kcs_wait(sc, KCS_IBF | KCS_OBF, KCS_OBF, "read_data");
677fbca7c3fSmarco 	if (sts != KCS_READ_STATE)
678fbca7c3fSmarco 		return (sts);
679fbca7c3fSmarco 
680fbca7c3fSmarco 	/* ASSERT: OBF is set read data, request next byte */
681fbca7c3fSmarco 	*data = bmc_read(sc, _KCS_DATAIN_REGISTER);
682fbca7c3fSmarco 	bmc_write(sc, _KCS_DATAOUT_REGISTER, KCS_READ_NEXT);
683fbca7c3fSmarco 
684f89aadebSmarco 	dbg_printf(50, "kcsreaddata: %.2x\n", *data);
685fbca7c3fSmarco 
686fbca7c3fSmarco 	return (sts);
687fbca7c3fSmarco }
688fbca7c3fSmarco 
689fbca7c3fSmarco /* Exported KCS functions */
690fbca7c3fSmarco int
kcs_sendmsg(struct ipmi_cmd * c)691779678c0Suebayasi kcs_sendmsg(struct ipmi_cmd *c)
692fbca7c3fSmarco {
693779678c0Suebayasi 	struct ipmi_softc *sc = c->c_sc;
694fbca7c3fSmarco 	int idx, sts;
695fbca7c3fSmarco 
696fbca7c3fSmarco 	/* ASSERT: IBF is clear */
697779678c0Suebayasi 	dbg_dump(50, "kcs sendmsg", c->c_txlen, sc->sc_buf);
698fbca7c3fSmarco 	sts = kcs_write_cmd(sc, KCS_WRITE_START);
699779678c0Suebayasi 	for (idx = 0; idx < c->c_txlen; idx++) {
700779678c0Suebayasi 		if (idx == c->c_txlen - 1)
701fbca7c3fSmarco 			sts = kcs_write_cmd(sc, KCS_WRITE_END);
702fbca7c3fSmarco 
703fbca7c3fSmarco 		if (sts != KCS_WRITE_STATE)
704fbca7c3fSmarco 			break;
705fbca7c3fSmarco 
706779678c0Suebayasi 		sts = kcs_write_data(sc, sc->sc_buf[idx]);
707fbca7c3fSmarco 	}
708fbca7c3fSmarco 	if (sts != KCS_READ_STATE) {
709779678c0Suebayasi 		dbg_printf(1, "kcs sendmsg = %d/%d <%.2x>\n", idx, c->c_txlen, sts);
710779678c0Suebayasi 		dbg_dump(1, "kcs_sendmsg", c->c_txlen, sc->sc_buf);
711b2bc3525Smarco 		return (-1);
712fbca7c3fSmarco 	}
713fbca7c3fSmarco 
714b2bc3525Smarco 	return (0);
715fbca7c3fSmarco }
716fbca7c3fSmarco 
717fbca7c3fSmarco int
kcs_recvmsg(struct ipmi_cmd * c)718779678c0Suebayasi kcs_recvmsg(struct ipmi_cmd *c)
719fbca7c3fSmarco {
720779678c0Suebayasi 	struct ipmi_softc *sc = c->c_sc;
721fbca7c3fSmarco 	int idx, sts;
722fbca7c3fSmarco 
723779678c0Suebayasi 	for (idx = 0; idx < c->c_maxrxlen; idx++) {
724779678c0Suebayasi 		sts = kcs_read_data(sc, &sc->sc_buf[idx]);
725fbca7c3fSmarco 		if (sts != KCS_READ_STATE)
726fbca7c3fSmarco 			break;
727fbca7c3fSmarco 	}
728fbca7c3fSmarco 	sts = kcs_wait(sc, KCS_IBF, 0, "recv");
729779678c0Suebayasi 	c->c_rxlen = idx;
730b2bc3525Smarco 	if (sts != KCS_IDLE_STATE) {
731779678c0Suebayasi 		dbg_printf(1, "kcs recvmsg = %d/%d <%.2x>\n", idx, c->c_maxrxlen, sts);
732b2bc3525Smarco 		return (-1);
733b2bc3525Smarco 	}
734fbca7c3fSmarco 
735779678c0Suebayasi 	dbg_dump(50, "kcs recvmsg", idx, sc->sc_buf);
736fbca7c3fSmarco 
737b2bc3525Smarco 	return (0);
738fbca7c3fSmarco }
739fbca7c3fSmarco 
740fbca7c3fSmarco int
kcs_reset(struct ipmi_softc * sc)741fbca7c3fSmarco kcs_reset(struct ipmi_softc *sc)
742fbca7c3fSmarco {
743fbca7c3fSmarco 	return (-1);
744fbca7c3fSmarco }
745fbca7c3fSmarco 
746fbca7c3fSmarco int
kcs_probe(struct ipmi_softc * sc)747fbca7c3fSmarco kcs_probe(struct ipmi_softc *sc)
748fbca7c3fSmarco {
749fbca7c3fSmarco 	u_int8_t v;
750fbca7c3fSmarco 
751fbca7c3fSmarco 	v = bmc_read(sc, _KCS_STATUS_REGISTER);
7528f330473Suebayasi 	if ((v & KCS_STATE_MASK) == KCS_ERROR_STATE)
7538f330473Suebayasi 		return (1);
754fbca7c3fSmarco #if 0
755fbca7c3fSmarco 	printf("kcs_probe: %2x\n", v);
756fbca7c3fSmarco 	printf(" STS: %2x\n", v & KCS_STATE_MASK);
757fbca7c3fSmarco 	printf(" ATN: %2x\n", v & KCS_SMS_ATN);
758fbca7c3fSmarco 	printf(" C/D: %2x\n", v & KCS_CD);
759fbca7c3fSmarco 	printf(" IBF: %2x\n", v & KCS_IBF);
760fbca7c3fSmarco 	printf(" OBF: %2x\n", v & KCS_OBF);
761fbca7c3fSmarco #endif
762fbca7c3fSmarco 	return (0);
763fbca7c3fSmarco }
764fbca7c3fSmarco 
765fbca7c3fSmarco /*
766fbca7c3fSmarco  * IPMI code
767fbca7c3fSmarco  */
768fbca7c3fSmarco #define READ_SMS_BUFFER		0x37
769fbca7c3fSmarco #define WRITE_I2C		0x50
770fbca7c3fSmarco 
771fbca7c3fSmarco #define GET_MESSAGE_CMD		0x33
772fbca7c3fSmarco #define SEND_MESSAGE_CMD	0x34
773fbca7c3fSmarco 
774fbca7c3fSmarco #define IPMB_CHANNEL_NUMBER	0
775fbca7c3fSmarco 
776fbca7c3fSmarco #define PUBLIC_BUS		0
777fbca7c3fSmarco 
778fbca7c3fSmarco #define MIN_I2C_PACKET_SIZE	3
779fbca7c3fSmarco #define MIN_IMB_PACKET_SIZE	7	/* one byte for cksum */
780fbca7c3fSmarco 
781fbca7c3fSmarco #define MIN_BTBMC_REQ_SIZE	4
782fbca7c3fSmarco #define MIN_BTBMC_RSP_SIZE	5
783fbca7c3fSmarco #define MIN_BMC_REQ_SIZE	2
784fbca7c3fSmarco #define MIN_BMC_RSP_SIZE	3
785fbca7c3fSmarco 
786fbca7c3fSmarco #define BMC_SA			0x20	/* BMC/ESM3 */
787fbca7c3fSmarco #define FPC_SA			0x22	/* front panel */
788fbca7c3fSmarco #define BP_SA			0xC0	/* Primary Backplane */
789fbca7c3fSmarco #define BP2_SA			0xC2	/* Secondary Backplane */
790fbca7c3fSmarco #define PBP_SA			0xC4	/* Peripheral Backplane */
791fbca7c3fSmarco #define DRAC_SA			0x28	/* DRAC-III */
792fbca7c3fSmarco #define DRAC3_SA		0x30	/* DRAC-III */
793fbca7c3fSmarco #define BMC_LUN			0
794fbca7c3fSmarco #define SMS_LUN			2
795fbca7c3fSmarco 
796b6d6d087Smarco struct ipmi_request {
797fbca7c3fSmarco 	u_int8_t	rsSa;
798fbca7c3fSmarco 	u_int8_t	rsLun;
799fbca7c3fSmarco 	u_int8_t	netFn;
800fbca7c3fSmarco 	u_int8_t	cmd;
801fbca7c3fSmarco 	u_int8_t	data_len;
802fbca7c3fSmarco 	u_int8_t	*data;
803b6d6d087Smarco };
804fbca7c3fSmarco 
805b6d6d087Smarco struct ipmi_response {
806fbca7c3fSmarco 	u_int8_t	cCode;
807fbca7c3fSmarco 	u_int8_t	data_len;
808fbca7c3fSmarco 	u_int8_t	*data;
809b6d6d087Smarco };
810fbca7c3fSmarco 
811b6d6d087Smarco struct ipmi_bmc_request {
812fbca7c3fSmarco 	u_int8_t	bmc_nfLn;
813fbca7c3fSmarco 	u_int8_t	bmc_cmd;
814fbca7c3fSmarco 	u_int8_t	bmc_data_len;
815fbca7c3fSmarco 	u_int8_t	bmc_data[1];
816b6d6d087Smarco };
817fbca7c3fSmarco 
818b6d6d087Smarco struct ipmi_bmc_response {
819fbca7c3fSmarco 	u_int8_t	bmc_nfLn;
820fbca7c3fSmarco 	u_int8_t	bmc_cmd;
821fbca7c3fSmarco 	u_int8_t	bmc_cCode;
822fbca7c3fSmarco 	u_int8_t	bmc_data_len;
823fbca7c3fSmarco 	u_int8_t	bmc_data[1];
824b6d6d087Smarco };
825fbca7c3fSmarco 
826fbca7c3fSmarco struct cfdriver ipmi_cd = {
827fbca7c3fSmarco 	NULL, "ipmi", DV_DULL
828fbca7c3fSmarco };
829fbca7c3fSmarco 
830fbca7c3fSmarco void
dumpb(const char * lbl,int len,const u_int8_t * data)831fbca7c3fSmarco dumpb(const char *lbl, int len, const u_int8_t *data)
832fbca7c3fSmarco {
833fbca7c3fSmarco 	int idx;
834fbca7c3fSmarco 
835fbca7c3fSmarco 	printf("%s: ", lbl);
836fbca7c3fSmarco 	for (idx = 0; idx < len; idx++)
837fbca7c3fSmarco 		printf("%.2x ", data[idx]);
838fbca7c3fSmarco 
839fbca7c3fSmarco 	printf("\n");
840fbca7c3fSmarco }
841fbca7c3fSmarco 
842fbca7c3fSmarco /*
843fbca7c3fSmarco  * bt_buildmsg builds an IPMI message from a nfLun, cmd, and data
844fbca7c3fSmarco  * This is used by BT protocol
845fbca7c3fSmarco  */
846235dcbf6Suebayasi void
bt_buildmsg(struct ipmi_cmd * c)847dd9acc24Suebayasi bt_buildmsg(struct ipmi_cmd *c)
848fbca7c3fSmarco {
849dd9acc24Suebayasi 	struct ipmi_softc *sc = c->c_sc;
850235dcbf6Suebayasi 	u_int8_t *buf = sc->sc_buf;
851fbca7c3fSmarco 
852779678c0Suebayasi 	buf[IPMI_BTMSG_LEN] = c->c_txlen + (IPMI_BTMSG_DATASND - 1);
853dd9acc24Suebayasi 	buf[IPMI_BTMSG_NFLN] = NETFN_LUN(c->c_netfn, c->c_rslun);
854fbca7c3fSmarco 	buf[IPMI_BTMSG_SEQ] = sc->sc_btseq++;
855dd9acc24Suebayasi 	buf[IPMI_BTMSG_CMD] = c->c_cmd;
856dd9acc24Suebayasi 	if (c->c_txlen && c->c_data)
857dd9acc24Suebayasi 		memcpy(buf + IPMI_BTMSG_DATASND, c->c_data, c->c_txlen);
858fbca7c3fSmarco }
859fbca7c3fSmarco 
860fbca7c3fSmarco /*
861fbca7c3fSmarco  * cmn_buildmsg builds an IPMI message from a nfLun, cmd, and data
862fbca7c3fSmarco  * This is used by both SMIC and KCS protocols
863fbca7c3fSmarco  */
864235dcbf6Suebayasi void
cmn_buildmsg(struct ipmi_cmd * c)865dd9acc24Suebayasi cmn_buildmsg(struct ipmi_cmd *c)
866fbca7c3fSmarco {
867235dcbf6Suebayasi 	struct ipmi_softc *sc = c->c_sc;
868235dcbf6Suebayasi 	u_int8_t *buf = sc->sc_buf;
869fbca7c3fSmarco 
870dd9acc24Suebayasi 	buf[IPMI_MSG_NFLN] = NETFN_LUN(c->c_netfn, c->c_rslun);
871dd9acc24Suebayasi 	buf[IPMI_MSG_CMD] = c->c_cmd;
872dd9acc24Suebayasi 	if (c->c_txlen && c->c_data)
873dd9acc24Suebayasi 		memcpy(buf + IPMI_MSG_DATASND, c->c_data, c->c_txlen);
874fbca7c3fSmarco }
875fbca7c3fSmarco 
876fbca7c3fSmarco /* Send an IPMI command */
877fbca7c3fSmarco int
ipmi_sendcmd(struct ipmi_cmd * c)8780fe7a73cSuebayasi ipmi_sendcmd(struct ipmi_cmd *c)
879fbca7c3fSmarco {
8800fe7a73cSuebayasi 	struct ipmi_softc	*sc = c->c_sc;
881cfe233edSmarco 	int		rc = -1;
882fbca7c3fSmarco 
883f89aadebSmarco 	dbg_printf(50, "ipmi_sendcmd: rssa=%.2x nfln=%.2x cmd=%.2x len=%.2x\n",
8840fe7a73cSuebayasi 	    c->c_rssa, NETFN_LUN(c->c_netfn, c->c_rslun), c->c_cmd, c->c_txlen);
8850fe7a73cSuebayasi 	dbg_dump(10, " send", c->c_txlen, c->c_data);
8860fe7a73cSuebayasi 	if (c->c_rssa != BMC_SA) {
887fbca7c3fSmarco #if 0
888235dcbf6Suebayasi 		sc->sc_if->buildmsg(c);
889fbca7c3fSmarco 		pI2C->bus = (sc->if_ver == 0x09) ?
890fbca7c3fSmarco 		    PUBLIC_BUS :
891fbca7c3fSmarco 		    IPMB_CHANNEL_NUMBER;
892fbca7c3fSmarco 
893fbca7c3fSmarco 		imbreq->rsSa = rssa;
894fbca7c3fSmarco 		imbreq->nfLn = NETFN_LUN(netfn, rslun);
895fbca7c3fSmarco 		imbreq->cSum1 = -(imbreq->rsSa + imbreq->nfLn);
896fbca7c3fSmarco 		imbreq->rqSa = BMC_SA;
897fbca7c3fSmarco 		imbreq->seqLn = NETFN_LUN(sc->imb_seq++, SMS_LUN);
898fbca7c3fSmarco 		imbreq->cmd = cmd;
8994dd6fa05Sderaadt 		if (txlen)
900fbca7c3fSmarco 			memcpy(imbreq->data, data, txlen);
901fbca7c3fSmarco 		/* Set message checksum */
902fbca7c3fSmarco 		imbreq->data[txlen] = cksum8(&imbreq->rqSa, txlen + 3);
903fbca7c3fSmarco #endif
904cfe233edSmarco 		goto done;
905fbca7c3fSmarco 	} else
906235dcbf6Suebayasi 		sc->sc_if->buildmsg(c);
907fbca7c3fSmarco 
908779678c0Suebayasi 	c->c_txlen += sc->sc_if->datasnd;
909779678c0Suebayasi 	rc = sc->sc_if->sendmsg(c);
910fbca7c3fSmarco 
911cfe233edSmarco done:
912fbca7c3fSmarco 	return (rc);
913fbca7c3fSmarco }
914fbca7c3fSmarco 
9150fe7a73cSuebayasi /* Receive an IPMI command */
916fbca7c3fSmarco int
ipmi_recvcmd(struct ipmi_cmd * c)9170fe7a73cSuebayasi ipmi_recvcmd(struct ipmi_cmd *c)
918fbca7c3fSmarco {
9190fe7a73cSuebayasi 	struct ipmi_softc *sc = c->c_sc;
920235dcbf6Suebayasi 	u_int8_t	*buf = sc->sc_buf, rc = 0;
921fbca7c3fSmarco 
922fbca7c3fSmarco 	/* Receive message from interface, copy out result data */
923779678c0Suebayasi 	c->c_maxrxlen += sc->sc_if->datarcv;
924779678c0Suebayasi 	if (sc->sc_if->recvmsg(c) ||
925779678c0Suebayasi 	    c->c_rxlen < sc->sc_if->datarcv) {
926bb0da4eaSmarco 		return (-1);
9279b4396f8Sderaadt 	}
928fbca7c3fSmarco 
929779678c0Suebayasi 	c->c_rxlen -= sc->sc_if->datarcv;
9300fe7a73cSuebayasi 	if (c->c_rxlen > 0 && c->c_data)
931779678c0Suebayasi 		memcpy(c->c_data, buf + sc->sc_if->datarcv, c->c_rxlen);
932fbca7c3fSmarco 
93327dce37dSreyk 	rc = buf[IPMI_MSG_CCODE];
93427dce37dSreyk #ifdef IPMI_DEBUG
93527dce37dSreyk 	if (rc != 0)
936580f10a8Suebayasi 		dbg_printf(1, "ipmi_recvcmd: nfln=%.2x cmd=%.2x err=%.2x\n",
937fbca7c3fSmarco 		    buf[IPMI_MSG_NFLN], buf[IPMI_MSG_CMD], buf[IPMI_MSG_CCODE]);
93827dce37dSreyk #endif
939987084d8Smarco 
940f89aadebSmarco 	dbg_printf(50, "ipmi_recvcmd: nfln=%.2x cmd=%.2x err=%.2x len=%.2x\n",
941fbca7c3fSmarco 	    buf[IPMI_MSG_NFLN], buf[IPMI_MSG_CMD], buf[IPMI_MSG_CCODE],
9420fe7a73cSuebayasi 	    c->c_rxlen);
9430fe7a73cSuebayasi 	dbg_dump(10, " recv", c->c_rxlen, c->c_data);
944fbca7c3fSmarco 
945fbca7c3fSmarco 	return (rc);
946fbca7c3fSmarco }
947fbca7c3fSmarco 
948cfe233edSmarco void
ipmi_cmd(struct ipmi_cmd * c)949f640b71bSuebayasi ipmi_cmd(struct ipmi_cmd *c)
9500fe7a73cSuebayasi {
951f640b71bSuebayasi 	if (cold || panicstr != NULL)
952f640b71bSuebayasi 		ipmi_cmd_poll(c);
953f640b71bSuebayasi 	else
954f640b71bSuebayasi 		ipmi_cmd_wait(c);
955f640b71bSuebayasi }
956f640b71bSuebayasi 
957f640b71bSuebayasi void
ipmi_cmd_poll(struct ipmi_cmd * c)958f640b71bSuebayasi ipmi_cmd_poll(struct ipmi_cmd *c)
959f640b71bSuebayasi {
96011e5289aSyasuoka 	if ((c->c_ccode = ipmi_sendcmd(c)))
96111e5289aSyasuoka 		printf("%s: sendcmd fails\n", DEVNAME(c->c_sc));
96211e5289aSyasuoka 	else
9630fe7a73cSuebayasi 		c->c_ccode = ipmi_recvcmd(c);
964f640b71bSuebayasi }
965f640b71bSuebayasi 
966f640b71bSuebayasi void
ipmi_cmd_wait(struct ipmi_cmd * c)967f640b71bSuebayasi ipmi_cmd_wait(struct ipmi_cmd *c)
968f640b71bSuebayasi {
969f640b71bSuebayasi 	struct task t;
970f640b71bSuebayasi 	int res;
971f640b71bSuebayasi 
972f640b71bSuebayasi 	task_set(&t, ipmi_cmd_wait_cb, c);
973f640b71bSuebayasi 	res = task_add(c->c_sc->sc_cmd_taskq, &t);
974f640b71bSuebayasi 	KASSERT(res == 1);
975f640b71bSuebayasi 
97603604742Smpi 	tsleep_nsec(c, PWAIT, "ipmicmd", INFSLP);
977f640b71bSuebayasi 
978f640b71bSuebayasi 	res = task_del(c->c_sc->sc_cmd_taskq, &t);
979f640b71bSuebayasi 	KASSERT(res == 0);
980f640b71bSuebayasi }
981f640b71bSuebayasi 
982f640b71bSuebayasi void
ipmi_cmd_wait_cb(void * arg)983f640b71bSuebayasi ipmi_cmd_wait_cb(void *arg)
984f640b71bSuebayasi {
985f640b71bSuebayasi 	struct ipmi_cmd *c = arg;
986f640b71bSuebayasi 
987f640b71bSuebayasi 	ipmi_cmd_poll(c);
988f640b71bSuebayasi 	wakeup(c);
9890fe7a73cSuebayasi }
9900fe7a73cSuebayasi 
991fbca7c3fSmarco /* Read a partial SDR entry */
992fbca7c3fSmarco int
get_sdr_partial(struct ipmi_softc * sc,u_int16_t recordId,u_int16_t reserveId,u_int8_t offset,u_int8_t length,void * buffer,u_int16_t * nxtRecordId)993fbca7c3fSmarco get_sdr_partial(struct ipmi_softc *sc, u_int16_t recordId, u_int16_t reserveId,
994fbca7c3fSmarco     u_int8_t offset, u_int8_t length, void *buffer, u_int16_t *nxtRecordId)
995fbca7c3fSmarco {
9969bf13170Suebayasi 	u_int8_t	cmd[IPMI_GET_WDOG_MAX + 255];	/* 8 + max of length */
997fbca7c3fSmarco 	int		len;
998fbca7c3fSmarco 
999fbca7c3fSmarco 	((u_int16_t *) cmd)[0] = reserveId;
1000fbca7c3fSmarco 	((u_int16_t *) cmd)[1] = recordId;
1001fbca7c3fSmarco 	cmd[4] = offset;
1002fbca7c3fSmarco 	cmd[5] = length;
10030fe7a73cSuebayasi 
10040fe7a73cSuebayasi 	struct ipmi_cmd	c;
10050fe7a73cSuebayasi 	c.c_sc = sc;
10060fe7a73cSuebayasi 	c.c_rssa = BMC_SA;
10070fe7a73cSuebayasi 	c.c_rslun = BMC_LUN;
10080fe7a73cSuebayasi 	c.c_netfn = STORAGE_NETFN;
10090fe7a73cSuebayasi 	c.c_cmd = STORAGE_GET_SDR;
10109bf13170Suebayasi 	c.c_txlen = IPMI_SET_WDOG_MAX;
10110fe7a73cSuebayasi 	c.c_rxlen = 0;
10128ad1e6fcStedu 	c.c_maxrxlen = 8 + length;
10138f330473Suebayasi 	c.c_data = cmd;
10140fe7a73cSuebayasi 	ipmi_cmd(&c);
10150fe7a73cSuebayasi 	len = c.c_rxlen;
10160fe7a73cSuebayasi 
10174dd6fa05Sderaadt 	if (nxtRecordId)
1018fbca7c3fSmarco 		*nxtRecordId = *(uint16_t *) cmd;
1019161b560cSuebayasi 	if (len > 2)
1020fbca7c3fSmarco 		memcpy(buffer, cmd + 2, len - 2);
10212409b544Suebayasi 	else
10222409b544Suebayasi 		return (1);
1023fbca7c3fSmarco 
1024fbca7c3fSmarco 	return (0);
1025fbca7c3fSmarco }
1026fbca7c3fSmarco 
1027fbca7c3fSmarco int maxsdrlen = 0x10;
1028fbca7c3fSmarco 
1029fbca7c3fSmarco /* Read an entire SDR; pass to add sensor */
1030fbca7c3fSmarco int
get_sdr(struct ipmi_softc * sc,u_int16_t recid,u_int16_t * nxtrec)1031fbca7c3fSmarco get_sdr(struct ipmi_softc *sc, u_int16_t recid, u_int16_t *nxtrec)
1032fbca7c3fSmarco {
10336bb54963Sjordan 	u_int16_t	resid = 0;
1034fbca7c3fSmarco 	int		len, sdrlen, offset;
1035fbca7c3fSmarco 	u_int8_t	*psdr;
1036b6d6d087Smarco 	struct sdrhdr	shdr;
1037fbca7c3fSmarco 
1038fbca7c3fSmarco 	/* Reserve SDR */
10390fe7a73cSuebayasi 	struct ipmi_cmd	c;
10400fe7a73cSuebayasi 	c.c_sc = sc;
10410fe7a73cSuebayasi 	c.c_rssa = BMC_SA;
10420fe7a73cSuebayasi 	c.c_rslun = BMC_LUN;
10430fe7a73cSuebayasi 	c.c_netfn = STORAGE_NETFN;
10440fe7a73cSuebayasi 	c.c_cmd = STORAGE_RESERVE_SDR;
10450fe7a73cSuebayasi 	c.c_txlen = 0;
10460fe7a73cSuebayasi 	c.c_maxrxlen = sizeof(resid);
10470fe7a73cSuebayasi 	c.c_rxlen = 0;
10480fe7a73cSuebayasi 	c.c_data = &resid;
10490fe7a73cSuebayasi 	ipmi_cmd(&c);
10500fe7a73cSuebayasi 
1051fbca7c3fSmarco 	/* Get SDR Header */
1052b6d6d087Smarco 	if (get_sdr_partial(sc, recid, resid, 0, sizeof shdr, &shdr, nxtrec)) {
10530c1677f8Sderaadt 		printf("%s: get header fails\n", DEVNAME(sc));
1054c79f7e70Smarco 		return (1);
1055fbca7c3fSmarco 	}
1056fbca7c3fSmarco 	/* Allocate space for entire SDR Length of SDR in header does not
1057fbca7c3fSmarco 	 * include header length */
1058fbca7c3fSmarco 	sdrlen = sizeof(shdr) + shdr.record_length;
1059e825885dSgsoares 	psdr = malloc(sdrlen, M_DEVBUF, M_NOWAIT);
1060fbca7c3fSmarco 	if (psdr == NULL)
1061c79f7e70Smarco 		return (1);
1062fbca7c3fSmarco 
1063fbca7c3fSmarco 	memcpy(psdr, &shdr, sizeof(shdr));
1064fbca7c3fSmarco 
1065fbca7c3fSmarco 	/* Read SDR Data maxsdrlen bytes at a time */
1066fbca7c3fSmarco 	for (offset = sizeof(shdr); offset < sdrlen; offset += maxsdrlen) {
1067fbca7c3fSmarco 		len = sdrlen - offset;
1068fbca7c3fSmarco 		if (len > maxsdrlen)
1069fbca7c3fSmarco 			len = maxsdrlen;
1070fbca7c3fSmarco 
1071fbca7c3fSmarco 		if (get_sdr_partial(sc, recid, resid, offset, len,
1072fbca7c3fSmarco 		    psdr + offset, NULL)) {
1073f1b21d5fScnst 			printf("%s: get chunk: %d,%d fails\n", DEVNAME(sc),
1074f1b21d5fScnst 			    offset, len);
1075697457d8Sderaadt 			free(psdr, M_DEVBUF, sdrlen);
1076c79f7e70Smarco 			return (1);
1077fbca7c3fSmarco 		}
1078fbca7c3fSmarco 	}
1079fbca7c3fSmarco 
1080fbca7c3fSmarco 	/* Add SDR to sensor list, if not wanted, free buffer */
10812409b544Suebayasi 	if (add_sdr_sensor(sc, psdr, sdrlen) == 0)
1082697457d8Sderaadt 		free(psdr, M_DEVBUF, sdrlen);
1083fbca7c3fSmarco 
1084fbca7c3fSmarco 	return (0);
1085fbca7c3fSmarco }
1086fbca7c3fSmarco 
1087fbca7c3fSmarco int
getbits(u_int8_t * bytes,int bitpos,int bitlen)1088fbca7c3fSmarco getbits(u_int8_t *bytes, int bitpos, int bitlen)
1089fbca7c3fSmarco {
1090fbca7c3fSmarco 	int	v;
1091fbca7c3fSmarco 	int	mask;
1092fbca7c3fSmarco 
1093fbca7c3fSmarco 	bitpos += bitlen - 1;
1094fbca7c3fSmarco 	for (v = 0; bitlen--;) {
1095fbca7c3fSmarco 		v <<= 1;
1096fbca7c3fSmarco 		mask = 1L << (bitpos & 7);
10974dd6fa05Sderaadt 		if (bytes[bitpos >> 3] & mask)
1098fbca7c3fSmarco 			v |= 1;
1099fbca7c3fSmarco 		bitpos--;
1100fbca7c3fSmarco 	}
1101fbca7c3fSmarco 
1102fbca7c3fSmarco 	return (v);
1103fbca7c3fSmarco }
1104fbca7c3fSmarco 
1105fbca7c3fSmarco /* Decode IPMI sensor name */
11062409b544Suebayasi int
ipmi_sensor_name(char * name,int len,u_int8_t typelen,u_int8_t * bits,int bitslen)11072409b544Suebayasi ipmi_sensor_name(char *name, int len, u_int8_t typelen, u_int8_t *bits,
11082409b544Suebayasi     int bitslen)
1109fbca7c3fSmarco {
1110fbca7c3fSmarco 	int	i, slen;
1111fbca7c3fSmarco 	char	bcdplus[] = "0123456789 -.:,_";
1112fbca7c3fSmarco 
1113fbca7c3fSmarco 	slen = typelen & 0x1F;
11148cdc17caSjordan 	switch (typelen >> 6) {
11158cdc17caSjordan 	case IPMI_NAME_UNICODE:
1116fbca7c3fSmarco 		//unicode
1117fbca7c3fSmarco 		break;
1118fbca7c3fSmarco 
11198cdc17caSjordan 	case IPMI_NAME_BCDPLUS:
1120fbca7c3fSmarco 		/* Characters are encoded in 4-bit BCDPLUS */
11211788be7eSjordan 		if (len < slen * 2 + 1)
11221788be7eSjordan 			slen = (len >> 1) - 1;
11232409b544Suebayasi 		if (slen > bitslen)
11242409b544Suebayasi 			return (0);
1125fbca7c3fSmarco 		for (i = 0; i < slen; i++) {
1126fbca7c3fSmarco 			*(name++) = bcdplus[bits[i] >> 4];
1127fbca7c3fSmarco 			*(name++) = bcdplus[bits[i] & 0xF];
1128fbca7c3fSmarco 		}
1129fbca7c3fSmarco 		break;
1130fbca7c3fSmarco 
11318cdc17caSjordan 	case IPMI_NAME_ASCII6BIT:
1132bb816960Sjordan 		/* Characters are encoded in 6-bit ASCII
1133bb816960Sjordan 		 *   0x00 - 0x3F maps to 0x20 - 0x5F */
11341788be7eSjordan 		/* XXX: need to calculate max len: slen = 3/4 * len */
11351788be7eSjordan 		if (len < slen + 1)
11361788be7eSjordan 			slen = len - 1;
11372409b544Suebayasi 		if (slen * 6 / 8 > bitslen)
11382409b544Suebayasi 			return (0);
11392409b544Suebayasi 		for (i = 0; i < slen * 8; i += 6) {
1140fbca7c3fSmarco 			*(name++) = getbits(bits, i, 6) + ' ';
11412409b544Suebayasi 		}
1142fbca7c3fSmarco 		break;
1143fbca7c3fSmarco 
11448cdc17caSjordan 	case IPMI_NAME_ASCII8BIT:
1145bb816960Sjordan 		/* Characters are 8-bit ascii */
11461788be7eSjordan 		if (len < slen + 1)
11471788be7eSjordan 			slen = len - 1;
11482409b544Suebayasi 		if (slen > bitslen)
11492409b544Suebayasi 			return (0);
1150fbca7c3fSmarco 		while (slen--)
1151fbca7c3fSmarco 			*(name++) = *(bits++);
1152fbca7c3fSmarco 		break;
1153fbca7c3fSmarco 	}
1154fbca7c3fSmarco 	*name = 0;
11552409b544Suebayasi 
11562409b544Suebayasi 	return (1);
1157fbca7c3fSmarco }
1158fbca7c3fSmarco 
1159fbca7c3fSmarco /* Calculate val * 10^exp */
1160fbca7c3fSmarco long
ipow(long val,int exp)1161fbca7c3fSmarco ipow(long val, int exp)
1162fbca7c3fSmarco {
1163fbca7c3fSmarco 	while (exp > 0) {
1164fbca7c3fSmarco 		val *= 10;
1165fbca7c3fSmarco 		exp--;
1166fbca7c3fSmarco 	}
1167fbca7c3fSmarco 
1168fbca7c3fSmarco 	while (exp < 0) {
1169fbca7c3fSmarco 		val /= 10;
1170fbca7c3fSmarco 		exp++;
1171fbca7c3fSmarco 	}
1172fbca7c3fSmarco 
1173fbca7c3fSmarco 	return (val);
1174fbca7c3fSmarco }
1175fbca7c3fSmarco 
11767a3397e7Sjordan /* Sign extend a n-bit value */
11777a3397e7Sjordan long
signextend(unsigned long val,int bits)11787a3397e7Sjordan signextend(unsigned long val, int bits)
11797a3397e7Sjordan {
11807a3397e7Sjordan 	long msk = (1L << (bits-1))-1;
11817a3397e7Sjordan 
11827a3397e7Sjordan 	return (-(val & ~msk) | val);
11837a3397e7Sjordan }
11847a3397e7Sjordan 
1185fbca7c3fSmarco /* Convert IPMI reading from sensor factors */
1186fbca7c3fSmarco long
ipmi_convert(u_int8_t v,struct sdrtype1 * s1,long adj)1187b6d6d087Smarco ipmi_convert(u_int8_t v, struct sdrtype1 *s1, long adj)
1188fbca7c3fSmarco {
1189aad52b95Skettenis 	int16_t	M, B;
1190aad52b95Skettenis 	int8_t	K1, K2;
1191fbca7c3fSmarco 	long	val;
1192fbca7c3fSmarco 
11937a3397e7Sjordan 	/* Calculate linear reading variables */
11947a3397e7Sjordan 	M  = signextend((((short)(s1->m_tolerance & 0xC0)) << 2) + s1->m, 10);
11957a3397e7Sjordan 	B  = signextend((((short)(s1->b_accuracy & 0xC0)) << 2) + s1->b, 10);
11967a3397e7Sjordan 	K1 = signextend(s1->rbexp & 0xF, 4);
11977a3397e7Sjordan 	K2 = signextend(s1->rbexp >> 4, 4);
1198fbca7c3fSmarco 
1199bb816960Sjordan 	/* Calculate sensor reading:
1200bb816960Sjordan 	 *  y = L((M * v + (B * 10^K1)) * 10^(K2+adj)
1201fbca7c3fSmarco 	 *
1202bb816960Sjordan 	 * This commutes out to:
1203bb816960Sjordan 	 *  y = L(M*v * 10^(K2+adj) + B * 10^(K1+K2+adj)); */
1204fbca7c3fSmarco 	val = ipow(M * v, K2 + adj) + ipow(B, K1 + K2 + adj);
1205fbca7c3fSmarco 
1206fbca7c3fSmarco 	/* Linearization function: y = f(x) 0 : y = x 1 : y = ln(x) 2 : y =
1207fbca7c3fSmarco 	 * log10(x) 3 : y = log2(x) 4 : y = e^x 5 : y = 10^x 6 : y = 2^x 7 : y
1208fbca7c3fSmarco 	 * = 1/x 8 : y = x^2 9 : y = x^3 10 : y = square root(x) 11 : y = cube
1209fbca7c3fSmarco 	 * root(x) */
1210fbca7c3fSmarco 	return (val);
1211fbca7c3fSmarco }
1212fbca7c3fSmarco 
1213fbca7c3fSmarco int
ipmi_sensor_status(struct ipmi_softc * sc,struct ipmi_sensor * psensor,u_int8_t * reading)1214bb816960Sjordan ipmi_sensor_status(struct ipmi_softc *sc, struct ipmi_sensor *psensor,
1215bb816960Sjordan     u_int8_t *reading)
1216bb816960Sjordan {
1217b6d6d087Smarco 	struct sdrtype1	*s1 = (struct sdrtype1 *)psensor->i_sdr;
1218a0c22b0bSuebayasi 	int		etype;
1219bb816960Sjordan 
1220a6b31359Sjordan 	/* Get reading of sensor */
1221a6b31359Sjordan 	switch (psensor->i_sensor.type) {
1222a6b31359Sjordan 	case SENSOR_TEMP:
1223a6b31359Sjordan 		psensor->i_sensor.value = ipmi_convert(reading[0], s1, 6);
1224a6b31359Sjordan 		psensor->i_sensor.value += 273150000;
1225a6b31359Sjordan 		break;
1226a6b31359Sjordan 
1227a6b31359Sjordan 	case SENSOR_VOLTS_DC:
122816dc177fSsthen 	case SENSOR_VOLTS_AC:
122916dc177fSsthen 	case SENSOR_AMPS:
123016dc177fSsthen 	case SENSOR_WATTS:
1231a6b31359Sjordan 		psensor->i_sensor.value = ipmi_convert(reading[0], s1, 6);
1232a6b31359Sjordan 		break;
1233a6b31359Sjordan 
1234a6b31359Sjordan 	case SENSOR_FANRPM:
1235a6b31359Sjordan 		psensor->i_sensor.value = ipmi_convert(reading[0], s1, 0);
123681a90f72Swilfried 		if (((s1->units1>>3)&0x7) == 0x3)
123781a90f72Swilfried 			psensor->i_sensor.value *= 60; // RPS -> RPM
1238a6b31359Sjordan 		break;
1239a6b31359Sjordan 	default:
1240a6b31359Sjordan 		break;
1241a6b31359Sjordan 	}
1242a6b31359Sjordan 
1243a6b31359Sjordan 	/* Return Sensor Status */
1244bfec59d3Sjordan 	etype = (psensor->etype << 8) + psensor->stype;
1245bb816960Sjordan 	switch (etype) {
1246bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_TEMP:
1247bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_VOLT:
124816dc177fSsthen 	case IPMI_SENSOR_TYPE_CURRENT:
1249bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_FAN:
1250a0c22b0bSuebayasi 		/* non-recoverable threshold */
1251a0c22b0bSuebayasi 		if (reading[2] & ((1 << 5) | (1 << 2)))
1252bb816960Sjordan 			return (SENSOR_S_CRIT);
1253a0c22b0bSuebayasi 		/* critical threshold */
1254a0c22b0bSuebayasi 		else if (reading[2] & ((1 << 4) | (1 << 1)))
1255bb816960Sjordan 			return (SENSOR_S_CRIT);
1256a0c22b0bSuebayasi 		/* non-critical threshold */
1257a0c22b0bSuebayasi 		else if (reading[2] & ((1 << 3) | (1 << 0)))
1258bb816960Sjordan 			return (SENSOR_S_WARN);
1259bb816960Sjordan 		break;
1260bb816960Sjordan 
1261bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_INTRUSION:
1262bb816960Sjordan 		psensor->i_sensor.value = (reading[2] & 1) ? 1 : 0;
1263fbd1c07fSjordan 		if (reading[2] & 0x1)
1264fbd1c07fSjordan 			return (SENSOR_S_CRIT);
1265bb816960Sjordan 		break;
1266bb816960Sjordan 
1267bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_PWRSUPPLY:
126853a40c1fSjordan 		/* Reading: 1 = present+powered, 0 = otherwise */
126953a40c1fSjordan 		psensor->i_sensor.value = (reading[2] & 1) ? 1 : 0;
127053a40c1fSjordan 		if (reading[2] & 0x10) {
127153a40c1fSjordan 			/* XXX: Need sysctl type for Power Supply types
127253a40c1fSjordan 			 *   ok: power supply installed && powered
127353a40c1fSjordan 			 * warn: power supply installed && !powered
127453a40c1fSjordan 			 * crit: power supply !installed
127553a40c1fSjordan 			 */
127653a40c1fSjordan 			return (SENSOR_S_CRIT);
127753a40c1fSjordan 		}
127853a40c1fSjordan 		if (reading[2] & 0x08) {
127953a40c1fSjordan 			/* Power supply AC lost */
128053a40c1fSjordan 			return (SENSOR_S_WARN);
128153a40c1fSjordan 		}
1282bb816960Sjordan 		break;
1283bb816960Sjordan 	}
1284bb816960Sjordan 
1285bb816960Sjordan 	return (SENSOR_S_OK);
1286bb816960Sjordan }
1287bb816960Sjordan 
1288bb816960Sjordan int
read_sensor(struct ipmi_softc * sc,struct ipmi_sensor * psensor)1289fbca7c3fSmarco read_sensor(struct ipmi_softc *sc, struct ipmi_sensor *psensor)
1290fbca7c3fSmarco {
1291b6d6d087Smarco 	struct sdrtype1	*s1 = (struct sdrtype1 *) psensor->i_sdr;
1292fbca7c3fSmarco 	u_int8_t	data[8];
12930fe7a73cSuebayasi 	int		rv = -1;
129457e29007Smarco 
1295fbca7c3fSmarco 	memset(data, 0, sizeof(data));
1296fbca7c3fSmarco 	data[0] = psensor->i_num;
1297fbca7c3fSmarco 
12980fe7a73cSuebayasi 	struct ipmi_cmd	c;
12990fe7a73cSuebayasi 	c.c_sc = sc;
13000fe7a73cSuebayasi 	c.c_rssa = s1->owner_id;
13010fe7a73cSuebayasi 	c.c_rslun = s1->owner_lun;
13020fe7a73cSuebayasi 	c.c_netfn = SE_NETFN;
13030fe7a73cSuebayasi 	c.c_cmd = SE_GET_SENSOR_READING;
13040fe7a73cSuebayasi 	c.c_txlen = 1;
13050fe7a73cSuebayasi 	c.c_maxrxlen = sizeof(data);
13060fe7a73cSuebayasi 	c.c_rxlen = 0;
13070fe7a73cSuebayasi 	c.c_data = data;
13080fe7a73cSuebayasi 	ipmi_cmd(&c);
1309fbca7c3fSmarco 
13105b08ab8aSyasuoka 	if (c.c_ccode != 0) {
13115b08ab8aSyasuoka 		dbg_printf(1, "sensor reading command for %s failed: %.2x\n",
13125b08ab8aSyasuoka 			psensor->i_sensor.desc, c.c_ccode);
13135b08ab8aSyasuoka 		return (rv);
13145b08ab8aSyasuoka 	}
13157788c9ecSmarco 	dbg_printf(10, "values=%.2x %.2x %.2x %.2x %s\n",
1316bb816960Sjordan 	    data[0],data[1],data[2],data[3], psensor->i_sensor.desc);
1317fbca7c3fSmarco 	psensor->i_sensor.flags &= ~SENSOR_FINVALID;
13188866e14aSuebayasi 	if ((data[1] & IPMI_INVALID_SENSOR) ||
13198866e14aSuebayasi 	    ((data[1] & IPMI_DISABLED_SENSOR) == 0 && data[0] == 0))
1320fbca7c3fSmarco 		psensor->i_sensor.flags |= SENSOR_FINVALID;
1321bb816960Sjordan 	psensor->i_sensor.status = ipmi_sensor_status(sc, psensor, data);
132257e29007Smarco 	rv = 0;
132357e29007Smarco 	return (rv);
1324fbca7c3fSmarco }
1325fbca7c3fSmarco 
1326fbca7c3fSmarco int
ipmi_sensor_type(int type,int ext_type,int units2,int entity)132716dc177fSsthen ipmi_sensor_type(int type, int ext_type, int units2, int entity)
1328fbca7c3fSmarco {
132916dc177fSsthen 	switch (units2) {
133016dc177fSsthen 	case IPMI_UNIT_TYPE_AMPS:
133116dc177fSsthen 		return (SENSOR_AMPS);
133216dc177fSsthen 
133316dc177fSsthen 	case IPMI_UNIT_TYPE_RPM:
133416dc177fSsthen 		return (SENSOR_FANRPM);
133516dc177fSsthen 
133616dc177fSsthen 	/* XXX sensors framework distinguishes AC/DC but ipmi does not */
133716dc177fSsthen 	case IPMI_UNIT_TYPE_VOLTS:
133816dc177fSsthen 		return (SENSOR_VOLTS_DC);
133916dc177fSsthen 
134016dc177fSsthen 	case IPMI_UNIT_TYPE_WATTS:
134116dc177fSsthen 		return (SENSOR_WATTS);
134216dc177fSsthen 	}
134316dc177fSsthen 
1344bb816960Sjordan 	switch (ext_type << 8L | type) {
1345bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_TEMP:
1346fbca7c3fSmarco 		return (SENSOR_TEMP);
1347fbca7c3fSmarco 
1348bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_PWRSUPPLY:
1349bb816960Sjordan 		if (entity == IPMI_ENTITY_PWRSUPPLY)
1350bb816960Sjordan 			return (SENSOR_INDICATOR);
1351bb816960Sjordan 		break;
1352bb816960Sjordan 
1353bfec59d3Sjordan 	case IPMI_SENSOR_TYPE_INTRUSION:
1354fbca7c3fSmarco 		return (SENSOR_INDICATOR);
1355fbca7c3fSmarco 	}
1356fbca7c3fSmarco 
1357fbca7c3fSmarco 	return (-1);
1358fbca7c3fSmarco }
1359fbca7c3fSmarco 
1360fbca7c3fSmarco /* Add Sensor to BSD Sysctl interface */
1361fbca7c3fSmarco int
add_sdr_sensor(struct ipmi_softc * sc,u_int8_t * psdr,int sdrlen)13622409b544Suebayasi add_sdr_sensor(struct ipmi_softc *sc, u_int8_t *psdr, int sdrlen)
1363fbca7c3fSmarco {
1364bb816960Sjordan 	int			rc;
1365b6d6d087Smarco 	struct sdrtype1		*s1 = (struct sdrtype1 *)psdr;
1366b6d6d087Smarco 	struct sdrtype2		*s2 = (struct sdrtype2 *)psdr;
1367fbca7c3fSmarco 	char			name[64];
1368fbca7c3fSmarco 
1369fbca7c3fSmarco 	switch (s1->sdrhdr.record_type) {
13708cdc17caSjordan 	case IPMI_SDR_TYPEFULL:
13712409b544Suebayasi 		rc = ipmi_sensor_name(name, sizeof(name), s1->typelen,
13722409b544Suebayasi 		    s1->name, sdrlen - (int)offsetof(struct sdrtype1, name));
13732409b544Suebayasi 		if (rc == 0)
13742409b544Suebayasi 			return (0);
1375bb816960Sjordan 		rc = add_child_sensors(sc, psdr, 1, s1->sensor_num,
1376bb816960Sjordan 		    s1->sensor_type, s1->event_code, 0, s1->entity_id, name);
1377fbca7c3fSmarco 		break;
1378fbca7c3fSmarco 
13798cdc17caSjordan 	case IPMI_SDR_TYPECOMPACT:
13802409b544Suebayasi 		rc = ipmi_sensor_name(name, sizeof(name), s2->typelen,
13812409b544Suebayasi 		    s2->name, sdrlen - (int)offsetof(struct sdrtype2, name));
13822409b544Suebayasi 		if (rc == 0)
13832409b544Suebayasi 			return (0);
13844dd6fa05Sderaadt 		rc = add_child_sensors(sc, psdr, s2->share1 & 0xF,
13854dd6fa05Sderaadt 		    s2->sensor_num, s2->sensor_type, s2->event_code,
13864dd6fa05Sderaadt 		    s2->share2 & 0x7F, s2->entity_id, name);
1387fbca7c3fSmarco 		break;
1388fbca7c3fSmarco 
1389fbca7c3fSmarco 	default:
1390fbca7c3fSmarco 		return (0);
1391fbca7c3fSmarco 	}
1392fbca7c3fSmarco 
1393bb816960Sjordan 	return rc;
1394bb816960Sjordan }
1395bb816960Sjordan 
1396bb816960Sjordan int
add_child_sensors(struct ipmi_softc * sc,u_int8_t * psdr,int count,int sensor_num,int sensor_type,int ext_type,int sensor_base,int entity,const char * name)1397bb816960Sjordan add_child_sensors(struct ipmi_softc *sc, u_int8_t *psdr, int count,
1398bb816960Sjordan     int sensor_num, int sensor_type, int ext_type, int sensor_base,
1399bb816960Sjordan     int entity, const char *name)
1400bb816960Sjordan {
14016a19ca32Smbuhl 	int			typ, idx, rc = 0;
1402bb816960Sjordan 	struct ipmi_sensor	*psensor;
1403b6d6d087Smarco 	struct sdrtype1		*s1 = (struct sdrtype1 *)psdr;
1404bb816960Sjordan 
140516dc177fSsthen 	typ = ipmi_sensor_type(sensor_type, ext_type, s1->units2, entity);
1406bb816960Sjordan 	if (typ == -1) {
14077788c9ecSmarco 		dbg_printf(5, "Unknown sensor type:%.2x et:%.2x sn:%.2x "
140816dc177fSsthen 		    "units2:%u name:%s\n", sensor_type, ext_type, sensor_num,
140916dc177fSsthen 		    s1->units2, name);
1410bb816960Sjordan 		return 0;
1411bb816960Sjordan 	}
1412bb816960Sjordan 	for (idx = 0; idx < count; idx++) {
1413e825885dSgsoares 		psensor = malloc(sizeof(*psensor), M_DEVBUF, M_NOWAIT | M_ZERO);
1414fbca7c3fSmarco 		if (psensor == NULL)
1415fbca7c3fSmarco 			break;
1416fbca7c3fSmarco 
1417fbca7c3fSmarco 		/* Initialize BSD Sensor info */
1418fbca7c3fSmarco 		psensor->i_sdr = psdr;
1419bb816960Sjordan 		psensor->i_num = sensor_num + idx;
1420bb816960Sjordan 		psensor->stype = sensor_type;
1421bb816960Sjordan 		psensor->etype = ext_type;
1422fbca7c3fSmarco 		psensor->i_sensor.type = typ;
1423bb816960Sjordan 		if (count > 1)
1424fbca7c3fSmarco 			snprintf(psensor->i_sensor.desc,
1425fbca7c3fSmarco 			    sizeof(psensor->i_sensor.desc),
1426bb816960Sjordan 			    "%s - %d", name, sensor_base + idx);
1427fbca7c3fSmarco 		else
1428fbca7c3fSmarco 			strlcpy(psensor->i_sensor.desc, name,
1429fbca7c3fSmarco 			    sizeof(psensor->i_sensor.desc));
1430fbca7c3fSmarco 
14317788c9ecSmarco 		dbg_printf(5, "add sensor:%.4x %.2x:%d ent:%.2x:%.2x %s\n",
1432fbca7c3fSmarco 		    s1->sdrhdr.record_id, s1->sensor_type,
1433fbca7c3fSmarco 		    typ, s1->entity_id, s1->entity_instance,
1434fbca7c3fSmarco 		    psensor->i_sensor.desc);
1435fbca7c3fSmarco 		if (read_sensor(sc, psensor) == 0) {
1436fbca7c3fSmarco 			SLIST_INSERT_HEAD(&ipmi_sensor_list, psensor, list);
143727515a6bSderaadt 			sensor_attach(&sc->sc_sensordev, &psensor->i_sensor);
14387788c9ecSmarco 			dbg_printf(5, "	 reading: %lld [%s]\n",
1439fbca7c3fSmarco 			    psensor->i_sensor.value,
1440fbca7c3fSmarco 			    psensor->i_sensor.desc);
14416a19ca32Smbuhl 			rc = 1;
14421be66a57Stracey 		} else
14431be66a57Stracey 			free(psensor, M_DEVBUF, sizeof(*psensor));
1444fbca7c3fSmarco 	}
1445fbca7c3fSmarco 
14466a19ca32Smbuhl 	return (rc);
1447fbca7c3fSmarco }
1448fbca7c3fSmarco 
1449fbca7c3fSmarco /* Handle IPMI Timer - reread sensor values */
1450fbca7c3fSmarco void
ipmi_refresh_sensors(struct ipmi_softc * sc)1451fbca7c3fSmarco ipmi_refresh_sensors(struct ipmi_softc *sc)
1452fbca7c3fSmarco {
1453987084d8Smarco 	if (SLIST_EMPTY(&ipmi_sensor_list))
1454987084d8Smarco 		return;
1455987084d8Smarco 
1456987084d8Smarco 	sc->current_sensor = SLIST_NEXT(sc->current_sensor, list);
1457987084d8Smarco 	if (sc->current_sensor == NULL)
1458987084d8Smarco 		sc->current_sensor = SLIST_FIRST(&ipmi_sensor_list);
1459987084d8Smarco 
146027dce37dSreyk 	if (read_sensor(sc, sc->current_sensor)) {
146193a89648Smarco 		dbg_printf(1, "%s: error reading: %s\n", DEVNAME(sc),
1462987084d8Smarco 		    sc->current_sensor->i_sensor.desc);
146327dce37dSreyk 		return;
146427dce37dSreyk 	}
1465fbca7c3fSmarco }
1466fbca7c3fSmarco 
1467fbca7c3fSmarco int
ipmi_map_regs(struct ipmi_softc * sc,struct ipmi_attach_args * ia)1468fbca7c3fSmarco ipmi_map_regs(struct ipmi_softc *sc, struct ipmi_attach_args *ia)
1469fbca7c3fSmarco {
147019146c2bSkettenis 	if (sc->sc_if && sc->sc_if->nregs == 0)
147119146c2bSkettenis 		return (0);
147219146c2bSkettenis 
1473fbca7c3fSmarco 	sc->sc_if = ipmi_get_if(ia->iaa_if_type);
1474fbca7c3fSmarco 	if (sc->sc_if == NULL)
1475fbca7c3fSmarco 		return (-1);
1476fbca7c3fSmarco 
1477fbca7c3fSmarco 	if (ia->iaa_if_iotype == 'i')
1478fbca7c3fSmarco 		sc->sc_iot = ia->iaa_iot;
1479fbca7c3fSmarco 	else
1480fbca7c3fSmarco 		sc->sc_iot = ia->iaa_memt;
1481fbca7c3fSmarco 
1482fbca7c3fSmarco 	sc->sc_if_rev = ia->iaa_if_rev;
1483cda7789cSkettenis 	sc->sc_if_iosize = ia->iaa_if_iosize;
1484fbca7c3fSmarco 	sc->sc_if_iospacing = ia->iaa_if_iospacing;
1485400a62caSmarco 	if (bus_space_map(sc->sc_iot, ia->iaa_if_iobase,
1486fbca7c3fSmarco 	    sc->sc_if->nregs * sc->sc_if_iospacing,
1487400a62caSmarco 	    0, &sc->sc_ioh)) {
14887518efcdSkettenis 		printf("%s: bus_space_map(%lx %lx %x 0 %p) failed\n",
14898deac126Smarco 		    DEVNAME(sc),
14908970610aSsf 		    (unsigned long)sc->sc_iot, ia->iaa_if_iobase,
1491400a62caSmarco 		    sc->sc_if->nregs * sc->sc_if_iospacing, &sc->sc_ioh);
1492400a62caSmarco 		return (-1);
1493400a62caSmarco 	}
1494fbca7c3fSmarco 	return (0);
1495fbca7c3fSmarco }
1496fbca7c3fSmarco 
1497fbca7c3fSmarco void
ipmi_unmap_regs(struct ipmi_softc * sc)14984b532468Smarco ipmi_unmap_regs(struct ipmi_softc *sc)
1499fbca7c3fSmarco {
150019146c2bSkettenis 	if (sc->sc_if->nregs > 0) {
1501fbca7c3fSmarco 		bus_space_unmap(sc->sc_iot, sc->sc_ioh,
1502fbca7c3fSmarco 		    sc->sc_if->nregs * sc->sc_if_iospacing);
1503fbca7c3fSmarco 	}
150419146c2bSkettenis }
1505fbca7c3fSmarco 
15061d18187bSjordan void
ipmi_poll_thread(void * arg)15071d18187bSjordan ipmi_poll_thread(void *arg)
15081d18187bSjordan {
15091d18187bSjordan 	struct ipmi_thread	*thread = arg;
15101d18187bSjordan 	struct ipmi_softc	*sc = thread->sc;
15114b532468Smarco 	u_int16_t		rec;
15124b532468Smarco 
15134b532468Smarco 	/* Scan SDRs, add sensors */
15144b532468Smarco 	for (rec = 0; rec != 0xFFFF;) {
15154b532468Smarco 		if (get_sdr(sc, rec, &rec)) {
15164b532468Smarco 			ipmi_unmap_regs(sc);
15174b532468Smarco 			printf("%s: no SDRs IPMI disabled\n", DEVNAME(sc));
15184b532468Smarco 			goto done;
15194b532468Smarco 		}
15209482363aScheloha 		tsleep_nsec(sc, PWAIT, "ipmirun", MSEC_TO_NSEC(1));
15214b532468Smarco 	}
15224b532468Smarco 
15234b532468Smarco 	/* initialize sensor list for thread */
1524c79f7e70Smarco 	if (SLIST_EMPTY(&ipmi_sensor_list))
1525c79f7e70Smarco 		goto done;
1526c79f7e70Smarco 	else
15274b532468Smarco 		sc->current_sensor = SLIST_FIRST(&ipmi_sensor_list);
15281d18187bSjordan 
1529c9cddc26Scnst 	strlcpy(sc->sc_sensordev.xname, sc->sc_dev.dv_xname,
1530c9cddc26Scnst 	    sizeof(sc->sc_sensordev.xname));
1531c9cddc26Scnst 	sensordev_install(&sc->sc_sensordev);
1532c9cddc26Scnst 
15331d18187bSjordan 	while (thread->running) {
15341d18187bSjordan 		ipmi_refresh_sensors(sc);
153554277de9Scheloha 		tsleep_nsec(thread, PWAIT, "ipmi_poll",
153654277de9Scheloha 		    SEC_TO_NSEC(SENSOR_REFRESH_RATE));
15371d18187bSjordan 	}
15381d18187bSjordan 
15394b532468Smarco done:
15401d18187bSjordan 	kthread_exit(0);
15411d18187bSjordan }
15421d18187bSjordan 
15431d18187bSjordan void
ipmi_create_thread(void * arg)15441d18187bSjordan ipmi_create_thread(void *arg)
15451d18187bSjordan {
15461d18187bSjordan 	struct ipmi_softc	*sc = arg;
15471d18187bSjordan 
15481d18187bSjordan 	if (kthread_create(ipmi_poll_thread, sc->sc_thread, NULL,
15492ed0e526Smarco 	    DEVNAME(sc)) != 0) {
15504b532468Smarco 		printf("%s: unable to create run thread, ipmi disabled\n",
15512ed0e526Smarco 		    DEVNAME(sc));
15522ed0e526Smarco 		return;
15532ed0e526Smarco 	}
15541d18187bSjordan }
15551d18187bSjordan 
1556fbca7c3fSmarco void
ipmi_attach_common(struct ipmi_softc * sc,struct ipmi_attach_args * ia)1557c08dc278Skettenis ipmi_attach_common(struct ipmi_softc *sc, struct ipmi_attach_args *ia)
1558fbca7c3fSmarco {
15598f330473Suebayasi 	struct ipmi_cmd		*c = &sc->sc_ioctl.cmd;
1560fbca7c3fSmarco 
15612b13db16Sderaadt 	/* Map registers */
15622b13db16Sderaadt 	ipmi_map_regs(sc, ia);
15632b13db16Sderaadt 
1564e825885dSgsoares 	sc->sc_thread = malloc(sizeof(struct ipmi_thread), M_DEVBUF, M_NOWAIT);
15651d18187bSjordan 	if (sc->sc_thread == NULL) {
15663c279471Sderaadt 		printf(": unable to allocate thread\n");
15671d18187bSjordan 		return;
15681d18187bSjordan 	}
15691d18187bSjordan 	sc->sc_thread->sc = sc;
15701d18187bSjordan 	sc->sc_thread->running = 1;
15711d18187bSjordan 
15721d18187bSjordan 	/* Setup threads */
15731d18187bSjordan 	kthread_create_deferred(ipmi_create_thread, sc);
1574fbca7c3fSmarco 
157519146c2bSkettenis 	printf(": version %d.%d interface %s",
157619146c2bSkettenis 	    ia->iaa_if_rev >> 4, ia->iaa_if_rev & 0xF, sc->sc_if->name);
157719146c2bSkettenis 	if (sc->sc_if->nregs > 0)
15787518efcdSkettenis 		printf(" %sbase 0x%lx/%x spacing %d",
157935701cd5Sderaadt 		    ia->iaa_if_iotype == 'i' ? "io" : "mem", ia->iaa_if_iobase,
158019146c2bSkettenis 		    ia->iaa_if_iospacing * sc->sc_if->nregs,
158119146c2bSkettenis 		    ia->iaa_if_iospacing);
158235701cd5Sderaadt 	if (ia->iaa_if_irq != -1)
158335701cd5Sderaadt 		printf(" irq %d", ia->iaa_if_irq);
158435701cd5Sderaadt 	printf("\n");
15857a3397e7Sjordan 
1586bd7ba471Smarco 	/* setup flag to exclude iic */
1587bd7ba471Smarco 	ipmi_enabled = 1;
1588bd7ba471Smarco 
15897a3397e7Sjordan 	/* Setup Watchdog timer */
15907a3397e7Sjordan 	sc->sc_wdog_period = 0;
159166590f60Suebayasi 	task_set(&sc->sc_wdog_tickle_task, ipmi_watchdog_tickle, sc);
15922bc62decSderaadt 	wdog_register(ipmi_watchdog, sc);
1593b2bc3525Smarco 
15948f330473Suebayasi 	rw_init(&sc->sc_ioctl.lock, DEVNAME(sc));
15958f330473Suebayasi 	sc->sc_ioctl.req.msgid = -1;
15968f330473Suebayasi 	c->c_sc = sc;
15978f330473Suebayasi 	c->c_ccode = -1;
1598f640b71bSuebayasi 
1599*5f3b3788Sgkoehler 	sc->sc_cmd_taskq = taskq_create("ipmicmd", 1, IPL_MPFLOOR,
1600*5f3b3788Sgkoehler 	    TASKQ_MPSAFE);
16017a3397e7Sjordan }
16027a3397e7Sjordan 
16037a3397e7Sjordan int
ipmi_activate(struct device * self,int act)16043b06f262Smikeb ipmi_activate(struct device *self, int act)
16053b06f262Smikeb {
16063b06f262Smikeb 	switch (act) {
16073b06f262Smikeb 	case DVACT_POWERDOWN:
16083b06f262Smikeb 		wdog_shutdown(self);
16093b06f262Smikeb 		break;
16103b06f262Smikeb 	}
16113b06f262Smikeb 
16123b06f262Smikeb 	return (0);
16133b06f262Smikeb }
16143b06f262Smikeb 
16158f330473Suebayasi struct ipmi_softc *
ipmilookup(dev_t dev)16168f330473Suebayasi ipmilookup(dev_t dev)
16178f330473Suebayasi {
16188f330473Suebayasi 	return (struct ipmi_softc *)device_lookup(&ipmi_cd, minor(dev));
16198f330473Suebayasi }
16208f330473Suebayasi 
16218f330473Suebayasi int
ipmiopen(dev_t dev,int flags,int mode,struct proc * p)16228f330473Suebayasi ipmiopen(dev_t dev, int flags, int mode, struct proc *p)
16238f330473Suebayasi {
16248f330473Suebayasi 	struct ipmi_softc	*sc = ipmilookup(dev);
16258f330473Suebayasi 
16268f330473Suebayasi 	if (sc == NULL)
16278f330473Suebayasi 		return (ENXIO);
16288f330473Suebayasi 	return (0);
16298f330473Suebayasi }
16308f330473Suebayasi 
16318f330473Suebayasi int
ipmiclose(dev_t dev,int flags,int mode,struct proc * p)16328f330473Suebayasi ipmiclose(dev_t dev, int flags, int mode, struct proc *p)
16338f330473Suebayasi {
16348f330473Suebayasi 	struct ipmi_softc	*sc = ipmilookup(dev);
16358f330473Suebayasi 
16368f330473Suebayasi 	if (sc == NULL)
16378f330473Suebayasi 		return (ENXIO);
16388f330473Suebayasi 	return (0);
16398f330473Suebayasi }
16408f330473Suebayasi 
16418f330473Suebayasi int
ipmiioctl(dev_t dev,u_long cmd,caddr_t data,int flag,struct proc * proc)16428f330473Suebayasi ipmiioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *proc)
16438f330473Suebayasi {
16448f330473Suebayasi 	struct ipmi_softc	*sc = ipmilookup(dev);
16458f330473Suebayasi 	struct ipmi_req		*req = (struct ipmi_req *)data;
16468f330473Suebayasi 	struct ipmi_recv	*recv = (struct ipmi_recv *)data;
16478f330473Suebayasi 	struct ipmi_cmd		*c = &sc->sc_ioctl.cmd;
16488f330473Suebayasi 	int			iv;
16498f330473Suebayasi 	int			len;
16508f330473Suebayasi 	u_char			ccode;
16518f330473Suebayasi 	int			rc = 0;
16528f330473Suebayasi 
16538f330473Suebayasi 	if (sc == NULL)
16548f330473Suebayasi 		return (ENXIO);
16558f330473Suebayasi 
16568f330473Suebayasi 	rw_enter_write(&sc->sc_ioctl.lock);
16578f330473Suebayasi 
16588f330473Suebayasi 	c->c_maxrxlen = sizeof(sc->sc_ioctl.buf);
16598f330473Suebayasi 	c->c_data = sc->sc_ioctl.buf;
16608f330473Suebayasi 
16618f330473Suebayasi 	switch (cmd) {
16628f330473Suebayasi 	case IPMICTL_SEND_COMMAND:
16638f330473Suebayasi 		if (req->msgid == -1) {
16648f330473Suebayasi 			rc = EINVAL;
16658f330473Suebayasi 			goto reset;
16668f330473Suebayasi 		}
16678f330473Suebayasi 		if (sc->sc_ioctl.req.msgid != -1) {
16688f330473Suebayasi 			rc = EBUSY;
16698f330473Suebayasi 			goto reset;
16708f330473Suebayasi 		}
16718f330473Suebayasi 		len = req->msg.data_len;
16728f330473Suebayasi 		if (len < 0) {
16738f330473Suebayasi 			rc = EINVAL;
16748f330473Suebayasi 			goto reset;
16758f330473Suebayasi 		}
16768f330473Suebayasi 		if (len > c->c_maxrxlen) {
16778f330473Suebayasi 			rc = E2BIG;
16788f330473Suebayasi 			goto reset;
16798f330473Suebayasi 		}
16808f330473Suebayasi 		sc->sc_ioctl.req = *req;
16818f330473Suebayasi 		c->c_ccode = -1;
16828f330473Suebayasi 		rc = copyin(req->msg.data, c->c_data, len);
16838f330473Suebayasi 		if (rc != 0)
16848f330473Suebayasi 			goto reset;
16858f330473Suebayasi 		KASSERT(c->c_ccode == -1);
16868f330473Suebayasi 
16878f330473Suebayasi 		/* Execute a command synchronously. */
16888f330473Suebayasi 		c->c_netfn = req->msg.netfn;
16898f330473Suebayasi 		c->c_cmd = req->msg.cmd;
16908f330473Suebayasi 		c->c_txlen = req->msg.data_len;
16918f330473Suebayasi 		c->c_rxlen = 0;
16928f330473Suebayasi 		ipmi_cmd(c);
16938f330473Suebayasi 		break;
16948f330473Suebayasi 	case IPMICTL_RECEIVE_MSG_TRUNC:
16958f330473Suebayasi 	case IPMICTL_RECEIVE_MSG:
16968f330473Suebayasi 		if (sc->sc_ioctl.req.msgid == -1) {
16978f330473Suebayasi 			rc = EINVAL;
16988f330473Suebayasi 			goto reset;
16998f330473Suebayasi 		}
17008f330473Suebayasi 		if (c->c_ccode == -1) {
17018f330473Suebayasi 			rc = EAGAIN;
17028f330473Suebayasi 			goto reset;
17038f330473Suebayasi 		}
17048f330473Suebayasi 		ccode = c->c_ccode & 0xff;
17058f330473Suebayasi 		rc = copyout(&ccode, recv->msg.data, 1);
17068f330473Suebayasi 		if (rc != 0)
17078f330473Suebayasi 			goto reset;
17088f330473Suebayasi 
17098f330473Suebayasi 		/* Return a command result. */
17108f330473Suebayasi 		recv->recv_type = IPMI_RESPONSE_RECV_TYPE;
17118f330473Suebayasi 		recv->msgid = sc->sc_ioctl.req.msgid;
17128f330473Suebayasi 		recv->msg.netfn = sc->sc_ioctl.req.msg.netfn;
17138f330473Suebayasi 		recv->msg.cmd = sc->sc_ioctl.req.msg.cmd;
17148f330473Suebayasi 		recv->msg.data_len = c->c_rxlen + 1;
17158f330473Suebayasi 
17168f330473Suebayasi 		rc = copyout(c->c_data, recv->msg.data + 1, c->c_rxlen);
1717eed8831cSuebayasi 		/* Always reset state after command completion. */
17188f330473Suebayasi 		goto reset;
17198f330473Suebayasi 	case IPMICTL_SET_MY_ADDRESS_CMD:
17208f330473Suebayasi 		iv = *(int *)data;
17218f330473Suebayasi 		if (iv < 0 || iv > RSSA_MASK) {
17228f330473Suebayasi 			rc = EINVAL;
17238f330473Suebayasi 			goto reset;
17248f330473Suebayasi 		}
17258f330473Suebayasi 		c->c_rssa = iv;
17268f330473Suebayasi 		break;
17278f330473Suebayasi 	case IPMICTL_GET_MY_ADDRESS_CMD:
17288f330473Suebayasi 		*(int *)data = c->c_rssa;
17298f330473Suebayasi 		break;
17308f330473Suebayasi 	case IPMICTL_SET_MY_LUN_CMD:
17318f330473Suebayasi 		iv = *(int *)data;
17328f330473Suebayasi 		if (iv < 0 || iv > LUN_MASK) {
17338f330473Suebayasi 			rc = EINVAL;
17348f330473Suebayasi 			goto reset;
17358f330473Suebayasi 		}
17368f330473Suebayasi 		c->c_rslun = iv;
17378f330473Suebayasi 		break;
17388f330473Suebayasi 	case IPMICTL_GET_MY_LUN_CMD:
17398f330473Suebayasi 		*(int *)data = c->c_rslun;
17408f330473Suebayasi 		break;
17418f330473Suebayasi 	case IPMICTL_SET_GETS_EVENTS_CMD:
17428f330473Suebayasi 		break;
17438f330473Suebayasi 	case IPMICTL_REGISTER_FOR_CMD:
17448f330473Suebayasi 	case IPMICTL_UNREGISTER_FOR_CMD:
17458f330473Suebayasi 	default:
17468f330473Suebayasi 		break;
17478f330473Suebayasi 	}
17488f330473Suebayasi done:
17498f330473Suebayasi 	rw_exit_write(&sc->sc_ioctl.lock);
17508f330473Suebayasi 	return (rc);
17518f330473Suebayasi reset:
17528f330473Suebayasi 	sc->sc_ioctl.req.msgid = -1;
17538f330473Suebayasi 	c->c_ccode = -1;
17548f330473Suebayasi 	goto done;
17558f330473Suebayasi }
17568f330473Suebayasi 
17577f5c5dbdSuebayasi #define		MIN_PERIOD	10
17587f5c5dbdSuebayasi 
17593b06f262Smikeb int
ipmi_watchdog(void * arg,int period)17607a3397e7Sjordan ipmi_watchdog(void *arg, int period)
17617a3397e7Sjordan {
17627a3397e7Sjordan 	struct ipmi_softc	*sc = arg;
17637a3397e7Sjordan 
17647a3397e7Sjordan 	if (sc->sc_wdog_period == period) {
17657a3397e7Sjordan 		if (period != 0) {
176666590f60Suebayasi 			struct task *t;
176766590f60Suebayasi 			int res;
176866590f60Suebayasi 
176966590f60Suebayasi 			t = &sc->sc_wdog_tickle_task;
1770e0bef428Suebayasi 			(void)task_del(systq, t);
177166590f60Suebayasi 			res = task_add(systq, t);
177266590f60Suebayasi 			KASSERT(res == 1);
1773e824610bSuebayasi 		}
1774e824610bSuebayasi 		return (period);
1775e824610bSuebayasi 	}
1776e824610bSuebayasi 
1777e824610bSuebayasi 	if (period < MIN_PERIOD && period > 0)
1778e824610bSuebayasi 		period = MIN_PERIOD;
1779e824610bSuebayasi 	sc->sc_wdog_period = period;
1780e824610bSuebayasi 	ipmi_watchdog_set(sc);
1781e824610bSuebayasi 	printf("%s: watchdog %sabled\n", DEVNAME(sc),
1782e824610bSuebayasi 	    (period == 0) ? "dis" : "en");
1783e824610bSuebayasi 	return (period);
1784e824610bSuebayasi }
1785e824610bSuebayasi 
1786e824610bSuebayasi void
ipmi_watchdog_tickle(void * arg)1787e824610bSuebayasi ipmi_watchdog_tickle(void *arg)
1788e824610bSuebayasi {
1789e824610bSuebayasi 	struct ipmi_softc	*sc = arg;
1790e824610bSuebayasi 	struct ipmi_cmd		c;
1791e824610bSuebayasi 
17920fe7a73cSuebayasi 	c.c_sc = sc;
17930fe7a73cSuebayasi 	c.c_rssa = BMC_SA;
17940fe7a73cSuebayasi 	c.c_rslun = BMC_LUN;
17950fe7a73cSuebayasi 	c.c_netfn = APP_NETFN;
17960fe7a73cSuebayasi 	c.c_cmd = APP_RESET_WATCHDOG;
17970fe7a73cSuebayasi 	c.c_txlen = 0;
17980fe7a73cSuebayasi 	c.c_maxrxlen = 0;
17990fe7a73cSuebayasi 	c.c_rxlen = 0;
18000fe7a73cSuebayasi 	c.c_data = NULL;
18010fe7a73cSuebayasi 	ipmi_cmd(&c);
18027a3397e7Sjordan }
18037a3397e7Sjordan 
1804e824610bSuebayasi void
ipmi_watchdog_set(void * arg)1805e824610bSuebayasi ipmi_watchdog_set(void *arg)
1806e824610bSuebayasi {
1807e824610bSuebayasi 	struct ipmi_softc	*sc = arg;
1808e824610bSuebayasi 	uint8_t			wdog[IPMI_GET_WDOG_MAX];
1809e824610bSuebayasi 	struct ipmi_cmd		c;
18107a3397e7Sjordan 
18110fe7a73cSuebayasi 	c.c_sc = sc;
18120fe7a73cSuebayasi 	c.c_rssa = BMC_SA;
18130fe7a73cSuebayasi 	c.c_rslun = BMC_LUN;
18140fe7a73cSuebayasi 	c.c_netfn = APP_NETFN;
18150fe7a73cSuebayasi 	c.c_cmd = APP_GET_WATCHDOG_TIMER;
18160fe7a73cSuebayasi 	c.c_txlen = 0;
18170fe7a73cSuebayasi 	c.c_maxrxlen = IPMI_GET_WDOG_MAX;
18180fe7a73cSuebayasi 	c.c_rxlen = 0;
18190fe7a73cSuebayasi 	c.c_data = wdog;
18200fe7a73cSuebayasi 	ipmi_cmd(&c);
18217a3397e7Sjordan 
18227a3397e7Sjordan 	/* Period is 10ths/sec */
1823e824610bSuebayasi 	uint16_t timo = htole16(sc->sc_wdog_period * 10);
1824fd99d266Syasuoka 
1825fd99d266Syasuoka 	memcpy(&wdog[IPMI_SET_WDOG_TIMOL], &timo, 2);
18260728ef28Suebayasi 	wdog[IPMI_SET_WDOG_TIMER] &= ~IPMI_WDOG_DONTSTOP;
1827e824610bSuebayasi 	wdog[IPMI_SET_WDOG_TIMER] |= (sc->sc_wdog_period == 0) ?
1828e824610bSuebayasi 	    0 : IPMI_WDOG_DONTSTOP;
1829fd99d266Syasuoka 	wdog[IPMI_SET_WDOG_ACTION] &= ~IPMI_WDOG_MASK;
1830e824610bSuebayasi 	wdog[IPMI_SET_WDOG_ACTION] |= (sc->sc_wdog_period == 0) ?
1831e824610bSuebayasi 	    IPMI_WDOG_DISABLED : IPMI_WDOG_REBOOT;
18327a3397e7Sjordan 
18330fe7a73cSuebayasi 	c.c_cmd = APP_SET_WATCHDOG_TIMER;
18340fe7a73cSuebayasi 	c.c_txlen = IPMI_SET_WDOG_MAX;
18350fe7a73cSuebayasi 	c.c_maxrxlen = 0;
18360fe7a73cSuebayasi 	c.c_rxlen = 0;
18370fe7a73cSuebayasi 	c.c_data = wdog;
18380fe7a73cSuebayasi 	ipmi_cmd(&c);
1839fbca7c3fSmarco }
1840c08dc278Skettenis 
1841c08dc278Skettenis #if defined(__amd64__) || defined(__i386__)
1842c08dc278Skettenis 
1843c08dc278Skettenis #include <dev/isa/isareg.h>
1844c08dc278Skettenis #include <dev/isa/isavar.h>
1845c08dc278Skettenis 
1846c08dc278Skettenis /*
1847c08dc278Skettenis  * Format of SMBIOS IPMI Flags
1848c08dc278Skettenis  *
1849c08dc278Skettenis  * bit0: interrupt trigger mode (1=level, 0=edge)
1850c08dc278Skettenis  * bit1: interrupt polarity (1=active high, 0=active low)
1851c08dc278Skettenis  * bit2: reserved
1852c08dc278Skettenis  * bit3: address LSB (1=odd,0=even)
1853c08dc278Skettenis  * bit4: interrupt (1=specified, 0=not specified)
1854c08dc278Skettenis  * bit5: reserved
1855c08dc278Skettenis  * bit6/7: register spacing (1,4,2,err)
1856c08dc278Skettenis  */
1857c08dc278Skettenis #define SMIPMI_FLAG_IRQLVL		(1L << 0)
1858c08dc278Skettenis #define SMIPMI_FLAG_IRQEN		(1L << 3)
1859c08dc278Skettenis #define SMIPMI_FLAG_ODDOFFSET		(1L << 4)
1860c08dc278Skettenis #define SMIPMI_FLAG_IFSPACING(x)	(((x)>>6)&0x3)
1861c08dc278Skettenis #define	 IPMI_IOSPACING_BYTE		 0
1862c08dc278Skettenis #define	 IPMI_IOSPACING_WORD		 2
1863c08dc278Skettenis #define	 IPMI_IOSPACING_DWORD		 1
1864c08dc278Skettenis 
1865c08dc278Skettenis struct dmd_ipmi {
1866c08dc278Skettenis 	u_int8_t	dmd_sig[4];		/* Signature 'IPMI' */
1867c08dc278Skettenis 	u_int8_t	dmd_i2c_address;	/* Address of BMC */
1868c08dc278Skettenis 	u_int8_t	dmd_nvram_address;	/* Address of NVRAM */
1869c08dc278Skettenis 	u_int8_t	dmd_if_type;		/* IPMI Interface Type */
1870c08dc278Skettenis 	u_int8_t	dmd_if_rev;		/* IPMI Interface Revision */
1871c08dc278Skettenis } __packed;
1872c08dc278Skettenis 
1873c08dc278Skettenis void	*scan_sig(long, long, int, int, const void *);
1874c08dc278Skettenis 
1875c08dc278Skettenis void	ipmi_smbios_probe(struct smbios_ipmi *, struct ipmi_attach_args *);
1876c08dc278Skettenis int	ipmi_match(struct device *, void *, void *);
1877c08dc278Skettenis void	ipmi_attach(struct device *, struct device *, void *);
1878c08dc278Skettenis 
1879471aeecfSnaddy const struct cfattach ipmi_ca = {
1880c08dc278Skettenis 	sizeof(struct ipmi_softc), ipmi_match, ipmi_attach,
1881c08dc278Skettenis 	NULL, ipmi_activate
1882c08dc278Skettenis };
1883c08dc278Skettenis 
1884c08dc278Skettenis int
ipmi_match(struct device * parent,void * match,void * aux)1885c08dc278Skettenis ipmi_match(struct device *parent, void *match, void *aux)
1886c08dc278Skettenis {
1887c08dc278Skettenis 	struct ipmi_softc	*sc;
1888c08dc278Skettenis 	struct ipmi_attach_args *ia = aux;
1889c08dc278Skettenis 	struct cfdata		*cf = match;
1890c08dc278Skettenis 	u_int8_t		cmd[32];
1891c08dc278Skettenis 	int			rv = 0;
1892c08dc278Skettenis 
1893c08dc278Skettenis 	if (strcmp(ia->iaa_name, cf->cf_driver->cd_name))
1894c08dc278Skettenis 		return (0);
1895c08dc278Skettenis 
1896c08dc278Skettenis 	/* XXX local softc is wrong wrong wrong */
1897c08dc278Skettenis 	sc = malloc(sizeof(*sc), M_TEMP, M_WAITOK | M_ZERO);
1898c08dc278Skettenis 	strlcpy(sc->sc_dev.dv_xname, "ipmi0", sizeof(sc->sc_dev.dv_xname));
1899c08dc278Skettenis 
1900c08dc278Skettenis 	/* Map registers */
1901c08dc278Skettenis 	if (ipmi_map_regs(sc, ia) == 0) {
1902c08dc278Skettenis 		sc->sc_if->probe(sc);
1903c08dc278Skettenis 
1904c08dc278Skettenis 		/* Identify BMC device early to detect lying bios */
1905c08dc278Skettenis 		struct ipmi_cmd c;
1906c08dc278Skettenis 		c.c_sc = sc;
1907c08dc278Skettenis 		c.c_rssa = BMC_SA;
1908c08dc278Skettenis 		c.c_rslun = BMC_LUN;
1909c08dc278Skettenis 		c.c_netfn = APP_NETFN;
1910c08dc278Skettenis 		c.c_cmd = APP_GET_DEVICE_ID;
1911c08dc278Skettenis 		c.c_txlen = 0;
1912c08dc278Skettenis 		c.c_maxrxlen = sizeof(cmd);
1913c08dc278Skettenis 		c.c_rxlen = 0;
1914c08dc278Skettenis 		c.c_data = cmd;
1915c08dc278Skettenis 		ipmi_cmd(&c);
1916c08dc278Skettenis 
1917c08dc278Skettenis 		dbg_dump(1, "bmc data", c.c_rxlen, cmd);
1918c08dc278Skettenis 		rv = 1; /* GETID worked, we got IPMI */
1919c08dc278Skettenis 		ipmi_unmap_regs(sc);
1920c08dc278Skettenis 	}
1921c08dc278Skettenis 
1922c08dc278Skettenis 	free(sc, M_TEMP, sizeof(*sc));
1923c08dc278Skettenis 
1924c08dc278Skettenis 	return (rv);
1925c08dc278Skettenis }
1926c08dc278Skettenis 
1927c08dc278Skettenis void
ipmi_attach(struct device * parent,struct device * self,void * aux)1928c08dc278Skettenis ipmi_attach(struct device *parent, struct device *self, void *aux)
1929c08dc278Skettenis {
1930c08dc278Skettenis 	ipmi_attach_common((struct ipmi_softc *)self, aux);
1931c08dc278Skettenis }
1932c08dc278Skettenis 
1933c08dc278Skettenis /* Scan memory for signature */
1934c08dc278Skettenis void *
scan_sig(long start,long end,int skip,int len,const void * data)1935c08dc278Skettenis scan_sig(long start, long end, int skip, int len, const void *data)
1936c08dc278Skettenis {
1937c08dc278Skettenis 	void *va;
1938c08dc278Skettenis 
1939c08dc278Skettenis 	while (start < end) {
1940c08dc278Skettenis 		va = ISA_HOLE_VADDR(start);
1941c08dc278Skettenis 		if (memcmp(va, data, len) == 0)
1942c08dc278Skettenis 			return (va);
1943c08dc278Skettenis 
1944c08dc278Skettenis 		start += skip;
1945c08dc278Skettenis 	}
1946c08dc278Skettenis 
1947c08dc278Skettenis 	return (NULL);
1948c08dc278Skettenis }
1949c08dc278Skettenis 
1950c08dc278Skettenis void
ipmi_smbios_probe(struct smbios_ipmi * pipmi,struct ipmi_attach_args * ia)1951c08dc278Skettenis ipmi_smbios_probe(struct smbios_ipmi *pipmi, struct ipmi_attach_args *ia)
1952c08dc278Skettenis {
1953c08dc278Skettenis 
1954c08dc278Skettenis 	dbg_printf(1, "ipmi_smbios_probe: %02x %02x %02x %02x %08llx %02x "
1955c08dc278Skettenis 	    "%02x\n",
1956c08dc278Skettenis 	    pipmi->smipmi_if_type,
1957c08dc278Skettenis 	    pipmi->smipmi_if_rev,
1958c08dc278Skettenis 	    pipmi->smipmi_i2c_address,
1959c08dc278Skettenis 	    pipmi->smipmi_nvram_address,
1960c08dc278Skettenis 	    pipmi->smipmi_base_address,
1961c08dc278Skettenis 	    pipmi->smipmi_base_flags,
1962c08dc278Skettenis 	    pipmi->smipmi_irq);
1963c08dc278Skettenis 
1964c08dc278Skettenis 	ia->iaa_if_type = pipmi->smipmi_if_type;
1965c08dc278Skettenis 	ia->iaa_if_rev = pipmi->smipmi_if_rev;
1966c08dc278Skettenis 	ia->iaa_if_irq = (pipmi->smipmi_base_flags & SMIPMI_FLAG_IRQEN) ?
1967c08dc278Skettenis 	    pipmi->smipmi_irq : -1;
1968c08dc278Skettenis 	ia->iaa_if_irqlvl = (pipmi->smipmi_base_flags & SMIPMI_FLAG_IRQLVL) ?
1969c08dc278Skettenis 	    IST_LEVEL : IST_EDGE;
1970cda7789cSkettenis 	ia->iaa_if_iosize = 1;
1971c08dc278Skettenis 
1972c08dc278Skettenis 	switch (SMIPMI_FLAG_IFSPACING(pipmi->smipmi_base_flags)) {
1973c08dc278Skettenis 	case IPMI_IOSPACING_BYTE:
1974c08dc278Skettenis 		ia->iaa_if_iospacing = 1;
1975c08dc278Skettenis 		break;
1976c08dc278Skettenis 
1977c08dc278Skettenis 	case IPMI_IOSPACING_DWORD:
1978c08dc278Skettenis 		ia->iaa_if_iospacing = 4;
1979c08dc278Skettenis 		break;
1980c08dc278Skettenis 
1981c08dc278Skettenis 	case IPMI_IOSPACING_WORD:
1982c08dc278Skettenis 		ia->iaa_if_iospacing = 2;
1983c08dc278Skettenis 		break;
1984c08dc278Skettenis 
1985c08dc278Skettenis 	default:
1986c08dc278Skettenis 		ia->iaa_if_iospacing = 1;
1987c08dc278Skettenis 		printf("ipmi: unknown register spacing\n");
1988c08dc278Skettenis 	}
1989c08dc278Skettenis 
1990c08dc278Skettenis 	/* Calculate base address (PCI BAR format) */
1991c08dc278Skettenis 	if (pipmi->smipmi_base_address & 0x1) {
1992c08dc278Skettenis 		ia->iaa_if_iotype = 'i';
1993c08dc278Skettenis 		ia->iaa_if_iobase = pipmi->smipmi_base_address & ~0x1;
1994c08dc278Skettenis 	} else {
1995c08dc278Skettenis 		ia->iaa_if_iotype = 'm';
1996c08dc278Skettenis 		ia->iaa_if_iobase = pipmi->smipmi_base_address & ~0xF;
1997c08dc278Skettenis 	}
1998c08dc278Skettenis 	if (pipmi->smipmi_base_flags & SMIPMI_FLAG_ODDOFFSET)
1999c08dc278Skettenis 		ia->iaa_if_iobase++;
2000c08dc278Skettenis 
2001c08dc278Skettenis 	if (pipmi->smipmi_base_flags == 0x7f) {
2002c08dc278Skettenis 		/* IBM 325 eServer workaround */
2003c08dc278Skettenis 		ia->iaa_if_iospacing = 1;
2004c08dc278Skettenis 		ia->iaa_if_iobase = pipmi->smipmi_base_address;
2005c08dc278Skettenis 		ia->iaa_if_iotype = 'i';
2006c08dc278Skettenis 		return;
2007c08dc278Skettenis 	}
2008c08dc278Skettenis }
2009c08dc278Skettenis 
2010c08dc278Skettenis int
ipmi_probe(void * aux)2011c08dc278Skettenis ipmi_probe(void *aux)
2012c08dc278Skettenis {
2013c08dc278Skettenis 	struct ipmi_attach_args *ia = aux;
2014c08dc278Skettenis 	struct dmd_ipmi *pipmi;
2015c08dc278Skettenis 	struct smbtable tbl;
2016c08dc278Skettenis 
2017c08dc278Skettenis 	tbl.cookie = 0;
2018c08dc278Skettenis 	if (smbios_find_table(SMBIOS_TYPE_IPMIDEV, &tbl))
2019c08dc278Skettenis 		ipmi_smbios_probe(tbl.tblhdr, ia);
2020c08dc278Skettenis 	else {
2021c08dc278Skettenis 		pipmi = (struct dmd_ipmi *)scan_sig(0xC0000L, 0xFFFFFL, 16, 4,
2022c08dc278Skettenis 		    "IPMI");
2023c08dc278Skettenis 		/* XXX hack to find Dell PowerEdge 8450 */
2024c08dc278Skettenis 		if (pipmi == NULL) {
2025c08dc278Skettenis 			/* no IPMI found */
2026c08dc278Skettenis 			return (0);
2027c08dc278Skettenis 		}
2028c08dc278Skettenis 
2029c08dc278Skettenis 		/* we have an IPMI signature, fill in attach arg structure */
2030c08dc278Skettenis 		ia->iaa_if_type = pipmi->dmd_if_type;
2031c08dc278Skettenis 		ia->iaa_if_rev = pipmi->dmd_if_rev;
2032c08dc278Skettenis 	}
2033c08dc278Skettenis 
2034c08dc278Skettenis 	return (1);
2035c08dc278Skettenis }
2036c08dc278Skettenis 
2037c08dc278Skettenis #endif
2038