summaryrefslogtreecommitdiff
path: root/libexec/ld.so/dl_printf.c
diff options
context:
space:
mode:
authorDale S. Rahn <rahnds@cvs.openbsd.org>2000-06-13 03:34:18 +0000
committerDale S. Rahn <rahnds@cvs.openbsd.org>2000-06-13 03:34:18 +0000
commitfe1bdbab88774dd0b581b085d5bd355f2b2ab86b (patch)
tree7fec591016d5f0ccde0ed562abed828e9c5c0a91 /libexec/ld.so/dl_printf.c
parent6e274593757db3e32991b92a21cdd60081006e3d (diff)
Initial import of ELF ld.so. This was written indepently of the gnu rtld
code and is fully BSD copyrighted. This initial import contains a mostly working mips and partially working powerpc version. No support for ld.so.hints or system run path exists.
Diffstat (limited to 'libexec/ld.so/dl_printf.c')
-rw-r--r--libexec/ld.so/dl_printf.c253
1 files changed, 253 insertions, 0 deletions
diff --git a/libexec/ld.so/dl_printf.c b/libexec/ld.so/dl_printf.c
new file mode 100644
index 00000000000..c39c81dbcb5
--- /dev/null
+++ b/libexec/ld.so/dl_printf.c
@@ -0,0 +1,253 @@
+/* $OpenBSD: dl_printf.c,v 1.1 2000/06/13 03:33:55 rahnds Exp $ */
+
+/*-
+ * Copyright (c) 1993
+ * The Regents of the University of California. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Berkeley and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * @(#)printf.c 8.1 (Berkeley) 6/11/93
+ */
+
+/*
+ * Scaled down version of printf(3).
+ *
+ * One additional format:
+ *
+ * The format %b is supported to decode error registers.
+ * Its usage is:
+ *
+ * printf("reg=%b\n", regval, "<base><arg>*");
+ *
+ * where <base> is the output base expressed as a control character, e.g.
+ * \10 gives octal; \20 gives hex. Each arg is a sequence of characters,
+ * the first of which gives the bit number to be inspected (origin 1), and
+ * the next characters (up to a control character, i.e. a character <= 32),
+ * give the name of the register. Thus:
+ *
+ * printf("reg=%b\n", 3, "\10\2BITTWO\1BITONE\n");
+ *
+ * would produce output:
+ *
+ * reg=3<BITTWO,BITONE>
+ */
+
+#include <sys/cdefs.h>
+#include <sys/types.h>
+#ifdef __STDC__
+#include <stdarg.h>
+#else
+#include <varargs.h>
+#endif
+#include "syscall.h"
+
+static void kprintn __P((void (*)(int), u_long, int));
+static void kdoprnt __P((void (*)(int), const char *, va_list));
+
+static void putchar __P((int));
+static void sputchar __P((int));
+static char *sbuf;
+
+static void
+putchar(c)
+ int c;
+{
+ char b;
+ b = c;
+ _dl_write(2, &b, 1);
+}
+
+static void
+sputchar(c)
+ int c;
+{
+ *sbuf++ = c;
+}
+
+void
+#ifdef __STDC__
+_dl_sprintf(char *buf, const char *fmt, ...)
+#else
+_dl_sprintf(buf, fmt, va_alist)
+ char *buf, *fmt;
+#endif
+{
+ va_list ap;
+
+ sbuf = buf;
+#ifdef __STDC__
+ va_start(ap, fmt);
+#else
+ va_start(ap);
+#endif
+ kdoprnt(sputchar, fmt, ap);
+ va_end(ap);
+ *sbuf = '\0';
+}
+
+void
+#ifdef __STDC__
+_dl_printf(const char *fmt, ...)
+#else
+_dl_printf(fmt, va_alist)
+ char *fmt;
+#endif
+{
+ va_list ap;
+
+#ifdef __STDC__
+ va_start(ap, fmt);
+#else
+ va_start(ap);
+#endif
+ kdoprnt(putchar, fmt, ap);
+ va_end(ap);
+}
+
+void
+_dl_vprintf(const char *fmt, va_list ap)
+{
+ kdoprnt(putchar, fmt, ap);
+}
+
+static void
+kdoprnt(put, fmt, ap)
+ void (*put)__P((int));
+ const char *fmt;
+ va_list ap;
+{
+ char *p;
+ int ch;
+ unsigned long ul;
+ int lflag;
+
+ for (;;) {
+ while ((ch = *fmt++) != '%') {
+ if (ch == '\0')
+ return;
+ put(ch);
+ }
+ lflag = 0;
+reswitch: switch (ch = *fmt++) {
+ case 'l':
+ lflag = 1;
+ goto reswitch;
+ case 'b':
+ {
+ int set, n;
+ ul = va_arg(ap, int);
+ p = va_arg(ap, char *);
+ kprintn(put, ul, *p++);
+
+ if (!ul)
+ break;
+
+ for (set = 0; (n = *p++);) {
+ if (ul & (1 << (n - 1))) {
+ put(set ? ',' : '<');
+ for (; (n = *p) > ' '; ++p)
+ put(n);
+ set = 1;
+ } else
+ for (; *p > ' '; ++p);
+ }
+ if (set)
+ put('>');
+ }
+ break;
+ case 'c':
+ ch = va_arg(ap, int);
+ put(ch & 0x7f);
+ break;
+ case 's':
+ p = va_arg(ap, char *);
+ while ((ch = *p++))
+ put(ch);
+ break;
+ case 'd':
+ ul = lflag ? va_arg(ap, long) : va_arg(ap, int);
+ if ((long)ul < 0) {
+ put('-');
+ ul = -(long)ul;
+ }
+ kprintn(put, ul, 10);
+ break;
+ case 'o':
+ ul = lflag ? va_arg(ap, u_long) : va_arg(ap, u_int);
+ kprintn(put, ul, 8);
+ break;
+ case 'u':
+ ul = lflag ? va_arg(ap, u_long) : va_arg(ap, u_int);
+ kprintn(put, ul, 10);
+ break;
+ case 'p':
+ put('0');
+ put('x');
+ lflag += sizeof(void *)==sizeof(u_long)? 1 : 0;
+ case 'x':
+ ul = lflag ? va_arg(ap, u_long) : va_arg(ap, u_int);
+ kprintn(put, ul, 16);
+ break;
+ case 'X':
+ {
+ int l = 28;
+ ul = lflag ? va_arg(ap, u_long) : va_arg(ap, u_int);
+ while(l >= 0) {
+ put("0123456789abcdef"[(ul >> l) & 0xf]);
+ l -= 4;
+ }
+ break;
+ }
+ default:
+ put('%');
+ if (lflag)
+ put('l');
+ put(ch);
+ }
+ }
+ va_end(ap);
+}
+
+static void
+kprintn(put, ul, base)
+ void (*put)__P((int));
+ unsigned long ul;
+ int base;
+{
+ /* hold a long in base 8 */
+ char *p, buf[(sizeof(long) * NBBY / 3) + 1];
+
+ p = buf;
+ do {
+ *p++ = "0123456789abcdef"[ul % base];
+ } while (ul /= base);
+ do {
+ put(*--p);
+ } while (p > buf);
+}