added io/seg utility

This commit is contained in:
aiju 2011-04-15 21:01:29 +02:00
parent 68d6b0808b
commit 2c6802e80f
4 changed files with 197 additions and 0 deletions

44
sys/man/1/io Normal file
View file

@ -0,0 +1,44 @@
.TH IO 1
.SH NAME
io \- access PC I/O registers
.SH SYNOPSIS
.B io
[
.B -WLMrw
]
.I address
[
.I value
]
.SH DESCRIPTION
.I io
accesses PC I/O space.
The operation to be performed is selected with
.B -r
or
.B -w
for reading or writing, respectively.
The default operation size is a byte.
C style notation for integers (e.g.
.B 0x42
or
.BR 023 )
is accepted for the
.I address
and
.I value
parameters.
.PP
.TP
.B -W
Perform a word (16 bit) operation.
.TP
.B -L
Perform a long / double word (32 bit) operation.
.TP
.B -M
Access a 64 bit wide machine specific register (MSR).
.SH SOURCE
.B /sys/src/cmd/io.c
.SH SEE ALSO
.IR seg (1)

51
sys/man/1/seg Normal file
View file

@ -0,0 +1,51 @@
.TH SEG 1
.SH NAME
seg \- access a named segment
.SH SYNOPSIS
.B seg
[
.B -WLrw
]
.I segment
.I segment-size
.I offset
[
.I value
]
.SH DESCRIPTION
.B seg
accesses a named segment as provided by e.g. certain drivers.
The operation to be performed is selected with
.B -r
and
.B -w
for reading and writing, respectively.
The default operation size is a byte.
C style notation for integers (e.g.
.B 0x42
or
.BR 023 )
is accepted for the
.IR segment-size ,
.IR offset
and
.I value
parameters.
.PP
.TP
.B -W
Perform a word (16 bit) operation
.TP
.B -L
Perform a long / double word (32 bit) operation.
.SH SOURCE
.B /sys/src/cmd/seg.c
.SH SEE ALSO
.IR io (1)
.SH BUGS
No check of the
.I segment-size
and
.I offset
parameters is performed whatsoever.
Odd values may cause the front to fall off.

55
sys/src/cmd/io.c Normal file
View file

@ -0,0 +1,55 @@
#include <u.h>
#include <libc.h>
const char const * datac[] = {0,"#P/iob","#P/iow",0,"#P/iol",0,0,0,"#P/msr"};
void
main(int argc, char** argv) {
int fd, size, op;
ulong port;
uvlong data;
uchar datab[8];
data = 0;
size = 1;
op = -1;
ARGBEGIN {
case 'W': size = 2; break;
case 'L': size = 4; break;
case 'M': size = 8; break;
case 'r': op = OREAD; break;
case 'w': op = OWRITE; break;
default: sysfatal("bad flag %c", ARGC());
} ARGEND;
if(op == -1) sysfatal("no operation selected");
if(argc < 1) sysfatal("no port selected");
if(op == OWRITE && argc < 2) sysfatal("no data selected");
port = strtoul(argv[0], 0, 0);
if(op == OWRITE) data = strtoull(argv[1], 0, 0);
fd = open(datac[size], op);
if(fd == -1) sysfatal("open: %r");
if(op == OWRITE) {
datab[0] = data;
datab[1] = data >> 8;
datab[2] = data >> 16;
datab[3] = data >> 24;
datab[4] = data >> 32;
datab[5] = data >> 40;
datab[6] = data >> 48;
datab[7] = data >> 56;
if(pwrite(fd, datab, size, port) != size)
sysfatal("pwrite: %r");
}
else {
memset(datab, 0, 8);
if(pread(fd, datab, size, port) != size)
sysfatal("pread: %r");
data = datab[0] | (datab[1] << 8) | (datab[2] << 16) |
(datab[3] << 24) | ((vlong)datab[4] << 32) |
((vlong)datab[5] << 40) | ((vlong)datab[6] << 48) |
((vlong)datab[7] << 56);
print("0x%llx\n", data);
}
}

47
sys/src/cmd/seg.c Normal file
View file

@ -0,0 +1,47 @@
#include <u.h>
#include <libc.h>
const char datac[] = {0,'b','w',0,'l'};
void
main(int argc, char** argv) {
ulong size, segsize, op, port;
char* segment;
ulong data;
uchar *seg;
data = 0;
size = 1;
op = -1;
ARGBEGIN {
case 'W': size = 2; break;
case 'L': size = 4; break;
case 'r': op = OREAD; break;
case 'w': op = OWRITE; break;
default: sysfatal("bad flag %c", ARGC());
} ARGEND;
if(op == -1) sysfatal("no operation selected");
if(argc < 3) sysfatal("no address, no segsize or no segment selected");
if(op == OWRITE && argc < 4) sysfatal("no data selected");
segment = argv[0];
segsize = strtoul(argv[1], 0, 0);
port = strtoul(argv[2], 0, 0);
if(op == OWRITE) data = strtoul(argv[3], 0, 0);
seg = segattach(0, segment, 0, segsize);
if(seg == (uchar*)-1) sysfatal("segattach: %r");
if(op == OWRITE) {
switch(size) {
case 1: *(uchar*)(seg+port) = data; break;
case 2: *(ushort*)(seg+port) = data; break;
case 4: *(ulong*)(seg+port) = data; break;
}
}
else {
data = seg[port];
if(size >= 2) data |= seg[port+1] << 8;
if(size >= 4) data |= (seg[port+2] << 16) | (seg[port+3] << 24);
print("0x%ulx\n", data);
}
}