]> git.ipfire.org Git - thirdparty/libbsd.git/commitdiff
Add strnvis and strnunvis functions
authorGuillem Jover <guillem@hadrons.org>
Fri, 23 Oct 2009 22:15:57 +0000 (00:15 +0200)
committerGuillem Jover <guillem@hadrons.org>
Fri, 23 Oct 2009 23:47:26 +0000 (01:47 +0200)
Taken from OpenBSD.

Versions
include/vis.h
src/unvis.c
src/vis.c

index 271839d1042a02540071f1c2e2f13ae66af8b9e6..3e99cf4cb14a2ea20d6cdbaab876336659fa514c 100644 (file)
--- a/Versions
+++ b/Versions
@@ -44,5 +44,8 @@ LIBBSD_0.1 {
 
 LIBBSD_0.2 {
     strtonum;
+
+    strnvis;
+    strnunvis;
 } LIBBSD_0.1;
 
index 84de6fc1131420a35600763ac86d4c41cd801473..835d2d6000a2fb0e7b409431b1549eb511182445 100644 (file)
@@ -78,8 +78,10 @@ __BEGIN_DECLS
 char   *vis(char *, int, int, int);
 int    strvis(char *, const char *, int);
 int    strvisx(char *, const char *, size_t, int);
+int    strnvis(char *, const char *, size_t, int);
 int    strunvis(char *, const char *);
 int    strunvisx(char *, const char *, int);
+ssize_t strnunvis(char *, const char *, size_t);
 int    unvis(char *, int, int *, int);
 __END_DECLS
 
index 66d74a5cf410804c39c817bb9bd2f0f79d79277a..188edca50588d83e0e2b414a4dee1839811a197a 100644 (file)
@@ -257,6 +257,47 @@ strunvis(char *dst, const char *src)
        return (dst - start);
 }
 
+ssize_t
+strnunvis(char *dst, const char *src, size_t sz)
+{
+       char c, p;
+       char *start = dst, *end = dst + sz - 1;
+       int state = 0;
+
+       if (sz > 0)
+               *end = '\0';
+       while ((c = *src++)) {
+       again:
+               switch (unvis(&p, c, &state, 0)) {
+               case UNVIS_VALID:
+                       if (dst < end)
+                               *dst = p;
+                       dst++;
+                       break;
+               case UNVIS_VALIDPUSH:
+                       if (dst < end)
+                               *dst = p;
+                       dst++;
+                       goto again;
+               case 0:
+               case UNVIS_NOCHAR:
+                       break;
+               default:
+                       if (dst <= end)
+                               *dst = '\0';
+                       return (-1);
+               }
+       }
+       if (unvis(&p, c, &state, UNVIS_END) == UNVIS_VALID) {
+               if (dst < end)
+                       *dst = p;
+               dst++;
+       }
+       if (dst <= end)
+               *dst = '\0';
+       return (dst - start);
+}
+
 int
 strunvisx(char *dst, const char *src, int flag)
 {
index 4ad31d5da649d7e265024e4dc5410341ba103d6d..189fde8f3cda5c1b0bf0c1e73e034f9f91d2417c 100644 (file)
--- a/src/vis.c
+++ b/src/vis.c
@@ -1,3 +1,4 @@
+/*     $OpenBSD: vis.c,v 1.18 2005/08/29 18:38:41 otto Exp $ */
 /*-
  * Copyright (c) 1989, 1993
  *     The Regents of the University of California.  All rights reserved.
 #include <sys/types.h>
 #include <limits.h>
 #include <ctype.h>
-#include <stdio.h>
+#include <string.h>
 #include <vis.h>
 
 #define        isoctal(c)      (((u_char)(c)) >= '0' && ((u_char)(c)) <= '7')
+#define        isvisible(c)                                                    \
+       (((u_int)(c) <= UCHAR_MAX && isascii((u_char)(c)) &&            \
+       (((c) != '*' && (c) != '?' && (c) != '[' && (c) != '#') ||      \
+               (flag & VIS_GLOB) == 0) && isgraph((u_char)(c))) ||     \
+       ((flag & VIS_SP) == 0 && (c) == ' ') ||                         \
+       ((flag & VIS_TAB) == 0 && (c) == '\t') ||                       \
+       ((flag & VIS_NL) == 0 && (c) == '\n') ||                        \
+       ((flag & VIS_SAFE) && ((c) == '\b' ||                           \
+               (c) == '\007' || (c) == '\r' ||                         \
+               isgraph((u_char)(c)))))
 
 /*
  * vis - visually encode characters
@@ -149,12 +160,15 @@ done:
 }
 
 /*
- * strvis, strvisx - visually encode characters from src into dst
+ * strvis, strnvis, strvisx - visually encode characters from src into dst
  *
  *     Dst must be 4 times the size of src to account for possible
  *     expansion.  The length of dst, not including the trailing NUL,
  *     is returned.
  *
+ *     Strnvis will write no more than siz-1 bytes (and will NULL terminate).
+ *     The number of bytes needed to fully encode the string is returned.
+ *
  *     Strvisx encodes exactly len bytes from src into dst.
  *     This is useful for encoding a block of data.
  */
@@ -174,6 +188,49 @@ strvis(dst, src, flag)
 }
 
 int
+strnvis(char *dst, const char *src, size_t siz, int flag)
+{
+       char *start, *end;
+       char tbuf[5];
+       int c, i;
+
+       i = 0;
+       for (start = dst, end = start + siz - 1; (c = *src) && dst < end; ) {
+               if (isvisible(c)) {
+                       i = 1;
+                       *dst++ = c;
+                       if (c == '\\' && (flag & VIS_NOSLASH) == 0) {
+                               /* need space for the extra '\\' */
+                               if (dst < end)
+                                       *dst++ = '\\';
+                               else {
+                                       dst--;
+                                       i = 2;
+                                       break;
+                               }
+                       }
+                       src++;
+               } else {
+                       i = vis(tbuf, c, flag, *++src) - tbuf;
+                       if (dst + i <= end) {
+                               memcpy(dst, tbuf, i);
+                               dst += i;
+                       } else {
+                               src--;
+                               break;
+                       }
+               }
+       }
+       if (siz > 0)
+               *dst = '\0';
+       if (dst + i > end) {
+               /* adjust return value for truncation */
+               while ((c = *src))
+                       dst += vis(tbuf, c, flag, *++src) - tbuf;
+       }
+       return (dst - start);
+}
+
 strvisx(dst, src, len, flag)
        char *dst;
        const char *src;