mirror of
				https://github.com/vbatts/tar-split.git
				synced 2025-10-26 16:30:57 +00:00 
			
		
		
		
	vis/unvis: pull in exact implementation from FreeBSD
Perhaps this is not completely ideal, because it brings in cgo. And with the flags, it can have tailored experience. I've added a basic test to ensure that the cases we're interested in are covered. This does not yet integrate the usage of Vis()/Unviz() into the manifest create and compare. Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
This commit is contained in:
		
							parent
							
								
									cc939615c7
								
							
						
					
					
						commit
						a63f83d94d
					
				
					 6 changed files with 649 additions and 0 deletions
				
			
		
							
								
								
									
										293
									
								
								unvis.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										293
									
								
								unvis.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,293 @@ | ||||||
|  | /*-
 | ||||||
|  |  * Copyright (c) 1989, 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. | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #if defined(LIBC_SCCS) && !defined(lint) | ||||||
|  | static char sccsid[] = "@(#)unvis.c     8.1 (Berkeley) 6/4/93"; | ||||||
|  | #endif /* LIBC_SCCS and not lint */ | ||||||
|  | 
 | ||||||
|  | #include <sys/cdefs.h> | ||||||
|  | 
 | ||||||
|  | #include <sys/types.h> | ||||||
|  | #include <ctype.h> | ||||||
|  | #include "vis.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * decode driven by state machine | ||||||
|  |  */ | ||||||
|  | #define S_GROUND        0       /* haven't seen escape char */ | ||||||
|  | #define S_START         1       /* start decoding special sequence */ | ||||||
|  | #define S_META          2       /* metachar started (M) */ | ||||||
|  | #define S_META1         3       /* metachar more, regular char (-) */ | ||||||
|  | #define S_CTRL          4       /* control char started (^) */ | ||||||
|  | #define S_OCTAL2        5       /* octal digit 2 */ | ||||||
|  | #define S_OCTAL3        6       /* octal digit 3 */ | ||||||
|  | #define S_HEX2          7       /* hex digit 2 */ | ||||||
|  | 
 | ||||||
|  | #define S_HTTP          0x080   /* %HEXHEX escape */ | ||||||
|  | 
 | ||||||
|  | #define isoctal(c)      (((u_char)(c)) >= '0' && ((u_char)(c)) <= '7') | ||||||
|  | #define ishex(c)        ((((u_char)(c)) >= '0' && ((u_char)(c)) <= '9') || (((u_char)(c)) >= 'a' && ((u_char)(c)) <= 'f')) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * unvis - decode characters previously encoded by vis | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | unvis(char *cp, int c, int *astate, int flag) | ||||||
|  | { | ||||||
|  | 
 | ||||||
|  |         if (flag & UNVIS_END) { | ||||||
|  |                 if (*astate == S_OCTAL2 || *astate == S_OCTAL3) { | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 } | ||||||
|  |                 return (*astate == S_GROUND ? UNVIS_NOCHAR : UNVIS_SYNBAD); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         switch (*astate & ~S_HTTP) { | ||||||
|  | 
 | ||||||
|  |         case S_GROUND: | ||||||
|  |                 *cp = 0; | ||||||
|  |                 if (c == '\\') { | ||||||
|  |                         *astate = S_START; | ||||||
|  |                         return (0); | ||||||
|  |                 } | ||||||
|  |                 if (flag & VIS_HTTPSTYLE && c == '%') { | ||||||
|  |                         *astate = S_START | S_HTTP; | ||||||
|  |                         return (0); | ||||||
|  |                 } | ||||||
|  |                 *cp = c; | ||||||
|  |                 return (UNVIS_VALID); | ||||||
|  | 
 | ||||||
|  |         case S_START: | ||||||
|  |                 if (*astate & S_HTTP) { | ||||||
|  |                     if (ishex(tolower(c))) { | ||||||
|  |                         *cp = isdigit(c) ? (c - '0') : (tolower(c) - 'a'); | ||||||
|  |                         *astate = S_HEX2; | ||||||
|  |                         return (0); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 switch(c) { | ||||||
|  |                 case '\\': | ||||||
|  |                         *cp = c; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case '0': case '1': case '2': case '3': | ||||||
|  |                 case '4': case '5': case '6': case '7': | ||||||
|  |                         *cp = (c - '0'); | ||||||
|  |                         *astate = S_OCTAL2; | ||||||
|  |                         return (0); | ||||||
|  |                 case 'M': | ||||||
|  |                         *cp = 0200; | ||||||
|  |                         *astate = S_META; | ||||||
|  |                         return (0); | ||||||
|  |                 case '^': | ||||||
|  |                         *astate = S_CTRL; | ||||||
|  |                         return (0); | ||||||
|  |                 case 'n': | ||||||
|  |                         *cp = '\n'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 'r': | ||||||
|  |                         *cp = '\r'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 'b': | ||||||
|  |                         *cp = '\b'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 'a': | ||||||
|  |                         *cp = '\007'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 'v': | ||||||
|  |                         *cp = '\v'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 't': | ||||||
|  |                         *cp = '\t'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 'f': | ||||||
|  |                         *cp = '\f'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 's': | ||||||
|  |                         *cp = ' '; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case 'E': | ||||||
|  |                         *cp = '\033'; | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 case '\n': | ||||||
|  |                         /*
 | ||||||
|  |                          * hidden newline | ||||||
|  |                          */ | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_NOCHAR); | ||||||
|  |                 case '$': | ||||||
|  |                         /*
 | ||||||
|  |                          * hidden marker | ||||||
|  |                          */ | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_NOCHAR); | ||||||
|  |                 } | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 return (UNVIS_SYNBAD); | ||||||
|  | 
 | ||||||
|  |         case S_META: | ||||||
|  |                 if (c == '-') | ||||||
|  |                         *astate = S_META1; | ||||||
|  |                 else if (c == '^') | ||||||
|  |                         *astate = S_CTRL; | ||||||
|  |                 else { | ||||||
|  |                         *astate = S_GROUND; | ||||||
|  |                         return (UNVIS_SYNBAD); | ||||||
|  |                 } | ||||||
|  |                 return (0); | ||||||
|  | 
 | ||||||
|  |         case S_META1: | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 *cp |= c; | ||||||
|  |                 return (UNVIS_VALID); | ||||||
|  | 
 | ||||||
|  |         case S_CTRL: | ||||||
|  |                 if (c == '?') | ||||||
|  |                         *cp |= 0177; | ||||||
|  |                 else | ||||||
|  |                         *cp |= c & 037; | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 return (UNVIS_VALID); | ||||||
|  | 
 | ||||||
|  |         case S_OCTAL2:  /* second possible octal digit */ | ||||||
|  |                 if (isoctal(c)) { | ||||||
|  |                         /*
 | ||||||
|  |                          * yes - and maybe a third | ||||||
|  |                          */ | ||||||
|  |                         *cp = (*cp << 3) + (c - '0'); | ||||||
|  |                         *astate = S_OCTAL3; | ||||||
|  |                         return (0); | ||||||
|  |                 } | ||||||
|  |                 /*
 | ||||||
|  |                  * no - done with current sequence, push back passed char | ||||||
|  |                  */ | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 return (UNVIS_VALIDPUSH); | ||||||
|  | 
 | ||||||
|  |         case S_OCTAL3:  /* third possible octal digit */ | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 if (isoctal(c)) { | ||||||
|  |                         *cp = (*cp << 3) + (c - '0'); | ||||||
|  |                         return (UNVIS_VALID); | ||||||
|  |                 } | ||||||
|  |                 /*
 | ||||||
|  |                  * we were done, push back passed char | ||||||
|  |                  */ | ||||||
|  |                 return (UNVIS_VALIDPUSH); | ||||||
|  | 
 | ||||||
|  |         case S_HEX2:    /* second mandatory hex digit */ | ||||||
|  |                 if (ishex(tolower(c))) { | ||||||
|  |                         *cp = (isdigit(c) ? (*cp << 4) + (c - '0') : (*cp << 4) + (tolower(c) - 'a' + 10)); | ||||||
|  |                 } | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 return (UNVIS_VALID); | ||||||
|  | 
 | ||||||
|  |         default: | ||||||
|  |                 /*
 | ||||||
|  |                  * decoder in unknown state - (probably uninitialized) | ||||||
|  |                  */ | ||||||
|  |                 *astate = S_GROUND; | ||||||
|  |                 return (UNVIS_SYNBAD); | ||||||
|  |         } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * strunvis - decode src into dst | ||||||
|  |  * | ||||||
|  |  *      Number of chars decoded into dst is returned, -1 on error. | ||||||
|  |  *      Dst is null terminated. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | strunvis(char *dst, const char *src) | ||||||
|  | { | ||||||
|  |         char c; | ||||||
|  |         char *start = dst; | ||||||
|  |         int state = 0; | ||||||
|  | 
 | ||||||
|  |         while ( (c = *src++) ) { | ||||||
|  |         again: | ||||||
|  |                 switch (unvis(dst, c, &state, 0)) { | ||||||
|  |                 case UNVIS_VALID: | ||||||
|  |                         dst++; | ||||||
|  |                         break; | ||||||
|  |                 case UNVIS_VALIDPUSH: | ||||||
|  |                         dst++; | ||||||
|  |                         goto again; | ||||||
|  |                 case 0: | ||||||
|  |                 case UNVIS_NOCHAR: | ||||||
|  |                         break; | ||||||
|  |                 default: | ||||||
|  |                         return (-1); | ||||||
|  |                 } | ||||||
|  |         } | ||||||
|  |         if (unvis(dst, c, &state, UNVIS_END) == UNVIS_VALID) | ||||||
|  |                 dst++; | ||||||
|  |         *dst = '\0'; | ||||||
|  |         return (dst - start); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | strunvisx(char *dst, const char *src, int flag) | ||||||
|  | { | ||||||
|  |         char c; | ||||||
|  |         char *start = dst; | ||||||
|  |         int state = 0; | ||||||
|  |      | ||||||
|  |         while ( (c = *src++) ) { | ||||||
|  |         again: | ||||||
|  |                 switch (unvis(dst, c, &state, flag)) { | ||||||
|  |                 case UNVIS_VALID: | ||||||
|  |                         dst++; | ||||||
|  |                         break; | ||||||
|  |                 case UNVIS_VALIDPUSH: | ||||||
|  |                         dst++; | ||||||
|  |                         goto again; | ||||||
|  |                 case 0: | ||||||
|  |                 case UNVIS_NOCHAR: | ||||||
|  |                         break; | ||||||
|  |                 default: | ||||||
|  |                         return (-1); | ||||||
|  |                 } | ||||||
|  |         } | ||||||
|  |         if (unvis(dst, c, &state, UNVIS_END) == UNVIS_VALID) | ||||||
|  |                 dst++; | ||||||
|  |         *dst = '\0'; | ||||||
|  |         return (dst - start); | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								unvis.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								unvis.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,15 @@ | ||||||
|  | package mtree | ||||||
|  | 
 | ||||||
|  | // #include "vis.h" | ||||||
|  | import "C" | ||||||
|  | import "fmt" | ||||||
|  | 
 | ||||||
|  | func Unvis(str string) (string, error) { | ||||||
|  | 	dst := new(C.char) | ||||||
|  | 	ret := C.strunvis(dst, C.CString(str)) | ||||||
|  | 	if ret == 0 { | ||||||
|  | 		return "", fmt.Errorf("failed to encode string") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return C.GoString(dst), nil | ||||||
|  | } | ||||||
							
								
								
									
										202
									
								
								vis.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								vis.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,202 @@ | ||||||
|  | /*-
 | ||||||
|  |  * Copyright (c) 1989, 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. | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #if defined(LIBC_SCCS) && !defined(lint) | ||||||
|  | static char sccsid[] = "@(#)vis.c       8.1 (Berkeley) 7/19/93"; | ||||||
|  | #endif /* LIBC_SCCS and not lint */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #include <sys/cdefs.h> | ||||||
|  | 
 | ||||||
|  | #include <sys/types.h> | ||||||
|  | #include <limits.h> | ||||||
|  | #include <ctype.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include "vis.h" | ||||||
|  | 
 | ||||||
|  | #define isoctal(c)      (((u_char)(c)) >= '0' && ((u_char)(c)) <= '7') | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * vis - visually encode characters | ||||||
|  |  */ | ||||||
|  | char * | ||||||
|  | vis(dst, c, flag, nextc) | ||||||
|  |         char *dst; | ||||||
|  |         int c, nextc; | ||||||
|  |         int flag; | ||||||
|  | { | ||||||
|  |         c = (unsigned char)c; | ||||||
|  | 
 | ||||||
|  |         if (flag & VIS_HTTPSTYLE) { | ||||||
|  |                 /* Described in RFC 1808 */ | ||||||
|  |                 if (!(isalnum(c) /* alpha-numeric */ | ||||||
|  |                     /* safe */ | ||||||
|  |                     || c == '$' || c == '-' || c == '_' || c == '.' || c == '+' | ||||||
|  |                     /* extra */ | ||||||
|  |                     || c == '!' || c == '*' || c == '\'' || c == '(' | ||||||
|  |                     || c == ')' || c == ',')) { | ||||||
|  |                         *dst++ = '%'; | ||||||
|  |                         snprintf(dst, 4, (c < 16 ? "0%X" : "%X"), c); | ||||||
|  |                         dst += 2; | ||||||
|  |                         goto done; | ||||||
|  |                 } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if ((flag & VIS_GLOB) && | ||||||
|  |             (c == '*' || c == '?' || c == '[' || c == '#')) | ||||||
|  |                 ; | ||||||
|  |         else if (isgraph(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'))) { | ||||||
|  |                 *dst++ = c; | ||||||
|  |                 if (c == '\\' && (flag & VIS_NOSLASH) == 0) | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                 *dst = '\0'; | ||||||
|  |                 return (dst); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (flag & VIS_CSTYLE) { | ||||||
|  |                 switch(c) { | ||||||
|  |                 case '\n': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 'n'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\r': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 'r'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\b': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 'b'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\a': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 'a'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\v': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 'v'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\t': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 't'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\f': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 'f'; | ||||||
|  |                         goto done; | ||||||
|  |                 case ' ': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = 's'; | ||||||
|  |                         goto done; | ||||||
|  |                 case '\0': | ||||||
|  |                         *dst++ = '\\'; | ||||||
|  |                         *dst++ = '0'; | ||||||
|  |                         if (isoctal(nextc)) { | ||||||
|  |                                 *dst++ = '0'; | ||||||
|  |                                 *dst++ = '0'; | ||||||
|  |                         } | ||||||
|  |                         goto done; | ||||||
|  |                 } | ||||||
|  |         } | ||||||
|  |         if (((c & 0177) == ' ') || isgraph(c) || (flag & VIS_OCTAL)) { | ||||||
|  |                 *dst++ = '\\'; | ||||||
|  |                 *dst++ = ((u_char)c >> 6 & 07) + '0'; | ||||||
|  |                 *dst++ = ((u_char)c >> 3 & 07) + '0'; | ||||||
|  |                 *dst++ = ((u_char)c & 07) + '0'; | ||||||
|  |                 goto done; | ||||||
|  |         } | ||||||
|  |         if ((flag & VIS_NOSLASH) == 0) | ||||||
|  |                 *dst++ = '\\'; | ||||||
|  |         if (c & 0200) { | ||||||
|  |                 c &= 0177; | ||||||
|  |                 *dst++ = 'M'; | ||||||
|  |         } | ||||||
|  |         if (iscntrl(c)) { | ||||||
|  |                 *dst++ = '^'; | ||||||
|  |                 if (c == 0177) | ||||||
|  |                         *dst++ = '?'; | ||||||
|  |                 else | ||||||
|  |                         *dst++ = c + '@'; | ||||||
|  |         } else { | ||||||
|  |                 *dst++ = '-'; | ||||||
|  |                 *dst++ = c; | ||||||
|  |         } | ||||||
|  | done: | ||||||
|  |         *dst = '\0'; | ||||||
|  |         return (dst); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * strvis, 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. | ||||||
|  |  * | ||||||
|  |  *      Strvisx encodes exactly len bytes from src into dst. | ||||||
|  |  *      This is useful for encoding a block of data. | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | strvis(dst, src, flag) | ||||||
|  |         char *dst; | ||||||
|  |         const char *src; | ||||||
|  |         int flag; | ||||||
|  | { | ||||||
|  |         char c; | ||||||
|  |         char *start; | ||||||
|  | 
 | ||||||
|  |         for (start = dst; (c = *src); ) | ||||||
|  |                 dst = vis(dst, c, flag, *++src); | ||||||
|  |         *dst = '\0'; | ||||||
|  |         return (dst - start); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | strvisx(dst, src, len, flag) | ||||||
|  |         char *dst; | ||||||
|  |         const char *src; | ||||||
|  |         size_t len; | ||||||
|  |         int flag; | ||||||
|  | { | ||||||
|  |         int c; | ||||||
|  |         char *start; | ||||||
|  | 
 | ||||||
|  |         for (start = dst; len > 1; len--) { | ||||||
|  |                 c = *src; | ||||||
|  |                 dst = vis(dst, c, flag, *++src); | ||||||
|  |         } | ||||||
|  |         if (len) | ||||||
|  |                 dst = vis(dst, *src, flag, '\0'); | ||||||
|  |         *dst = '\0'; | ||||||
|  | 
 | ||||||
|  |         return (dst - start); | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								vis.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								vis.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,14 @@ | ||||||
|  | package mtree | ||||||
|  | 
 | ||||||
|  | // #include "vis.h" | ||||||
|  | import "C" | ||||||
|  | import "fmt" | ||||||
|  | 
 | ||||||
|  | func Vis(str string) (string, error) { | ||||||
|  | 	dst := new(C.char) | ||||||
|  | 	ret := C.strvis(dst, C.CString(str), C.VIS_WHITE|C.VIS_OCTAL|C.VIS_GLOB) | ||||||
|  | 	if ret == 0 { | ||||||
|  | 		return "", fmt.Errorf("failed to encode string") | ||||||
|  | 	} | ||||||
|  | 	return C.GoString(dst), nil | ||||||
|  | } | ||||||
							
								
								
									
										90
									
								
								vis.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								vis.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,90 @@ | ||||||
|  | /*-
 | ||||||
|  |  * Copyright (c) 1990, 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. | ||||||
|  |  * | ||||||
|  |  *      @(#)vis.h       8.1 (Berkeley) 6/2/93 | ||||||
|  |  * $FreeBSD$ | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _VIS_H_ | ||||||
|  | #define _VIS_H_ | ||||||
|  | 
 | ||||||
|  | #include <sys/types.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * to select alternate encoding format | ||||||
|  |  */ | ||||||
|  | #define VIS_OCTAL       0x01    /* use octal \ddd format */ | ||||||
|  | #define VIS_CSTYLE      0x02    /* use \[nrft0..] where appropriate */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * to alter set of characters encoded (default is to encode all | ||||||
|  |  * non-graphic except space, tab, and newline). | ||||||
|  |  */ | ||||||
|  | #define VIS_SP          0x04    /* also encode space */ | ||||||
|  | #define VIS_TAB         0x08    /* also encode tab */ | ||||||
|  | #define VIS_NL          0x10    /* also encode newline */ | ||||||
|  | #define VIS_WHITE       (VIS_SP | VIS_TAB | VIS_NL) | ||||||
|  | #define VIS_SAFE        0x20    /* only encode "unsafe" characters */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * other | ||||||
|  |  */ | ||||||
|  | #define VIS_NOSLASH     0x40    /* inhibit printing '\' */ | ||||||
|  | #define VIS_HTTPSTYLE   0x80    /* http-style escape % HEX HEX */ | ||||||
|  | #define VIS_GLOB        0x100   /* encode glob(3) magics */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * unvis return codes | ||||||
|  |  */ | ||||||
|  | #define UNVIS_VALID      1      /* character valid */ | ||||||
|  | #define UNVIS_VALIDPUSH  2      /* character valid, push back passed char */ | ||||||
|  | #define UNVIS_NOCHAR     3      /* valid sequence, no character produced */ | ||||||
|  | #define UNVIS_SYNBAD    -1      /* unrecognized escape sequence */ | ||||||
|  | #define UNVIS_ERROR     -2      /* decoder in unknown state (unrecoverable) */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * unvis flags | ||||||
|  |  */ | ||||||
|  | #define UNVIS_END       1       /* no more characters */ | ||||||
|  | 
 | ||||||
|  | #include <sys/cdefs.h> | ||||||
|  | 
 | ||||||
|  | __BEGIN_DECLS | ||||||
|  | char    *vis(char *, int, int, int); | ||||||
|  | int     strvis(char *, const char *, int); | ||||||
|  | int     strvisx(char *, const char *, size_t, int); | ||||||
|  | int     strunvis(char *, const char *); | ||||||
|  | int     strunvisx(char *, const char *, int); | ||||||
|  | int     unvis(char *, int, int *, int); | ||||||
|  | __END_DECLS | ||||||
|  | 
 | ||||||
|  | #endif /* !_VIS_H_ */ | ||||||
							
								
								
									
										35
									
								
								vis_test.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								vis_test.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,35 @@ | ||||||
|  | package mtree | ||||||
|  | 
 | ||||||
|  | import "testing" | ||||||
|  | 
 | ||||||
|  | func TestVis(t *testing.T) { | ||||||
|  | 	testset := []struct { | ||||||
|  | 		Src, Dest string | ||||||
|  | 	}{ | ||||||
|  | 		{"[", "\\133"}, | ||||||
|  | 		{" ", "\\040"}, | ||||||
|  | 		{"	", "\\011"}, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for i := range testset { | ||||||
|  | 		got, err := Vis(testset[i].Src) | ||||||
|  | 		if err != nil { | ||||||
|  | 			t.Errorf("working with %q: %s", testset[i].Src, err) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if got != testset[i].Dest { | ||||||
|  | 			t.Errorf("expected %#v; got %#v", testset[i].Dest, got) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		got, err = Unvis(got) | ||||||
|  | 		if err != nil { | ||||||
|  | 			t.Errorf("working with %q: %s", testset[i].Src, err) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if got != testset[i].Src { | ||||||
|  | 			t.Errorf("expected %#v; got %#v", testset[i].Src, got) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue