| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Helpers for formatting and printing strings | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 31 August 2008 James Bottomley | 
					
						
							| 
									
										
										
										
											2013-04-30 15:27:30 -07:00
										 |  |  |  * Copyright (C) 2013, Intel Corporation | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  |  */ | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/math64.h>
 | 
					
						
							| 
									
										
										
										
											2011-11-16 21:29:17 -05:00
										 |  |  | #include <linux/export.h>
 | 
					
						
							| 
									
										
										
										
											2013-04-30 15:27:30 -07:00
										 |  |  | #include <linux/ctype.h>
 | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | #include <linux/string_helpers.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * string_get_size - get the size in the specified units | 
					
						
							|  |  |  |  * @size:	The size to be converted | 
					
						
							|  |  |  |  * @units:	units to use (powers of 1000 or 1024) | 
					
						
							|  |  |  |  * @buf:	buffer to format to | 
					
						
							|  |  |  |  * @len:	length of buffer | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function returns a string formatted to 3 significant figures | 
					
						
							|  |  |  |  * giving the size in the required units.  Returns 0 on success or | 
					
						
							|  |  |  |  * error on failure.  @buf is always zero terminated. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int string_get_size(u64 size, const enum string_size_units units, | 
					
						
							|  |  |  | 		    char *buf, int len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-08-06 16:09:31 -07:00
										 |  |  | 	static const char *const units_10[] = { | 
					
						
							|  |  |  | 		"B", "kB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB", NULL | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	static const char *const units_2[] = { | 
					
						
							|  |  |  | 		"B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB", | 
					
						
							|  |  |  | 		NULL | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	static const char *const *const units_str[] = { | 
					
						
							|  |  |  | 		[STRING_UNITS_10] = units_10, | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | 		[STRING_UNITS_2] = units_2, | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2012-05-29 15:07:32 -07:00
										 |  |  | 	static const unsigned int divisor[] = { | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | 		[STRING_UNITS_10] = 1000, | 
					
						
							|  |  |  | 		[STRING_UNITS_2] = 1024, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	int i, j; | 
					
						
							|  |  |  | 	u64 remainder = 0, sf_cap; | 
					
						
							|  |  |  | 	char tmp[8]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tmp[0] = '\0'; | 
					
						
							| 
									
										
										
										
											2008-10-14 11:34:21 -07:00
										 |  |  | 	i = 0; | 
					
						
							|  |  |  | 	if (size >= divisor[units]) { | 
					
						
							|  |  |  | 		while (size >= divisor[units] && units_str[units][i]) { | 
					
						
							|  |  |  | 			remainder = do_div(size, divisor[units]); | 
					
						
							|  |  |  | 			i++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-14 11:34:21 -07:00
										 |  |  | 		sf_cap = size; | 
					
						
							|  |  |  | 		for (j = 0; sf_cap*10 < 1000; j++) | 
					
						
							|  |  |  | 			sf_cap *= 10; | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-14 11:34:21 -07:00
										 |  |  | 		if (j) { | 
					
						
							|  |  |  | 			remainder *= 1000; | 
					
						
							|  |  |  | 			do_div(remainder, divisor[units]); | 
					
						
							|  |  |  | 			snprintf(tmp, sizeof(tmp), ".%03lld", | 
					
						
							|  |  |  | 				 (unsigned long long)remainder); | 
					
						
							|  |  |  | 			tmp[j+1] = '\0'; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-14 11:34:21 -07:00
										 |  |  | 	snprintf(buf, len, "%lld%s %s", (unsigned long long)size, | 
					
						
							| 
									
										
										
										
											2008-08-31 10:13:54 -05:00
										 |  |  | 		 tmp, units_str[units][i]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL(string_get_size); | 
					
						
							| 
									
										
										
										
											2013-04-30 15:27:30 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static bool unescape_space(char **src, char **dst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *p = *dst, *q = *src; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (*q) { | 
					
						
							|  |  |  | 	case 'n': | 
					
						
							|  |  |  | 		*p = '\n'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 'r': | 
					
						
							|  |  |  | 		*p = '\r'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 't': | 
					
						
							|  |  |  | 		*p = '\t'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 'v': | 
					
						
							|  |  |  | 		*p = '\v'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 'f': | 
					
						
							|  |  |  | 		*p = '\f'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*dst += 1; | 
					
						
							|  |  |  | 	*src += 1; | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool unescape_octal(char **src, char **dst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *p = *dst, *q = *src; | 
					
						
							|  |  |  | 	u8 num; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (isodigit(*q) == 0) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	num = (*q++) & 7; | 
					
						
							|  |  |  | 	while (num < 32 && isodigit(*q) && (q - *src < 3)) { | 
					
						
							|  |  |  | 		num <<= 3; | 
					
						
							|  |  |  | 		num += (*q++) & 7; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*p = num; | 
					
						
							|  |  |  | 	*dst += 1; | 
					
						
							|  |  |  | 	*src = q; | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool unescape_hex(char **src, char **dst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *p = *dst, *q = *src; | 
					
						
							|  |  |  | 	int digit; | 
					
						
							|  |  |  | 	u8 num; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (*q++ != 'x') | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	num = digit = hex_to_bin(*q++); | 
					
						
							|  |  |  | 	if (digit < 0) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	digit = hex_to_bin(*q); | 
					
						
							|  |  |  | 	if (digit >= 0) { | 
					
						
							|  |  |  | 		q++; | 
					
						
							|  |  |  | 		num = (num << 4) | digit; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*p = num; | 
					
						
							|  |  |  | 	*dst += 1; | 
					
						
							|  |  |  | 	*src = q; | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool unescape_special(char **src, char **dst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *p = *dst, *q = *src; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (*q) { | 
					
						
							|  |  |  | 	case '\"': | 
					
						
							|  |  |  | 		*p = '\"'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case '\\': | 
					
						
							|  |  |  | 		*p = '\\'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 'a': | 
					
						
							|  |  |  | 		*p = '\a'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 'e': | 
					
						
							|  |  |  | 		*p = '\e'; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*dst += 1; | 
					
						
							|  |  |  | 	*src += 1; | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int string_unescape(char *src, char *dst, size_t size, unsigned int flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *out = dst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (*src && --size) { | 
					
						
							|  |  |  | 		if (src[0] == '\\' && src[1] != '\0' && size > 1) { | 
					
						
							|  |  |  | 			src++; | 
					
						
							|  |  |  | 			size--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (flags & UNESCAPE_SPACE && | 
					
						
							|  |  |  | 					unescape_space(&src, &out)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (flags & UNESCAPE_OCTAL && | 
					
						
							|  |  |  | 					unescape_octal(&src, &out)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (flags & UNESCAPE_HEX && | 
					
						
							|  |  |  | 					unescape_hex(&src, &out)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (flags & UNESCAPE_SPECIAL && | 
					
						
							|  |  |  | 					unescape_special(&src, &out)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			*out++ = '\\'; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		*out++ = *src++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*out = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return out - dst; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL(string_unescape); |