#include <mdb/mdb_types.h>
#include <mdb/mdb_target.h>
#include <mdb/mdb_io.h>
#include <mdb/mdb_err.h>
#include <mdb/mdb_string.h>
#include <mdb/mdb_modapi.h>
#include <mdb/mdb.h>
#define FUNCP(p) ((void *)(p))
#define SZ_NONE ((size_t)-1L)
typedef mdb_tgt_addr_t mdb_fmt_func_f(mdb_tgt_t *,
mdb_tgt_as_t, mdb_tgt_addr_t, size_t);
#define FMT_NONE 0x0
#define FMT_FUNC 0x1
#define FMT_PRINTF 0x2
#define FMT_MATCH 0x4
#define FMT_WRITE 0x8
#define FMT_NOAUTOWRAP 0x10
#define FMT_TYPE(x) ((x) & 0x7)
typedef struct mdb_fmt_desc {
int f_type;
void *f_ptr;
const char *f_help;
size_t f_size;
boolean_t f_float;
} mdb_fmt_desc_t;
static const char help_plus[] = "increment dot by the count";
static const char help_minus[] = "decrement dot by the count";
static const char help_escchr[] = "character using C character notation";
static const char help_swapint[] = "swap bytes and shorts";
static const char help_dotinstr[] = "address and disassembled instruction";
static const char help_instr[] = "disassembled instruction";
static const char help_escstr[] = "string using C string notation";
static const char help_time32[] = "decoded time32_t";
static const char help_carat[] = "decrement dot by increment * count";
static const char help_dot[] = "dot as symbol+offset";
#ifndef _KMDB
static const char help_f[] = "float";
#endif
static const char help_swapshort[] = "swap bytes";
static const char help_nl[] = "newline";
static const char help_ws[] = "whitespace";
static const char help_rawstr[] = "raw string";
static const char help_tab[] = "horizontal tab";
static const char help_sdbyte[] = "decimal signed int";
static const char help_time64[] = "decoded time64_t";
static const char help_binary[] = "binary unsigned long long";
static const char help_hex64[] = "hexadecimal long long";
static const char help_match32[] = "int";
static const char help_match64[] = "long long";
static const char help_match16[] = "short";
static const char help_uintptr[] = "hexadecimal uintptr_t";
static const char help_ctf[] = "whose size is inferred by CTF info";
static const char help_jazzed[] = "jazzed-up binary unsigned long long";
static mdb_tgt_addr_t
fmt_dot(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint_t oflags = mdb_iob_getflags(mdb.m_out) & MDB_IOB_INDENT;
char buf[24];
mdb_iob_clrflags(mdb.m_out, oflags);
if (mdb.m_flags & MDB_FL_PSYM) {
while (cnt-- != 0)
mdb_iob_printf(mdb.m_out, "%-#16lla%16T", addr);
} else {
(void) mdb_iob_snprintf(buf, sizeof (buf),
"%#llx:", (u_longlong_t)addr);
while (cnt-- != 0)
mdb_iob_printf(mdb.m_out, "%-16s%16T", buf);
}
mdb_iob_setflags(mdb.m_out, oflags);
mdb_nv_set_value(mdb.m_rvalue, addr);
return (addr);
}
#ifndef _KMDB
static mdb_tgt_addr_t
fmt_float(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
float f;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &f, sizeof (f), addr) != sizeof (f)) {
warn("failed to read data from target");
break;
}
mdb_iob_printf(mdb.m_out, "%e", f);
addr += sizeof (f);
}
return (addr);
}
#endif
static mdb_tgt_addr_t
fmt_plus(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
return (addr + cnt);
}
static mdb_tgt_addr_t
fmt_minus(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
return (addr - cnt);
}
static mdb_tgt_addr_t
fmt_carat(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
return (addr - (mdb.m_incr * cnt));
}
static mdb_tgt_addr_t
fmt_nl(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
while (cnt-- != 0)
mdb_iob_nl(mdb.m_out);
return (addr);
}
static mdb_tgt_addr_t
fmt_ws(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
mdb_iob_ws(mdb.m_out, cnt);
return (addr);
}
static mdb_tgt_addr_t
fmt_tab(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
size_t ts = mdb_iob_gettabstop(mdb.m_out);
mdb_iob_tabstop(mdb.m_out, cnt);
mdb_iob_tab(mdb.m_out);
mdb_iob_tabstop(mdb.m_out, ts);
return (addr);
}
static mdb_tgt_addr_t
fmt_rawstr(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint_t oflags = mdb_iob_getflags(mdb.m_out) & MDB_IOB_INDENT;
char buf[BUFSIZ];
ssize_t nbytes;
mdb_iob_clrflags(mdb.m_out, oflags);
for (; cnt-- != 0; addr++) {
do {
nbytes = mdb_tgt_readstr(t, as, buf, BUFSIZ, addr);
if (nbytes > 0) {
mdb_iob_puts(mdb.m_out, buf);
addr += MIN(nbytes, BUFSIZ - 1);
} else if (nbytes < 0) {
warn("failed to read data from target");
goto out;
}
} while (nbytes == BUFSIZ);
if (cnt != 0)
mdb_iob_puts(mdb.m_out, "\\0");
}
out:
mdb_iob_setflags(mdb.m_out, oflags);
return (addr);
}
static mdb_tgt_addr_t
fmt_escstr(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint_t oflags = mdb_iob_getflags(mdb.m_out) & MDB_IOB_INDENT;
char buf[BUFSIZ];
ssize_t nbytes;
char *s;
mdb_iob_clrflags(mdb.m_out, oflags);
for (; cnt-- != 0; addr++) {
do {
nbytes = mdb_tgt_readstr(t, as, buf, BUFSIZ, addr);
if (nbytes > 0) {
s = strchr2esc(buf, strlen(buf));
mdb_iob_puts(mdb.m_out, s);
strfree(s);
addr += MIN(nbytes, BUFSIZ - 1);
} else if (nbytes < 0) {
warn("failed to read data from target");
goto out;
}
} while (nbytes == BUFSIZ);
if (cnt != 0)
mdb_iob_puts(mdb.m_out, "\\0");
}
out:
mdb_iob_setflags(mdb.m_out, oflags);
return (addr);
}
static mdb_tgt_addr_t
fmt_escchr(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
char *(*convert)(const char *, size_t);
ssize_t nbytes;
char *buf, *s;
if (mdb.m_flags & MDB_FL_ADB)
convert = &strchr2adb;
else
convert = &strchr2esc;
buf = mdb_alloc(cnt + 1, UM_SLEEP);
buf[cnt] = 0;
if ((nbytes = mdb_tgt_aread(t, as, buf, cnt, addr)) > 0) {
s = convert(buf, nbytes);
mdb_iob_puts(mdb.m_out, s);
strfree(s);
addr += nbytes;
}
mdb_free(buf, cnt + 1);
return (addr);
}
static mdb_tgt_addr_t
fmt_swapshort(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
ushort_t x;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
x = (x << 8) | (x >> 8);
mdb_iob_printf(mdb.m_out, "%-8x", x);
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_swapint(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint_t x;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
x = ((x << 24) | ((x << 8) & 0xff0000) |
((x >> 8) & 0xff00) | ((x >> 24) & 0xff));
mdb_iob_printf(mdb.m_out, "%-16x", x);
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_time32(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
int32_t x;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
mdb_iob_printf(mdb.m_out, "%-24Y", (time_t)x);
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_time64(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
int64_t x;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
if ((time_t)x == x)
mdb_iob_printf(mdb.m_out, "%-24Y", (time_t)x);
else
mdb_iob_printf(mdb.m_out, "%-24llR", x);
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_sdbyte(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
int8_t x;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
mdb_iob_printf(mdb.m_out, "%-8d", (int32_t)x);
mdb_nv_set_value(mdb.m_rvalue, (uint8_t)x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_instr(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
char buf[BUFSIZ];
uintptr_t naddr;
if (as == MDB_TGT_AS_VIRT)
as = MDB_TGT_AS_VIRT_I;
while (cnt-- != 0) {
naddr = mdb_dis_ins2str(mdb.m_disasm, t, as,
buf, sizeof (buf), addr);
if (naddr == addr)
return (addr);
mdb_iob_printf(mdb.m_out, "%s\n", buf);
addr = naddr;
}
return (addr);
}
static mdb_tgt_addr_t
fmt_dotinstr(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint_t oflags = mdb_iob_getflags(mdb.m_out) & MDB_IOB_INDENT;
char buf[BUFSIZ];
uintptr_t naddr;
uint32_t i;
if (as == MDB_TGT_AS_VIRT)
as = MDB_TGT_AS_VIRT_I;
for (mdb_iob_clrflags(mdb.m_out, oflags); cnt-- != 0; addr = naddr) {
if (mdb_tgt_aread(t, as, &i, sizeof (i), addr) != sizeof (i)) {
warn("failed to read data from target");
break;
}
naddr = mdb_dis_ins2str(mdb.m_disasm, t, as,
buf, sizeof (buf), addr);
if (naddr == addr)
break;
mdb_iob_printf(mdb.m_out, "%lx %x: %s\n", (long)addr, i, buf);
}
mdb_iob_setflags(mdb.m_out, oflags);
return (addr);
}
static mdb_tgt_addr_t
fmt_binary(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint64_t x;
const char *fmts[] = { "%-64s", "%-65s" };
const uint64_t mask = 0x8000000000000000ull;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
mdb_iob_printf(mdb.m_out, fmts[(x & mask) != 0],
numtostr(x, 2, NTOS_UNSIGNED));
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_jazzed(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
uint64_t x;
char buf[256];
while (cnt-- != 0) {
boolean_t header = B_TRUE;
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) != sizeof (x)) {
warn("failed to read data from target");
break;
}
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
mdb_iob_printf(mdb.m_out, "%s\n",
numtostr(x, 2, NTOS_UNSIGNED));
while (x != 0) {
int b = 63, forearm;
int i = 0, highbit;
while (!(x & (1ULL << b)))
b--;
highbit = b;
while (x & ((1ULL << b) - 1)) {
buf[i++] = x & (1ULL << b) ? '|' : ' ';
b--;
}
if (header) {
header = B_FALSE;
buf[i] = '\0';
mdb_iob_printf(mdb.m_out, "%s|\n", buf);
continue;
}
buf[i++] = '+';
for (forearm = b; forearm > -2; forearm--)
buf[i++] = '-';
buf[i] = '\0';
mdb_iob_printf(mdb.m_out, "%s bit %d %smask 0x%0*llx\n",
buf, b, b < 10 && highbit >= 10 ? " " : "",
(highbit / 4) + 1, 1ULL << b);
x &= ~(1ULL << b);
}
}
return (addr);
}
static mdb_tgt_addr_t
fmt_hex64(mdb_tgt_t *t, mdb_tgt_as_t as, mdb_tgt_addr_t addr, size_t cnt)
{
const char *fmts[] = { "%-16llx", "%-17llx" };
const uint64_t mask = 0xf000000000000000ull;
uint64_t x;
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, &x, sizeof (x), addr) == sizeof (x)) {
mdb_iob_printf(mdb.m_out, fmts[(x & mask) != 0], x);
mdb_nv_set_value(mdb.m_rvalue, x);
addr += sizeof (x);
} else {
warn("failed to read data from target");
break;
}
}
return (addr);
}
static const mdb_fmt_desc_t fmttab[] = {
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_FUNC, FUNCP(fmt_plus), help_plus, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_FUNC, FUNCP(fmt_minus), help_minus, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_PRINTF, "%-8x", NULL, 1 },
{ FMT_FUNC, FUNCP(fmt_escchr), help_escchr, 1 },
{ FMT_PRINTF, "%-16d", NULL, 4 },
{ FMT_PRINTF, "%-21llu", NULL, 8 },
#ifdef _KMDB
{ FMT_NONE, NULL, NULL, 0 },
#else
{ FMT_PRINTF, "%g", NULL, sizeof (double), B_TRUE },
#endif
{ FMT_PRINTF, "%-23llo", NULL, 8 },
{ FMT_FUNC, FUNCP(fmt_swapint), help_swapint, 4 },
{ FMT_FUNC, FUNCP(fmt_dotinstr), help_dotinstr, 0 },
{ FMT_FUNC, FUNCP(fmt_hex64), help_hex64, 8 },
#ifdef _LP64
{ FMT_FUNC, FUNCP(fmt_hex64), help_uintptr, 8 },
#else
{ FMT_PRINTF, "%-16x", help_uintptr, 4 },
#endif
{ FMT_MATCH, NULL, help_match32, 4 },
{ FMT_MATCH, NULL, help_match64, 8 },
{ FMT_FUNC, FUNCP(fmt_nl), help_nl, SZ_NONE },
{ FMT_PRINTF, "%-#16o", NULL, 4 },
{ FMT_PRINTF, "%-19a", NULL, sizeof (uintptr_t) },
{ FMT_PRINTF, "%-#16q", NULL, 4 },
{ FMT_FUNC, FUNCP(fmt_binary), help_binary, 8 },
{ FMT_FUNC, FUNCP(fmt_escstr), help_escstr, 0 },
{ FMT_FUNC, FUNCP(fmt_tab), help_tab, SZ_NONE },
{ FMT_PRINTF, "%-16u", NULL, 4 },
{ FMT_PRINTF, "%-8u", NULL, 1 },
{ FMT_PRINTF|FMT_WRITE, "%-16r", NULL, 4 },
{ FMT_PRINTF, "%-16x", NULL, 4 },
{ FMT_FUNC, FUNCP(fmt_time32), help_time32, 4 },
{ FMT_FUNC|FMT_WRITE, FUNCP(fmt_hex64), help_hex64, 8 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_FUNC, FUNCP(fmt_carat), help_carat, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_FUNC, FUNCP(fmt_dot), help_dot, SZ_NONE },
{ FMT_PRINTF, "%-#8o", NULL, 1 },
{ FMT_PRINTF, "%c", NULL, 1 },
{ FMT_PRINTF, "%-8hd", NULL, 2 },
{ FMT_PRINTF, "%-21lld", NULL, 8 },
#ifdef _KMDB
{ FMT_NONE, NULL, NULL, 0 },
#else
{ FMT_FUNC, FUNCP(fmt_float), help_f, sizeof (float),
B_TRUE },
#endif
{ FMT_PRINTF, "%-24llq", NULL, 8 },
{ FMT_FUNC, FUNCP(fmt_swapshort), help_swapshort, 2 },
{ FMT_FUNC, FUNCP(fmt_instr), help_instr, 0 },
{ FMT_FUNC|FMT_NOAUTOWRAP,
FUNCP(fmt_jazzed), help_jazzed, 8 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_MATCH, NULL, help_match16, 2 },
{ FMT_NONE, NULL, NULL, 0 },
{ FMT_FUNC, FUNCP(fmt_nl), help_nl, SZ_NONE },
{ FMT_PRINTF, "%-#8ho", NULL, 2 },
{ FMT_PRINTF, "%-19a", NULL, sizeof (uintptr_t) },
{ FMT_PRINTF, "%-#8hq", NULL, 2 },
{ FMT_FUNC, FUNCP(fmt_ws), help_ws, SZ_NONE },
{ FMT_FUNC, FUNCP(fmt_rawstr), help_rawstr, 0 },
{ FMT_FUNC, FUNCP(fmt_tab), help_tab, SZ_NONE },
{ FMT_PRINTF, "%-8hu", NULL, 2 },
{ FMT_FUNC|FMT_WRITE, FUNCP(fmt_sdbyte), help_sdbyte, 1 },
{ FMT_PRINTF|FMT_WRITE, "%-8hr", NULL, 2 },
{ FMT_PRINTF, "%-8hx", NULL, 2 },
{ FMT_FUNC, FUNCP(fmt_time64), help_time64, 8 },
{ FMT_WRITE, NULL, help_ctf, 0 },
};
mdb_tgt_addr_t
mdb_fmt_print(mdb_tgt_t *t, mdb_tgt_as_t as,
mdb_tgt_addr_t addr, size_t cnt, char fmt)
{
const mdb_fmt_desc_t *fp = &fmttab[fmt];
mdb_fmt_func_f *funcp;
uintmax_t rvalue;
void *buf;
uint_t oflags = mdb.m_flags;
union {
uint64_t i8;
uint32_t i4;
uint16_t i2;
uint8_t i1;
double d;
} u;
if (fmt < 0 || fmt > (sizeof (fmttab) / sizeof (fmttab[0]))) {
warn("invalid format character -- '%c'\n", fmt);
return (addr);
}
if (!(fp->f_type & FMT_NOAUTOWRAP)) {
mdb_iob_set_autowrap(mdb.m_out);
}
switch (FMT_TYPE(fp->f_type)) {
case FMT_FUNC:
funcp = (mdb_fmt_func_f *)fp->f_ptr;
addr = funcp(t, as, addr, cnt);
break;
case FMT_PRINTF:
switch (fp->f_size) {
case 1:
buf = &u.i1;
break;
case 2:
buf = &u.i2;
break;
case 4:
buf = &u.i4;
break;
case 8:
buf = &u.i8;
break;
default:
fail("format %c is defined using illegal size\n", fmt);
}
if (fp->f_float == B_TRUE) {
if (fp->f_size != 8) {
fail("format %c is using illegal fp size\n",
fmt);
}
buf = &u.d;
}
while (cnt-- != 0) {
if (mdb_tgt_aread(t, as, buf, fp->f_size, addr) !=
fp->f_size) {
warn("failed to read data from target");
return (addr);
}
switch (fp->f_size) {
case 1:
mdb_iob_printf(mdb.m_out, fp->f_ptr, u.i1);
rvalue = u.i1;
break;
case 2:
mdb_iob_printf(mdb.m_out, fp->f_ptr, u.i2);
rvalue = u.i2;
break;
case 4:
mdb_iob_printf(mdb.m_out, fp->f_ptr, u.i4);
rvalue = u.i4;
break;
case 8:
if (fp->f_float) {
mdb_iob_printf(mdb.m_out, fp->f_ptr,
u.d);
} else {
mdb_iob_printf(mdb.m_out, fp->f_ptr,
u.i8);
}
rvalue = u.i8;
break;
default:
rvalue = 0;
break;
}
mdb_nv_set_value(mdb.m_rvalue, rvalue);
addr += fp->f_size;
}
break;
default:
warn("invalid format character -- '%c'\n", fmt);
}
mdb.m_flags = oflags;
return (addr);
}
int
cmd_formats(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
const mdb_fmt_desc_t *fp = &fmttab[0];
int i;
const char *write;
if ((flags & DCMD_ADDRSPEC) || argc != 0)
return (DCMD_USAGE);
for (i = 0; i < (sizeof (fmttab) / sizeof (fmttab[0])); i++, fp++) {
if (fp->f_type == FMT_NONE)
continue;
write = (fp->f_type & FMT_WRITE) ? "write " : "";
if (fp->f_type & FMT_FUNC)
mdb_printf("%c - %s%s", i, write, fp->f_help);
else if (fp->f_type & FMT_MATCH)
mdb_printf("%c - match %s", i, fp->f_help);
else if (fp->f_help != NULL)
mdb_printf("%c - %s%s", i, write, fp->f_help);
else
mdb_printf("%c - %s%s", i, write,
mdb_iob_format2str(fp->f_ptr));
switch (fp->f_size) {
case SZ_NONE:
mdb_printf("\n");
break;
case 0:
mdb_printf(" (variable size)\n");
break;
case 1:
mdb_printf(" (1 byte)\n");
break;
default:
mdb_printf(" (%lu bytes)\n", fp->f_size);
}
}
return (DCMD_OK);
}