69 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			69 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* vi: set sw=4 ts=4: */
 | 
						|
/*
 | 
						|
 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
 | 
						|
 */
 | 
						|
//kbuild:lib-y += percent_decode.o
 | 
						|
 | 
						|
#include "libbb.h"
 | 
						|
 | 
						|
static unsigned hex_to_bin(unsigned char c)
 | 
						|
{
 | 
						|
	unsigned v;
 | 
						|
 | 
						|
	v = c - '0';
 | 
						|
	if (v <= 9)
 | 
						|
		return v;
 | 
						|
	/* c | 0x20: letters to lower case, non-letters
 | 
						|
	 * to (potentially different) non-letters */
 | 
						|
	v = (unsigned)(c | 0x20) - 'a';
 | 
						|
	if (v <= 5)
 | 
						|
		return v + 10;
 | 
						|
	return ~0;
 | 
						|
/* For testing:
 | 
						|
void t(char c) { printf("'%c'(%u) %u\n", c, c, hex_to_bin(c)); }
 | 
						|
int main() { t(0x10); t(0x20); t('0'); t('9'); t('A'); t('F'); t('a'); t('f');
 | 
						|
t('0'-1); t('9'+1); t('A'-1); t('F'+1); t('a'-1); t('f'+1); return 0; }
 | 
						|
*/
 | 
						|
}
 | 
						|
 | 
						|
char* FAST_FUNC percent_decode_in_place(char *str, int strict)
 | 
						|
{
 | 
						|
	/* note that decoded string is always shorter than original */
 | 
						|
	char *src = str;
 | 
						|
	char *dst = str;
 | 
						|
	char c;
 | 
						|
 | 
						|
	while ((c = *src++) != '\0') {
 | 
						|
		unsigned v;
 | 
						|
 | 
						|
		if (!strict && c == '+') {
 | 
						|
			*dst++ = ' ';
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		if (c != '%') {
 | 
						|
			*dst++ = c;
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		v = hex_to_bin(src[0]);
 | 
						|
		if (v > 15) {
 | 
						|
 bad_hex:
 | 
						|
			if (strict)
 | 
						|
				return NULL;
 | 
						|
			*dst++ = '%';
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		v = (v * 16) | hex_to_bin(src[1]);
 | 
						|
		if (v > 255)
 | 
						|
			goto bad_hex;
 | 
						|
		if (strict && (v == '/' || v == '\0')) {
 | 
						|
			/* caller takes it as indication of invalid
 | 
						|
			 * (dangerous wrt exploits) chars */
 | 
						|
			return str + 1;
 | 
						|
		}
 | 
						|
		*dst++ = v;
 | 
						|
		src += 2;
 | 
						|
	}
 | 
						|
	*dst = '\0';
 | 
						|
	return str;
 | 
						|
}
 |