Radek Vokál 13963e
--- bsd-finger-0.17/finger/finger.c.widechar	2005-12-15 09:14:18.000000000 +0100
Radek Vokál 13963e
+++ bsd-finger-0.17/finger/finger.c	2005-12-15 09:14:18.000000000 +0100
Radek Vokál 13963e
@@ -77,6 +77,7 @@
Radek Vokál 13963e
 #include <getopt.h>
Radek Vokál 13963e
 #include <signal.h>
Radek Vokál 13963e
 #include <errno.h>
Radek Vokál 13963e
+#include <locale.h>
Radek Vokál 13963e
 #include "finger.h"
Radek Vokál 13963e
 #ifdef _USAGI
Radek Vokál 13963e
 #include "version.h"
Radek Vokál 13963e
@@ -286,6 +287,9 @@
Radek Vokál 13963e
 	struct utmp *uptr;
Radek Vokál 13963e
 	int dolocal, *used;
Radek Vokál 13963e
 
Radek Vokál 13963e
+	if (setlocale (LC_ALL, "") != NULL)
Radek Vokál 13963e
+		set_haslocale();
Radek Vokál 13963e
+
Radek Vokál 13963e
 	used = calloc(argc, sizeof(int));
Radek Vokál 13963e
 	if (!used) {
Radek Vokál 13963e
 		eprintf("finger: out of space.\n");
Radek Vokál 13963e
--- bsd-finger-0.17/finger/finger.h.widechar	2005-12-15 09:14:17.000000000 +0100
Radek Vokál 13963e
+++ bsd-finger-0.17/finger/finger.h	2005-12-15 09:14:18.000000000 +0100
Radek Vokál 13963e
@@ -117,3 +117,7 @@
Radek Vokál 13963e
 /* terminal inquiries */
Radek Vokál 13963e
 int is8bit(void);
Radek Vokál 13963e
 int getscreenwidth(void);
Radek Vokál 13963e
+
Radek Vokál 13963e
+/* locale support */
Radek Vokál 13963e
+void set_haslocale(void);
Radek Vokál 13963e
+
Radek Vokál 13963e
--- bsd-finger-0.17/finger/display.c.widechar	1999-09-29 00:53:58.000000000 +0200
Radek Vokál 13963e
+++ bsd-finger-0.17/finger/display.c	2005-12-15 10:05:40.000000000 +0100
Radek Vokál 13963e
@@ -40,8 +40,19 @@
Radek Vokál 13963e
 #include <stdlib.h>
Radek Vokál 13963e
 #include <string.h>
Radek Vokál 13963e
 #include <stdarg.h>
Radek Vokál 13963e
+#include <inttypes.h>
Radek Vokál 13963e
 #include "finger.h"
Radek Vokál 13963e
 
Radek Vokál 13963e
+#define HAVE_WCHAR_H 1
Radek Vokál 13963e
+#define HAVE_MBRTOWC 1
Radek Vokál 13963e
+#define HAVE_WCWIDTH 1
Radek Vokál 13963e
+
Radek Vokál 13963e
+#if defined(HAVE_WCHAR_H) && defined(HAVE_MBRTOWC) && defined(HAVE_WCWIDTH)
Radek Vokál 13963e
+#include <wchar.h>
Radek Vokál 13963e
+#include <wctype.h>
Radek Vokál 13963e
+#include <assert.h>
Radek Vokál 13963e
+#endif
Radek Vokál 13963e
+
Radek Vokál 13963e
 int
Radek Vokál 13963e
 getscreenwidth(void)
Radek Vokál 13963e
 {
Radek Vokál 13963e
@@ -147,9 +158,105 @@
Radek Vokál 13963e
 	fxputc(stdout, ch);
Radek Vokál 13963e
 }
Radek Vokál 13963e
 
Radek Vokál 13963e
+static int has_locale = 0;
Radek Vokál 13963e
+
Radek Vokál 13963e
+void
Radek Vokál 13963e
+set_haslocale (void)
Radek Vokál 13963e
+{
Radek Vokál 13963e
+	has_locale = 1;
Radek Vokál 13963e
+}
Radek Vokál 13963e
+
Radek Vokál 13963e
+#if defined(HAVE_WCHAR_H) && defined(HAVE_MBRTOWC) && defined(HAVE_WCWIDTH)
Radek Vokál 13963e
+static int verifymultibyte(const char *buf) {
Radek Vokál 13963e
+	mbstate_t state;
Radek Vokál 13963e
+	wchar_t nextchar;
Radek Vokál 13963e
+	size_t bytesconsumed;
Radek Vokál 13963e
+	char *eop, *op;
Radek Vokál 13963e
+	(void)memset(&state, 0, sizeof(mbstate_t));
Radek Vokál 13963e
+
Radek Vokál 13963e
+	eop = (char *) (buf + strlen(buf));
Radek Vokál 13963e
+	op = (char *) buf;
Radek Vokál 13963e
+	while (op < eop) {
Radek Vokál 13963e
+		bytesconsumed = mbrtowc(&nextchar, op, eop - op, &state);
Radek Vokál 13963e
+		if (bytesconsumed == (size_t)(-1) ||
Radek Vokál 13963e
+		    bytesconsumed == (size_t)(-2)) {
Radek Vokál 13963e
+			return 0;
Radek Vokál 13963e
+		}
Radek Vokál 13963e
+		op += bytesconsumed;
Radek Vokál 13963e
+	}
Radek Vokál 13963e
+
Radek Vokál 13963e
+	return 1;
Radek Vokál 13963e
+}
Radek Vokál 13963e
+
Radek Vokál 13963e
+#define OCTALIFY(n, o) \
Radek Vokál 13963e
+	*(n)++ = '\\', \
Radek Vokál 13963e
+	*(n)++ = (((uint32_t)*(o) >> 6) & 3) + '0', \
Radek Vokál 13963e
+	*(n)++ = (((uint32_t)*(o) >> 3) & 7) + '0', \
Radek Vokál 13963e
+	*(n)++ = (((uint32_t)*(o) >> 0) & 7) + '0', \
Radek Vokál 13963e
+	(o)++
Radek Vokál 13963e
+
Radek Vokál 13963e
+#endif
Radek Vokál 13963e
+
Radek Vokál 13963e
 static void fxputs(FILE *f, const char *buf) {
Radek Vokál 13963e
-	int i;
Radek Vokál 13963e
-	for (i=0; buf[i]; i++) fxputc(f, buf[i]);
Radek Vokál 13963e
+	int widechars;
Radek Vokál 13963e
+
Radek Vokál 13963e
+#if defined(HAVE_WCHAR_H) && defined(HAVE_MBRTOWC) && defined(HAVE_WCWIDTH)
Radek Vokál 13963e
+	if (has_locale)
Radek Vokál 13963e
+		widechars = verifymultibyte (buf);
Radek Vokál 13963e
+	else
Radek Vokál 13963e
+		widechars = 0;
Radek Vokál 13963e
+#else
Radek Vokál 13963e
+	widechars = 0;
Radek Vokál 13963e
+#endif
Radek Vokál 13963e
+
Radek Vokál 13963e
+	/* on 7-bit terminals, without wide-chars support, or string
Radek Vokál 13963e
+	 * isn't parseable, print char * by char */
Radek Vokál 13963e
+	if (!is8bit() || !widechars) {
Radek Vokál 13963e
+		unsigned int i;
Radek Vokál 13963e
+		char ch;
Radek Vokál 13963e
+		for (i = 0; i < strlen (buf); i++) {
Radek Vokál 13963e
+			ch = buf[i];
Radek Vokál 13963e
+			fxputc(f, ch);
Radek Vokál 13963e
+		}
Radek Vokál 13963e
+		return;
Radek Vokál 13963e
+	}
Radek Vokál 13963e
+
Radek Vokál 13963e
+#if defined(HAVE_WCHAR_H) && defined(HAVE_MBRTOWC) && defined(HAVE_WCWIDTH)
Radek Vokál 13963e
+	{
Radek Vokál 13963e
+		mbstate_t state;
Radek Vokál 13963e
+		wchar_t nextchar;
Radek Vokál 13963e
+		size_t bytesconsumed;
Radek Vokál 13963e
+		char *eop, *op, buffer[256];
Radek Vokál 13963e
+		(void)memset(&state, 0, sizeof(mbstate_t));
Radek Vokál 13963e
+		char* op1;
Radek Vokál 13963e
+		eop = (char *) (buf + strlen(buf));
Radek Vokál 13963e
+		op = (char *) buf;
Radek Vokál 13963e
+		op1 = op;
Radek Vokál 13963e
+		while (op < eop) {
Radek Vokál 13963e
+			bytesconsumed = mbrtowc(&nextchar, op,
Radek Vokál 13963e
+					eop - op, &state);
Radek Vokál 13963e
+			/* This isn't supposed to happen as we verified the
Radek Vokál 13963e
+			 * string before hand */
Radek Vokál 13963e
+			assert(bytesconsumed != (size_t)(-1) && bytesconsumed != (size_t)(-2));
Radek Vokál 13963e
+
Radek Vokál 13963e
+			if (iswprint(nextchar)) {
Radek Vokál 13963e
+				(void)memcpy(buffer, op, bytesconsumed);
Radek Vokál 13963e
+				buffer[bytesconsumed] = '\0';
Radek Vokál 13963e
+				op += bytesconsumed;
Radek Vokál 13963e
+			} else if (bytesconsumed == 1) {
Radek Vokál 13963e
+				op++;
Radek Vokál 13963e
+			} else {
Radek Vokál 13963e
+				char *tmp;
Radek Vokál 13963e
+				tmp = buffer;
Radek Vokál 13963e
+				buffer[bytesconsumed] = '\0';
Radek Vokál 13963e
+				while (bytesconsumed-- > 0) {
Radek Vokál 13963e
+					OCTALIFY(tmp, op);
Radek Vokál 13963e
+				}
Radek Vokál 13963e
+			}
Radek Vokál 13963e
+		}
Radek Vokál 13963e
+		fprintf(f,"%s",op1);
Radek Vokál 13963e
+	}
Radek Vokál 13963e
+#endif
Radek Vokál 13963e
 }
Radek Vokál 13963e
 
Radek Vokál 13963e
 int xprintf(const char *fmt, ...) {