diff options
Diffstat (limited to 'sys')
-rw-r--r-- | sys/dev/pci/arc.c | 103 |
1 files changed, 100 insertions, 3 deletions
diff --git a/sys/dev/pci/arc.c b/sys/dev/pci/arc.c index 3598ac2d553..baeb104bd32 100644 --- a/sys/dev/pci/arc.c +++ b/sys/dev/pci/arc.c @@ -1,4 +1,4 @@ -/* $OpenBSD: arc.c,v 1.26 2006/08/18 10:40:04 dlg Exp $ */ +/* $OpenBSD: arc.c,v 1.27 2006/08/18 13:55:52 dlg Exp $ */ /* * Copyright (c) 2006 David Gwynne <dlg@openbsd.org> @@ -203,6 +203,7 @@ struct arc_fw_hdr { /* the fw header must always equal this */ struct arc_fw_hdr arc_fw_hdr = { 0x5e, 0x01, 0x61 }; +#define ARC_FW_SYSINFO 0x23 /* opcode. reply is fw_sysinfo */ #define ARC_FW_MUTE_ALARM 0x30 /* opcode only */ #define ARC_FW_SET_ALARM 0x31 /* opcode + 1 byte for setting */ #define ARC_FW_SET_ALARM_DISABLE 0x00 @@ -221,6 +222,54 @@ struct { \ } __packed #define ARC_FW_MSGBUF(_msg) ((void *)(_msg)->msg) +struct arc_fw_comminfo { + u_int8_t baud_rate; + u_int8_t data_bits; + u_int8_t stop_bits; + u_int8_t parity; + u_int8_t flow_control; +} __packed; + +struct arc_fw_sysinfo { + u_int8_t vendor_name[40]; + u_int8_t serial_number[16]; + u_int8_t firmware_version[16]; + u_int8_t boot_version[16]; + u_int8_t mb_version[16]; + u_int8_t model_name[8]; + + u_int8_t local_ip[4]; + u_int8_t current_ip[4]; + + u_int32_t time_tick; + u_int32_t cpu_speed; + u_int32_t icache; + u_int32_t dcache; + u_int32_t scache; + u_int32_t memory_size; + u_int32_t memory_speed; + u_int32_t events; + + u_int8_t gsiMacAddress[6]; + u_int8_t gsiDhcp; + + u_int8_t alarm; + u_int8_t channel_usage; + u_int8_t max_ata_mode; + u_int8_t sdram_ecc; + u_int8_t rebuild_priority; + struct arc_fw_comminfo comm_a; + struct arc_fw_comminfo comm_b; + u_int8_t ide_channels; + u_int8_t scsi_host_channels; + u_int8_t ide_host_channels; + u_int8_t max_volume_set; + u_int8_t max_raid_set; + u_int8_t ether_port; + u_int8_t raid6_engine; + u_int8_t reserved[75]; +} __packed; + int arc_match(struct device *, void *, void *); void arc_attach(struct device *, struct device *, void *); int arc_detach(struct device *, int); @@ -346,6 +395,8 @@ int arc_msgbuf(struct arc_softc *, void *, size_t, #if NBIO > 0 int arc_bioctl(struct device *, u_long, caddr_t); int arc_bio_alarm(struct arc_softc *, struct bioc_alarm *); +int arc_bio_alarm_state(struct arc_softc *, + struct bioc_alarm *); #endif int @@ -842,9 +893,12 @@ arc_bio_alarm(struct arc_softc *sc, struct bioc_alarm *ba) break; + case BIOC_GASTATUS: + /* system info is too big/ugly to deal with here */ + return (arc_bio_alarm_state(sc, ba)); + default: - error = EOPNOTSUPP; - break; + return (EOPNOTSUPP); } arc_lock(sc); @@ -861,6 +915,49 @@ arc_bio_alarm(struct arc_softc *sc, struct bioc_alarm *ba) return (0); } + +int +arc_bio_alarm_state(struct arc_softc *sc, struct bioc_alarm *ba) +{ + ARC_FW_MSG(1) request; + ARC_FW_MSG(sizeof(struct arc_fw_sysinfo)) *reply; + struct arc_fw_sysinfo *sysinfo; + int error = 0; + + reply = malloc(sizeof(*reply), M_TEMP, M_WAITOK); + if (reply == NULL) + return (ENOMEM); + + sysinfo = ARC_FW_MSGBUF(reply); + + request.hdr = arc_fw_hdr; + request.len = htole16(sizeof(request.msg)); + request.msg[0] = ARC_FW_SYSINFO; + request.cksum = arc_msg_cksum(&request, sizeof(request)); + + arc_lock(sc); + error = arc_msgbuf(sc, &request, sizeof(request), + reply, sizeof(*reply)); + arc_unlock(sc); + + if (error != 0) + goto out; + + if (memcmp(&reply->hdr, &arc_fw_hdr, sizeof(reply->hdr)) != 0 || + reply->cksum != arc_msg_cksum(reply, sizeof(*reply))) { + error = EIO; + goto out; + } + + printf("%s: %s value: 0x%02x\n", DEVNAME(sc), __func__, + sysinfo->alarm); + + ba->ba_status = sysinfo->alarm; + +out: + free(reply, M_TEMP); + return (error); +} #endif /* NBIO > 0 */ u_int8_t |