summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/sgi/conf/files.sgi3
-rw-r--r--sys/arch/sgi/pci/ioc.c50
-rw-r--r--sys/arch/sgi/sgi/l1.c865
-rw-r--r--sys/arch/sgi/sgi/l1.h73
-rw-r--r--sys/arch/sgi/xbow/hub.h5
5 files changed, 974 insertions, 22 deletions
diff --git a/sys/arch/sgi/conf/files.sgi b/sys/arch/sgi/conf/files.sgi
index a3a2461f72d..3ff414df6ca 100644
--- a/sys/arch/sgi/conf/files.sgi
+++ b/sys/arch/sgi/conf/files.sgi
@@ -1,4 +1,4 @@
-# $OpenBSD: files.sgi,v 1.38 2009/11/07 18:56:55 miod Exp $
+# $OpenBSD: files.sgi,v 1.39 2009/11/08 22:44:14 miod Exp $
#
# maxpartitions must be first item in files.${ARCH}
#
@@ -16,6 +16,7 @@ file arch/sgi/sgi/disksubr.c disk
file arch/sgi/sgi/ip27_machdep.c tgt_origin
file arch/sgi/sgi/ip30_machdep.c tgt_octane
file arch/sgi/sgi/ip32_machdep.c tgt_o2
+file arch/sgi/sgi/l1.c tgt_origin
file arch/sgi/sgi/machdep.c
file arch/sgi/sgi/mainbus.c
file arch/sgi/sgi/mutex.c
diff --git a/sys/arch/sgi/pci/ioc.c b/sys/arch/sgi/pci/ioc.c
index 2c40a01b4ea..79559c6b4fb 100644
--- a/sys/arch/sgi/pci/ioc.c
+++ b/sys/arch/sgi/pci/ioc.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ioc.c,v 1.27 2009/11/08 13:10:03 miod Exp $ */
+/* $OpenBSD: ioc.c,v 1.28 2009/11/08 22:44:16 miod Exp $ */
/*
* Copyright (c) 2008 Joel Sing.
@@ -31,7 +31,10 @@
#include <machine/autoconf.h>
#include <machine/bus.h>
+#ifdef TGT_ORIGIN
#include <sgi/sgi/ip27.h>
+#include <sgi/sgi/l1.h>
+#endif
#include <dev/pci/pcireg.h>
#include <dev/pci/pcivar.h>
@@ -106,6 +109,7 @@ int iocow_read_byte(void *);
int iocow_triplet(void *, int);
int iocow_pulse(struct ioc_softc *, int, int);
+#ifdef TGT_ORIGIN
/*
* A mask of nodes on which an ioc driver has attached.
* We use this to prevent attaching a pci IOC3 card which NIC has failed,
@@ -114,6 +118,7 @@ int iocow_pulse(struct ioc_softc *, int, int);
* XXX This obviously will not work in N mode...
*/
static uint64_t ioc_nodemask = 0;
+#endif
int
ioc_match(struct device *parent, void *match, void *aux)
@@ -258,6 +263,7 @@ ioc_attach(struct device *parent, struct device *self, void *aux)
goto unknown;
}
} else {
+#ifdef TGT_ORIGIN
/*
* If no owserial device has been found, then it is
* very likely that we are the on-board IOC3 found
@@ -267,7 +273,7 @@ ioc_attach(struct device *parent, struct device *self, void *aux)
if ((sys_config.system_type == SGI_IP27 ||
sys_config.system_type == SGI_IP35) &&
!ISSET(ioc_nodemask, 1UL << currentnasid)) {
- SET(currentnasid, 1UL << currentnasid);
+ SET(ioc_nodemask, 1UL << currentnasid);
device_mask = (1 << IOCDEV_SERIAL_A) |
(1 << IOCDEV_SERIAL_B) | (1 << IOCDEV_LPT) |
@@ -286,13 +292,9 @@ ioc_attach(struct device *parent, struct device *self, void *aux)
rtcbase = IOC3_BYTEBUS_0;
dual_irq = 1;
- /*
- * XXX On IP35 class machines, there are no
- * XXX Number-In-a-Can chips to tell us the
- * XXX Ethernet address, we need to query
- * XXX the L1 controller.
- */
- } else {
+ } else
+#endif
+ {
unknown:
/*
* Well, we don't really know what kind of device
@@ -464,6 +466,8 @@ ioc_attach_child(struct device *ioc, const char *name, bus_addr_t base, int dev)
struct ioc_softc *sc = (struct ioc_softc *)ioc;
struct ioc_attach_args iaa;
+ memset(&iaa, 0, sizeof iaa);
+
iaa.iaa_name = name;
iaa.iaa_memt = sc->sc_memt;
iaa.iaa_memh = sc->sc_memh;
@@ -471,17 +475,23 @@ ioc_attach_child(struct device *ioc, const char *name, bus_addr_t base, int dev)
iaa.iaa_base = base;
iaa.iaa_dev = dev;
- if (sc->sc_owmac != NULL)
- memcpy(iaa.iaa_enaddr, sc->sc_owmac->sc_enaddr, 6);
- else {
- /*
- * XXX On IP35, there is no Number-In-a-Can attached to
- * XXX the onboard IOC3; instead, the Ethernet address
- * XXX is stored in the machine eeprom and can be
- * XXX queried by sending the appropriate L1 command
- * XXX to the L1 UART. This L1 code is not written yet.
- */
- memset(iaa.iaa_enaddr, 0xff, 6);
+ if (dev == IOCDEV_EF) {
+ if (sc->sc_owmac != NULL)
+ memcpy(iaa.iaa_enaddr, sc->sc_owmac->sc_enaddr, 6);
+ else {
+#ifdef TGT_ORIGIN
+ /*
+ * On IP35 class machines, there are no
+ * Number-In-a-Can attached to the onboard
+ * IOC3; instead, the Ethernet address is
+ * stored in the Brick EEPROM, and can be
+ * retrieved with an L1 controller query.
+ */
+ if (l1_get_brick_ethernet_address(currentnasid,
+ iaa.iaa_enaddr) != 0)
+#endif
+ memset(iaa.iaa_enaddr, 0xff, 6);
+ }
}
return config_found_sm(ioc, &iaa, ioc_print, ioc_search_mundane);
diff --git a/sys/arch/sgi/sgi/l1.c b/sys/arch/sgi/sgi/l1.c
new file mode 100644
index 00000000000..9357b08e0e9
--- /dev/null
+++ b/sys/arch/sgi/sgi/l1.c
@@ -0,0 +1,865 @@
+/* $OpenBSD: l1.c,v 1.1 2009/11/08 22:44:16 miod Exp $ */
+
+/*
+ * Copyright (c) 2009 Miodrag Vallat.
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
+ * Communication with the L1 controller, on IP35 systems.
+ * We use a direct 57600 bps serial link from each processor to the L1 chip.
+ * Information is sent as ppp-encoded packets.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/malloc.h>
+
+#include <machine/autoconf.h>
+#include <machine/mnode.h>
+#include <sgi/xbow/hub.h>
+
+#include <dev/ic/ns16550reg.h>
+#include <dev/ic/comreg.h>
+
+#include <net/ppp_defs.h>
+
+#include <sgi/sgi/l1.h>
+
+/*
+ * L1 communication defines
+ */
+
+/* packet types */
+#define L1PKT_REQUEST 0x00
+#define L1PKT_RESPONSE 0x20
+#define L1PKT_EVENT 0x40
+
+/* packet subchannels */
+#define L1CH_CPU0 0x00 /* exclusive channel for cpu 0 */
+#define L1CH_CPU1 0x01 /* exclusive channel for cpu 1 */
+#define L1CH_CPU2 0x02 /* exclusive channel for cpu 2 */
+#define L1CH_CPU3 0x03 /* exclusive channel for cpu 3 */
+#define L1CH_CONSOLE 0x04 /* L1 console */
+/* 05..0f reserved */
+/* 10..1f available for operating system */
+#define L1CH_MISC 0x10
+
+/* argument encoding */
+#define L1_ARG_INT 0x00 /* followed by 32 bit BE value */
+#define L1_ARG_ASCII 0x01 /* followed by NUL terminated string */
+#define L1_ARG_BINARY 0x80 /* length in low 7 bits */
+
+int l1_serial_getc(int16_t);
+int l1_serial_putc(int16_t, u_char);
+int l1_serial_ppp_write(int16_t, uint16_t *, u_char, int);
+int l1_serial_ppp_read(int16_t, int);
+int l1_packet_put(int16_t, u_char *, size_t);
+int l1_packet_get(int16_t, u_char *, size_t);
+static inline
+void l1_packet_put_be32(u_char *, uint32_t);
+static inline
+void l1_packet_put_be16(u_char *, uint16_t);
+static inline
+uint32_t l1_packet_get_be32(u_char *);
+int l1_packet_get_int(u_char **, size_t *, uint32_t *);
+int l1_packet_get_ascii(u_char **, size_t *, char **);
+int l1_packet_get_binary(u_char **, size_t *, u_char **, size_t *);
+size_t l1_command_build(u_char *, size_t, uint32_t, uint16_t, int, ...);
+int l1_receive_response(int16_t, u_char *, size_t *);
+
+static inline
+size_t ia_skip(u_char *, size_t);
+
+/* l1_packet_get() return values */
+#define L1PG_TIMEOUT -1
+#define L1PG_BADCRC -2
+#define L1PG_SHORTPACKET -3
+
+/*
+ * Basic serial routines (polled)
+ */
+
+#define L1_UART_ADDRESS(nasid, r) \
+ IP27_RHSPEC_ADDR(nasid, HSPEC_L1_UART(r))
+
+int
+l1_serial_getc(int16_t nasid)
+{
+ uint64_t lsr;
+ int n;
+
+ for (n = 1000000; n != 0; n--) {
+ lsr = *(volatile uint64_t *)L1_UART_ADDRESS(nasid, com_lsr);
+ if ((lsr & LSR_RXRDY) != 0)
+ break;
+ }
+
+ if (n == 0) {
+#ifdef L1_DEBUG
+ printf("%s: RX timeout, lsr %02x\n", __func__, lsr);
+#endif
+ return -1;
+ }
+
+ return *(volatile uint64_t *)L1_UART_ADDRESS(nasid, com_data) & 0xff;
+
+}
+
+int
+l1_serial_putc(int16_t nasid, u_char val)
+{
+ uint64_t lsr;
+ int n;
+
+ for (n = 1000000; n != 0; n--) {
+ lsr = *(volatile uint64_t *)L1_UART_ADDRESS(nasid, com_lsr);
+ if ((lsr & LSR_TXRDY) != 0)
+ break;
+ }
+
+ if (n == 0) {
+#ifdef L1_DEBUG
+ printf("%s: TX timeout, lsr %02x\n", __func__, lsr);
+#endif
+ return EWOULDBLOCK;
+ }
+
+ *(volatile uint64_t *)L1_UART_ADDRESS(nasid, com_data) = (uint64_t)val;
+ return 0;
+}
+
+/*
+ * Single character routines, with optional ppp frame escaping, and optional
+ * ppp crc update.
+ */
+
+/*
+ * FCS lookup table as calculated by genfcstab.
+ * Straight from <net/ppp_tty.c>, only 512 bytes long; probably not worth
+ * trying to share with outsmart config(8) machinery...
+ */
+static const u_int16_t fcstab[256] = {
+ 0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
+ 0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,
+ 0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,
+ 0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,
+ 0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,
+ 0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,
+ 0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,
+ 0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,
+ 0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,
+ 0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,
+ 0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,
+ 0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,
+ 0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,
+ 0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,
+ 0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,
+ 0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,
+ 0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,
+ 0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,
+ 0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,
+ 0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,
+ 0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,
+ 0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,
+ 0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,
+ 0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,
+ 0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,
+ 0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,
+ 0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,
+ 0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,
+ 0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,
+ 0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,
+ 0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,
+ 0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78
+};
+
+int
+l1_serial_ppp_write(int16_t nasid, uint16_t *crc, u_char data, int escape)
+{
+ /* update crc if necessary */
+ if (crc != NULL)
+ *crc = PPP_FCS(*crc, data);
+
+ /* escape data if necessary */
+ if (escape && (data == PPP_FLAG || data == PPP_ESCAPE)) {
+ if (l1_serial_putc(nasid, PPP_ESCAPE) != 0)
+ return EWOULDBLOCK;
+ data ^= PPP_TRANS;
+ }
+
+ return l1_serial_putc(nasid, data);
+}
+
+int
+l1_serial_ppp_read(int16_t nasid, int unescape)
+{
+ int data;
+
+ if ((data = l1_serial_getc(nasid)) < 0)
+ return data;
+
+ /* unescape data if necessary */
+ if (unescape && data == PPP_ESCAPE)
+ data = l1_serial_getc(nasid) ^ PPP_TRANS;
+
+ return data;
+}
+
+/*
+ * Complete ppp packet emission and reception.
+ */
+
+int
+l1_packet_put(int16_t nasid, u_char *packet, size_t len)
+{
+ uint16_t crc = PPP_INITFCS;
+
+ /* send incoming packet flag */
+ if (l1_serial_ppp_write(nasid, NULL, PPP_FLAG, 0) != 0)
+ return EWOULDBLOCK;
+
+ /* send packet data */
+ while (len-- != 0)
+ if (l1_serial_ppp_write(nasid, &crc, *packet++, 1) != 0)
+ return EWOULDBLOCK;
+
+ /* send crc */
+ crc ^= PPP_INITFCS;
+ if (l1_serial_ppp_write(nasid, NULL, crc & 0xff, 1) != 0)
+ return EWOULDBLOCK;
+ if (l1_serial_ppp_write(nasid, NULL, (crc >> 8) & 0xff, 1) != 0)
+ return EWOULDBLOCK;
+
+ /* send final packet byte flag */
+ if (l1_serial_ppp_write(nasid, &crc, PPP_FLAG, 0) != 0)
+ return EWOULDBLOCK;
+
+ return 0;
+}
+
+int
+l1_packet_get(int16_t nasid, u_char *buf, size_t buflen)
+{
+ uint16_t crc;
+ size_t rcvlen;
+ int data;
+
+ /* wait for incoming packet flag */
+ for (;;) {
+ data = l1_serial_ppp_read(nasid, 0);
+ if (data < 0)
+ return L1PG_TIMEOUT;
+ if (data == PPP_FLAG)
+ break;
+ }
+
+ /* read packet */
+ rcvlen = 0;
+ crc = PPP_INITFCS;
+ for (;;) {
+ data = l1_serial_ppp_read(nasid, 1);
+ if (data < 0)
+ return L1PG_TIMEOUT;
+ if (data == PPP_FLAG) /* end of packet */
+ break;
+ if (rcvlen < buflen)
+ buf[rcvlen] = data;
+ rcvlen++;
+ crc = PPP_FCS(crc, data);
+ }
+
+ if (rcvlen < 2) {
+#ifdef L1_DEBUG
+ printf("%s: short packet\n", __func__);
+#endif
+ return L1PG_SHORTPACKET;
+ }
+
+ /* check CRC */
+ rcvlen -= 2; /* crc bytes */
+ if (crc != PPP_GOODFCS) {
+#ifdef L1_DEBUG
+ printf("%s: CRC error (%04x)\n", __func__, crc);
+#endif
+ return L1PG_BADCRC;
+ }
+
+ return rcvlen;
+}
+
+/*
+ * L1 packet construction and deconstruction helpers
+ */
+
+static inline void
+l1_packet_put_be32(u_char *buf, uint32_t data)
+{
+ *buf++ = data >> 24;
+ *buf++ = data >> 16;
+ *buf++ = data >> 8;
+ *buf++ = data;
+}
+
+static inline void
+l1_packet_put_be16(u_char *buf, uint16_t data)
+{
+ *buf++ = data >> 8;
+ *buf++ = data;
+}
+
+static inline uint32_t
+l1_packet_get_be32(u_char *buf)
+{
+ uint32_t data;
+
+ data = *buf++;
+ data <<= 8;
+ data |= *buf++;
+ data <<= 8;
+ data |= *buf++;
+ data <<= 8;
+ data |= *buf++;
+
+ return data;
+}
+
+int
+l1_packet_get_int(u_char **buf, size_t *buflen, uint32_t *rval)
+{
+ u_char *b = *buf;
+
+ if (*buflen < 5)
+ return ENOMEM;
+
+ if (*b++ != L1_ARG_INT)
+ return EINVAL;
+
+ *rval = l1_packet_get_be32(b);
+
+ b += 4;
+ *buf = b;
+ *buflen -= 5;
+
+ return 0;
+}
+
+int
+l1_packet_get_ascii(u_char **buf, size_t *buflen, char **rval)
+{
+ u_char *b = *buf;
+ u_char *s, *e;
+
+ if (*buflen < 2)
+ return ENOMEM;
+
+ if (*b != L1_ARG_ASCII)
+ return EINVAL;
+
+ /* check for a terminating NUL within the given bounds */
+ e = b + *buflen;
+ for (s = b + 1; s != e; s++)
+ if (*s == '\0')
+ break;
+ if (s == e)
+ return ENOMEM;
+
+ *rval = (char *)b + 1;
+
+ s++;
+ *buflen -= s - b;
+ *buf = s;
+
+ return 0;
+}
+
+int
+l1_packet_get_binary(u_char **buf, size_t *buflen, u_char **rval, size_t *rlen)
+{
+ u_char *b = *buf;
+ size_t datalen;
+
+ if (*buflen < 1)
+ return ENOMEM;
+
+ if ((*b & L1_ARG_BINARY) == 0)
+ return EINVAL;
+
+ datalen = *b & ~L1_ARG_BINARY;
+ if (*buflen < 1 + datalen)
+ return EINVAL;
+
+ b++;
+ *rval = b;
+ *rlen = datalen;
+ *buflen -= 1 + datalen;
+ b += datalen;
+ *buf = b;
+
+ return 0;
+}
+
+/*
+ * Build a whole request packet.
+ */
+
+size_t
+l1_command_build(u_char *buf, size_t buflen, uint32_t address, uint16_t request,
+ int nargs, ...)
+{
+ va_list ap;
+ uint32_t data;
+ size_t len = 0;
+ int argtype;
+ const char *str;
+
+ /*
+ * Setup packet header (type, channel, address, request)
+ */
+
+ if (buflen >= 1) {
+ *buf++ = L1PKT_REQUEST | L1CH_MISC;
+ buflen--;
+ }
+ len++;
+
+ if (buflen >= 4) {
+ l1_packet_put_be32(buf, address);
+ buf += 4;
+ buflen -= 4;
+ }
+ len += 4;
+
+ if (buflen >= 2) {
+ l1_packet_put_be16(buf, request);
+ buf += 2;
+ buflen -= 2;
+ }
+ len += 2;
+
+ /*
+ * Setup command arguments
+ */
+
+ if (buflen >= 1) {
+ *buf++ = nargs;
+ buflen--;
+ }
+ len++;
+
+ va_start(ap, nargs);
+ while (nargs-- != 0) {
+ argtype = va_arg(ap, int);
+ switch (argtype) {
+ case L1_ARG_INT:
+ data = (uint32_t)va_arg(ap, int);
+ if (buflen >= 5) {
+ *buf++ = L1_ARG_INT;
+ l1_packet_put_be32(buf, data);
+ buf += 4;
+ buflen -= 5;
+ }
+ len += 5;
+ break;
+ case L1_ARG_ASCII:
+ str = va_arg(ap, const char *);
+ data = strlen(str);
+ if (buflen >= data + 2) {
+ *buf++ = L1_ARG_ASCII;
+ memcpy(buf, str, data + 1);
+ buf += data + 1;
+ buflen -= data + 2;
+ }
+ len += data + 2;
+ break;
+ case L1_ARG_BINARY:
+ data = (uint32_t)va_arg(ap, size_t); /* size */
+ str = va_arg(ap, const char *); /* data */
+ if (buflen >= 1 + data) {
+ *buf++ = L1_ARG_BINARY | data;
+ memcpy(buf, str, data);
+ buf += data;
+ buflen -= data + 1;
+ }
+ len += data + 1;
+ break;
+ }
+ }
+ va_end(ap);
+
+ return len;
+}
+
+/*
+ * Get a packet response, ignoring any other packet seen in between.
+ * Note that despite being the only user of L1 in the kernel, we may
+ * receive event packets from the console.
+ */
+int
+l1_receive_response(int16_t nasid, u_char *pkt, size_t *pktlen)
+{
+ int rc;
+
+ for (;;) {
+ rc = l1_packet_get(nasid, pkt, *pktlen);
+ if (rc == L1PG_TIMEOUT)
+ return EWOULDBLOCK;
+
+ if (rc < 0) /* bad packet */
+ continue;
+
+ if (pkt[0] != (L1PKT_RESPONSE | L1CH_MISC))
+ continue; /* it's not our response */
+
+ *pktlen = (size_t)rc;
+ return 0;
+ }
+}
+
+/*
+ * Read a board IA information record from EEPROM
+ */
+
+#define EEPROM_CHUNK 0x40
+
+int
+l1_read_board_ia(int16_t nasid, u_char **ria, size_t *rialen)
+{
+ u_char pkt[64 + EEPROM_CHUNK]; /* command and response packet buffer */
+ u_char *pktbuf, *chunk, *ia = NULL;
+ size_t pktlen, chunklen, ialen, iapos;
+ uint32_t data;
+ int rc;
+
+ /*
+ * Build a first packet, asking for 0 bytes to be read.
+ */
+ pktlen = l1_command_build(pkt, sizeof pkt,
+ L1_ADDRESS(L1_TYPE_L1, L1_ADDRESS_LOCAL | L1_TASK_GENERAL),
+ L1_REQ_EEPROM, 4,
+ L1_ARG_INT, L1_EEP_LOGIC, L1_ARG_INT, L1_EEP_BOARD,
+ L1_ARG_INT, 0 /* offset */, L1_ARG_INT, 0 /* size */);
+ if (pktlen > sizeof pkt) {
+#ifdef DIAGNOSTIC
+ panic("L1 command packet too large (%zu) for buffer", pktlen);
+#endif
+ return ENOMEM;
+ }
+
+ if (l1_packet_put(nasid, pkt, pktlen) != 0)
+ return EWOULDBLOCK;
+
+ pktlen = sizeof pkt;
+ if (l1_receive_response(nasid, pkt, &pktlen) != 0)
+ return EWOULDBLOCK;
+
+ if (pktlen <= 6) {
+#ifdef L1_DEBUG
+ printf("truncated response (length %d)\n", pktlen);
+#endif
+ return ENXIO;
+ }
+
+ /*
+ * Check the response code.
+ */
+
+ data = l1_packet_get_be32(&pkt[1]);
+ if (data != L1_RESP_OK) {
+#ifdef L1_DEBUG
+ printf("unexpected L1 response code: %08x\n", data);
+#endif
+ return ENXIO;
+ }
+
+ /*
+ * EEPROM read commands should return either one or two values:
+ * the first value is the size of the remaining EEPROM data, and
+ * the second value is the data read itself, if we asked for a
+ * nonzero size in the command (that size might be shorter than
+ * the data we asked for).
+ */
+
+ if (pkt[5] != 1) {
+#ifdef L1_DEBUG
+ printf("unexpected L1 response: %d values\n", pkt[5]);
+#endif
+ return ENXIO;
+ }
+
+ pktbuf = pkt + 6;
+ pktlen -= 6;
+
+ if (l1_packet_get_int(&pktbuf, &pktlen, &data) != 0) {
+#ifdef L1_DEBUG
+ printf("unable to parse response as integer\n");
+#endif
+ return ENXIO;
+ }
+
+ /*
+ * Now that we know the size of the IA entry, allocate memory for it.
+ */
+
+ ialen = (size_t)data;
+ ia = (u_char *)malloc(ialen, M_DEVBUF, M_NOWAIT);
+ if (ia == NULL)
+ return ENOMEM;
+
+ /*
+ * Read the EEPROM contents in small chunks, so as not to keep L1
+ * busy for too long.
+ */
+
+ iapos = 0;
+ while (iapos < ialen) {
+ /*
+ * Build a command packet, this time actually reading data.
+ */
+ pktlen = l1_command_build(pkt, sizeof pkt,
+ L1_ADDRESS(L1_TYPE_L1, L1_ADDRESS_LOCAL | L1_TASK_GENERAL),
+ L1_REQ_EEPROM, 4,
+ L1_ARG_INT, L1_EEP_LOGIC, L1_ARG_INT, L1_EEP_BOARD,
+ L1_ARG_INT, iapos, L1_ARG_INT, EEPROM_CHUNK);
+ /* no need to check size again, it's the same size as earlier */
+
+ if (l1_packet_put(nasid, pkt, pktlen) != 0) {
+ rc = EWOULDBLOCK;
+ goto fail;
+ }
+
+ pktlen = sizeof pkt;
+ if (l1_receive_response(nasid, pkt, &pktlen) != 0) {
+ rc = EWOULDBLOCK;
+ goto fail;
+ }
+
+ if (pktlen <= 6) {
+#ifdef L1_DEBUG
+ printf("truncated response (length %d)\n", pktlen);
+#endif
+ rc = ENXIO;
+ goto fail;
+ }
+
+ /*
+ * Check the response code.
+ */
+
+ data = l1_packet_get_be32(&pkt[1]);
+ if (data != L1_RESP_OK) {
+#ifdef L1_DEBUG
+ printf("unexpected L1 response code: %08x\n", data);
+#endif
+ rc = ENXIO;
+ goto fail;
+ }
+
+ if (pkt[5] != 2) {
+#ifdef L1_DEBUG
+ printf("unexpected L1 response: %d values\n", pkt[5]);
+#endif
+ rc = ENXIO;
+ goto fail;
+ }
+
+ pktbuf = pkt + 6;
+ pktlen -= 6;
+
+ if (l1_packet_get_int(&pktbuf, &pktlen, &data) != 0) {
+#ifdef L1_DEBUG
+ printf("unable to parse first response as integer\n");
+#endif
+ rc = ENXIO;
+ goto fail;
+ }
+
+ if (l1_packet_get_binary(&pktbuf, &pktlen,
+ &chunk, &chunklen) != 0) {
+#ifdef L1_DEBUG
+ printf("unable to parse second response as binary\n");
+#endif
+ rc = ENXIO;
+ goto fail;
+ }
+
+ /* should not happen, but we don't like infinite loops */
+ if (chunklen == 0) {
+#ifdef L1_DEBUG
+ printf("read command returned 0 bytes\n");
+#endif
+ rc = ENXIO;
+ goto fail;
+ }
+
+ memcpy(ia + iapos, chunk, chunklen);
+ iapos += chunklen;
+#ifdef L1_DEBUG
+ printf("got %02x bytes of eeprom, %x/%x\n",
+ chunklen, iapos, ialen);
+#endif
+ }
+
+ /*
+ * Verify the checksum
+ */
+
+ chunk = ia;
+ iapos = ialen;
+ data = 0;
+ while (iapos-- != 0)
+ data += *chunk++;
+ if ((data & 0xff) != 0) {
+#ifdef L1_DEBUG
+ printf("wrong IA checksum\n");
+#endif
+ rc = EINVAL;
+ goto fail;
+ }
+
+ *ria = ia;
+ *rialen = ialen;
+ return 0;
+
+fail:
+ if (ia != NULL)
+ free(ia, M_DEVBUF);
+ return rc;
+}
+
+/*
+ * Information Area (IA) decoding helpers
+ *
+ * The format of an Information Area is as follows:
+ * B format byte (00)
+ * B length in 8 byte units
+ * B language (00 = english)
+ * 3B manufacturing date, minutes since 1/1/1996
+ * B type/length of manufacturer name string (up to 20 chars)
+ * #B manufacturer name
+ * B type/length of product name string (up to 14 chars)
+ * #B product name
+ * B type/length of serial number (up to 6 chars)
+ * #B serial number
+ * B type/length of part number (up to 10 chars)
+ * #B part number
+ * B FRU file id
+ * B type/length of board rev (always 0xC2)
+ * 2B board reviison
+ * B type/length of eeprom size field (0x01)
+ * 1B size code for eeprom (02)
+ * B type/length of temp waiver field (0xC2)
+ * 2B temp waiver
+ *
+ * and then in main boards only:
+ * G, P and Y encryption keys, each being
+ * B type/length (0x04)
+ * 4B key
+ *
+ * and then on main boards being I-Bricks only:
+ * B type/length of mac address (as ascii string)
+ * 12B mac address
+ * followed by IEEE1394 configuration info, as type/length followed
+ * by data again.
+ *
+ * A 0xC1 byte is the EOF record, and the last byte is a checksum.
+ *
+ * Type/length encoding is done as follows:
+ * bits 7-6 are the type:
+ * 00 binary data
+ * 01 BCD
+ * 02 packed 6 bit ascii
+ * 03 regular 8 bit ascii
+ * bits 5-0 are the length.
+ */
+
+#define IA_TYPE_SHIFT 6
+#define IA_TYPE_BINARY 0
+#define IA_TYPE_BCD 1
+#define IA_TYPE_PACKED 2
+#define IA_TYPE_ASCII 3
+#define IA_LENGTH_MASK 0x3f
+
+#define IA_TL(t,l) (((t) << IA_TYPE_SHIFT) | (l))
+
+#define IA_EOF IA_TL(IA_TYPE_ASCII, 1)
+
+static inline size_t
+ia_skip(u_char *ia, size_t iapos)
+{
+ ia += iapos;
+
+ /* don't go past EOF marker */
+ if (*ia == IA_EOF)
+ return iapos;
+
+ iapos += 1 + (*ia & IA_LENGTH_MASK);
+ return iapos;
+}
+
+int
+l1_get_brick_ethernet_address(int16_t nasid, uint8_t *enaddr)
+{
+ u_char *ia, *mac;
+ size_t iapos, ialen;
+ char hexaddr[18];
+ int rc;
+
+ /* read the Board IA of this node */
+ rc = l1_read_board_ia(nasid, &ia, &ialen);
+ if (rc != 0)
+ return rc;
+
+ /* simple sanity checks */
+ if (ia[0] != 0 || ia[1] * 8 > ialen) {
+ rc = EINVAL;
+ goto out;
+ }
+
+ /* skip fixed part */
+ iapos = 6;
+ /* skip 4 records */
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ /* skip FRU */
+ iapos++;
+ /* skip 3 records */
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ /* skip encryption key records if applicable */
+ if (ia[iapos] == IA_TL(IA_TYPE_BINARY, 4)) {
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ iapos = ia_skip(ia, iapos);
+ }
+ /* check the next record looks like an Ethernet address */
+ if (ia[iapos] != IA_TL(IA_TYPE_ASCII, 12)) {
+ rc = EINVAL;
+ goto out;
+ }
+
+ iapos++;
+ mac = ia + iapos;
+ snprintf(hexaddr, sizeof hexaddr, "%c%c:%c%c:%c%c:%c%c:%c%c:%c%c",
+ mac[0], mac[1], mac[2], mac[3], mac[4], mac[5],
+ mac[6], mac[7], mac[8], mac[9], mac[10], mac[11]);
+ enaddr_aton(hexaddr, enaddr);
+
+out:
+ free(ia, M_DEVBUF);
+ return rc;
+}
diff --git a/sys/arch/sgi/sgi/l1.h b/sys/arch/sgi/sgi/l1.h
new file mode 100644
index 00000000000..db400ed7370
--- /dev/null
+++ b/sys/arch/sgi/sgi/l1.h
@@ -0,0 +1,73 @@
+/* $OpenBSD: l1.h,v 1.1 2009/11/08 22:44:16 miod Exp $ */
+
+/*
+ * Copyright (c) 2009 Miodrag Vallat.
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
+ * High-level L1 communication defines
+ */
+
+/* L1 command types and destination addresses */
+#define L1_ADDRESS(type,addr) (((type) << 28) | (addr))
+
+#define L1_TYPE_L1 0x00
+#define L1_TYPE_L2 0x01
+#define L1_TYPE_L3 0x02
+#define L1_TYPE_CBRICK 0x03
+#define L1_TYPE_IOBRICK 0x04
+
+#define L1_ADDRESS_RACK_MASK 0x0ffc0000
+#define L1_ADDRESS_RACK_SHIFT 18
+#define L1_ADDRESS_RACK_LOCAL 0x3ff
+#define L1_ADDRESS_BAY_MASK 0x0003f000
+#define L1_ADDRESS_BAY_SHIFT 12
+#define L1_ADDRESS_BAY_LOCAL 0x3f
+#define L1_ADDRESS_TASK_MASK 0x0000001f
+#define L1_ADDRESS_TASK_SHIFT 0
+
+#define L1_ADDRESS_LOCAL \
+ ((L1_ADDRESS_RACK_LOCAL << L1_ADDRESS_RACK_SHIFT) | \
+ (L1_ADDRESS_BAY_LOCAL << L1_ADDRESS_BAY_SHIFT))
+
+#define L1_TASK_INVALID 0x00
+#define L1_TASK_ROUTER 0x01
+#define L1_TASK_SYSMGMT 0x02
+#define L1_TASK_COMMAND 0x03
+#define L1_TASK_ENVIRONMENT 0x04
+#define L1_TASK_BEDROCK 0x05
+#define L1_TASK_GENERAL 0x06
+
+/* response codes */
+#define L1_RESP_OK 0x00000000
+
+/*
+ * Various commands (very incomplete list)
+ */
+
+/* L1_TASK_COMMAND requests */
+#define L1_REQ_EXEC_CMD 0x0000 /* interpret plaintext command */
+
+/* L1_TASK_GENERAL requests */
+#define L1_REQ_EEPROM 0x0006 /* access eeprom */
+#define L1_REQ_DISP1 0x1004 /* display text on LCD first line */
+#define L1_REQ_DISP2 0x1005 /* display text on LCD second line */
+
+/* L1_REQ_EEPROM additional argument value */
+#define L1_EEP_LOGIC 0x01 /* component: logic board */
+#define L1_EEP_BOARD 0x02 /* ia code: board */
+
+int l1_read_board_ia(int16_t, u_char **, size_t *);
+int l1_get_brick_ethernet_address(int16_t, uint8_t *);
diff --git a/sys/arch/sgi/xbow/hub.h b/sys/arch/sgi/xbow/hub.h
index 63dfed9fb54..e0dcc2d75a0 100644
--- a/sys/arch/sgi/xbow/hub.h
+++ b/sys/arch/sgi/xbow/hub.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: hub.h,v 1.6 2009/10/22 22:08:54 miod Exp $ */
+/* $OpenBSD: hub.h,v 1.7 2009/11/08 22:44:16 miod Exp $ */
/*
* Copyright (c) 2009 Miodrag Vallat.
@@ -65,9 +65,12 @@
#define LBOOTBASE_IP35 0x18000000
#define LBOOTSIZE_IP35 0x08000000
+#define HSPEC_L1_UARTBASE 0x00000080
#define HSPEC_SYNERGY0 0x04000000 /* synergy #0 base */
#define HSPEC_SYNERGY1 0x05000000 /* synergy #1 base */
+#define HSPEC_L1_UART(r) \
+ (LREGBASE_IP35 + HSPEC_L1_UARTBASE + ((r) << 3))
#define HSPEC_SYNERGY(s,r) \
(LREGBASE_IP35 + ((s) ? HSPEC_SYNERGY1 : HSPEC_SYNERGY0) + ((r) << 3))