ref: 7b698ef54a538ff2cb27fe4a76c73faf4fd1987a
parent: c6a3e6b464ba23d8c99355b77c49690d6d07a6c4
author: aiju <aiju@phicode.de>
date: Mon May 16 18:56:52 EDT 2011
added aux/icanhasmsi
--- /dev/null
+++ b/sys/src/cmd/aux/icanhasmsi.c
@@ -1,0 +1,60 @@
+#include <u.h>
+#include <libc.h>
+
+int
+pcicfg16r(int fd, ushort *s, vlong offset)
+{+ char buf[2];
+
+ if(pread(fd, buf, 2, offset) < 2) return -1;
+ *s = buf[0] | (buf[1] << 8);
+ return 0;
+}
+
+void
+main()
+{+ int fd;
+ long n;
+ Dir *dir;
+ char *p, *s;
+ uchar cap, c;
+ ushort sh;
+
+ fd = open("/dev/pci", OREAD);+ if(fd < 0) sysfatal("open /dev/pci: %r");+ n = dirreadall(fd, &dir);
+ if(n < 0) sysfatal("dirreadall /dev/pci: %r");+ close(fd);
+ for(; n--; dir++) {+ p = dir->name + strlen(dir->name) - 3;
+ if(strcmp(p, "raw") != 0)
+ continue;
+ s = smprint("/dev/pci/%s", dir->name);+ fd = open(s, OREAD);
+ if(fd < 0) {+ fprint(2, "open %s: %r", s);
+ free(s);
+ continue;
+ }
+ if(pcicfg16r(fd, &sh, 0) < 0) goto err;
+ if(sh == 0xFFFF) goto end;
+ if(pcicfg16r(fd, &sh, 0x06) < 0) goto err;
+ if((sh & (1<<4)) == 0) goto end;
+ cap = 0x33;
+ for(;;) {+ if(pread(fd, &cap, 1, cap+1) < 0) goto err;
+ if(cap == 0) goto end;
+ if(pread(fd, &c, 1, cap) < 0) goto err;
+ if(c == 0x05) break;
+ }
+ s[strlen(s) - 3] = 0;
+ print("%s\n", s+9);+ goto end;
+ err:
+ fprint(2, "read %s: %r", s);
+ end:
+ free(s);
+ close(fd);
+ }
+}
--
⑨